mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-12-17 02:05:48 +00:00
ad16347b08
In case one cannot rely on QFileSystemWatcher. see https://www.mail-archive.com/lyx-devel@lists.lyx.org/msg199112.html
224 lines
4.9 KiB
C++
224 lines
4.9 KiB
C++
/**
|
|
* \file RenderGraphic.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Angus Leeming
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "RenderGraphic.h"
|
|
|
|
#include "insets/Inset.h"
|
|
|
|
#include "Buffer.h"
|
|
#include "LyX.h"
|
|
#include "LyXRC.h"
|
|
#include "MetricsInfo.h"
|
|
|
|
#include "frontends/FontMetrics.h"
|
|
#include "frontends/Painter.h"
|
|
|
|
#include "graphics/GraphicsImage.h"
|
|
|
|
#include "support/FileName.h"
|
|
#include "support/filetools.h"
|
|
#include "support/gettext.h"
|
|
|
|
#include "support/bind.h"
|
|
|
|
using namespace std;
|
|
|
|
namespace lyx {
|
|
|
|
|
|
RenderGraphic::RenderGraphic(Inset const * inset)
|
|
: loader_(inset->buffer().fileName())
|
|
{
|
|
loader_.connect(bind(&Inset::updateFrontend, inset));
|
|
}
|
|
|
|
|
|
RenderGraphic::RenderGraphic(RenderGraphic const & other, Inset const * inset)
|
|
: RenderBase(other), loader_(other.loader_), params_(other.params_)
|
|
{
|
|
loader_.connect(bind(&Inset::updateFrontend, inset));
|
|
}
|
|
|
|
|
|
RenderBase * RenderGraphic::clone(Inset const * inset) const
|
|
{
|
|
return new RenderGraphic(*this, inset);
|
|
}
|
|
|
|
void RenderGraphic::reload() const
|
|
{
|
|
loader_.reload();
|
|
}
|
|
|
|
void RenderGraphic::update(graphics::Params const & params)
|
|
{
|
|
params_ = params;
|
|
|
|
if (!params_.filename.empty())
|
|
loader_.reset(params_.filename, params_);
|
|
}
|
|
|
|
|
|
namespace {
|
|
|
|
bool displayGraphic(graphics::Params const & params)
|
|
{
|
|
return params.display && lyxrc.display_graphics;
|
|
}
|
|
|
|
|
|
docstring const statusMessage(graphics::Params const & params,
|
|
graphics::ImageStatus status)
|
|
{
|
|
docstring ret;
|
|
|
|
if (!displayGraphic(params))
|
|
ret = _("Not shown.");
|
|
else {
|
|
switch (status) {
|
|
case graphics::WaitingToLoad:
|
|
ret = _("Not shown.");
|
|
break;
|
|
case graphics::Loading:
|
|
ret = _("Loading...");
|
|
break;
|
|
case graphics::Converting:
|
|
ret = _("Converting to loadable format...");
|
|
break;
|
|
case graphics::Loaded:
|
|
ret = _("Loaded into memory. Generating pixmap...");
|
|
break;
|
|
case graphics::ScalingEtc:
|
|
ret = _("Scaling etc...");
|
|
break;
|
|
case graphics::Ready:
|
|
ret = _("Ready to display");
|
|
break;
|
|
case graphics::ErrorNoFile:
|
|
ret = _("No file found!");
|
|
break;
|
|
case graphics::ErrorConverting:
|
|
ret = _("Error converting to loadable format");
|
|
break;
|
|
case graphics::ErrorLoading:
|
|
ret = _("Error loading file into memory");
|
|
break;
|
|
case graphics::ErrorGeneratingPixmap:
|
|
ret = _("Error generating the pixmap");
|
|
break;
|
|
case graphics::ErrorUnknown:
|
|
ret = _("No image");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
bool readyToDisplay(graphics::Loader const & loader)
|
|
{
|
|
if (!loader.image() || loader.status() != graphics::Ready)
|
|
return false;
|
|
return loader.image()->isDrawable();
|
|
}
|
|
|
|
} // namespace anon
|
|
|
|
|
|
void RenderGraphic::metrics(MetricsInfo & mi, Dimension & dim) const
|
|
{
|
|
if (displayGraphic(params_)) {
|
|
if (loader_.status() == graphics::WaitingToLoad)
|
|
loader_.startLoading();
|
|
if (!loader_.monitoring())
|
|
loader_.startMonitoring();
|
|
loader_.checkModifiedAsync();
|
|
}
|
|
|
|
bool const image_ready = displayGraphic(params_) && readyToDisplay(loader_);
|
|
if (image_ready) {
|
|
dim.wid = loader_.image()->width() + 2 * Inset::TEXT_TO_INSET_OFFSET;
|
|
dim.asc = loader_.image()->height();
|
|
dim_ = dim;
|
|
return;
|
|
}
|
|
|
|
dim.asc = 50;
|
|
dim.des = 0;
|
|
|
|
int font_width = 0;
|
|
|
|
FontInfo msgFont(mi.base.font);
|
|
msgFont.setFamily(SANS_FAMILY);
|
|
|
|
// FIXME UNICODE
|
|
docstring const justname = from_utf8(params_.filename.onlyFileName());
|
|
if (!justname.empty()) {
|
|
msgFont.setSize(FONT_SIZE_FOOTNOTE);
|
|
font_width = theFontMetrics(msgFont).width(justname);
|
|
}
|
|
|
|
docstring const msg = statusMessage(params_, loader_.status());
|
|
if (!msg.empty()) {
|
|
msgFont.setSize(FONT_SIZE_TINY);
|
|
font_width = max(font_width,
|
|
theFontMetrics(msgFont).width(msg));
|
|
}
|
|
|
|
dim.wid = max(50, font_width + 15);
|
|
|
|
dim_ = dim;
|
|
}
|
|
|
|
|
|
void RenderGraphic::draw(PainterInfo & pi, int x, int y) const
|
|
{
|
|
// This will draw the graphics. If the graphics has not been
|
|
// loaded yet, we draw just a rectangle.
|
|
int const x1 = x + Inset::TEXT_TO_INSET_OFFSET;
|
|
int const y1 = y - dim_.asc;
|
|
int const w = dim_.wid - 2 * Inset::TEXT_TO_INSET_OFFSET;
|
|
int const h = dim_.height();
|
|
|
|
if (displayGraphic(params_) && readyToDisplay(loader_))
|
|
pi.pain.image(x1, y1, w, h, *loader_.image());
|
|
|
|
else {
|
|
Color c = pi.change_.changed() ? pi.change_.color() : Color_foreground;
|
|
pi.pain.rectangle(x1, y1, w, h, c);
|
|
|
|
// Print the file name.
|
|
FontInfo msgFont = pi.base.font;
|
|
msgFont.setPaintColor(c);
|
|
msgFont.setFamily(SANS_FAMILY);
|
|
string const justname = params_.filename.onlyFileName();
|
|
|
|
if (!justname.empty()) {
|
|
msgFont.setSize(FONT_SIZE_FOOTNOTE);
|
|
pi.pain.text(x1 + 6, y - theFontMetrics(msgFont).maxAscent() - 4,
|
|
from_utf8(justname), msgFont);
|
|
}
|
|
|
|
// Print the message.
|
|
docstring const msg = statusMessage(params_, loader_.status());
|
|
if (!msg.empty()) {
|
|
msgFont.setSize(FONT_SIZE_TINY);
|
|
pi.pain.text(x1 + 6, y - 4, msg, msgFont);
|
|
}
|
|
}
|
|
pi.change_.paintCue(pi, x1, y1, x1 + w, y1 + h);
|
|
}
|
|
|
|
|
|
} // namespace lyx
|