]> git.lyx.org Git - features.git/blob - src/insets/InsetFlex.cpp
XHTML/DocBook: merge code duplicates for vertical alignment.
[features.git] / src / insets / InsetFlex.cpp
1 /**
2  * \file InsetFlex.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Angus Leeming
7  * \author Martin Vermeer
8  * \author Jürgen Spitzmüller
9  *
10  * Full author contact details are available in file CREDITS.
11  */
12
13 #include <config.h>
14
15 #include "InsetFlex.h"
16
17 #include "Buffer.h"
18 #include "BufferParams.h"
19 #include "Cursor.h"
20 #include "FuncRequest.h"
21 #include "FuncStatus.h"
22 #include "Language.h"
23 #include "Lexer.h"
24 #include "ParIterator.h"
25 #include "TextClass.h"
26
27 #include "support/gettext.h"
28 #include "support/lstrings.h"
29
30 #include <ostream>
31
32 using namespace std;
33
34 namespace lyx {
35
36
37 InsetFlex::InsetFlex(Buffer * buf, string const & layoutName)
38         : InsetCollapsible(buf), name_(layoutName)
39 {}
40
41
42 InsetFlex::InsetFlex(InsetFlex const & in)
43         : InsetCollapsible(in), name_(in.name_)
44 {}
45
46
47 // special code for InsetFlex when there is not the explicit Flex:: prefix
48 InsetLayout const & InsetFlex::getLayout() const
49 {
50         if (!buffer_)
51                 return DocumentClass::plainInsetLayout();
52
53         DocumentClass const & dc = buffer().params().documentClass();
54         docstring const dname = from_utf8(name_);
55         if (dc.hasInsetLayout(dname))
56                 return dc.insetLayout(dname);
57         return dc.insetLayout(from_utf8("Flex:" + name_));
58 }
59
60
61 InsetDecoration InsetFlex::decoration() const
62 {
63         InsetDecoration const dec = getLayout().decoration();
64         return dec == InsetDecoration::DEFAULT ? InsetDecoration::CONGLOMERATE : dec;
65 }
66
67
68 void InsetFlex::write(ostream & os) const
69 {
70         os << "Flex ";
71         string name;
72         if (name_.empty())
73                 name = "undefined";
74         else {
75                 InsetLayout const & il = getLayout();
76                 // use il.name(), since this resolves obsoleted InsetLayout names
77                 if (il.name() == "undefined")
78                         // This is the name of the plain_insetlayout_. We assume that the
79                         // name resolution has failed.
80                         name = name_;
81                 else {
82                         name = to_utf8(il.name());
83                         // Remove the "Flex:" prefix, if it is present
84                         if (support::prefixIs(name, "Flex:"))
85                                 name = support::split(name, ':');
86                 }
87         }
88         os << name << "\n";
89         InsetCollapsible::write(os);
90 }
91
92
93 bool InsetFlex::getStatus(Cursor & cur, FuncRequest const & cmd,
94                 FuncStatus & flag) const
95 {
96         switch (cmd.action()) {
97         case LFUN_INSET_SPLIT:
98         case LFUN_INSET_DISSOLVE:
99                 if (!cmd.argument().empty()) {
100                         InsetLayout const & il = getLayout();
101                         InsetLyXType const type =
102                                 translateLyXType(to_utf8(cmd.argument()));
103                         if (il.lyxtype() == type
104                             || (il.name() == DocumentClass::plainInsetLayout().name()
105                                     && type == InsetLyXType::CHARSTYLE)) {
106                                 FuncRequest temp_cmd(cmd.action());
107                                 return InsetCollapsible::getStatus(cur, temp_cmd, flag);
108                         } else
109                                 return false;
110                 }
111                 // fall-through
112         default:
113                 return InsetCollapsible::getStatus(cur, cmd, flag);
114         }
115 }
116
117
118 void InsetFlex::doDispatch(Cursor & cur, FuncRequest & cmd)
119 {
120         switch (cmd.action()) {
121         case LFUN_INSET_SPLIT:
122         case LFUN_INSET_DISSOLVE:
123                 if (!cmd.argument().empty()) {
124                         InsetLayout const & il = getLayout();
125                         InsetLyXType const type =
126                                 translateLyXType(to_utf8(cmd.argument()));
127
128                         if (il.lyxtype() == type
129                             || (il.name() == DocumentClass::plainInsetLayout().name()
130                                     && type == InsetLyXType::CHARSTYLE)) {
131                                 FuncRequest temp_cmd(cmd.action());
132                                 InsetCollapsible::doDispatch(cur, temp_cmd);
133                         } else
134                                 cur.undispatched();
135                         break;
136                 }
137                 // fall-through
138         default:
139                 InsetCollapsible::doDispatch(cur, cmd);
140                 break;
141         }
142 }
143
144
145 void InsetFlex::updateBuffer(ParIterator const & it, UpdateType utype, bool const deleted)
146 {
147         BufferParams const & bp = buffer().masterBuffer()->params();
148         InsetLayout const & il = getLayout();
149         docstring custom_label = translateIfPossible(il.labelstring());
150
151         Counters & cnts = bp.documentClass().counters();
152         docstring const & count = il.counter();
153         bool const have_counter = cnts.hasCounter(count);
154         if (have_counter) {
155                 Paragraph const & par = it.paragraph();
156                 if (!par.isDeleted(it.pos())) {
157                         cnts.step(count, utype);
158                         custom_label += ' ' +
159                                 cnts.theCounter(count, it.paragraph().getParLanguage(bp)->code());
160                 } else
161                         custom_label += ' ' + from_ascii("#");
162         }
163         setLabel(custom_label);
164
165         bool const save_counter = have_counter && utype == OutputUpdate;
166         if (save_counter) {
167                 // we assume the counter is local to this inset
168                 // if this turns out to be wrong in some case, we will
169                 // need a layout flag
170                 cnts.saveLastCounter();
171         }
172         InsetCollapsible::updateBuffer(it, utype, deleted);
173         if (save_counter)
174                 cnts.restoreLastCounter();
175 }
176
177
178 } // namespace lyx