]> git.lyx.org Git - lyx.git/blob - src/insets/insetcollapsable.C
dc5dd625817f1e11b3cb17024ce067fbd793bde4
[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 "cursor.h"
20 #include "debug.h"
21 #include "dispatchresult.h"
22 #include "LColor.h"
23 #include "lyxlex.h"
24 #include "funcrequest.h"
25 #include "metricsinfo.h"
26 #include "paragraph.h"
27
28 #include "frontends/font_metrics.h"
29 #include "frontends/Painter.h"
30 #include "frontends/LyXView.h"
31
32
33 using lyx::graphics::PreviewLoader;
34
35 using std::endl;
36 using std::string;
37 using std::max;
38 using std::min;
39 using std::ostream;
40
41
42 InsetCollapsable::InsetCollapsable(BufferParams const & bp,
43         CollapseStatus status)
44         : inset(bp), label("Label"), status_(status)
45 {
46         inset.setOwner(this);
47         inset.setAutoBreakRows(true);
48         inset.setDrawFrame(InsetText::ALWAYS);
49         inset.setFrameColor(LColor::collapsableframe);
50         setInsetName("Collapsable");
51         setButtonLabel();
52 }
53
54
55 InsetCollapsable::InsetCollapsable(InsetCollapsable const & in)
56         : UpdatableInset(in), inset(in.inset),
57           labelfont_(in.labelfont_), label(in.label), status_(in.status_)
58 {
59         inset.setOwner(this);
60         setButtonLabel();
61 }
62
63
64 void InsetCollapsable::write(Buffer const & buf, ostream & os) const
65 {
66         os << "status ";
67         switch (status_) {
68         case Open:
69                 os << "open";
70                 break;
71         case Collapsed:
72                 os << "collapsed";
73                 break;
74         case Inlined:
75                 os << "inlined";
76                 break;
77         }
78         os << "\n";
79         inset.text_.write(buf, os);
80 }
81
82
83 void InsetCollapsable::read(Buffer const & buf, LyXLex & lex)
84 {
85         bool token_found = false;
86         if (lex.isOK()) {
87                 lex.next();
88                 string const token = lex.getString();
89                 if (token == "status") {
90                         lex.next();
91                         string const tmp_token = lex.getString();
92
93                         if (tmp_token == "inlined") {
94                                 status_ = Inlined;
95                                 token_found = true;
96                         } else if (tmp_token == "collapsed") {
97                                 status_ = Collapsed;
98                                 token_found = true;
99                         } else if (tmp_token == "open") {
100                                 status_ = Open;
101                                 token_found = true;
102                         } else {
103                                 lyxerr << "InsetCollapsable::read: Missing status!"
104                                        << endl;
105                                 // Take countermeasures
106                                 lex.pushToken(token);
107                         }
108                 } else {
109                         lyxerr << "InsetCollapsable::Read: Missing 'status'-tag!"
110                                    << endl;
111                         // take countermeasures
112                         lex.pushToken(token);
113                 }
114         }
115         inset.read(buf, lex);
116
117         if (!token_found)
118                 status_ = isOpen() ? Open : Collapsed;
119
120         setButtonLabel();
121 }
122
123
124 void InsetCollapsable::dimension_collapsed(Dimension & dim) const
125 {
126         font_metrics::buttonText(label, labelfont_, dim.wid, dim.asc, dim.des);
127 }
128
129
130 int InsetCollapsable::height_collapsed() const
131 {
132         Dimension dim;
133         font_metrics::buttonText(label, labelfont_, dim.wid, dim.asc, dim.des);
134         return dim.asc + dim.des;
135 }
136
137
138 void InsetCollapsable::metrics(MetricsInfo & mi, Dimension & dim) const
139 {
140         if (status_ == Inlined) {
141                 inset.metrics(mi, dim);
142         } else {
143                 dimension_collapsed(dim);
144                 if (status_ == Open) {
145                         Dimension insetdim;
146                         inset.metrics(mi, insetdim);
147                         dim.des += insetdim.height() + TEXT_TO_BOTTOM_OFFSET;
148                         dim.wid = max(dim.wid, insetdim.wid);
149                 }
150         }
151         dim_ = dim;
152 }
153
154
155 void InsetCollapsable::draw_collapsed(PainterInfo & pi, int x, int y) const
156 {
157         pi.pain.buttonText(x, y, label, labelfont_);
158 }
159
160
161 void InsetCollapsable::draw(PainterInfo & pi, int x, int y) const
162 {
163         setPosCache(pi, x, y);
164
165         if (status_ == Inlined) {
166                 inset.draw(pi, x, y);
167         } else {
168                 Dimension dimc;
169                 dimension_collapsed(dimc);
170                 int const aa  = ascent();
171                 button_dim.x1 = x + 0;
172                 button_dim.x2 = x + dimc.width();
173                 button_dim.y1 = y - aa + pi.base.bv->top_y();
174                 button_dim.y2 = y - aa + pi.base.bv->top_y() + dimc.height();
175
176                 draw_collapsed(pi, x, y);
177                 if (status_ == Open) {
178                         if (!owner())
179                                 x += scroll();
180                         inset.draw(pi, x, y - aa + dimc.height() + inset.ascent());
181                 }
182         }
183 }
184
185
186 void InsetCollapsable::drawSelection(PainterInfo & pi, int x, int y) const
187 {
188         inset.drawSelection(pi, x, y);
189 }
190
191
192 InsetOld::EDITABLE InsetCollapsable::editable() const
193 {
194         return status_ != Collapsed ? HIGHLY_EDITABLE : IS_EDITABLE;
195 }
196
197
198 bool InsetCollapsable::descendable() const
199 {
200         return status_ != Collapsed;
201 }
202
203
204 void InsetCollapsable::lfunMouseRelease(LCursor & cur, FuncRequest const & cmd)
205 {
206         if (cmd.button() == mouse_button::button3)
207                 showInsetDialog(&cur.bv());
208
209         switch (status_) {
210
211         case Collapsed:
212                 lyxerr << "InsetCollapsable::lfunMouseRelease 1" << endl;
213                 setStatus(Open);
214                 edit(cur, true);
215                 break;
216
217         case Open: {
218                 FuncRequest cmd1 = cmd;
219 //              cmd1.y -= cur.bv().top_y();
220                 if (hitButton(cmd1)) {
221                         lyxerr << "InsetCollapsable::lfunMouseRelease 2" << endl;
222                         setStatus(Collapsed);
223                         cur.dispatched(FINISHED_RIGHT);
224                         break;
225                 }
226                 lyxerr << "InsetCollapsable::lfunMouseRelease 3" << endl;
227                 inset.dispatch(cur, cmd);
228                 break;
229         }
230
231         case Inlined:
232                 inset.dispatch(cur, cmd);
233                 break;
234         }
235 }
236
237
238 int InsetCollapsable::latex(Buffer const & buf, ostream & os,
239                             OutputParams const & runparams) const
240 {
241         return inset.latex(buf, os, runparams);
242 }
243
244
245 int InsetCollapsable::plaintext(Buffer const & buf, ostream & os,
246                             OutputParams const & runparams) const
247 {
248         return inset.plaintext(buf, os, runparams);
249 }
250
251
252 int InsetCollapsable::linuxdoc(Buffer const & buf, ostream & os,
253                                OutputParams const & runparams) const
254 {
255         return inset.linuxdoc(buf, os, runparams);
256 }
257
258
259 int InsetCollapsable::docbook(Buffer const & buf, ostream & os,
260                               OutputParams const & runparams) const
261 {
262         return inset.docbook(buf, os, runparams);
263 }
264
265
266 bool InsetCollapsable::hitButton(FuncRequest const & cmd) const
267 {
268         return button_dim.contains(cmd.x, cmd.y);
269 }
270
271
272 string const InsetCollapsable::getNewLabel(string const & l) const
273 {
274         string la;
275         pos_type const max_length = 15;
276         pos_type const p_siz = inset.paragraphs().begin()->size();
277         pos_type const n = min(max_length, p_siz);
278         pos_type i = 0;
279         pos_type j = 0;
280         for( ; i < n && j < p_siz; ++j) {
281                 if (inset.paragraphs().begin()->isInset(j))
282                         continue;
283                 la += inset.paragraphs().begin()->getChar(j);
284                 ++i;
285         }
286         if (inset.paragraphs().size() > 1 || (i > 0 && j < p_siz)) {
287                 la += "...";
288         }
289         return la.empty() ? l : la;
290 }
291
292
293 void InsetCollapsable::edit(LCursor & cur, bool left)
294 {
295         //lyxerr << "InsetCollapsable: edit left/right" << endl;
296         cur.push(this);
297         inset.edit(cur, left);
298         open();
299 }
300
301
302 InsetBase * InsetCollapsable::editXY(LCursor & cur, int x, int y)
303 {
304         cur.push(this);
305         //lyxerr << "InsetCollapsable: edit xy" << endl;
306         if (status_ == Collapsed) {
307                 setStatus(Open);
308                 inset.edit(cur, true);
309 #warning look here
310 //we are not calling edit(x,y) because there are no coordinates in the
311 //inset yet. I personally think it's ok. (ab)
312                 return this;
313         }
314
315 //if (y <= yo() + inset.ascent() + button_dim.y2)
316 //      y = yo();
317 //else
318 //      y += inset.ascent() - height_collapsed();
319         return inset.editXY(cur, x, y);
320 }
321
322
323 void InsetCollapsable::priv_dispatch(LCursor & cur, FuncRequest const & cmd)
324 {
325         //lyxerr << "\nInsetCollapsable::priv_dispatch (begin): cmd: " << cmd
326         //      << "  button y: " << button_dim.y2 << endl;
327         switch (cmd.action) {
328                 case LFUN_MOUSE_PRESS:
329                         if (status_ == Inlined)
330                                 inset.dispatch(cur, cmd);
331                         else if (status_ == Open && cmd.y > button_dim.y2)
332                                 inset.dispatch(cur, cmd);
333                         break;
334
335                 case LFUN_MOUSE_MOTION:
336                         if (status_ == Inlined)
337                                 inset.dispatch(cur, cmd);
338                         else if (status_ == Open && cmd.y > button_dim.y2)
339                                 inset.dispatch(cur, cmd);
340                         break;
341
342                 case LFUN_MOUSE_RELEASE:
343                         lfunMouseRelease(cur, cmd);
344                         break;
345
346                 case LFUN_INSET_TOGGLE:
347                         if (inset.text_.toggleInset(cur))
348                                 break;
349                         if (status_ == Open) {
350                                 setStatus(Inlined);
351                                 break;
352                         }
353                         setStatus(Collapsed);
354                         cur.dispatched(FINISHED_RIGHT);
355                         break;
356
357                 default:
358                         inset.dispatch(cur, cmd);
359                         break;
360         }
361 }
362
363
364 void InsetCollapsable::validate(LaTeXFeatures & features) const
365 {
366         inset.validate(features);
367 }
368
369
370 void InsetCollapsable::getCursorPos(CursorSlice const & cur,
371         int & x, int & y) const
372 {
373         inset.getCursorPos(cur, x, y);
374 }
375
376
377 void InsetCollapsable::getLabelList(Buffer const & buffer,
378                                     std::vector<string> & list) const
379 {
380         inset.getLabelList(buffer, list);
381 }
382
383
384 int InsetCollapsable::scroll(bool recursive) const
385 {
386         int sx = UpdatableInset::scroll(false);
387
388         if (recursive)
389                 sx += inset.scroll(false);
390
391         return sx;
392 }
393
394
395 size_t InsetCollapsable::nargs() const
396 {
397         return inset.nargs();
398 }
399
400
401 LyXText * InsetCollapsable::getText(int i) const
402 {
403         return inset.getText(i);
404 }
405
406
407 void InsetCollapsable::open()
408 {
409         if (status_ == Collapsed)   // ...but not inlined
410                 setStatus(Open);
411 }
412
413
414 void InsetCollapsable::close()
415 {
416         setStatus(Collapsed);
417 }
418
419
420 void InsetCollapsable::setLabel(string const & l)
421 {
422         label = l;
423 }
424
425
426 void InsetCollapsable::setStatus(CollapseStatus st)
427 {
428         status_ = st;
429         setButtonLabel();
430 }
431
432
433 void InsetCollapsable::markErased()
434 {
435         inset.markErased();
436 }
437
438
439 void InsetCollapsable::addPreview(PreviewLoader & loader) const
440 {
441         inset.addPreview(loader);
442 }
443
444
445 bool InsetCollapsable::insetAllowed(InsetOld::Code code) const
446 {
447         return inset.insetAllowed(code);
448 }
449
450
451 void InsetCollapsable::setLabelFont(LyXFont & font)
452 {
453         labelfont_ = font;
454 }
455
456
457 void InsetCollapsable::scroll(BufferView & bv, float sx) const
458 {
459         UpdatableInset::scroll(bv, sx);
460 }
461
462
463 void InsetCollapsable::scroll(BufferView & bv, int offset) const
464 {
465         UpdatableInset::scroll(bv, offset);
466 }
467
468
469 Box const & InsetCollapsable::buttonDim() const
470 {
471         return button_dim;
472 }
473
474
475 void InsetCollapsable::setBackgroundColor(LColor_color color)
476 {
477         InsetOld::setBackgroundColor(color);
478         inset.setBackgroundColor(color);
479 }