lyx_mirror/src/insets/RenderGraphic.cpp
Juergen Spitzmueller 148748b0c2 Make InsetInfo math/IPA icons visible in dark mode
This does not show the real colors, but the icons are at least readable.
2020-12-13 09:51:32 +01:00

230 lines
5.2 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)
: inset_(inset), loader_(inset->buffer().fileName())
{
loader_.connect(bind(&Inset::updateFrontend, inset));
}
RenderGraphic::RenderGraphic(RenderGraphic const & other, Inset const * inset)
: RenderBase(other), inset_(inset), 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
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() + inset_->leftOffset(mi.base.bv)
+ inset_->rightOffset(mi.base.bv);
dim.asc = loader_.image()->height();
dim_ = dim;
return;
}
dim.asc = 50;
dim.des = 0;
int font_width = 0;
int font_height = 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(FOOTNOTE_SIZE);
font_width = theFontMetrics(msgFont).width(justname);
font_height = theFontMetrics(msgFont).maxHeight();
}
docstring const msg = statusMessage(params_, loader_.status());
if (!msg.empty()) {
msgFont.setSize(TINY_SIZE);
font_width = max(font_width,
theFontMetrics(msgFont).width(msg));
font_height += theFontMetrics(msgFont).maxAscent();
dim.des = theFontMetrics(msgFont).maxDescent();
}
dim.wid = max(50, font_width + 15);
dim.asc = max(50, font_height + 15);
dim_ = dim;
}
void RenderGraphic::draw(PainterInfo & pi, int x, int y, bool const darkmode) const
{
// This will draw the graphics. If the graphics has not been
// loaded yet, we draw just a rectangle.
int const x1 = x + inset_->leftOffset(pi.base.bv);
int const y1 = y - dim_.asc;
int const w = dim_.wid - inset_->leftOffset(pi.base.bv) - inset_->rightOffset(pi.base.bv);
int const h = dim_.height();
if (displayGraphic(params_) && readyToDisplay(loader_))
pi.pain.image(x1, y1, w, h, *loader_.image(), darkmode);
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(FOOTNOTE_SIZE);
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(TINY_SIZE);
pi.pain.text(x1 + 6, y - 4, msg, msgFont);
}
}
pi.change.paintCue(pi, x1, y1, x1 + w, y1 + h);
}
} // namespace lyx