#pragma once
#include "FloatSize.h"
#include "Glyph.h"
#include <climits>
#include <wtf/Vector.h>
#if USE(CG)
#include <CoreGraphics/CGGeometry.h>
#endif
namespace WebCore {
class Font;
#if USE(WINGDI)
typedef wchar_t GlyphBufferGlyph;
#else
typedef Glyph GlyphBufferGlyph;
#endif
#if USE(CG)
struct GlyphBufferAdvance : CGSize {
public:
GlyphBufferAdvance() : CGSize(CGSizeZero) { }
GlyphBufferAdvance(CGSize size)
: CGSize(size)
{
}
GlyphBufferAdvance(float width, float height)
: CGSize(CGSizeMake(width, height))
{
}
void setWidth(CGFloat width) { this->CGSize::width = width; }
void setHeight(CGFloat height) { this->CGSize::height = height; }
CGFloat width() const { return this->CGSize::width; }
CGFloat height() const { return this->CGSize::height; }
};
#else
typedef FloatSize GlyphBufferAdvance;
#endif
class GlyphBuffer {
public:
bool isEmpty() const { return m_font.isEmpty(); }
unsigned size() const { return m_font.size(); }
void clear()
{
m_font.clear();
m_glyphs.clear();
m_advances.clear();
if (m_offsetsInString)
m_offsetsInString->clear();
}
GlyphBufferGlyph* glyphs(unsigned from) { return m_glyphs.data() + from; }
GlyphBufferAdvance* advances(unsigned from) { return m_advances.data() + from; }
const GlyphBufferGlyph* glyphs(unsigned from) const { return m_glyphs.data() + from; }
const GlyphBufferAdvance* advances(unsigned from) const { return m_advances.data() + from; }
size_t advancesCount() const { return m_advances.size(); }
const Font* fontAt(unsigned index) const { return m_font[index]; }
void setInitialAdvance(GlyphBufferAdvance initialAdvance) { m_initialAdvance = initialAdvance; }
const GlyphBufferAdvance& initialAdvance() const { return m_initialAdvance; }
Glyph glyphAt(unsigned index) const
{
return m_glyphs[index];
}
GlyphBufferAdvance advanceAt(unsigned index) const
{
return m_advances[index];
}
static const unsigned noOffset = UINT_MAX;
void add(Glyph glyph, const Font* font, float width, unsigned offsetInString = noOffset)
{
GlyphBufferAdvance advance;
advance.setWidth(width);
advance.setHeight(0);
add(glyph, font, advance, offsetInString);
}
void add(Glyph glyph, const Font* font, GlyphBufferAdvance advance, unsigned offsetInString = noOffset)
{
m_font.append(font);
m_glyphs.append(glyph);
m_advances.append(advance);
if (offsetInString != noOffset && m_offsetsInString)
m_offsetsInString->append(offsetInString);
}
void reverse(unsigned from, unsigned length)
{
for (unsigned i = from, end = from + length - 1; i < end; ++i, --end)
swap(i, end);
}
void expandLastAdvance(float width)
{
ASSERT(!isEmpty());
GlyphBufferAdvance& lastAdvance = m_advances.last();
lastAdvance.setWidth(lastAdvance.width() + width);
}
void expandLastAdvance(GlyphBufferAdvance expansion)
{
ASSERT(!isEmpty());
GlyphBufferAdvance& lastAdvance = m_advances.last();
lastAdvance.setWidth(lastAdvance.width() + expansion.width());
lastAdvance.setHeight(lastAdvance.height() + expansion.height());
}
void saveOffsetsInString()
{
m_offsetsInString.reset(new Vector<unsigned, 2048>());
}
int offsetInString(unsigned index) const
{
ASSERT(m_offsetsInString);
return (*m_offsetsInString)[index];
}
void shrink(unsigned truncationPoint)
{
m_font.shrink(truncationPoint);
m_glyphs.shrink(truncationPoint);
m_advances.shrink(truncationPoint);
if (m_offsetsInString)
m_offsetsInString->shrink(truncationPoint);
}
private:
void swap(unsigned index1, unsigned index2)
{
const Font* f = m_font[index1];
m_font[index1] = m_font[index2];
m_font[index2] = f;
GlyphBufferGlyph g = m_glyphs[index1];
m_glyphs[index1] = m_glyphs[index2];
m_glyphs[index2] = g;
GlyphBufferAdvance s = m_advances[index1];
m_advances[index1] = m_advances[index2];
m_advances[index2] = s;
}
Vector<const Font*, 2048> m_font;
Vector<GlyphBufferGlyph, 2048> m_glyphs;
Vector<GlyphBufferAdvance, 2048> m_advances;
GlyphBufferAdvance m_initialAdvance;
std::unique_ptr<Vector<unsigned, 2048>> m_offsetsInString;
};
}