1
0
Fork 0
mirror of https://github.com/juce-framework/JUCE.git synced 2026-01-11 23:54:18 +00:00
JUCE/src/gui/graphics/contexts/juce_LowLevelGraphicsSoftwareRenderer.cpp
2010-03-27 18:29:09 +00:00

1658 lines
62 KiB
C++

/*
==============================================================================
This file is part of the JUCE library - "Jules' Utility Class Extensions"
Copyright 2004-9 by Raw Material Software Ltd.
------------------------------------------------------------------------------
JUCE can be redistributed and/or modified under the terms of the GNU General
Public License (Version 2), as published by the Free Software Foundation.
A copy of the license is included in the JUCE distribution, or can be found
online at www.gnu.org/licenses.
JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
------------------------------------------------------------------------------
To release a closed-source product which uses JUCE, commercial licenses are
available: visit www.rawmaterialsoftware.com/juce for more information.
==============================================================================
*/
#include "../../../core/juce_StandardHeader.h"
BEGIN_JUCE_NAMESPACE
#include "juce_LowLevelGraphicsSoftwareRenderer.h"
#include "juce_EdgeTable.h"
#include "../imaging/juce_Image.h"
#include "../colour/juce_PixelFormats.h"
#include "../geometry/juce_PathStrokeType.h"
#include "../geometry/juce_Rectangle.h"
#include "../../../core/juce_SystemStats.h"
#include "../../../core/juce_Singleton.h"
#include "../../../utilities/juce_DeletedAtShutdown.h"
#if (JUCE_WINDOWS || JUCE_LINUX) && ! JUCE_64BIT
#define JUCE_USE_SSE_INSTRUCTIONS 1
#endif
#if JUCE_MSVC && JUCE_DEBUG
#pragma warning (disable: 4714) // warning about forcedinline methods not being inlined
#endif
#if JUCE_MSVC
#pragma warning (push)
#pragma warning (disable: 4127) // "expression is constant" warning
#endif
//==============================================================================
template <class PixelType, bool replaceExisting = false>
class SolidColourEdgeTableRenderer
{
public:
SolidColourEdgeTableRenderer (const Image::BitmapData& data_, const PixelARGB& colour) throw()
: data (data_),
sourceColour (colour)
{
if (sizeof (PixelType) == 3)
{
areRGBComponentsEqual = sourceColour.getRed() == sourceColour.getGreen()
&& sourceColour.getGreen() == sourceColour.getBlue();
filler[0].set (sourceColour);
filler[1].set (sourceColour);
filler[2].set (sourceColour);
filler[3].set (sourceColour);
}
}
forcedinline void setEdgeTableYPos (const int y) throw()
{
linePixels = (PixelType*) data.getLinePointer (y);
}
forcedinline void handleEdgeTablePixel (const int x, const int alphaLevel) const throw()
{
if (replaceExisting)
linePixels[x].set (sourceColour);
else
linePixels[x].blend (sourceColour, alphaLevel);
}
forcedinline void handleEdgeTableLine (const int x, int width, const int alphaLevel) const throw()
{
PixelARGB p (sourceColour);
p.multiplyAlpha (alphaLevel);
PixelType* dest = linePixels + x;
if (replaceExisting || p.getAlpha() >= 0xff)
replaceLine (dest, p, width);
else
blendLine (dest, p, width);
}
private:
const Image::BitmapData& data;
PixelType* linePixels;
PixelARGB sourceColour;
PixelRGB filler [4];
bool areRGBComponentsEqual;
inline void blendLine (PixelType* dest, const PixelARGB& colour, int width) const throw()
{
do
{
dest->blend (colour);
++dest;
} while (--width > 0);
}
forcedinline void replaceLine (PixelRGB* dest, const PixelARGB& colour, int width) const throw()
{
if (areRGBComponentsEqual) // if all the component values are the same, we can cheat..
{
memset (dest, colour.getRed(), width * 3);
}
else
{
if (width >> 5)
{
const int* const intFiller = (const int*) filler;
while (width > 8 && (((pointer_sized_int) dest) & 7) != 0)
{
dest->set (colour);
++dest;
--width;
}
while (width > 4)
{
((int*) dest) [0] = intFiller[0];
((int*) dest) [1] = intFiller[1];
((int*) dest) [2] = intFiller[2];
dest = (PixelRGB*) (((uint8*) dest) + 12);
width -= 4;
}
}
while (--width >= 0)
{
dest->set (colour);
++dest;
}
}
}
forcedinline void replaceLine (PixelAlpha* dest, const PixelARGB& colour, int width) const throw()
{
memset (dest, colour.getAlpha(), width);
}
forcedinline void replaceLine (PixelARGB* dest, const PixelARGB& colour, int width) const throw()
{
do
{
dest->set (colour);
++dest;
} while (--width > 0);
}
SolidColourEdgeTableRenderer (const SolidColourEdgeTableRenderer&);
SolidColourEdgeTableRenderer& operator= (const SolidColourEdgeTableRenderer&);
};
//==============================================================================
class LinearGradientPixelGenerator
{
public:
LinearGradientPixelGenerator (const ColourGradient& gradient, const AffineTransform& transform, const PixelARGB* const lookupTable_, const int numEntries_)
: lookupTable (lookupTable_), numEntries (numEntries_)
{
jassert (numEntries_ >= 0);
float x1 = gradient.x1;
float y1 = gradient.y1;
float x2 = gradient.x2;
float y2 = gradient.y2;
if (! transform.isIdentity())
{
const Line l (x2, y2, x1, y1);
const Point<float> p3 = l.getPointAlongLine (0.0f, 100.0f);
float x3 = p3.getX();
float y3 = p3.getY();
transform.transformPoint (x1, y1);
transform.transformPoint (x2, y2);
transform.transformPoint (x3, y3);
const Line l2 (x2, y2, x3, y3);
const float prop = l2.findNearestPointTo (x1, y1);
const Point<float> newP2 (l2.getPointAlongLineProportionally (prop));
x2 = newP2.getX();
y2 = newP2.getY();
}
vertical = fabs (x1 - x2) < 0.001f;
horizontal = fabs (y1 - y2) < 0.001f;
if (vertical)
{
scale = roundToInt ((numEntries << (int) numScaleBits) / (double) (y2 - y1));
start = roundToInt (y1 * scale);
}
else if (horizontal)
{
scale = roundToInt ((numEntries << (int) numScaleBits) / (double) (x2 - x1));
start = roundToInt (x1 * scale);
}
else
{
grad = (y2 - y1) / (double) (x1 - x2);
yTerm = y1 - x1 / grad;
scale = roundToInt ((numEntries << (int) numScaleBits) / (yTerm * grad - (y2 * grad - x2)));
grad *= scale;
}
}
forcedinline void setY (const int y) throw()
{
if (vertical)
linePix = lookupTable [jlimit (0, numEntries, (y * scale - start) >> (int) numScaleBits)];
else if (! horizontal)
start = roundToInt ((y - yTerm) * grad);
}
inline const PixelARGB getPixel (const int x) const throw()
{
return vertical ? linePix
: lookupTable [jlimit (0, numEntries, (x * scale - start) >> (int) numScaleBits)];
}
private:
const PixelARGB* const lookupTable;
const int numEntries;
PixelARGB linePix;
int start, scale;
double grad, yTerm;
bool vertical, horizontal;
enum { numScaleBits = 12 };
LinearGradientPixelGenerator (const LinearGradientPixelGenerator&);
LinearGradientPixelGenerator& operator= (const LinearGradientPixelGenerator&);
};
//==============================================================================
class RadialGradientPixelGenerator
{
public:
RadialGradientPixelGenerator (const ColourGradient& gradient, const AffineTransform&,
const PixelARGB* const lookupTable_, const int numEntries_) throw()
: lookupTable (lookupTable_),
numEntries (numEntries_),
gx1 (gradient.x1),
gy1 (gradient.y1)
{
jassert (numEntries_ >= 0);
const float gdx = gradient.x1 - gradient.x2;
const float gdy = gradient.y1 - gradient.y2;
maxDist = gdx * gdx + gdy * gdy;
invScale = numEntries / sqrt (maxDist);
jassert (roundToInt (sqrt (maxDist) * invScale) <= numEntries);
}
forcedinline void setY (const int y) throw()
{
dy = y - gy1;
dy *= dy;
}
inline const PixelARGB getPixel (const int px) const throw()
{
double x = px - gx1;
x *= x;
x += dy;
return lookupTable [x >= maxDist ? numEntries : roundToInt (sqrt (x) * invScale)];
}
protected:
const PixelARGB* const lookupTable;
const int numEntries;
const double gx1, gy1;
double maxDist, invScale, dy;
RadialGradientPixelGenerator (const RadialGradientPixelGenerator&);
RadialGradientPixelGenerator& operator= (const RadialGradientPixelGenerator&);
};
//==============================================================================
class TransformedRadialGradientPixelGenerator : public RadialGradientPixelGenerator
{
public:
TransformedRadialGradientPixelGenerator (const ColourGradient& gradient, const AffineTransform& transform,
const PixelARGB* const lookupTable_, const int numEntries_) throw()
: RadialGradientPixelGenerator (gradient, transform, lookupTable_, numEntries_),
inverseTransform (transform.inverted())
{
tM10 = inverseTransform.mat10;
tM00 = inverseTransform.mat00;
}
forcedinline void setY (const int y) throw()
{
lineYM01 = inverseTransform.mat01 * y + inverseTransform.mat02 - gx1;
lineYM11 = inverseTransform.mat11 * y + inverseTransform.mat12 - gy1;
}
inline const PixelARGB getPixel (const int px) const throw()
{
double x = px;
const double y = tM10 * x + lineYM11;
x = tM00 * x + lineYM01;
x *= x;
x += y * y;
if (x >= maxDist)
return lookupTable [numEntries];
else
return lookupTable [jmin (numEntries, roundToInt (sqrt (x) * invScale))];
}
private:
double tM10, tM00, lineYM01, lineYM11;
const AffineTransform inverseTransform;
TransformedRadialGradientPixelGenerator (const TransformedRadialGradientPixelGenerator&);
TransformedRadialGradientPixelGenerator& operator= (const TransformedRadialGradientPixelGenerator&);
};
//==============================================================================
template <class PixelType, class GradientType>
class GradientEdgeTableRenderer : public GradientType
{
public:
GradientEdgeTableRenderer (const Image::BitmapData& destData_, const ColourGradient& gradient, const AffineTransform& transform,
const PixelARGB* const lookupTable_, const int numEntries_) throw()
: GradientType (gradient, transform, lookupTable_, numEntries_ - 1),
destData (destData_)
{
}
forcedinline void setEdgeTableYPos (const int y) throw()
{
linePixels = (PixelType*) destData.getLinePointer (y);
GradientType::setY (y);
}
forcedinline void handleEdgeTablePixel (const int x, const int alphaLevel) const throw()
{
linePixels[x].blend (GradientType::getPixel (x), alphaLevel);
}
forcedinline void handleEdgeTableLine (int x, int width, const int alphaLevel) const throw()
{
PixelType* dest = linePixels + x;
if (alphaLevel < 0xff)
{
do
{
(dest++)->blend (GradientType::getPixel (x++), alphaLevel);
} while (--width > 0);
}
else
{
do
{
(dest++)->blend (GradientType::getPixel (x++));
} while (--width > 0);
}
}
private:
const Image::BitmapData& destData;
PixelType* linePixels;
GradientEdgeTableRenderer (const GradientEdgeTableRenderer&);
GradientEdgeTableRenderer& operator= (const GradientEdgeTableRenderer&);
};
//==============================================================================
static forcedinline int safeModulo (int n, const int divisor) throw()
{
jassert (divisor > 0);
n %= divisor;
return (n < 0) ? (n + divisor) : n;
}
//==============================================================================
template <class DestPixelType, class SrcPixelType, bool repeatPattern>
class ImageFillEdgeTableRenderer
{
public:
ImageFillEdgeTableRenderer (const Image::BitmapData& destData_,
const Image::BitmapData& srcData_,
const int extraAlpha_,
const int x, const int y) throw()
: destData (destData_),
srcData (srcData_),
extraAlpha (extraAlpha_ + 1),
xOffset (repeatPattern ? safeModulo (x, srcData_.width) - srcData_.width : x),
yOffset (repeatPattern ? safeModulo (y, srcData_.height) - srcData_.height : y)
{
}
forcedinline void setEdgeTableYPos (int y) throw()
{
linePixels = (DestPixelType*) destData.getLinePointer (y);
y -= yOffset;
if (repeatPattern)
{
jassert (y >= 0);
y %= srcData.height;
}
sourceLineStart = (SrcPixelType*) srcData.getLinePointer (y);
}
forcedinline void handleEdgeTablePixel (int x, int alphaLevel) const throw()
{
alphaLevel = (alphaLevel * extraAlpha) >> 8;
linePixels[x].blend (sourceLineStart [repeatPattern ? ((x - xOffset) % srcData.width) : (x - xOffset)], alphaLevel);
}
forcedinline void handleEdgeTableLine (int x, int width, int alphaLevel) const throw()
{
DestPixelType* dest = linePixels + x;
alphaLevel = (alphaLevel * extraAlpha) >> 8;
x -= xOffset;
jassert (repeatPattern || (x >= 0 && x + width <= srcData.width));
if (alphaLevel < 0xfe)
{
do
{
dest++ ->blend (sourceLineStart [repeatPattern ? (x++ % srcData.width) : x++], alphaLevel);
} while (--width > 0);
}
else
{
if (repeatPattern)
{
do
{
dest++ ->blend (sourceLineStart [x++ % srcData.width]);
} while (--width > 0);
}
else
{
copyRow (dest, sourceLineStart + x, width);
}
}
}
void clipEdgeTableLine (EdgeTable& et, int x, int y, int width) throw()
{
jassert (x - xOffset >= 0 && x + width - xOffset <= srcData.width);
SrcPixelType* s = (SrcPixelType*) srcData.getLinePointer (y - yOffset);
uint8* mask = (uint8*) (s + x - xOffset);
if (sizeof (SrcPixelType) == sizeof (PixelARGB))
mask += PixelARGB::indexA;
et.clipLineToMask (x, y, mask, sizeof (SrcPixelType), width);
}
private:
const Image::BitmapData& destData;
const Image::BitmapData& srcData;
const int extraAlpha, xOffset, yOffset;
DestPixelType* linePixels;
SrcPixelType* sourceLineStart;
template <class PixelType1, class PixelType2>
forcedinline static void copyRow (PixelType1* dest, PixelType2* src, int width) throw()
{
do
{
dest++ ->blend (*src++);
} while (--width > 0);
}
forcedinline static void copyRow (PixelRGB* dest, PixelRGB* src, int width) throw()
{
memcpy (dest, src, width * sizeof (PixelRGB));
}
ImageFillEdgeTableRenderer (const ImageFillEdgeTableRenderer&);
ImageFillEdgeTableRenderer& operator= (const ImageFillEdgeTableRenderer&);
};
//==============================================================================
template <class DestPixelType, class SrcPixelType, bool repeatPattern>
class TransformedImageFillEdgeTableRenderer
{
public:
TransformedImageFillEdgeTableRenderer (const Image::BitmapData& destData_,
const Image::BitmapData& srcData_,
const AffineTransform& transform,
const int extraAlpha_,
const bool betterQuality_) throw()
: interpolator (transform),
destData (destData_),
srcData (srcData_),
extraAlpha (extraAlpha_ + 1),
betterQuality (betterQuality_),
pixelOffset (betterQuality_ ? 0.5f : 0.0f),
pixelOffsetInt (betterQuality_ ? -128 : 0),
maxX (srcData_.width - 1),
maxY (srcData_.height - 1),
scratchSize (2048)
{
scratchBuffer.malloc (scratchSize);
}
~TransformedImageFillEdgeTableRenderer() throw()
{
}
forcedinline void setEdgeTableYPos (const int newY) throw()
{
y = newY;
linePixels = (DestPixelType*) destData.getLinePointer (newY);
}
forcedinline void handleEdgeTablePixel (const int x, int alphaLevel) throw()
{
alphaLevel *= extraAlpha;
alphaLevel >>= 8;
SrcPixelType p;
generate (&p, x, 1);
linePixels[x].blend (p, alphaLevel);
}
forcedinline void handleEdgeTableLine (const int x, int width, int alphaLevel) throw()
{
if (width > scratchSize)
{
scratchSize = width;
scratchBuffer.malloc (scratchSize);
}
SrcPixelType* span = scratchBuffer;
generate (span, x, width);
DestPixelType* dest = linePixels + x;
alphaLevel *= extraAlpha;
alphaLevel >>= 8;
if (alphaLevel < 0xfe)
{
do
{
dest++ ->blend (*span++, alphaLevel);
} while (--width > 0);
}
else
{
do
{
dest++ ->blend (*span++);
} while (--width > 0);
}
}
void clipEdgeTableLine (EdgeTable& et, int x, int y_, int width) throw()
{
if (width > scratchSize)
{
scratchSize = width;
scratchBuffer.malloc (scratchSize);
}
y = y_;
generate (scratchBuffer, x, width);
et.clipLineToMask (x, y_,
reinterpret_cast<uint8*> (scratchBuffer.getData()) + SrcPixelType::indexA,
sizeof (SrcPixelType), width);
}
private:
//==============================================================================
void generate (PixelARGB* dest, const int x, int numPixels) throw()
{
this->interpolator.setStartOfLine (x + pixelOffset, y + pixelOffset, numPixels);
do
{
int hiResX, hiResY;
this->interpolator.next (hiResX, hiResY);
hiResX += pixelOffsetInt;
hiResY += pixelOffsetInt;
int loResX = hiResX >> 8;
int loResY = hiResY >> 8;
if (repeatPattern)
{
loResX = safeModulo (loResX, srcData.width);
loResY = safeModulo (loResY, srcData.height);
}
if (betterQuality
&& ((unsigned int) loResX) < (unsigned int) maxX
&& ((unsigned int) loResY) < (unsigned int) maxY)
{
uint32 c[4] = { 256 * 128, 256 * 128, 256 * 128, 256 * 128 };
hiResX &= 255;
hiResY &= 255;
const uint8* src = this->srcData.getPixelPointer (loResX, loResY);
uint32 weight = (256 - hiResX) * (256 - hiResY);
c[0] += weight * src[0];
c[1] += weight * src[1];
c[2] += weight * src[2];
c[3] += weight * src[3];
weight = hiResX * (256 - hiResY);
c[0] += weight * src[4];
c[1] += weight * src[5];
c[2] += weight * src[6];
c[3] += weight * src[7];
src += this->srcData.lineStride;
weight = (256 - hiResX) * hiResY;
c[0] += weight * src[0];
c[1] += weight * src[1];
c[2] += weight * src[2];
c[3] += weight * src[3];
weight = hiResX * hiResY;
c[0] += weight * src[4];
c[1] += weight * src[5];
c[2] += weight * src[6];
c[3] += weight * src[7];
dest->setARGB ((uint8) (c[PixelARGB::indexA] >> 16),
(uint8) (c[PixelARGB::indexR] >> 16),
(uint8) (c[PixelARGB::indexG] >> 16),
(uint8) (c[PixelARGB::indexB] >> 16));
}
else
{
if (! repeatPattern)
{
// Beyond the edges, just repeat the edge pixels and leave the anti-aliasing to be handled by the edgetable
if (loResX < 0) loResX = 0;
if (loResY < 0) loResY = 0;
if (loResX > maxX) loResX = maxX;
if (loResY > maxY) loResY = maxY;
}
dest->set (*(const PixelARGB*) this->srcData.getPixelPointer (loResX, loResY));
}
++dest;
} while (--numPixels > 0);
}
void generate (PixelRGB* dest, const int x, int numPixels) throw()
{
this->interpolator.setStartOfLine (x + pixelOffset, y + pixelOffset, numPixels);
do
{
int hiResX, hiResY;
this->interpolator.next (hiResX, hiResY);
hiResX += pixelOffsetInt;
hiResY += pixelOffsetInt;
int loResX = hiResX >> 8;
int loResY = hiResY >> 8;
if (repeatPattern)
{
loResX = safeModulo (loResX, srcData.width);
loResY = safeModulo (loResY, srcData.height);
}
if (betterQuality
&& ((unsigned int) loResX) < (unsigned int) maxX
&& ((unsigned int) loResY) < (unsigned int) maxY)
{
uint32 c[3] = { 256 * 128, 256 * 128, 256 * 128 };
hiResX &= 255;
hiResY &= 255;
const uint8* src = this->srcData.getPixelPointer (loResX, loResY);
unsigned int weight = (256 - hiResX) * (256 - hiResY);
c[0] += weight * src[0];
c[1] += weight * src[1];
c[2] += weight * src[2];
weight = hiResX * (256 - hiResY);
c[0] += weight * src[3];
c[1] += weight * src[4];
c[2] += weight * src[5];
src += this->srcData.lineStride;
weight = (256 - hiResX) * hiResY;
c[0] += weight * src[0];
c[1] += weight * src[1];
c[2] += weight * src[2];
weight = hiResX * hiResY;
c[0] += weight * src[3];
c[1] += weight * src[4];
c[2] += weight * src[5];
dest->setARGB ((uint8) 255,
(uint8) (c[PixelRGB::indexR] >> 16),
(uint8) (c[PixelRGB::indexG] >> 16),
(uint8) (c[PixelRGB::indexB] >> 16));
}
else
{
if (! repeatPattern)
{
// Beyond the edges, just repeat the edge pixels and leave the anti-aliasing to be handled by the edgetable
if (loResX < 0) loResX = 0;
if (loResY < 0) loResY = 0;
if (loResX > maxX) loResX = maxX;
if (loResY > maxY) loResY = maxY;
}
dest->set (*(const PixelRGB*) this->srcData.getPixelPointer (loResX, loResY));
}
++dest;
} while (--numPixels > 0);
}
void generate (PixelAlpha* dest, const int x, int numPixels) throw()
{
this->interpolator.setStartOfLine (x + pixelOffset, y + pixelOffset, numPixels);
do
{
int hiResX, hiResY;
this->interpolator.next (hiResX, hiResY);
hiResX += pixelOffsetInt;
hiResY += pixelOffsetInt;
int loResX = hiResX >> 8;
int loResY = hiResY >> 8;
if (repeatPattern)
{
loResX = safeModulo (loResX, srcData.width);
loResY = safeModulo (loResY, srcData.height);
}
if (betterQuality
&& ((unsigned int) loResX) < (unsigned int) maxX
&& ((unsigned int) loResY) < (unsigned int) maxY)
{
hiResX &= 255;
hiResY &= 255;
const uint8* src = this->srcData.getPixelPointer (loResX, loResY);
uint32 c = 256 * 128;
c += src[0] * ((256 - hiResX) * (256 - hiResY));
c += src[1] * (hiResX * (256 - hiResY));
src += this->srcData.lineStride;
c += src[0] * ((256 - hiResX) * hiResY);
c += src[1] * (hiResX * hiResY);
*((uint8*) dest) = (uint8) c;
}
else
{
if (! repeatPattern)
{
// Beyond the edges, just repeat the edge pixels and leave the anti-aliasing to be handled by the edgetable
if (loResX < 0) loResX = 0;
if (loResY < 0) loResY = 0;
if (loResX > maxX) loResX = maxX;
if (loResY > maxY) loResY = maxY;
}
*((uint8*) dest) = *(this->srcData.getPixelPointer (loResX, loResY));
}
++dest;
} while (--numPixels > 0);
}
//==============================================================================
class TransformedImageSpanInterpolator
{
public:
TransformedImageSpanInterpolator (const AffineTransform& transform) throw()
: inverseTransform (transform.inverted())
{}
void setStartOfLine (float x, float y, const int numPixels) throw()
{
float x1 = x, y1 = y;
inverseTransform.transformPoint (x1, y1);
x += numPixels;
inverseTransform.transformPoint (x, y);
xBresenham.set ((int) (x1 * 256.0f), (int) (x * 256.0f), numPixels);
yBresenham.set ((int) (y1 * 256.0f), (int) (y * 256.0f), numPixels);
}
void next (int& x, int& y) throw()
{
x = xBresenham.n;
xBresenham.stepToNext();
y = yBresenham.n;
yBresenham.stepToNext();
}
private:
class BresenhamInterpolator
{
public:
BresenhamInterpolator() throw() {}
void set (const int n1, const int n2, const int numSteps_) throw()
{
numSteps = jmax (1, numSteps_);
step = (n2 - n1) / numSteps;
remainder = modulo = (n2 - n1) % numSteps;
n = n1;
if (modulo <= 0)
{
modulo += numSteps;
remainder += numSteps;
--step;
}
modulo -= numSteps;
}
forcedinline void stepToNext() throw()
{
modulo += remainder;
n += step;
if (modulo > 0)
{
modulo -= numSteps;
++n;
}
}
int n;
private:
int numSteps, step, modulo, remainder;
};
const AffineTransform inverseTransform;
BresenhamInterpolator xBresenham, yBresenham;
TransformedImageSpanInterpolator (const TransformedImageSpanInterpolator&);
TransformedImageSpanInterpolator& operator= (const TransformedImageSpanInterpolator&);
};
//==============================================================================
TransformedImageSpanInterpolator interpolator;
const Image::BitmapData& destData;
const Image::BitmapData& srcData;
const int extraAlpha;
const bool betterQuality;
const float pixelOffset;
const int pixelOffsetInt, maxX, maxY;
int y;
DestPixelType* linePixels;
HeapBlock <SrcPixelType> scratchBuffer;
int scratchSize;
TransformedImageFillEdgeTableRenderer (const TransformedImageFillEdgeTableRenderer&);
TransformedImageFillEdgeTableRenderer& operator= (const TransformedImageFillEdgeTableRenderer&);
};
//==============================================================================
class LLGCSavedState
{
public:
LLGCSavedState (const Rectangle<int>& clip_, const int xOffset_, const int yOffset_,
const Font& font_, const FillType& fillType_,
const Graphics::ResamplingQuality interpolationQuality_) throw()
: edgeTable (new EdgeTableHolder (EdgeTable (clip_))),
xOffset (xOffset_), yOffset (yOffset_),
font (font_), fillType (fillType_),
interpolationQuality (interpolationQuality_)
{
}
LLGCSavedState (const LLGCSavedState& other) throw()
: edgeTable (other.edgeTable), xOffset (other.xOffset),
yOffset (other.yOffset), font (other.font),
fillType (other.fillType), interpolationQuality (other.interpolationQuality)
{
}
~LLGCSavedState() throw()
{
}
bool clipToRectangle (const Rectangle<int>& r) throw()
{
dupeEdgeTableIfMultiplyReferenced();
edgeTable->edgeTable.clipToRectangle (r.translated (xOffset, yOffset));
return ! edgeTable->edgeTable.isEmpty();
}
bool clipToRectangleList (const RectangleList& r) throw()
{
dupeEdgeTableIfMultiplyReferenced();
RectangleList offsetList (r);
offsetList.offsetAll (xOffset, yOffset);
EdgeTable e2 (offsetList);
edgeTable->edgeTable.clipToEdgeTable (e2);
return ! edgeTable->edgeTable.isEmpty();
}
bool excludeClipRectangle (const Rectangle<int>& r) throw()
{
dupeEdgeTableIfMultiplyReferenced();
edgeTable->edgeTable.excludeRectangle (r.translated (xOffset, yOffset));
return ! edgeTable->edgeTable.isEmpty();
}
void clipToPath (const Path& p, const AffineTransform& transform) throw()
{
dupeEdgeTableIfMultiplyReferenced();
EdgeTable et (edgeTable->edgeTable.getMaximumBounds(), p, transform.translated ((float) xOffset, (float) yOffset));
edgeTable->edgeTable.clipToEdgeTable (et);
}
//==============================================================================
void fillEdgeTable (Image& image, EdgeTable& et, const bool replaceContents = false) throw()
{
et.clipToEdgeTable (edgeTable->edgeTable);
Image::BitmapData destData (image, 0, 0, image.getWidth(), image.getHeight(), true);
if (fillType.isGradient())
{
jassert (! replaceContents); // that option is just for solid colours
ColourGradient g2 (*(fillType.gradient));
g2.multiplyOpacity (fillType.getOpacity());
g2.x1 -= 0.5f; g2.y1 -= 0.5f;
g2.x2 -= 0.5f; g2.y2 -= 0.5f;
AffineTransform transform (fillType.transform.translated ((float) xOffset, (float) yOffset));
const bool isIdentity = transform.isOnlyTranslation();
if (isIdentity)
{
// If our translation doesn't involve any distortion, we can speed it up..
transform.transformPoint (g2.x1, g2.y1);
transform.transformPoint (g2.x2, g2.y2);
transform = AffineTransform::identity;
}
HeapBlock <PixelARGB> lookupTable;
const int numLookupEntries = g2.createLookupTable (transform, lookupTable);
jassert (numLookupEntries > 0);
switch (image.getFormat())
{
case Image::ARGB: renderGradient (et, destData, g2, transform, lookupTable, numLookupEntries, isIdentity, (PixelARGB*) 0); break;
case Image::RGB: renderGradient (et, destData, g2, transform, lookupTable, numLookupEntries, isIdentity, (PixelRGB*) 0); break;
default: renderGradient (et, destData, g2, transform, lookupTable, numLookupEntries, isIdentity, (PixelAlpha*) 0); break;
}
}
else if (fillType.isTiledImage())
{
renderImage (image, *(fillType.image), fillType.image->getBounds(), fillType.transform, &et);
}
else
{
const PixelARGB fillColour (fillType.colour.getPixelARGB());
switch (image.getFormat())
{
case Image::ARGB: renderSolidFill (et, destData, fillColour, replaceContents, (PixelARGB*) 0); break;
case Image::RGB: renderSolidFill (et, destData, fillColour, replaceContents, (PixelRGB*) 0); break;
default: renderSolidFill (et, destData, fillColour, replaceContents, (PixelAlpha*) 0); break;
}
}
}
//==============================================================================
void renderImage (Image& destImage, const Image& sourceImage, const Rectangle<int>& srcClip,
const AffineTransform& t, const EdgeTable* const tiledFillClipRegion) throw()
{
const AffineTransform transform (t.translated ((float) xOffset, (float) yOffset));
const Image::BitmapData destData (destImage, 0, 0, destImage.getWidth(), destImage.getHeight(), true);
const Image::BitmapData srcData (sourceImage, srcClip.getX(), srcClip.getY(), srcClip.getWidth(), srcClip.getHeight());
const int alpha = fillType.colour.getAlpha();
const bool betterQuality = (interpolationQuality != Graphics::lowResamplingQuality);
if (transform.isOnlyTranslation())
{
// If our translation doesn't involve any distortion, just use a simple blit..
int tx = (int) (transform.getTranslationX() * 256.0f);
int ty = (int) (transform.getTranslationY() * 256.0f);
if ((! betterQuality) || ((tx | ty) & 224) == 0)
{
tx = ((tx + 128) >> 8);
ty = ((ty + 128) >> 8);
if (tiledFillClipRegion != 0)
{
blittedRenderImage (sourceImage, destImage, *tiledFillClipRegion, destData, srcData, alpha, tx, ty, true);
}
else
{
EdgeTable et (Rectangle<int> (tx, ty, srcClip.getWidth(), srcClip.getHeight()).getIntersection (destImage.getBounds()));
et.clipToEdgeTable (edgeTable->edgeTable);
if (! et.isEmpty())
blittedRenderImage (sourceImage, destImage, et, destData, srcData, alpha, tx, ty, false);
}
return;
}
}
if (transform.isSingularity())
return;
if (tiledFillClipRegion != 0)
{
transformedRenderImage (sourceImage, destImage, *tiledFillClipRegion, destData, srcData, alpha, transform, betterQuality, true);
}
else
{
Path p;
p.addRectangle (0.0f, 0.0f, (float) srcClip.getWidth(), (float) srcClip.getHeight());
EdgeTable et (edgeTable->edgeTable.getMaximumBounds(), p, transform);
et.clipToEdgeTable (edgeTable->edgeTable);
if (! et.isEmpty())
transformedRenderImage (sourceImage, destImage, et, destData, srcData, alpha, transform, betterQuality, false);
}
}
//==============================================================================
void clipToImageAlpha (const Image& image, const Rectangle<int>& srcClip, const AffineTransform& t) throw()
{
if (! image.hasAlphaChannel())
{
Path p;
p.addRectangle (srcClip);
clipToPath (p, t);
return;
}
dupeEdgeTableIfMultiplyReferenced();
const AffineTransform transform (t.translated ((float) xOffset, (float) yOffset));
const Image::BitmapData srcData (image, srcClip.getX(), srcClip.getY(), srcClip.getWidth(), srcClip.getHeight());
const bool betterQuality = (interpolationQuality != Graphics::lowResamplingQuality);
EdgeTable& et = edgeTable->edgeTable;
if (transform.isOnlyTranslation())
{
// If our translation doesn't involve any distortion, just use a simple blit..
const int tx = (int) (transform.getTranslationX() * 256.0f);
const int ty = (int) (transform.getTranslationY() * 256.0f);
if ((! betterQuality) || ((tx | ty) & 224) == 0)
{
const int imageX = ((tx + 128) >> 8);
const int imageY = ((ty + 128) >> 8);
if (image.getFormat() == Image::ARGB)
straightClipImage (et, srcData, imageX, imageY, (PixelARGB*)0);
else
straightClipImage (et, srcData, imageX, imageY, (PixelAlpha*)0);
return;
}
}
if (transform.isSingularity())
{
et.clipToRectangle (Rectangle<int>());
return;
}
{
Path p;
p.addRectangle (0, 0, (float) srcData.width, (float) srcData.height);
EdgeTable et2 (et.getMaximumBounds(), p, transform);
et.clipToEdgeTable (et2);
}
if (! et.isEmpty())
{
if (image.getFormat() == Image::ARGB)
transformedClipImage (et, srcData, transform, betterQuality, (PixelARGB*)0);
else
transformedClipImage (et, srcData, transform, betterQuality, (PixelAlpha*)0);
}
}
template <class SrcPixelType>
void transformedClipImage (EdgeTable& et, const Image::BitmapData& srcData, const AffineTransform& transform, const bool betterQuality, const SrcPixelType *) throw()
{
TransformedImageFillEdgeTableRenderer <SrcPixelType, SrcPixelType, false> renderer (srcData, srcData, transform, 255, betterQuality);
for (int y = 0; y < et.getMaximumBounds().getHeight(); ++y)
renderer.clipEdgeTableLine (et, et.getMaximumBounds().getX(), y + et.getMaximumBounds().getY(),
et.getMaximumBounds().getWidth());
}
template <class SrcPixelType>
void straightClipImage (EdgeTable& et, const Image::BitmapData& srcData, int imageX, int imageY, const SrcPixelType *) throw()
{
Rectangle<int> r (imageX, imageY, srcData.width, srcData.height);
et.clipToRectangle (r);
ImageFillEdgeTableRenderer <SrcPixelType, SrcPixelType, false> renderer (srcData, srcData, 255, imageX, imageY);
for (int y = 0; y < r.getHeight(); ++y)
renderer.clipEdgeTableLine (et, r.getX(), y + r.getY(), r.getWidth());
}
//==============================================================================
class EdgeTableHolder : public ReferenceCountedObject
{
public:
EdgeTableHolder (const EdgeTable& e) throw() : edgeTable (e) {}
EdgeTable edgeTable;
};
ReferenceCountedObjectPtr<EdgeTableHolder> edgeTable;
int xOffset, yOffset;
Font font;
FillType fillType;
Graphics::ResamplingQuality interpolationQuality;
private:
LLGCSavedState& operator= (const LLGCSavedState&);
void dupeEdgeTableIfMultiplyReferenced() throw()
{
if (edgeTable->getReferenceCount() > 1)
edgeTable = new EdgeTableHolder (edgeTable->edgeTable);
}
//==============================================================================
template <class DestPixelType>
void renderGradient (EdgeTable& et, const Image::BitmapData& destData, const ColourGradient& g, const AffineTransform& transform,
const PixelARGB* const lookupTable, const int numLookupEntries, const bool isIdentity, DestPixelType*) throw()
{
jassert (destData.pixelStride == sizeof (DestPixelType));
if (g.isRadial)
{
if (isIdentity)
{
GradientEdgeTableRenderer <DestPixelType, RadialGradientPixelGenerator> renderer (destData, g, transform, lookupTable, numLookupEntries);
et.iterate (renderer);
}
else
{
GradientEdgeTableRenderer <DestPixelType, TransformedRadialGradientPixelGenerator> renderer (destData, g, transform, lookupTable, numLookupEntries);
et.iterate (renderer);
}
}
else
{
GradientEdgeTableRenderer <DestPixelType, LinearGradientPixelGenerator> renderer (destData, g, transform, lookupTable, numLookupEntries);
et.iterate (renderer);
}
}
//==============================================================================
template <class DestPixelType>
void renderSolidFill (EdgeTable& et, const Image::BitmapData& destData, const PixelARGB& fillColour, const bool replaceContents, DestPixelType*) throw()
{
jassert (destData.pixelStride == sizeof (DestPixelType));
if (replaceContents)
{
SolidColourEdgeTableRenderer <DestPixelType, true> r (destData, fillColour);
et.iterate (r);
}
else
{
SolidColourEdgeTableRenderer <DestPixelType, false> r (destData, fillColour);
et.iterate (r);
}
}
//==============================================================================
void transformedRenderImage (const Image& srcImage, Image& destImage, const EdgeTable& et, const Image::BitmapData& destData, const Image::BitmapData& srcData,
const int alpha, const AffineTransform& transform, const bool betterQuality, const bool repeatPattern) throw()
{
switch (destImage.getFormat())
{
case Image::ARGB:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelARGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelARGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelRGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelRGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
default:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelAlpha, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelARGB, PixelAlpha, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
}
break;
case Image::RGB:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelARGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelARGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelRGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelRGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
default:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelAlpha, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelRGB, PixelAlpha, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
}
break;
default:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelARGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelARGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelRGB, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelRGB, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
default:
if (repeatPattern) { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelAlpha, true> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
else { TransformedImageFillEdgeTableRenderer <PixelAlpha, PixelAlpha, false> r (destData, srcData, transform, alpha, betterQuality); et.iterate (r); }
break;
}
break;
}
}
//==============================================================================
void blittedRenderImage (const Image& srcImage, Image& destImage, const EdgeTable& et, const Image::BitmapData& destData,
const Image::BitmapData& srcData, const int alpha, int x, int y, const bool repeatPattern) throw()
{
switch (destImage.getFormat())
{
case Image::ARGB:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelARGB, PixelARGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelARGB, PixelARGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelARGB, PixelRGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelARGB, PixelRGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
default:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelARGB, PixelAlpha, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelARGB, PixelAlpha, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
}
break;
case Image::RGB:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelRGB, PixelARGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelRGB, PixelARGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelRGB, PixelRGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelRGB, PixelRGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
default:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelRGB, PixelAlpha, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelRGB, PixelAlpha, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
}
break;
default:
switch (srcImage.getFormat())
{
case Image::ARGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelAlpha, PixelARGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelAlpha, PixelARGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
case Image::RGB:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelAlpha, PixelRGB, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelAlpha, PixelRGB, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
default:
if (repeatPattern) { ImageFillEdgeTableRenderer <PixelAlpha, PixelAlpha, true> r (destData, srcData, alpha, x, y); et.iterate (r); }
else { ImageFillEdgeTableRenderer <PixelAlpha, PixelAlpha, false> r (destData, srcData, alpha, x, y); et.iterate (r); }
break;
}
break;
}
}
};
//==============================================================================
LowLevelGraphicsSoftwareRenderer::LowLevelGraphicsSoftwareRenderer (Image& image_)
: image (image_)
{
currentState = new LLGCSavedState (image_.getBounds(), 0, 0, Font(),
FillType(), Graphics::mediumResamplingQuality);
}
LowLevelGraphicsSoftwareRenderer::~LowLevelGraphicsSoftwareRenderer()
{
}
bool LowLevelGraphicsSoftwareRenderer::isVectorDevice() const
{
return false;
}
//==============================================================================
void LowLevelGraphicsSoftwareRenderer::setOrigin (int x, int y)
{
currentState->xOffset += x;
currentState->yOffset += y;
}
bool LowLevelGraphicsSoftwareRenderer::clipToRectangle (const Rectangle<int>& r)
{
return currentState->clipToRectangle (r);
}
bool LowLevelGraphicsSoftwareRenderer::clipToRectangleList (const RectangleList& clipRegion)
{
return currentState->clipToRectangleList (clipRegion);
}
void LowLevelGraphicsSoftwareRenderer::excludeClipRectangle (const Rectangle<int>& r)
{
currentState->excludeClipRectangle (r);
}
void LowLevelGraphicsSoftwareRenderer::clipToPath (const Path& path, const AffineTransform& transform)
{
currentState->clipToPath (path, transform);
}
void LowLevelGraphicsSoftwareRenderer::clipToImageAlpha (const Image& sourceImage, const Rectangle<int>& srcClip, const AffineTransform& transform)
{
currentState->clipToImageAlpha (sourceImage, srcClip, transform);
}
bool LowLevelGraphicsSoftwareRenderer::clipRegionIntersects (const Rectangle<int>& r)
{
return currentState->edgeTable->edgeTable.getMaximumBounds()
.intersects (r.translated (currentState->xOffset, currentState->yOffset));
}
const Rectangle<int> LowLevelGraphicsSoftwareRenderer::getClipBounds() const
{
return currentState->edgeTable->edgeTable.getMaximumBounds().translated (-currentState->xOffset, -currentState->yOffset);
}
bool LowLevelGraphicsSoftwareRenderer::isClipEmpty() const
{
return currentState->edgeTable->edgeTable.isEmpty();
}
//==============================================================================
void LowLevelGraphicsSoftwareRenderer::saveState()
{
stateStack.add (new LLGCSavedState (*currentState));
}
void LowLevelGraphicsSoftwareRenderer::restoreState()
{
LLGCSavedState* const top = stateStack.getLast();
if (top != 0)
{
currentState = top;
stateStack.removeLast (1, false);
}
else
{
jassertfalse // trying to pop with an empty stack!
}
}
//==============================================================================
void LowLevelGraphicsSoftwareRenderer::setFill (const FillType& fillType)
{
currentState->fillType = fillType;
}
void LowLevelGraphicsSoftwareRenderer::setOpacity (float newOpacity)
{
currentState->fillType.setOpacity (newOpacity);
}
void LowLevelGraphicsSoftwareRenderer::setInterpolationQuality (Graphics::ResamplingQuality quality)
{
currentState->interpolationQuality = quality;
}
//==============================================================================
void LowLevelGraphicsSoftwareRenderer::fillRect (const Rectangle<int>& r, const bool replaceExistingContents)
{
const Rectangle<int>& totalClip = currentState->edgeTable->edgeTable.getMaximumBounds();
const Rectangle<int> clipped (totalClip.getIntersection (r.translated (currentState->xOffset, currentState->yOffset)));
if (! clipped.isEmpty())
{
EdgeTable et (clipped);
currentState->fillEdgeTable (image, et, replaceExistingContents);
}
}
void LowLevelGraphicsSoftwareRenderer::fillPath (const Path& path, const AffineTransform& transform)
{
EdgeTable et (currentState->edgeTable->edgeTable.getMaximumBounds(),
path, transform.translated ((float) currentState->xOffset,
(float) currentState->yOffset));
currentState->fillEdgeTable (image, et);
}
void LowLevelGraphicsSoftwareRenderer::drawImage (const Image& sourceImage, const Rectangle<int>& srcClip,
const AffineTransform& transform, const bool fillEntireClipAsTiles)
{
jassert (sourceImage.getBounds().contains (srcClip));
currentState->renderImage (image, sourceImage, srcClip, transform,
fillEntireClipAsTiles ? &(currentState->edgeTable->edgeTable) : 0);
}
//==============================================================================
void LowLevelGraphicsSoftwareRenderer::drawLine (double x1, double y1, double x2, double y2)
{
Path p;
p.addLineSegment ((float) x1, (float) y1, (float) x2, (float) y2, 1.0f);
fillPath (p, AffineTransform::identity);
}
void LowLevelGraphicsSoftwareRenderer::drawVerticalLine (const int x, double top, double bottom)
{
if (bottom > top)
{
EdgeTable et ((float) (x + currentState->xOffset), (float) (top + currentState->yOffset), 1.0f, (float) (bottom - top));
currentState->fillEdgeTable (image, et);
}
}
void LowLevelGraphicsSoftwareRenderer::drawHorizontalLine (const int y, double left, double right)
{
if (right > left)
{
EdgeTable et ((float) (left + currentState->xOffset), (float) (y + currentState->yOffset),
(float) (right - left), 1.0f);
currentState->fillEdgeTable (image, et);
}
}
//==============================================================================
class GlyphCache : private DeletedAtShutdown
{
public:
GlyphCache() throw()
: accessCounter (0), hits (0), misses (0)
{
for (int i = 120; --i >= 0;)
glyphs.add (new CachedGlyph());
}
~GlyphCache() throw()
{
clearSingletonInstance();
}
juce_DeclareSingleton_SingleThreaded_Minimal (GlyphCache);
//==============================================================================
void drawGlyph (LLGCSavedState& state, Image& image, const Font& font, const int glyphNumber, float x, float y) throw()
{
++accessCounter;
int oldestCounter = std::numeric_limits<int>::max();
CachedGlyph* oldest = 0;
for (int i = glyphs.size(); --i >= 0;)
{
CachedGlyph* const glyph = glyphs.getUnchecked (i);
if (glyph->glyph == glyphNumber && glyph->font == font)
{
++hits;
glyph->lastAccessCount = accessCounter;
glyph->draw (state, image, x, y);
return;
}
if (glyph->lastAccessCount <= oldestCounter)
{
oldestCounter = glyph->lastAccessCount;
oldest = glyph;
}
}
if (hits + ++misses > (glyphs.size() << 4))
{
if (misses * 2 > hits)
{
for (int i = 32; --i >= 0;)
glyphs.add (new CachedGlyph());
}
hits = misses = 0;
oldest = glyphs.getLast();
}
jassert (oldest != 0);
oldest->lastAccessCount = accessCounter;
oldest->generate (font, glyphNumber);
oldest->draw (state, image, x, y);
}
//==============================================================================
class CachedGlyph
{
public:
CachedGlyph() : glyph (0), lastAccessCount (0) {}
~CachedGlyph() {}
void draw (LLGCSavedState& state, Image& image, const float x, const float y) const throw()
{
if (edgeTable != 0)
{
EdgeTable et (*edgeTable);
et.translate (x, roundToInt (y));
state.fillEdgeTable (image, et, false);
}
}
void generate (const Font& newFont, const int glyphNumber) throw()
{
font = newFont;
glyph = glyphNumber;
edgeTable = 0;
Path glyphPath;
font.getTypeface()->getOutlineForGlyph (glyphNumber, glyphPath);
if (! glyphPath.isEmpty())
{
const float fontHeight = font.getHeight();
const AffineTransform transform (AffineTransform::scale (fontHeight * font.getHorizontalScale(), fontHeight)
.translated (0.0f, -0.5f));
edgeTable = new EdgeTable (glyphPath.getBoundsTransformed (transform).getSmallestIntegerContainer().expanded (1, 0),
glyphPath, transform);
}
}
int glyph, lastAccessCount;
Font font;
//==============================================================================
juce_UseDebuggingNewOperator
private:
ScopedPointer <EdgeTable> edgeTable;
CachedGlyph (const CachedGlyph&);
CachedGlyph& operator= (const CachedGlyph&);
};
//==============================================================================
juce_UseDebuggingNewOperator
private:
OwnedArray <CachedGlyph> glyphs;
int accessCounter, hits, misses;
GlyphCache (const GlyphCache&);
GlyphCache& operator= (const GlyphCache&);
};
juce_ImplementSingleton_SingleThreaded (GlyphCache);
void LowLevelGraphicsSoftwareRenderer::setFont (const Font& newFont)
{
currentState->font = newFont;
}
const Font LowLevelGraphicsSoftwareRenderer::getFont()
{
return currentState->font;
}
void LowLevelGraphicsSoftwareRenderer::drawGlyph (int glyphNumber, const AffineTransform& transform)
{
Font& f = currentState->font;
if (transform.isOnlyTranslation())
{
GlyphCache::getInstance()->drawGlyph (*currentState, image, f, glyphNumber,
transform.getTranslationX() + (float) currentState->xOffset,
transform.getTranslationY() + (float) currentState->yOffset);
}
else
{
Path p;
f.getTypeface()->getOutlineForGlyph (glyphNumber, p);
fillPath (p, AffineTransform::scale (f.getHeight() * f.getHorizontalScale(), f.getHeight()).followedBy (transform));
}
}
#if JUCE_MSVC
#pragma warning (pop)
#endif
END_JUCE_NAMESPACE