Coverage Report

Created: 2026-04-29 07:28

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libetonyek/src/lib/contexts/KEY1DivElement.cpp
Line
Count
Source
1
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2
/*
3
 * This file is part of the libetonyek project.
4
 *
5
 * This Source Code Form is subject to the terms of the Mozilla Public
6
 * License, v. 2.0. If a copy of the MPL was not distributed with this
7
 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8
 */
9
10
#include <boost/spirit/include/qi.hpp>
11
#include <boost/phoenix.hpp>
12
#include <memory>
13
14
#include "KEY1DivElement.h"
15
16
#include "IWORKProperties.h"
17
#include "IWORKText.h"
18
#include "IWORKTokenizer.h"
19
#include "KEY1ParserState.h"
20
#include "KEY1StringConverter.h"
21
#include "KEY1Token.h"
22
#include "KEY1XMLContextBase.h"
23
24
#include "libetonyek_xml.h"
25
26
namespace libetonyek
27
{
28
KEY1DivStyle::KEY1DivStyle(KEY1ParserState &state, IWORKStylePtr_t parentStyle)
29
0
  : m_state(state)
30
0
  , m_style()
31
0
  , m_parentStyle(parentStyle)
32
0
  , m_propMap()
33
0
{
34
0
}
35
36
bool KEY1DivStyle::readAttribute(const int name, const char *const value)
37
0
{
38
0
  switch (name)
39
0
  {
40
0
  case KEY1Token::paragraph_alignment:
41
0
  {
42
0
    switch (m_state.getTokenizer().getId(value))
43
0
    {
44
0
    case KEY1Token::center :
45
0
      m_propMap.put<property::Alignment>(IWORK_ALIGNMENT_CENTER);
46
0
      break;
47
0
    case KEY1Token::justified :
48
0
      m_propMap.put<property::Alignment>(IWORK_ALIGNMENT_JUSTIFY);
49
0
      break;
50
0
    case KEY1Token::left :
51
0
      m_propMap.put<property::Alignment>(IWORK_ALIGNMENT_LEFT);
52
0
      break;
53
0
    case KEY1Token::right :
54
0
      m_propMap.put<property::Alignment>(IWORK_ALIGNMENT_RIGHT);
55
0
      break;
56
0
    default:
57
0
      ETONYEK_DEBUG_MSG(("KEY1DivStyle::readAttribute[KEY1DivElement.cpp]: unknown alignment %s\n", value));
58
0
      break;
59
0
    }
60
0
    break;
61
0
  }
62
0
  case KEY1Token::paragraph_first_line_indent :
63
0
  {
64
0
    boost::optional<double> size=try_double_cast(value);
65
0
    if (size)
66
0
      m_propMap.put<property::FirstLineIndent>(get(size));
67
0
    else
68
0
    {
69
0
      ETONYEK_DEBUG_MSG(("KEY1DivStyle::readAttribute[KEY1DivElement.cpp]: unknown first line indent %s\n", value));
70
0
    }
71
0
    break;
72
0
  }
73
0
  case KEY1Token::paragraph_head_indent :
74
0
  {
75
0
    boost::optional<double> size=try_double_cast(value);
76
0
    if (size)
77
0
      m_propMap.put<property::LeftIndent>(get(size));
78
0
    else
79
0
    {
80
0
      ETONYEK_DEBUG_MSG(("KEY1DivStyle::readAttribute[KEY1DivElement.cpp]: unknown left line indent %s\n", value));
81
0
    }
82
0
    break;
83
0
  }
84
0
  case KEY1Token::paragraph_tail_indent :
85
0
  {
86
0
    boost::optional<double> size=try_double_cast(value);
87
0
    if (size)
88
0
      m_propMap.put<property::RightIndent>(get(size));
89
0
    else
90
0
    {
91
0
      ETONYEK_DEBUG_MSG(("KEY1DivStyle::readAttribute[KEY1DivElement.cpp]: unknown left line indent %s\n", value));
92
0
    }
93
0
    break;
94
0
  }
95
0
  case KEY1Token::tab_stops :
96
0
  {
97
0
    namespace spirit = boost::spirit;
98
0
    namespace qi =  boost::spirit::qi;
99
0
    using spirit::ascii::space;
100
0
    using spirit::qi::double_;
101
0
    using boost::phoenix::construct;
102
103
0
    qi::rule<std::string::const_iterator, IWORKTabStop(), spirit::ascii::space_type> tabStopRule =
104
0
      (qi::lit('C') >> qi::double_)[qi::_val=construct<IWORKTabStop>(IWORK_TABULATION_CENTER,qi::_1)] |
105
0
      (qi::lit('D') >> qi::double_)[qi::_val=construct<IWORKTabStop>(IWORK_TABULATION_DECIMAL,qi::_1)] |
106
0
      (qi::lit('L') >> qi::double_)[qi::_val=construct<IWORKTabStop>(IWORK_TABULATION_LEFT,qi::_1)] |
107
0
      (qi::lit('R') >> qi::double_)[qi::_val=construct<IWORKTabStop>(IWORK_TABULATION_RIGHT,qi::_1)];
108
0
    std::string val(value);
109
0
    std::string::const_iterator it = val.begin(), end = val.end();
110
0
    IWORKTabStops_t tabs;
111
0
    if (spirit::qi::phrase_parse(it, end, tabStopRule >> *(tabStopRule), space, tabs) && it==end)
112
0
      m_propMap.put<property::Tabs>(tabs);
113
0
    else
114
0
    {
115
0
      ETONYEK_DEBUG_MSG(("KEY1DivStyle::readAttribute[KEY1DivElement.cpp]: unknown tabs %s\n", value));
116
0
    }
117
0
    break;
118
0
  }
119
0
  default:
120
0
    return false;
121
0
  }
122
0
  return true;
123
0
}
124
125
IWORKStylePtr_t KEY1DivStyle::getStyle()
126
0
{
127
0
  if (!m_style)
128
0
    m_style = std::make_shared<IWORKStyle>(m_propMap, boost::none, m_parentStyle);
129
0
  return m_style;
130
0
}
131
132
133
KEY1DivElement::KEY1DivElement(KEY1ParserState &state, IWORKStylePtr_t spanStyle, IWORKStylePtr_t parentStyle, bool &delayedLineBreak)
134
0
  : KEY1XMLElementContextBase(state)
135
0
  , m_spanStyle(state, spanStyle)
136
0
  , m_style(state, parentStyle)
137
0
  , m_opened(false)
138
0
  , m_delayedLineBreak(delayedLineBreak)
139
0
{
140
0
}
141
142
void KEY1DivElement::attribute(const int name, const char *const value)
143
0
{
144
0
  if (m_style.readAttribute(name, value) || m_spanStyle.readAttribute(name, value))
145
0
    return;
146
0
  switch (name)
147
0
  {
148
0
  case KEY1Token::id :
149
0
    setId(value);
150
0
    break;
151
0
  default:
152
0
    ETONYEK_DEBUG_MSG(("KEY1DivElement::attribute: unknown attribute with value=%s\n", value));
153
0
  }
154
0
}
155
156
IWORKXMLContextPtr_t KEY1DivElement::element(const int name)
157
0
{
158
0
  switch (name)
159
0
  {
160
0
  case KEY1Token::span | KEY1Token::NS_URI_KEY :
161
0
    return std::make_shared<KEY1SpanElement>(getState(), m_spanStyle.getStyle(), m_delayedLineBreak);
162
0
  default:
163
0
    ETONYEK_DEBUG_MSG(("KEY1DivElement::element: unknown element\n"));
164
0
    break;
165
0
  }
166
0
  return IWORKXMLContextPtr_t();
167
0
}
168
169
void KEY1DivElement::CDATA(const char *value)
170
0
{
171
0
  ensureOpened();
172
0
  KEY1SpanElement::sendCDATA(value, getState().m_currentText, m_spanStyle.getStyle(), m_delayedLineBreak);
173
0
}
174
175
void KEY1DivElement::text(const char *const value)
176
0
{
177
0
  ensureOpened();
178
0
  if (bool(getState().m_currentText))
179
0
  {
180
0
    getState().m_currentText->setSpanStyle(m_spanStyle.getStyle());
181
0
    getState().m_currentText->insertText(value);
182
0
  }
183
0
}
184
185
void KEY1DivElement::endOfElement()
186
0
{
187
0
  m_delayedLineBreak=false;
188
0
  ensureOpened();
189
0
  if (bool(getState().m_currentText))
190
0
    getState().m_currentText->flushParagraph();
191
0
}
192
193
void KEY1DivElement::ensureOpened()
194
0
{
195
0
  if (m_delayedLineBreak)
196
0
  {
197
0
    if (bool(getState().m_currentText))
198
0
      getState().m_currentText->flushParagraph();
199
0
    m_delayedLineBreak=false;
200
0
  }
201
0
  if (!m_opened)
202
0
  {
203
0
    if (bool(getState().m_currentText))
204
0
      getState().m_currentText->setParagraphStyle(m_style.getStyle());
205
0
    m_opened = true;
206
0
  }
207
0
}
208
}
209
210
/* vim:set shiftwidth=2 softtabstop=2 expandtab: */