This keyword is not expanded by Git which means it's not replaced with the correct revision value in the releases made using git-based scripts and it's confusing to have lines with unexpanded "$Id$" in the released files. As expanding them with Git is not that simple (it could be done with git archive and export-subst attribute) and there are not many benefits in having them in the first place, just remove all these lines. If nothing else, this will make an eventual transition to Git simpler. Closes #14487. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@74602 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
338 lines
8.5 KiB
C++
338 lines
8.5 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: src/common/fontmgrcmn.cpp
|
|
// Purpose: font management for ports that don't have their own
|
|
// Author: Vaclav Slavik
|
|
// Created: 2006-11-18
|
|
// Copyright: (c) 2001-2002 SciTech Software, Inc. (www.scitechsoft.com)
|
|
// (c) 2006 REA Elektronik GmbH
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// For compilers that support precompilation, includes "wx.h".
|
|
#include "wx/wxprec.h"
|
|
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
|
|
#include "wx/private/fontmgr.h"
|
|
|
|
#include "wx/listimpl.cpp"
|
|
#include "wx/hashmap.h"
|
|
|
|
WX_DECLARE_LIST(wxFontInstance, wxFontInstanceList);
|
|
WX_DEFINE_LIST(wxFontInstanceList)
|
|
WX_DEFINE_LIST(wxFontBundleList)
|
|
|
|
WX_DECLARE_HASH_MAP(wxString, wxFontBundle*,
|
|
wxStringHash, wxStringEqual,
|
|
wxFontBundleHashBase);
|
|
// in STL build, hash class is typedef to a template, so it can't be forward
|
|
// declared, as we do; solve it by having a dummy class:
|
|
class wxFontBundleHash : public wxFontBundleHashBase
|
|
{
|
|
};
|
|
|
|
// ============================================================================
|
|
// implementation
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxFontFaceBase
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxFontFaceBase::wxFontFaceBase()
|
|
: m_refCnt(0)
|
|
{
|
|
m_instances = new wxFontInstanceList;
|
|
m_instances->DeleteContents(true);
|
|
}
|
|
|
|
wxFontFaceBase::~wxFontFaceBase()
|
|
{
|
|
delete m_instances;
|
|
}
|
|
|
|
void wxFontFaceBase::Acquire()
|
|
{
|
|
m_refCnt++;
|
|
}
|
|
|
|
void wxFontFaceBase::Release()
|
|
{
|
|
if ( --m_refCnt == 0 )
|
|
{
|
|
m_instances->Clear();
|
|
}
|
|
}
|
|
|
|
wxFontInstance *wxFontFaceBase::GetFontInstance(float ptSize, bool aa)
|
|
{
|
|
wxASSERT_MSG( m_refCnt > 0, wxT("font library not loaded!") );
|
|
|
|
for ( wxFontInstanceList::const_iterator i = m_instances->begin();
|
|
i != m_instances->end(); ++i )
|
|
{
|
|
if ( (*i)->GetPointSize() == ptSize && (*i)->IsAntiAliased() == aa )
|
|
return *i;
|
|
}
|
|
|
|
wxFontInstance *i = CreateFontInstance(ptSize, aa);
|
|
m_instances->Append(i);
|
|
return i;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxFontBundleBase
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxFontBundleBase::wxFontBundleBase()
|
|
{
|
|
for (int i = 0; i < FaceType_Max; i++)
|
|
m_faces[i] = NULL;
|
|
}
|
|
|
|
wxFontBundleBase::~wxFontBundleBase()
|
|
{
|
|
for (int i = 0; i < FaceType_Max; i++)
|
|
delete m_faces[i];
|
|
}
|
|
|
|
wxFontFace *wxFontBundleBase::GetFace(FaceType type) const
|
|
{
|
|
wxFontFace *f = m_faces[type];
|
|
|
|
wxCHECK_MSG( f, NULL, wxT("no such face in font bundle") );
|
|
|
|
f->Acquire();
|
|
|
|
return f;
|
|
}
|
|
|
|
wxFontFace *
|
|
wxFontBundleBase::GetFaceForFont(const wxFontMgrFontRefData& font) const
|
|
{
|
|
wxASSERT_MSG( font.GetFaceName().empty() ||
|
|
GetName().CmpNoCase(font.GetFaceName()) == 0,
|
|
wxT("calling GetFaceForFont for incompatible font") );
|
|
|
|
int type = FaceType_Regular;
|
|
|
|
if ( font.GetWeight() == wxBOLD )
|
|
type |= FaceType_Bold;
|
|
|
|
// FIXME -- this should read "if ( font->GetStyle() == wxITALIC )",
|
|
// but since DFB doesn't support slant, we try to display it with italic
|
|
// face (better than nothing...)
|
|
if ( font.GetStyle() == wxITALIC || font.GetStyle() == wxSLANT )
|
|
{
|
|
if ( HasFace((FaceType)(type | FaceType_Italic)) )
|
|
type |= FaceType_Italic;
|
|
}
|
|
|
|
if ( !HasFace((FaceType)type) )
|
|
{
|
|
// if we can't get the exact font requested, substitute it with
|
|
// some other variant:
|
|
for (int i = 0; i < FaceType_Max; i++)
|
|
{
|
|
if ( HasFace((FaceType)i) )
|
|
return GetFace((FaceType)i);
|
|
}
|
|
|
|
wxFAIL_MSG( wxT("no face") );
|
|
return NULL;
|
|
}
|
|
|
|
return GetFace((FaceType)type);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxFontsManagerBase
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxFontsManager *wxFontsManagerBase::ms_instance = NULL;
|
|
|
|
wxFontsManagerBase::wxFontsManagerBase()
|
|
{
|
|
m_hash = new wxFontBundleHash();
|
|
m_list = new wxFontBundleList;
|
|
m_list->DeleteContents(true);
|
|
}
|
|
|
|
wxFontsManagerBase::~wxFontsManagerBase()
|
|
{
|
|
delete m_hash;
|
|
delete m_list;
|
|
}
|
|
|
|
/* static */
|
|
wxFontsManager *wxFontsManagerBase::Get()
|
|
{
|
|
if ( !ms_instance )
|
|
ms_instance = new wxFontsManager();
|
|
return ms_instance;
|
|
}
|
|
|
|
/* static */
|
|
void wxFontsManagerBase::CleanUp()
|
|
{
|
|
wxDELETE(ms_instance);
|
|
}
|
|
|
|
wxFontBundle *wxFontsManagerBase::GetBundle(const wxString& name) const
|
|
{
|
|
return (*m_hash)[name.Lower()];
|
|
}
|
|
|
|
wxFontBundle *
|
|
wxFontsManagerBase::GetBundleForFont(const wxFontMgrFontRefData& font) const
|
|
{
|
|
wxFontBundle *bundle = NULL;
|
|
|
|
wxString facename = font.GetFaceName();
|
|
if ( !facename.empty() )
|
|
bundle = GetBundle(facename);
|
|
|
|
if ( !bundle )
|
|
{
|
|
facename = GetDefaultFacename((wxFontFamily)font.GetFamily());
|
|
if ( !facename.empty() )
|
|
bundle = GetBundle(facename);
|
|
}
|
|
|
|
if ( !bundle )
|
|
{
|
|
if ( m_list->GetFirst() )
|
|
bundle = m_list->GetFirst()->GetData();
|
|
else
|
|
wxFAIL_MSG(wxT("Fatal error, no fonts available!"));
|
|
}
|
|
|
|
return bundle;
|
|
}
|
|
|
|
void wxFontsManagerBase::AddBundle(wxFontBundle *bundle)
|
|
{
|
|
(*m_hash)[bundle->GetName().Lower()] = bundle;
|
|
m_list->Append(bundle);
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxFontMgrFontRefData
|
|
// ----------------------------------------------------------------------------
|
|
|
|
wxFontMgrFontRefData::wxFontMgrFontRefData(int size,
|
|
wxFontFamily family,
|
|
wxFontStyle style,
|
|
wxFontWeight weight,
|
|
bool underlined,
|
|
const wxString& faceName,
|
|
wxFontEncoding encoding)
|
|
{
|
|
if ( family == wxFONTFAMILY_DEFAULT )
|
|
family = wxFONTFAMILY_SWISS;
|
|
if ( size == wxDEFAULT )
|
|
size = 12;
|
|
|
|
m_info.family = (wxFontFamily)family;
|
|
m_info.faceName = faceName;
|
|
m_info.style = (wxFontStyle)style;
|
|
m_info.weight = (wxFontWeight)weight;
|
|
m_info.pointSize = size;
|
|
m_info.underlined = underlined;
|
|
m_info.encoding = encoding;
|
|
|
|
m_fontFace = NULL;
|
|
m_fontBundle = NULL;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
wxFontMgrFontRefData::wxFontMgrFontRefData(const wxFontMgrFontRefData& data)
|
|
{
|
|
m_info = data.m_info;
|
|
|
|
m_fontFace = data.m_fontFace;
|
|
m_fontBundle = data.m_fontBundle;
|
|
m_fontValid = data.m_fontValid;
|
|
if ( m_fontFace )
|
|
m_fontFace->Acquire();
|
|
}
|
|
|
|
wxFontMgrFontRefData::~wxFontMgrFontRefData()
|
|
{
|
|
if ( m_fontFace )
|
|
m_fontFace->Release();
|
|
}
|
|
|
|
wxFontBundle *wxFontMgrFontRefData::GetFontBundle() const
|
|
{
|
|
wxConstCast(this, wxFontMgrFontRefData)->EnsureValidFont();
|
|
return m_fontBundle;
|
|
}
|
|
|
|
wxFontInstance *
|
|
wxFontMgrFontRefData::GetFontInstance(float scale, bool antialiased) const
|
|
{
|
|
wxConstCast(this, wxFontMgrFontRefData)->EnsureValidFont();
|
|
return m_fontFace->GetFontInstance(m_info.pointSize * scale,
|
|
antialiased);
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetPointSize(int pointSize)
|
|
{
|
|
m_info.pointSize = pointSize;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetFamily(wxFontFamily family)
|
|
{
|
|
m_info.family = family;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetStyle(wxFontStyle style)
|
|
{
|
|
m_info.style = style;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetWeight(wxFontWeight weight)
|
|
{
|
|
m_info.weight = weight;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetFaceName(const wxString& faceName)
|
|
{
|
|
m_info.faceName = faceName;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetUnderlined(bool underlined)
|
|
{
|
|
m_info.underlined = underlined;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::SetEncoding(wxFontEncoding encoding)
|
|
{
|
|
m_info.encoding = encoding;
|
|
m_fontValid = false;
|
|
}
|
|
|
|
void wxFontMgrFontRefData::EnsureValidFont()
|
|
{
|
|
if ( !m_fontValid )
|
|
{
|
|
wxFontFace *old = m_fontFace;
|
|
|
|
m_fontBundle = wxFontsManager::Get()->GetBundleForFont(*this);
|
|
m_fontFace = m_fontBundle->GetFaceForFont(*this);
|
|
|
|
if ( old )
|
|
old->Release();
|
|
}
|
|
}
|