]> git.lyx.org Git - lyx.git/blob - src/insets/InsetERT.cpp
3a39b96a748a4036aa7dfdaecc30f8f02f6b1755
[lyx.git] / src / insets / InsetERT.cpp
1 /**
2  * \file InsetERT.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Jürgen Vigna
7  * \author Lars Gullik Bjønnes
8  *
9  * Full author contact details are available in file CREDITS.
10  */
11
12 #include <config.h>
13
14 #include "InsetERT.h"
15
16 #include "Buffer.h"
17 #include "BufferParams.h"
18 #include "BufferView.h"
19 #include "Cursor.h"
20 #include "debug.h"
21 #include "DispatchResult.h"
22 #include "FuncRequest.h"
23 #include "FuncStatus.h"
24 #include "gettext.h"
25 #include "Language.h"
26 #include "Color.h"
27 #include "LyXAction.h"
28 #include "Lexer.h"
29 #include "TextClass.h"
30 #include "MetricsInfo.h"
31 #include "ParagraphParameters.h"
32 #include "Paragraph.h"
33
34 #include "frontends/alert.h"
35
36 #include <sstream>
37
38
39 namespace lyx {
40
41 using support::token;
42
43 using std::endl;
44 using std::min;
45
46 using std::auto_ptr;
47 using std::istringstream;
48 using std::ostream;
49 using std::ostringstream;
50 using std::string;
51
52
53 void InsetERT::init()
54 {
55         setButtonLabel();
56         setLabelFont(layout_.labelfont);
57         text_.current_font.setLanguage(latex_language);
58         text_.real_current_font.setLanguage(latex_language);
59 }
60
61
62 InsetERT::InsetERT(BufferParams const & bp, CollapseStatus status)
63         : InsetCollapsable(bp, status)
64 {
65         setLayout(bp);
66         init();
67 }
68
69
70 InsetERT::InsetERT(InsetERT const & in)
71         : InsetCollapsable(in)
72 {
73         init();
74 }
75
76
77 auto_ptr<Inset> InsetERT::doClone() const
78 {
79         return auto_ptr<Inset>(new InsetERT(*this));
80 }
81
82
83 #if 0
84 InsetERT::InsetERT(BufferParams const & bp,
85                    Language const *, string const & contents, CollapseStatus status)
86         : InsetCollapsable(bp, status)
87 {
88         Font font(Font::ALL_INHERIT, latex_language);
89         paragraphs().begin()->insert(0, contents, font);
90
91         // the init has to be after the initialization of the paragraph
92         // because of the label settings (draw_label for ert insets).
93         init();
94 }
95 #endif
96
97
98 InsetERT::~InsetERT()
99 {
100         InsetERTMailer(*this).hideDialog();
101 }
102
103
104 void InsetERT::write(Buffer const & buf, ostream & os) const
105 {
106         os << "ERT" << "\n";
107         InsetCollapsable::write(buf, os);
108 }
109
110
111 void InsetERT::read(Buffer const & buf, Lexer & lex)
112 {
113         InsetCollapsable::read(buf, lex);
114
115         // Force default font
116         // This avoids paragraphs in buffer language that would have a
117         // foreign language after a document langauge change, and it ensures
118         // that all new text in ERT gets the "latex" language, since new text
119         // inherits the language from the last position of the existing text.
120         // As a side effect this makes us also robust against bugs in LyX
121         // that might lead to font changes in ERT in .lyx files.
122         Font font(Font::ALL_INHERIT, latex_language);
123         ParagraphList::iterator par = paragraphs().begin();
124         ParagraphList::iterator const end = paragraphs().end();
125         while (par != end) {
126                 pos_type siz = par->size();
127                 for (pos_type i = 0; i <= siz; ++i) {
128                         par->setFont(i, font);
129                 }
130                 ++par;
131         }
132 }
133
134
135 docstring const InsetERT::editMessage() const
136 {
137         return _("Opened ERT Inset");
138 }
139
140
141 int InsetERT::latex(Buffer const &, odocstream & os,
142                     OutputParams const &) const
143 {
144         ParagraphList::const_iterator par = paragraphs().begin();
145         ParagraphList::const_iterator end = paragraphs().end();
146
147         int lines = 0;
148         while (par != end) {
149                 pos_type siz = par->size();
150                 for (pos_type i = 0; i < siz; ++i) {
151                         // ignore all struck out text
152                         if (par->isDeleted(i))
153                                 continue;
154
155                         os.put(par->getChar(i));
156                 }
157                 ++par;
158                 if (par != end) {
159                         os << "\n";
160                         ++lines;
161                 }
162         }
163
164         return lines;
165 }
166
167
168 int InsetERT::plaintext(Buffer const &, odocstream &,
169                         OutputParams const &) const
170 {
171         return 0; // do not output TeX code
172 }
173
174
175 int InsetERT::docbook(Buffer const &, odocstream & os,
176                       OutputParams const &) const
177 {
178         ParagraphList::const_iterator par = paragraphs().begin();
179         ParagraphList::const_iterator end = paragraphs().end();
180
181         int lines = 0;
182         while (par != end) {
183                 pos_type siz = par->size();
184                 for (pos_type i = 0; i < siz; ++i)
185                         os.put(par->getChar(i));
186                 ++par;
187                 if (par != end) {
188                         os << "\n";
189                         ++lines;
190                 }
191         }
192
193         return lines;
194 }
195
196
197 void InsetERT::doDispatch(Cursor & cur, FuncRequest & cmd)
198 {
199         //lyxerr << "\nInsetERT::doDispatch (begin): cmd: " << cmd << endl;
200         switch (cmd.action) {
201
202         case LFUN_QUOTE_INSERT: {
203                 // We need to bypass the fancy quotes in Text
204                 FuncRequest f(LFUN_SELF_INSERT, "\"");
205                 dispatch(cur, f);
206                 break;
207         }
208         case LFUN_INSET_MODIFY: {
209                 InsetCollapsable::CollapseStatus st;
210                 InsetERTMailer::string2params(to_utf8(cmd.argument()), st);
211                 setStatus(cur, st);
212                 break;
213         }
214         case LFUN_PASTE:
215         case LFUN_CLIPBOARD_PASTE:
216         case LFUN_PRIMARY_SELECTION_PASTE: {
217                 InsetCollapsable::doDispatch(cur, cmd);
218
219                 // Since we can only store plain text, we must reset all
220                 // attributes.
221                 // FIXME: Change only the pasted paragraphs
222
223                 BufferParams const & bp = cur.buffer().params();
224                 Layout_ptr const layout =
225                         bp.getTextClass().defaultLayout();
226                 Font font = layout->font;
227                 // ERT contents has always latex_language
228                 font.setLanguage(latex_language);
229                 ParagraphList::iterator const end = paragraphs().end();
230                 for (ParagraphList::iterator par = paragraphs().begin();
231                      par != end; ++par) {
232                         // in case par had a manual label
233                         par->setBeginOfBody();
234                         pos_type const siz = par->size();
235                         for (pos_type i = 0; i < siz; ++i) {
236                                 par->setFont(i, font);
237                         }
238                         par->params().clear();
239                 }
240                 break;
241         }
242         default:
243                 // Force any new text to latex_language
244                 // FIXME: This should only be necessary in init(), but
245                 // new paragraphs that are created by pressing enter at the
246                 // start of an existing paragraph get the buffer language
247                 // and not latex_language, so we take this brute force
248                 // approach.
249                 text_.current_font.setLanguage(latex_language);
250                 text_.real_current_font.setLanguage(latex_language);
251
252                 InsetCollapsable::doDispatch(cur, cmd);
253                 break;
254         }
255 }
256
257
258 bool InsetERT::getStatus(Cursor & cur, FuncRequest const & cmd,
259         FuncStatus & status) const
260 {
261         switch (cmd.action) {
262                 // suppress these
263                 case LFUN_ACCENT_ACUTE:
264                 case LFUN_ACCENT_BREVE:
265                 case LFUN_ACCENT_CARON:
266                 case LFUN_ACCENT_CEDILLA:
267                 case LFUN_ACCENT_CIRCLE:
268                 case LFUN_ACCENT_CIRCUMFLEX:
269                 case LFUN_ACCENT_DOT:
270                 case LFUN_ACCENT_GRAVE:
271                 case LFUN_ACCENT_HUNGARIAN_UMLAUT:
272                 case LFUN_ACCENT_MACRON:
273                 case LFUN_ACCENT_OGONEK:
274                 case LFUN_ACCENT_SPECIAL_CARON:
275                 case LFUN_ACCENT_TIE:
276                 case LFUN_ACCENT_TILDE:
277                 case LFUN_ACCENT_UMLAUT:
278                 case LFUN_ACCENT_UNDERBAR:
279                 case LFUN_ACCENT_UNDERDOT:
280                 case LFUN_APPENDIX:
281                 case LFUN_BREAK_LINE:
282                 case LFUN_CAPTION_INSERT:
283                 case LFUN_DEPTH_DECREMENT:
284                 case LFUN_DEPTH_INCREMENT:
285                 case LFUN_DOTS_INSERT:
286                 case LFUN_END_OF_SENTENCE_PERIOD_INSERT:
287                 case LFUN_ENVIRONMENT_INSERT:
288                 case LFUN_ERT_INSERT:
289                 case LFUN_FILE_INSERT:
290                 case LFUN_FLOAT_INSERT:
291                 case LFUN_FLOAT_WIDE_INSERT:
292                 case LFUN_WRAP_INSERT:
293                 case LFUN_FONT_BOLD:
294                 case LFUN_FONT_CODE:
295                 case LFUN_FONT_DEFAULT:
296                 case LFUN_FONT_EMPH:
297                 case LFUN_FONT_FREE_APPLY:
298                 case LFUN_FONT_FREE_UPDATE:
299                 case LFUN_FONT_NOUN:
300                 case LFUN_FONT_ROMAN:
301                 case LFUN_FONT_SANS:
302                 case LFUN_FONT_FRAK:
303                 case LFUN_FONT_ITAL:
304                 case LFUN_FONT_SIZE:
305                 case LFUN_FONT_STATE:
306                 case LFUN_FONT_UNDERLINE:
307                 case LFUN_FOOTNOTE_INSERT:
308                 case LFUN_HFILL_INSERT:
309                 case LFUN_HTML_INSERT:
310                 case LFUN_HYPHENATION_POINT_INSERT:
311                 case LFUN_LIGATURE_BREAK_INSERT:
312                 case LFUN_INDEX_INSERT:
313                 case LFUN_INDEX_PRINT:
314                 case LFUN_LABEL_INSERT:
315                 case LFUN_OPTIONAL_INSERT:
316                 case LFUN_BIBITEM_INSERT:
317                 case LFUN_LINE_INSERT:
318                 case LFUN_PAGEBREAK_INSERT:
319                 case LFUN_CLEARPAGE_INSERT:
320                 case LFUN_CLEARDOUBLEPAGE_INSERT:
321                 case LFUN_LANGUAGE:
322                 case LFUN_LAYOUT:
323                 case LFUN_LAYOUT_PARAGRAPH:
324                 case LFUN_LAYOUT_TABULAR:
325                 case LFUN_MARGINALNOTE_INSERT:
326                 case LFUN_MATH_DISPLAY:
327                 case LFUN_MATH_INSERT:
328                 case LFUN_MATH_MATRIX:
329                 case LFUN_MATH_MODE:
330                 case LFUN_MENU_OPEN:
331                 case LFUN_MENU_SEPARATOR_INSERT:
332                 case LFUN_BRANCH_INSERT:
333                 case LFUN_CHARSTYLE_INSERT:
334                 case LFUN_NOTE_INSERT:
335                 case LFUN_BOX_INSERT:
336                 case LFUN_NOTE_NEXT:
337                 case LFUN_PARAGRAPH_SPACING:
338                 case LFUN_LABEL_GOTO:
339                 case LFUN_REFERENCE_NEXT:
340                 case LFUN_SPACE_INSERT:
341                 case LFUN_SERVER_GOTO_FILE_ROW:
342                 case LFUN_SERVER_NOTIFY:
343                 case LFUN_SERVER_SET_XY:
344                 case LFUN_TABULAR_INSERT:
345                 case LFUN_TOC_INSERT:
346                 case LFUN_URL_INSERT:
347                 case LFUN_FLOAT_LIST:
348                 case LFUN_INSET_INSERT:
349                 case LFUN_PARAGRAPH_PARAMS:
350                 case LFUN_PARAGRAPH_PARAMS_APPLY:
351                 case LFUN_PARAGRAPH_UPDATE:
352                 case LFUN_NOMENCL_INSERT:
353                 case LFUN_NOMENCL_PRINT:
354                 case LFUN_NOACTION:
355                         status.enabled(false);
356                         return true;
357
358                 case LFUN_QUOTE_INSERT:
359                 case LFUN_INSET_MODIFY:
360                 case LFUN_PASTE:
361                 case LFUN_CLIPBOARD_PASTE:
362                 case LFUN_PRIMARY_SELECTION_PASTE:
363                         status.enabled(true);
364                         return true;
365
366                 // this one is difficult to get right. As a half-baked
367                 // solution, we consider only the first action of the sequence
368                 case LFUN_COMMAND_SEQUENCE: {
369                         // argument contains ';'-terminated commands
370                         string const firstcmd = token(to_utf8(cmd.argument()), ';', 0);
371                         FuncRequest func(lyxaction.lookupFunc(firstcmd));
372                         func.origin = cmd.origin;
373                         return getStatus(cur, func, status);
374                 }
375
376                 default:
377                         return InsetCollapsable::getStatus(cur, cmd, status);
378         }
379 }
380
381
382 void InsetERT::setButtonLabel()
383 {
384         // FIXME UNICODE
385         setLabel(isOpen() ?  _("ERT") : getNewLabel(_("ERT")));
386 }
387
388
389 bool InsetERT::insetAllowed(Inset::Code /* code */) const
390 {
391         return false;
392 }
393
394
395 bool InsetERT::metrics(MetricsInfo & mi, Dimension & dim) const
396 {
397         Font tmpfont = mi.base.font;
398         getDrawFont(mi.base.font);
399         mi.base.font.realize(tmpfont);
400         InsetCollapsable::metrics(mi, dim);
401         mi.base.font = tmpfont;
402         bool const changed = dim_ != dim;
403         dim_ = dim;
404         return changed;
405 }
406
407
408 void InsetERT::draw(PainterInfo & pi, int x, int y) const
409 {
410         Font tmpfont = pi.base.font;
411         getDrawFont(pi.base.font);
412         pi.base.font.realize(tmpfont);
413         const_cast<InsetERT &>(*this).setButtonLabel();
414         InsetCollapsable::draw(pi, x, y);
415         pi.base.font = tmpfont;
416 }
417
418
419 bool InsetERT::showInsetDialog(BufferView * bv) const
420 {
421         InsetERTMailer(const_cast<InsetERT &>(*this)).showDialog(bv);
422         return true;
423 }
424
425
426 void InsetERT::getDrawFont(Font & font) const
427 {
428         font = Font(Font::ALL_INHERIT, latex_language);
429         font.realize(layout_.font);
430 }
431
432
433 string const InsetERTMailer::name_("ert");
434
435 InsetERTMailer::InsetERTMailer(InsetERT & inset)
436         : inset_(inset)
437 {}
438
439
440 string const InsetERTMailer::inset2string(Buffer const &) const
441 {
442         return params2string(inset_.status());
443 }
444
445
446 void InsetERTMailer::string2params(string const & in,
447                                    InsetCollapsable::CollapseStatus & status)
448 {
449         status = InsetCollapsable::Collapsed;
450         if (in.empty())
451                 return;
452
453         istringstream data(in);
454         Lexer lex(0,0);
455         lex.setStream(data);
456
457         string name;
458         lex >> name;
459         if (name != name_)
460                 return print_mailer_error("InsetERTMailer", in, 1, name_);
461
462         int s;
463         lex >> s;
464         if (lex)
465                 status = static_cast<InsetCollapsable::CollapseStatus>(s);
466 }
467
468
469 string const
470 InsetERTMailer::params2string(InsetCollapsable::CollapseStatus status)
471 {
472         ostringstream data;
473         data << name_ << ' ' << status;
474         return data.str();
475 }
476
477
478 } // namespace lyx