/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: */ |