]> git.lyx.org Git - lyx.git/blob - src/insets/insetcollapsable.C
Final touch 'inset display()'; fix 'is a bit silly' bug
[lyx.git] / src / insets / insetcollapsable.C
1 /**
2  * \file insetcollapsable.C
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Alejandro Aguilar Sierra
7  * \author Jürgen Vigna
8  * \author Lars Gullik Bjønnes
9  *
10  * Full author contact details are available in file CREDITS.
11  */
12
13 #include <config.h>
14
15 #include "insetcollapsable.h"
16
17 #include "buffer.h"
18 #include "BufferView.h"
19 #include "debug.h"
20 #include "LColor.h"
21 #include "lyxlex.h"
22 #include "funcrequest.h"
23 #include "metricsinfo.h"
24 #include "paragraph.h"
25 #include "WordLangTuple.h"
26
27 #include "frontends/font_metrics.h"
28 #include "frontends/Painter.h"
29 #include "frontends/LyXView.h"
30
31
32 using lyx::graphics::PreviewLoader;
33
34 using std::endl;
35 using std::string;
36 using std::max;
37 using std::ostream;
38
39
40 InsetCollapsable::InsetCollapsable(BufferParams const & bp, bool collapsed)
41         : UpdatableInset(), inset(bp), collapsed_(collapsed),
42           button_dim(0, 0, 0, 0), label("Label"),
43 #if 0
44         autocollapse(false),
45 #endif
46           in_update(false), first_after_edit(false)
47 {
48         inset.setOwner(this);
49         inset.setAutoBreakRows(true);
50         inset.setDrawFrame(InsetText::ALWAYS);
51         inset.setFrameColor(LColor::collapsableframe);
52         setInsetName("Collapsable");
53 }
54
55
56 InsetCollapsable::InsetCollapsable(InsetCollapsable const & in)
57         : UpdatableInset(in), inset(in.inset), collapsed_(in.collapsed_),
58           labelfont_(in.labelfont_), button_dim(0, 0, 0, 0), label(in.label),
59 #if 0
60           autocollapse(in.autocollapse),
61 #endif
62           in_update(false), first_after_edit(false)
63 {
64         inset.setOwner(this);
65 }
66
67
68 bool InsetCollapsable::insertInset(BufferView * bv, InsetOld * in)
69 {
70         if (!insetAllowed(in->lyxCode())) {
71                 lyxerr << "InsetCollapsable::InsertInset: "
72                         "Unable to insert inset." << endl;
73                 return false;
74         }
75         return inset.insertInset(bv, in);
76 }
77
78
79 void InsetCollapsable::write(Buffer const & buf, ostream & os) const
80 {
81         os << "collapsed " << (collapsed_ ? "true" : "false") << "\n";
82         inset.writeParagraphData(buf, os);
83 }
84
85
86 void InsetCollapsable::read(Buffer const & buf, LyXLex & lex)
87 {
88         if (lex.isOK()) {
89                 lex.next();
90                 string const token = lex.getString();
91                 if (token == "collapsed") {
92                         lex.next();
93                         collapsed_ = lex.getBool();
94                 } else {
95                         lyxerr << "InsetCollapsable::Read: Missing collapsed!"
96                                << endl;
97                         // Take countermeasures
98                         lex.pushToken(token);
99                 }
100         }
101         inset.read(buf, lex);
102 }
103
104
105 void InsetCollapsable::dimension_collapsed(Dimension & dim) const
106 {
107         font_metrics::buttonText(label, labelfont_, dim.wid, dim.asc, dim.des);
108 }
109
110
111 int InsetCollapsable::height_collapsed() const
112 {
113         Dimension dim;
114         font_metrics::buttonText(label, labelfont_, dim.wid, dim.asc, dim.des);
115         return dim.asc + dim.des;
116 }
117
118
119 void InsetCollapsable::metrics(MetricsInfo & mi, Dimension & dim) const
120 {
121         //lyxerr << "InsetCollapsable::metrics:  width: " << mi.base.textwidth << endl;
122         dimension_collapsed(dim);
123         if (!collapsed_) {
124                 Dimension insetdim;
125                 inset.metrics(mi, insetdim);
126                 dim.des += insetdim.height() + TEXT_TO_BOTTOM_OFFSET;
127                 dim.wid = max(dim.wid, insetdim.wid);
128         }
129         dim_ = dim;
130 }
131
132
133 void InsetCollapsable::draw_collapsed(PainterInfo & pi, int x, int y) const
134 {
135         pi.pain.buttonText(x, y, label, labelfont_);
136 }
137
138
139 void InsetCollapsable::draw(PainterInfo & pi, int x, int y, bool inlined) const
140 {
141         Dimension dim_collapsed;
142         dimension_collapsed(dim_collapsed);
143
144         int const aa  = ascent();
145         button_dim.x1 = 0;
146         button_dim.x2 = dim_collapsed.width();
147         button_dim.y1 = -aa;
148         button_dim.y2 = -aa + dim_collapsed.height();
149
150         if (!isOpen()) {
151                 draw_collapsed(pi, x, y);
152                 return;
153         }
154
155         int old_x = x;
156
157         if (!owner())
158                 x += scroll();
159
160         top_x = x;
161         top_baseline = y;
162
163         int const bl = y - aa + dim_collapsed.ascent();
164
165         if (inlined) {
166                 inset.draw(pi, x, y);
167         } else {
168                 draw_collapsed(pi, old_x, bl);
169                 inset.draw(pi, x, bl + dim_collapsed.descent() + inset.ascent());
170         }
171 }
172
173
174 void InsetCollapsable::draw(PainterInfo & pi, int x, int y) const
175 {
176         // by default, we are not inlined-drawing
177         draw(pi, x, y, false);
178 }
179
180
181 InsetOld::EDITABLE InsetCollapsable::editable() const
182 {
183         return collapsed_ ? IS_EDITABLE : HIGHLY_EDITABLE;
184 }
185
186
187 void InsetCollapsable::insetUnlock(BufferView * bv)
188 {
189 #if 0
190         if (autocollapse) {
191                 if (change_label_with_text) {
192                         draw_label = get_new_label();
193                 } else {
194                         draw_label = label;
195                 }
196                 collapsed_ = true;
197         }
198 #endif
199         inset.insetUnlock(bv);
200         if (scroll())
201                 scroll(bv, 0.0F);
202         bv->updateInset(this);
203 }
204
205
206 FuncRequest InsetCollapsable::adjustCommand(FuncRequest const & cmd)
207 {
208         FuncRequest cmd1 = cmd;
209         cmd1.y = ascent() + cmd.y - (height_collapsed() + inset.ascent());
210         return cmd1;
211 }
212
213
214 void InsetCollapsable::lfunMouseRelease(FuncRequest const & cmd)
215 {
216         bool ret = false;
217         BufferView * bv = cmd.view();
218
219         if (collapsed_ && cmd.button() != mouse_button::button3) {
220                 collapsed_ = false;
221                 bv->updateInset(this);
222                 bv->buffer()->markDirty();
223                 return;
224         }
225
226         if (cmd.button() != mouse_button::button3 && hitButton(cmd)) {
227                 if (collapsed_) {
228                         collapsed_ = false;
229                 } else {
230                         collapsed_ = true;
231                         bv->unlockInset(this);
232                 }
233                 bv->updateInset(this);
234                 bv->buffer()->markDirty();
235         } else if (!collapsed_ && cmd.y > button_dim.y2) {
236                 ret = inset.dispatch(adjustCommand(cmd)) == DISPATCHED;
237         }
238         if (cmd.button() == mouse_button::button3 && !ret)
239                 showInsetDialog(bv);
240 }
241
242
243 int InsetCollapsable::latex(Buffer const & buf, ostream & os,
244                             LatexRunParams const & runparams) const
245 {
246         return inset.latex(buf, os, runparams);
247 }
248
249
250 int InsetCollapsable::ascii(Buffer const & buf, ostream & os, int ll) const
251 {
252         return inset.ascii(buf, os, ll);
253 }
254
255
256 int InsetCollapsable::linuxdoc(Buffer const & buf, ostream & os) const
257 {
258         return inset.linuxdoc(buf, os);
259 }
260
261
262 int InsetCollapsable::docbook(Buffer const & buf, ostream & os, bool mixcont) const
263 {
264         return inset.docbook(buf, os, mixcont);
265 }
266
267
268 bool InsetCollapsable::hitButton(FuncRequest const & cmd) const
269 {
270         return button_dim.contains(cmd.x, cmd.y);
271 }
272
273
274 void InsetCollapsable::edit(BufferView * bv, int index)
275 {
276         lyxerr << "InsetCollapsable: edit" << endl;
277         if (!bv->lockInset(this))
278                 lyxerr << "InsetCollapsable: can't lock index " << index << endl;
279         inset.dispatch(FuncRequest(bv, LFUN_INSET_EDIT, "left"));
280         first_after_edit = true;
281 }
282
283
284 dispatch_result
285 InsetCollapsable::priv_dispatch(FuncRequest const & cmd,
286                                 idx_type & idx, pos_type & pos)
287 {
288         //lyxerr << "InsetCollapsable::localDispatch: "
289         //      << cmd.action << " '" << cmd.argument << "'\n";
290         BufferView * bv = cmd.view();
291         switch (cmd.action) {
292                 case LFUN_INSET_EDIT: {
293                         if (!cmd.argument.empty()) {
294                                 UpdatableInset::priv_dispatch(cmd, idx, pos);
295                                 if (collapsed_) {
296                                         lyxerr << "branch collapsed_" << endl;
297                                         collapsed_ = false;
298                                         if (bv->lockInset(this)) {
299                                                 bv->updateInset(this);
300                                                 bv->buffer()->markDirty();
301                                                 inset.dispatch(cmd);
302                                                 first_after_edit = true;
303                                         }
304                                 } else {
305                                         lyxerr << "branch not collapsed_" << endl;
306                                         if (bv->lockInset(this))
307                                                 inset.dispatch(cmd);
308                                 }
309                                 return DISPATCHED;
310                         }
311
312 #ifdef WITH_WARNINGS
313 #warning Fix this properly in BufferView_pimpl::workAreaButtonRelease
314 #endif
315                         if (cmd.button() == mouse_button::button3)
316                                 return DISPATCHED;
317
318                         UpdatableInset::priv_dispatch(cmd, idx, pos);
319
320                         if (collapsed_) {
321                                 collapsed_ = false;
322                                 // set this only here as it should be recollapsed only if
323                                 // it was already collapsed!
324                                 first_after_edit = true;
325                                 if (!bv->lockInset(this))
326                                         return DISPATCHED;
327                                 bv->updateInset(this);
328                                 bv->buffer()->markDirty();
329                                 inset.dispatch(cmd);
330                         } else {
331                                 if (!bv->lockInset(this))
332                                         return DISPATCHED;
333                                 if (cmd.y <= button_dim.y2) {
334                                         FuncRequest cmd1 = cmd;
335                                         cmd1.y = 0;
336                                         inset.dispatch(cmd1);
337                                 } else
338                                         inset.dispatch(adjustCommand(cmd));
339                         }
340                         return DISPATCHED;
341                 }
342
343                 case LFUN_MOUSE_PRESS:
344                         if (!collapsed_ && cmd.y > button_dim.y2)
345                                 inset.dispatch(adjustCommand(cmd));
346                         return DISPATCHED;
347
348                 case LFUN_MOUSE_MOTION:
349                         if (!collapsed_ && cmd.y > button_dim.y2)
350                                 inset.dispatch(adjustCommand(cmd));
351                         return DISPATCHED;
352
353                 case LFUN_MOUSE_RELEASE:
354                         lfunMouseRelease(cmd);
355                         return DISPATCHED;
356
357                 default:
358                         dispatch_result result = inset.dispatch(cmd);
359                         if (result >= FINISHED)
360                                 bv->unlockInset(this);
361                         first_after_edit = false;
362                         return result;
363         }
364 }
365
366
367 bool InsetCollapsable::lockInsetInInset(BufferView * bv, UpdatableInset * in)
368 {
369         if (&inset == in)
370                 return true;
371         return inset.lockInsetInInset(bv, in);
372 }
373
374
375 bool InsetCollapsable::unlockInsetInInset(BufferView * bv, UpdatableInset * in,
376                                           bool lr)
377 {
378         if (&inset == in) {
379                 bv->unlockInset(this);
380                 return true;
381         }
382         return inset.unlockInsetInInset(bv, in, lr);
383 }
384
385
386 int InsetCollapsable::insetInInsetY() const
387 {
388         return inset.insetInInsetY() - (top_baseline - inset.y());
389 }
390
391
392 void InsetCollapsable::validate(LaTeXFeatures & features) const
393 {
394         inset.validate(features);
395 }
396
397
398 void InsetCollapsable::getCursor(BufferView & bv, int & x, int & y) const
399 {
400         inset.getCursor(bv, x, y);
401 }
402
403
404 void InsetCollapsable::getCursorPos(BufferView * bv, int & x, int & y) const
405 {
406         inset.getCursorPos(bv, x , y);
407 }
408
409
410 UpdatableInset * InsetCollapsable::getLockingInset() const
411 {
412         UpdatableInset * in = inset.getLockingInset();
413         if (&inset == in)
414                 return const_cast<InsetCollapsable *>(this);
415         return in;
416 }
417
418
419 UpdatableInset * InsetCollapsable::getFirstLockingInsetOfType(InsetOld::Code c)
420 {
421         if (c == lyxCode())
422                 return this;
423         return inset.getFirstLockingInsetOfType(c);
424 }
425
426
427 void InsetCollapsable::setFont(BufferView * bv, LyXFont const & font,
428                                bool toggleall, bool selectall)
429 {
430         inset.setFont(bv, font, toggleall, selectall);
431 }
432
433
434 LyXText * InsetCollapsable::getLyXText(BufferView const * bv,
435                                        bool const recursive) const
436 {
437         return inset.getLyXText(bv, recursive);
438 }
439
440
441 void InsetCollapsable::deleteLyXText(BufferView * bv, bool recursive) const
442 {
443         inset.deleteLyXText(bv, recursive);
444 }
445
446
447 void InsetCollapsable::getLabelList(Buffer const & buffer,
448                                     std::vector<string> & list) const
449 {
450         inset.getLabelList(buffer, list);
451 }
452
453
454 int InsetCollapsable::scroll(bool recursive) const
455 {
456         int sx = UpdatableInset::scroll(false);
457
458         if (recursive)
459                 sx += inset.scroll(recursive);
460
461         return sx;
462 }
463
464
465 ParagraphList * InsetCollapsable::getParagraphs(int i) const
466 {
467         return inset.getParagraphs(i);
468 }
469
470
471 LyXText * InsetCollapsable::getText(int i) const
472 {
473         return inset.getText(i);
474 }
475
476
477 LyXCursor const & InsetCollapsable::cursor(BufferView * bv) const
478 {
479         return inset.cursor(bv);
480 }
481
482
483 InsetOld * InsetCollapsable::getInsetFromID(int id_arg) const
484 {
485         if (id_arg == id())
486                 return const_cast<InsetCollapsable *>(this);
487         return inset.getInsetFromID(id_arg);
488 }
489
490
491 void InsetCollapsable::open(BufferView * bv)
492 {
493         if (!collapsed_)
494                 return;
495
496         collapsed_ = false;
497         bv->updateInset(this);
498 }
499
500
501 void InsetCollapsable::close(BufferView * bv) const
502 {
503         if (collapsed_)
504                 return;
505
506         collapsed_ = true;
507         bv->updateInset(this);
508 }
509
510
511 void InsetCollapsable::setLabel(string const & l) const
512 {
513         label = l;
514 }
515
516
517 void InsetCollapsable::setCollapsed(bool c) const
518 {
519         collapsed_ = c;
520 }
521
522
523 void InsetCollapsable::markErased()
524 {
525         inset.markErased();
526 }
527
528
529 bool InsetCollapsable::nextChange(BufferView * bv, lyx::pos_type & length)
530 {
531         bool found = inset.nextChange(bv, length);
532
533         if (first_after_edit && !found)
534                 close(bv);
535         else if (!found)
536                 first_after_edit = false;
537         return found;
538 }
539
540
541 bool InsetCollapsable::searchForward(BufferView * bv, string const & str,
542                                      bool cs, bool mw)
543 {
544         bool found = inset.searchForward(bv, str, cs, mw);
545         if (first_after_edit && !found)
546                 close(bv);
547         else if (!found)
548                 first_after_edit = false;
549         return found;
550 }
551
552
553 bool InsetCollapsable::searchBackward(BufferView * bv, string const & str,
554                                       bool cs, bool mw)
555 {
556         bool found = inset.searchBackward(bv, str, cs, mw);
557         if (first_after_edit && !found)
558                 close(bv);
559         else if (!found)
560                 first_after_edit = false;
561         return found;
562 }
563
564
565 WordLangTuple const
566 InsetCollapsable::selectNextWordToSpellcheck(BufferView * bv, float & value) const
567 {
568         WordLangTuple word = inset.selectNextWordToSpellcheck(bv, value);
569         if (first_after_edit && word.word().empty())
570                 close(bv);
571         first_after_edit = false;
572         return word;
573 }
574
575
576 void InsetCollapsable::addPreview(PreviewLoader & loader) const
577 {
578         inset.addPreview(loader);
579 }