]> git.lyx.org Git - lyx.git/blob - src/insets/RenderPreview.cpp
0b17d2f83c5ff40d9d938ba66642d01df4c4d87a
[lyx.git] / src / insets / RenderPreview.cpp
1 /**
2  * \file RenderPreview.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  *
8  * Full author contact details are available in file CREDITS.
9  */
10
11 #include <config.h>
12
13 #include "insets/RenderPreview.h"
14 #include "insets/Inset.h"
15
16 #include "Buffer.h"
17 #include "BufferView.h"
18 #include "Dimension.h"
19 #include "LyX.h"
20 #include "LyXRC.h"
21 #include "MetricsInfo.h"
22
23 #include "frontends/FontMetrics.h"
24 #include "frontends/Painter.h"
25
26 #include "graphics/PreviewImage.h"
27 #include "graphics/PreviewLoader.h"
28
29 #include "support/FileName.h"
30 #include "support/gettext.h"
31 #include "support/lassert.h"
32 #include "support/lstrings.h"
33
34 using namespace std;
35 using namespace lyx::support;
36
37 namespace lyx {
38
39
40 bool RenderPreview::previewText()
41 {
42         // Use a switch to trigger a warning if the enum is changed.
43         switch(lyxrc.preview) {
44         case LyXRC::PREVIEW_ON:
45         case LyXRC::PREVIEW_NO_MATH:
46                 return true;
47         case LyXRC::PREVIEW_OFF:
48                 break;
49         }
50
51         return false;
52 }
53
54
55 bool RenderPreview::previewMath()
56 {
57         // Use a switch to trigger a warning if the enum is changed.
58         switch(lyxrc.preview) {
59         case LyXRC::PREVIEW_ON:
60                 return true;
61         case LyXRC::PREVIEW_NO_MATH:
62         case LyXRC::PREVIEW_OFF:
63                 break;
64         }
65
66         return false;
67 }
68
69
70 RenderPreview::RenderPreview(Inset const * inset)
71         : parent_(inset)
72 {}
73
74
75 RenderPreview::RenderPreview(RenderPreview const & other,
76                              Inset const * inset)
77         : RenderBase(other),
78           snippet_(other.snippet_),
79           parent_(inset)
80 {}
81
82
83 RenderBase * RenderPreview::clone(Inset const * inset) const
84 {
85         return new RenderPreview(*this, inset);
86 }
87
88
89 namespace {
90
91 docstring const statusMessage(BufferView const * bv, string const & snippet)
92 {
93         LASSERT(bv, return docstring());
94
95         Buffer const & buffer = bv->buffer();
96         graphics::PreviewLoader const * loader = buffer.loader();
97         // please coverity (probably worth the check anyway)
98         if (!loader)
99                 return docstring();
100         graphics::PreviewLoader::Status const status = loader->status(snippet);
101
102         docstring message;
103         switch (status) {
104         case graphics::PreviewLoader::InQueue:
105         case graphics::PreviewLoader::Processing:
106                 message = _("Preview loading");
107                 break;
108         case graphics::PreviewLoader::Ready:
109                 message = _("Preview ready");
110                 break;
111         case graphics::PreviewLoader::NotFound:
112                 message = _("Preview failed");
113                 break;
114         }
115
116         return message;
117 }
118
119 } // namespace
120
121
122 graphics::PreviewImage const *
123 RenderPreview::getPreviewImage(Buffer const & buffer) const
124 {
125         graphics::PreviewLoader const * loader = buffer.loader();
126         LASSERT(loader, return 0);
127         return loader->preview(snippet_);
128 }
129
130
131 void RenderPreview::metrics(MetricsInfo & mi, Dimension & dim) const
132 {
133         LBUFERR(mi.base.bv);
134
135         graphics::PreviewImage const * const pimage =
136                 getPreviewImage(mi.base.bv->buffer());
137
138         if (pimage) {
139                 // If prepared, load the picture before dim calculation. See bug #5627.
140                 pimage->image();
141                 dim = pimage->dim();
142         } else {
143                 dim.asc = 50;
144                 dim.des = 0;
145
146                 FontInfo font(mi.base.font);
147                 font.setFamily(SANS_FAMILY);
148                 font.setSize(FOOTNOTE_SIZE);
149                 docstring const stat = statusMessage(mi.base.bv, snippet_);
150                 dim.wid = 15 + theFontMetrics(font).width(stat);
151         }
152
153         dim_ = dim;
154 }
155
156
157 void RenderPreview::draw(PainterInfo & pi, int x, int y) const
158 {
159         LBUFERR(pi.base.bv);
160
161         graphics::PreviewImage const * const pimage =
162                 getPreviewImage(pi.base.bv->buffer());
163         graphics::Image const * const image = pimage ? pimage->image() : 0;
164
165         if (image) {
166                 pi.pain.image(x, y - dim_.asc, dim_.wid, dim_.height(),
167                               *image);
168         } else {
169                 int const offset = Inset::textOffset(pi.base.bv);
170
171                 pi.pain.rectangle(x + offset,
172                                   y - dim_.asc,
173                                   dim_.wid - 2 * offset,
174                                   dim_.asc + dim_.des,
175                                   Color_foreground);
176
177                 FontInfo font(pi.base.font);
178                 font.setFamily(SANS_FAMILY);
179                 font.setSize(FOOTNOTE_SIZE);
180
181                 docstring const stat = statusMessage(pi.base.bv, snippet_);
182                 pi.pain.text(x + offset + 6,
183                              y - theFontMetrics(font).maxAscent() - 4,
184                              stat, font);
185         }
186         pi.change.paintCue(pi, x, y - dim_.asc,
187                            x + dim_.width(), y - dim_.asc + dim_.height());
188 }
189
190
191 void RenderPreview::startLoading(Buffer const & buffer, bool forexport) const
192 {
193         if (!forexport && (lyxrc.preview == LyXRC::PREVIEW_OFF || snippet_.empty()))
194                 return;
195
196         graphics::PreviewLoader * loader = buffer.loader();
197         LASSERT(loader, return);
198         loader->startLoading(forexport);
199 }
200
201
202 void RenderPreview::addPreview(docstring const & latex_snippet,
203                                Buffer const & buffer,
204                                bool ignore_lyxrc)
205 {
206         if (lyxrc.preview == LyXRC::PREVIEW_OFF && !ignore_lyxrc)
207                 return;
208
209         graphics::PreviewLoader * loader = buffer.loader();
210         LASSERT(loader, return);
211         addPreview(latex_snippet, *loader, ignore_lyxrc);
212 }
213
214
215 void RenderPreview::addPreview(docstring const & latex_snippet,
216                                graphics::PreviewLoader & ploader,
217                                bool ignore_lyxrc)
218 {
219         if (lyxrc.preview == LyXRC::PREVIEW_OFF && !ignore_lyxrc)
220                 return;
221
222         // FIXME UNICODE
223         // We have to make sure that we call latex with the right encoding
224         snippet_ = support::trim(to_utf8(latex_snippet));
225         if (snippet_.empty())
226                 return;
227
228         if (ploader.preview(snippet_))
229                 return;
230
231         // If this is the first time of calling, connect to the
232         // PreviewLoader signal that'll inform us when the preview image
233         // is ready for loading.
234         if (!ploader_connection_.connected())
235                 // This is a scoped connection.
236                 ploader_connection_ =
237                         ploader.connect([this](graphics::PreviewImage const & pi){
238                                 imageReady(pi);
239                         });
240
241         ploader.add(snippet_);
242 }
243
244
245 void RenderPreview::removePreview(Buffer const & buffer)
246 {
247         if (snippet_.empty())
248                 return;
249
250         graphics::PreviewLoader * loader = buffer.loader();
251         LASSERT(loader, return);
252         loader->remove(snippet_);
253         snippet_.erase();
254 }
255
256
257 void RenderPreview::imageReady(graphics::PreviewImage const & pimage)
258 {
259         // Check the current snippet is the same as that previewed.
260         if (snippet_ == pimage.snippet())
261                 parent_->updateFrontend();
262 }
263
264
265 RenderMonitoredPreview::RenderMonitoredPreview(Inset const * inset)
266         : RenderPreview(inset)
267 {
268         setAbsFile(FileName());
269 }
270
271
272 void RenderMonitoredPreview::setAbsFile(FileName const & file)
273 {
274         bool mon = monitoring();
275         if (mon)
276                 stopMonitoring();
277         filename_ = file;
278         if (mon)
279                 startMonitoring();
280 }
281
282
283 void RenderMonitoredPreview::draw(PainterInfo & pi, int x, int y) const
284 {
285         RenderPreview::draw(pi, x, y);
286         startMonitoring();
287         monitor_->checkModifiedAsync();
288 }
289
290
291 signals2::connection RenderMonitoredPreview::connect(slot const & slot)
292 {
293         return changed_.connect(slot);
294 }
295
296
297 bool RenderMonitoredPreview::monitoring() const
298 {
299         return (bool) monitor_;
300 }
301
302
303 void RenderMonitoredPreview::startMonitoring() const
304 {
305         if (!monitoring()) {
306                 monitor_ = FileSystemWatcher::activeMonitor(filename_);
307                 // Disconnected at the same time as this is destroyed.
308                 monitor_->connect([this](bool /* exists */){ changed_(); });
309         }
310 }
311
312
313 void RenderMonitoredPreview::stopMonitoring() const
314 {
315         monitor_ = nullptr;
316 }
317
318
319
320
321 } // namespace lyx