#ifndef FontCache_h
#define FontCache_h
#include "FontDescription.h"
#include <limits.h>
#include <wtf/Forward.h>
#include <wtf/PassRefPtr.h>
#include <wtf/RefPtr.h>
#include <wtf/Vector.h>
#include <wtf/text/WTFString.h>
#include <wtf/unicode/Unicode.h>
#if OS(WINDOWS)
#include <windows.h>
#include <objidl.h>
#include <mlang.h>
#endif
namespace WebCore {
class Font;
class FontPlatformData;
class FontData;
class FontSelector;
class OpenTypeVerticalData;
class SimpleFontData;
#if PLATFORM(WIN)
#if USE(IMLANG_FONT_LINK2)
typedef IMLangFontLink2 IMLangFontLinkType;
#else
typedef IMLangFontLink IMLangFontLinkType;
#endif
#endif
struct FontDescriptionFontDataCacheKey {
explicit FontDescriptionFontDataCacheKey(unsigned size = 0)
: size(size)
, weight(0)
, flags(0)
{ }
FontDescriptionFontDataCacheKey(const FontDescription& description)
: size(description.computedPixelSize())
, weight(description.weight())
, flags(makeFlagKey(description))
{ }
static unsigned makeFlagKey(const FontDescription& description)
{
return static_cast<unsigned>(description.widthVariant()) << 4
| static_cast<unsigned>(description.orientation()) << 3
| static_cast<unsigned>(description.italic()) << 2
| static_cast<unsigned>(description.usePrinterFont()) << 1
| static_cast<unsigned>(description.renderingMode());
}
bool operator==(const FontDescriptionFontDataCacheKey& other) const
{
return size == other.size && weight == other.weight && flags == other.flags;
}
bool operator!=(const FontDescriptionFontDataCacheKey& other) const
{
return !(*this == other);
}
inline unsigned computeHash() const
{
return StringHasher::hashMemory<sizeof(FontDescriptionFontDataCacheKey)>(this);
}
unsigned size;
unsigned weight;
unsigned flags;
};
class FontCache {
friend class FontCachePurgePreventer;
WTF_MAKE_NONCOPYABLE(FontCache); WTF_MAKE_FAST_ALLOCATED;
public:
friend FontCache* fontCache();
enum ShouldRetain { Retain, DoNotRetain };
PassRefPtr<FontData> getFontData(const FontDescription&, int& familyIndex, FontSelector*);
void releaseFontData(const SimpleFontData*);
PassRefPtr<SimpleFontData> systemFallbackForCharacters(const FontDescription&, const SimpleFontData* originalFontData, bool isPlatformFont, const UChar* characters, int length);
void platformInit();
#if PLATFORM(WIN)
IMLangFontLinkType* getFontLinkInterface();
static void comInitialize();
static void comUninitialize();
static IMultiLanguage* getMultiLanguageInterface();
#endif
void getTraitsInFamily(const AtomicString&, Vector<unsigned>&);
PassRefPtr<SimpleFontData> getCachedFontData(const FontDescription&, const AtomicString&, bool checkingAlternateName = false, ShouldRetain = Retain);
PassRefPtr<SimpleFontData> getLastResortFallbackFont(const FontDescription&, ShouldRetain = Retain);
SimpleFontData* getNonRetainedLastResortFallbackFont(const FontDescription&);
void addClient(FontSelector*);
void removeClient(FontSelector*);
unsigned short generation();
void invalidate();
size_t fontDataCount();
size_t inactiveFontDataCount();
void purgeInactiveFontData(int count = INT_MAX);
#if PLATFORM(WIN)
PassRefPtr<SimpleFontData> fontDataFromDescriptionAndLogFont(const FontDescription&, ShouldRetain, const LOGFONT&, AtomicString& outFontFamilyName);
#endif
#if ENABLE(OPENTYPE_VERTICAL)
typedef AtomicString FontFileKey;
PassRefPtr<OpenTypeVerticalData> getVerticalData(const FontFileKey&, const FontPlatformData&);
#endif
struct SimpleFontFamily {
String name;
bool isBold;
bool isItalic;
};
#if PLATFORM(BLACKBERRY)
static void getFontFamilyForCharacters(const UChar* characters, size_t numCharacters, const char* preferredLocale, const FontDescription&, SimpleFontFamily*);
#else
static void getFontFamilyForCharacters(const UChar* characters, size_t numCharacters, const char* preferredLocale, SimpleFontFamily*);
#endif
private:
FontCache();
~FontCache();
void disablePurging() { m_purgePreventCount++; }
void enablePurging()
{
ASSERT(m_purgePreventCount);
if (!--m_purgePreventCount)
purgeInactiveFontDataIfNeeded();
}
void purgeInactiveFontDataIfNeeded();
FontPlatformData* getCachedFontPlatformData(const FontDescription&, const AtomicString& family, bool checkingAlternateName = false);
PassOwnPtr<FontPlatformData> createFontPlatformData(const FontDescription&, const AtomicString& family);
#if PLATFORM(MAC)
PassRefPtr<SimpleFontData> similarFontPlatformData(const FontDescription&);
#endif
PassRefPtr<SimpleFontData> getCachedFontData(const FontPlatformData*, ShouldRetain = Retain);
int m_purgePreventCount;
#if PLATFORM(MAC)
friend class ComplexTextController;
#endif
friend class SimpleFontData; friend class FontGlyphs;
};
FontCache* fontCache();
class FontCachePurgePreventer {
public:
FontCachePurgePreventer() { fontCache()->disablePurging(); }
~FontCachePurgePreventer() { fontCache()->enablePurging(); }
};
}
#endif