blob: c3ddc5c326b18fb89a7f2e9d85ac27ab1d74e1be [file] [log] [blame] [edit]
//========================================================================
//
// PreScanOutputDev.cc
//
// Copyright 2005 Glyph & Cog, LLC
//
//========================================================================
//========================================================================
//
// Modified under the Poppler project - http://poppler.freedesktop.org
//
// All changes made under the Poppler project to this file are licensed
// under GPL version 2 or later
//
// Copyright (C) 2009 Carlos Garcia Campos <carlosgc@gnome.org>
// Copyright (C) 2010, 2011, 2018-2021 Albert Astals Cid <aacid@kde.org>
// Copyright (C) 2011, 2014 William Bader <williambader@hotmail.com>
// Copyright (C) 2011, 2013 Thomas Freitag <Thomas.Freitag@alfa.de>
// Copyright (C) 2011 Adrian Johnson <ajohnson@redneon.com>
// Copyright (C) 2022 Oliver Sander <oliver.sander@tu-dresden.de>
//
// To see a description of the changes please see the Changelog file that
// came with your tarball or type make ChangeLog if you are building from git
//
//========================================================================
#include <config.h>
#include <cmath>
#include "GlobalParams.h"
#include "Gfx.h"
#include "GfxFont.h"
#include "Link.h"
#include "Catalog.h"
#include "Page.h"
#include "PreScanOutputDev.h"
//------------------------------------------------------------------------
// PreScanOutputDev
//------------------------------------------------------------------------
PreScanOutputDev::PreScanOutputDev(PSLevel levelA) : level(levelA)
{
clearStats();
}
PreScanOutputDev::~PreScanOutputDev() { }
void PreScanOutputDev::startPage(int /*pageNum*/, GfxState * /*state*/, XRef * /*xref*/) { }
void PreScanOutputDev::endPage() { }
void PreScanOutputDev::stroke(GfxState *state)
{
double dashStart;
check(state->getStrokeColorSpace(), state->getStrokeColor(), state->getStrokeOpacity(), state->getBlendMode());
const std::vector<double> &dash = state->getLineDash(&dashStart);
if (dash.size() != 0) {
gdi = false;
}
}
void PreScanOutputDev::fill(GfxState *state)
{
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
}
void PreScanOutputDev::eoFill(GfxState *state)
{
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
}
bool PreScanOutputDev::tilingPatternFill(GfxState *state, Gfx *gfx, Catalog *catalog, GfxTilingPattern *tPat, const double *mat, int x0, int y0, int x1, int y1, double xStep, double yStep)
{
if (tPat->getPaintType() == 1) {
bool tilingNeeded = (x1 - x0 != 1 || y1 - y0 != 1);
if (tilingNeeded) {
inTilingPatternFill++;
}
gfx->drawForm(tPat->getContentStream(), tPat->getResDict(), mat, tPat->getBBox());
if (tilingNeeded) {
inTilingPatternFill--;
}
} else {
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
}
return true;
}
bool PreScanOutputDev::functionShadedFill(GfxState *state, GfxFunctionShading *shading)
{
if (shading->getColorSpace()->getMode() != csDeviceGray && shading->getColorSpace()->getMode() != csCalGray) {
gray = false;
}
mono = false;
if (state->getFillOpacity() != 1 || state->getBlendMode() != gfxBlendNormal) {
transparency = true;
}
return true;
}
bool PreScanOutputDev::axialShadedFill(GfxState *state, GfxAxialShading *shading, double /*tMin*/, double /*tMax*/)
{
if (shading->getColorSpace()->getMode() != csDeviceGray && shading->getColorSpace()->getMode() != csCalGray) {
gray = false;
}
mono = false;
if (state->getFillOpacity() != 1 || state->getBlendMode() != gfxBlendNormal) {
transparency = true;
}
return true;
}
bool PreScanOutputDev::radialShadedFill(GfxState *state, GfxRadialShading *shading, double /*sMin*/, double /*sMax*/)
{
if (shading->getColorSpace()->getMode() != csDeviceGray && shading->getColorSpace()->getMode() != csCalGray) {
gray = false;
}
mono = false;
if (state->getFillOpacity() != 1 || state->getBlendMode() != gfxBlendNormal) {
transparency = true;
}
return true;
}
void PreScanOutputDev::clip(GfxState * /*state*/)
{
//~ check for a rectangle "near" the edge of the page;
//~ else set gdi to false
}
void PreScanOutputDev::eoClip(GfxState * /*state*/)
{
//~ see clip()
}
void PreScanOutputDev::beginStringOp(GfxState *state)
{
int render;
double m11, m12, m21, m22;
bool simpleTTF;
render = state->getRender();
if (!(render & 1)) {
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
}
if ((render & 3) == 1 || (render & 3) == 2) {
check(state->getStrokeColorSpace(), state->getStrokeColor(), state->getStrokeOpacity(), state->getBlendMode());
}
std::shared_ptr<const GfxFont> font = state->getFont();
state->getFontTransMat(&m11, &m12, &m21, &m22);
//~ this should check for external fonts that are non-TrueType
simpleTTF = fabs(m11 + m22) < 0.01 && m11 > 0 && fabs(m12) < 0.01 && fabs(m21) < 0.01 && fabs(state->getHorizScaling() - 1) < 0.001 && (font->getType() == fontTrueType || font->getType() == fontTrueTypeOT);
if (simpleTTF) {
//~ need to create a FoFiTrueType object, and check for a Unicode cmap
}
if (state->getRender() != 0 || !simpleTTF) {
gdi = false;
}
}
void PreScanOutputDev::endStringOp(GfxState * /*state*/) { }
bool PreScanOutputDev::beginType3Char(GfxState * /*state*/, double /*x*/, double /*y*/, double /*dx*/, double /*dy*/, CharCode /*code*/, const Unicode * /*u*/, int /*uLen*/)
{
// return false so all Type 3 chars get rendered (no caching)
return false;
}
void PreScanOutputDev::endType3Char(GfxState * /*state*/) { }
void PreScanOutputDev::drawImageMask(GfxState *state, Object * /*ref*/, Stream *str, int width, int height, bool /*invert*/, bool /*interpolate*/, bool inlineImg)
{
int i, j;
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
gdi = false;
if ((level == psLevel1 || level == psLevel1Sep) && (state->getFillColorSpace()->getMode() == csPattern || inTilingPatternFill > 0)) {
patternImgMask = true;
}
if (inlineImg) {
str->reset();
j = height * ((width + 7) / 8);
for (i = 0; i < j; ++i) {
str->getChar();
}
str->close();
}
}
void PreScanOutputDev::drawImage(GfxState *state, Object * /*ref*/, Stream *str, int width, int height, GfxImageColorMap *colorMap, bool /*interpolate*/, const int * /*maskColors*/, bool inlineImg)
{
GfxColorSpace *colorSpace;
int i, j;
colorSpace = colorMap->getColorSpace();
if (colorSpace->getMode() == csIndexed) {
colorSpace = ((GfxIndexedColorSpace *)colorSpace)->getBase();
}
if (colorSpace->getMode() == csDeviceGray || colorSpace->getMode() == csCalGray) {
if (colorMap->getBits() > 1) {
mono = false;
}
} else {
gray = false;
mono = false;
}
if (state->getFillOpacity() != 1 || state->getBlendMode() != gfxBlendNormal) {
transparency = true;
}
gdi = false;
if ((level == psLevel1 || level == psLevel1Sep) && inTilingPatternFill > 0) {
patternImgMask = true;
}
if (inlineImg) {
str->reset();
j = height * ((width * colorMap->getNumPixelComps() * colorMap->getBits() + 7) / 8);
for (i = 0; i < j; ++i) {
str->getChar();
}
str->close();
}
}
void PreScanOutputDev::drawMaskedImage(GfxState *state, Object * /*ref*/, Stream * /*str*/, int /*width*/, int /*height*/, GfxImageColorMap *colorMap, bool /*interpolate*/, Stream * /*maskStr*/, int /*maskWidth*/, int /*maskHeight*/,
bool /*maskInvert*/, bool /*maskInterpolate*/)
{
GfxColorSpace *colorSpace;
colorSpace = colorMap->getColorSpace();
if (colorSpace->getMode() == csIndexed) {
colorSpace = ((GfxIndexedColorSpace *)colorSpace)->getBase();
}
if (colorSpace->getMode() == csDeviceGray || colorSpace->getMode() == csCalGray) {
if (colorMap->getBits() > 1) {
mono = false;
}
} else {
gray = false;
mono = false;
}
if (state->getFillOpacity() != 1 || state->getBlendMode() != gfxBlendNormal) {
transparency = true;
}
gdi = false;
}
void PreScanOutputDev::drawSoftMaskedImage(GfxState * /*state*/, Object * /*ref*/, Stream * /*str*/, int /*width*/, int /*height*/, GfxImageColorMap *colorMap, bool /*interpolate*/, Stream * /*maskStr*/, int /*maskWidth*/,
int /*maskHeight*/, GfxImageColorMap * /*maskColorMap*/, bool /*maskInterpolate*/)
{
GfxColorSpace *colorSpace;
colorSpace = colorMap->getColorSpace();
if (colorSpace->getMode() == csIndexed) {
colorSpace = ((GfxIndexedColorSpace *)colorSpace)->getBase();
}
if (colorSpace->getMode() != csDeviceGray && colorSpace->getMode() != csCalGray) {
gray = false;
}
mono = false;
transparency = true;
gdi = false;
}
void PreScanOutputDev::beginTransparencyGroup(GfxState * /*state*/, const double * /*bbox*/, GfxColorSpace * /*blendingColorSpace*/, bool /*isolated*/, bool /*knockout*/, bool /*forSoftMask*/)
{
gdi = false;
}
void PreScanOutputDev::paintTransparencyGroup(GfxState *state, const double * /*bbox*/)
{
check(state->getFillColorSpace(), state->getFillColor(), state->getFillOpacity(), state->getBlendMode());
}
void PreScanOutputDev::setSoftMask(GfxState * /*state*/, const double * /*bbox*/, bool /*alpha*/, Function * /*transferFunc*/, GfxColor * /*backdropColor*/)
{
transparency = true;
}
void PreScanOutputDev::check(GfxColorSpace *colorSpace, const GfxColor *color, double opacity, GfxBlendMode blendMode)
{
GfxRGB rgb;
if (colorSpace->getMode() == csPattern) {
mono = false;
gray = false;
gdi = false;
} else {
colorSpace->getRGB(color, &rgb);
if (rgb.r != rgb.g || rgb.g != rgb.b || rgb.b != rgb.r) {
mono = false;
gray = false;
} else if (!((rgb.r == 0 && rgb.g == 0 && rgb.b == 0) || (rgb.r == gfxColorComp1 && rgb.g == gfxColorComp1 && rgb.b == gfxColorComp1))) {
mono = false;
}
}
if (opacity != 1 || blendMode != gfxBlendNormal) {
transparency = true;
}
}
void PreScanOutputDev::clearStats()
{
mono = true;
gray = true;
transparency = false;
gdi = true;
patternImgMask = false;
inTilingPatternFill = 0;
}