mirror of
https://github.com/snowie2000/mactype.git
synced 2025-01-08 11:47:43 +08:00
1788 lines
56 KiB
C++
1788 lines
56 KiB
C++
#include "settings.h"
|
||
#include "strtoken.h"
|
||
#include <math.h> //pow
|
||
#include "supinfo.h"
|
||
#include "fteng.h"
|
||
#include <stdlib.h>
|
||
#ifdef INFINALITY
|
||
#include <freetype/ftenv.h>
|
||
#endif
|
||
|
||
CControlCenter* g_ControlCenter = NULL;
|
||
|
||
inline BOOL IsFolder(LPCTSTR pszPath) {
|
||
return pszPath && *pszPath && *(pszPath + wcslen(pszPath) - 1) == '\\';
|
||
}
|
||
|
||
wstring LowerCase(wstring str) {
|
||
transform(str.begin(), str.end(), str.begin(), ::tolower);
|
||
return str;
|
||
}
|
||
|
||
const wstring GetAppDir() {
|
||
static wstring AppDir;
|
||
if (AppDir.length()) {
|
||
return AppDir;
|
||
}
|
||
WCHAR name[MAX_PATH] = { 0 };
|
||
|
||
int nSize = GetModuleFileName(NULL, name, MAX_PATH + 1);
|
||
PathRemoveFileSpec(name);
|
||
AppDir = wstring(name) + L"\\"; // path should always end with a "\"
|
||
AppDir = LowerCase(AppDir);
|
||
return AppDir;
|
||
}
|
||
|
||
CGdippSettings* CGdippSettings::s_pInstance;
|
||
CParseIni CGdippSettings::m_Config;
|
||
CHashedStringList FontNameCache;
|
||
|
||
static const TCHAR c_szGeneral[] = _T("General");
|
||
static const TCHAR c_szFreeType[] = _T("FreeType");
|
||
static const TCHAR c_szDirectWrite[] = _T("DirectWrite");
|
||
#define HINTING_MIN 0
|
||
#define HINTING_MAX 2
|
||
#define AAMODE_MIN -1
|
||
#define AAMODE_MAX 6
|
||
#define GAMMAVALUE_MIN 0.0625f
|
||
#define GAMMAVALUE_MAX 20.0f
|
||
#define CONTRAST_MIN 0.0625f
|
||
#define CONTRAST_MAX 10.0f
|
||
#define RENDERWEIGHT_MIN 0.0625f
|
||
#define RENDERWEIGHT_MAX 10.0f
|
||
#define NWEIGHT_MIN -64
|
||
#define NWEIGHT_MAX +64
|
||
#define BWEIGHT_MIN -32
|
||
#define BWEIGHT_MAX +32
|
||
#define SLANT_MIN -32
|
||
#define SLANT_MAX +32
|
||
|
||
CGdippSettings* CGdippSettings::CreateInstance()
|
||
{
|
||
CCriticalSectionLock __lock(CCriticalSectionLock::CS_SETTING);
|
||
CGdippSettings* pSettings = new CGdippSettings;
|
||
CGdippSettings* pOldSettings = reinterpret_cast<CGdippSettings*>(InterlockedExchangePointer(reinterpret_cast<void**>(&s_pInstance), pSettings));
|
||
_ASSERTE(pOldSettings == NULL);
|
||
int nSize = GetModuleFileName(NULL, pSettings->m_szexeName, MAX_PATH);
|
||
for (int i = nSize; i > 0; --i) {
|
||
if (pSettings->m_szexeName[i] == _T('\\')) {
|
||
StringCchCopy(pSettings->m_szexeName, nSize - i, pSettings->m_szexeName + i + 1);
|
||
break;
|
||
}
|
||
}
|
||
return pSettings;
|
||
}
|
||
|
||
void CGdippSettings::DestroyInstance()
|
||
{
|
||
CCriticalSectionLock __lock(CCriticalSectionLock::CS_SETTING);
|
||
|
||
CGdippSettings* pSettings = reinterpret_cast<CGdippSettings*>(InterlockedExchangePointer(reinterpret_cast<void**>(&s_pInstance), NULL));
|
||
if (pSettings) {
|
||
delete pSettings;
|
||
}
|
||
}
|
||
|
||
CGdippSettings* CGdippSettings::GetInstance()
|
||
{
|
||
CCriticalSectionLock __lock(CCriticalSectionLock::CS_SETTING);
|
||
CGdippSettings* pSettings = s_pInstance;
|
||
_ASSERTE(pSettings != NULL);
|
||
|
||
if (!pSettings->m_bDelayedInit) {
|
||
pSettings->DelayedInit();
|
||
}
|
||
return pSettings;
|
||
}
|
||
|
||
const CGdippSettings* CGdippSettings::GetInstanceNoInit()
|
||
{
|
||
CCriticalSectionLock __lock(CCriticalSectionLock::CS_SETTING);
|
||
CGdippSettings* pSettings = s_pInstance;
|
||
_ASSERTE(pSettings != NULL);
|
||
return pSettings;
|
||
}
|
||
|
||
void CGdippSettings::DelayedInit()
|
||
{
|
||
if (!g_pFTEngine) {
|
||
return;
|
||
}
|
||
if (IsBadCodePtr((FARPROC)RegOpenKeyExW) || *(DWORD_PTR*)RegOpenKeyExW==0)
|
||
return;
|
||
/*
|
||
In Windows 8, this call will fail in restricted environment
|
||
if GetDC failed, we are in a restricted environment where features like font subsitutations doesn't work properly.
|
||
Because these features requires DC to get the real font name and we don't have access to any DC, even CreateCompatibleDC(null) fails (it will succeed, but DeleteDC will fail)
|
||
So it is better exit than doing initialization.
|
||
*/
|
||
m_bDelayedInit = true;
|
||
|
||
HDC hdcScreen = GetDC(NULL);
|
||
if (!hdcScreen) {
|
||
return;
|
||
}
|
||
|
||
//ForceChangeFont
|
||
//if (m_szForceChangeFont[0]) {
|
||
// EnumFontFamilies(hdcScreen, m_szForceChangeFont, EnumFontFamProc, reinterpret_cast<LPARAM>(this));
|
||
//}
|
||
//fetch screen dpi
|
||
m_nScreenDpi = GetDeviceCaps(hdcScreen, LOGPIXELSX);
|
||
ReleaseDC(NULL, hdcScreen);
|
||
|
||
// //FontLink
|
||
// if (FontLink()) {
|
||
// m_fontlinkinfo.init();
|
||
// }
|
||
|
||
|
||
const int nTextTuning = _GetFreeTypeProfileInt(_T("TextTuning"), 0, NULL),
|
||
nTextTuningR = _GetFreeTypeProfileInt(_T("TextTuningR"), 0, NULL),
|
||
nTextTuningG = _GetFreeTypeProfileInt(_T("TextTuningG"), 0, NULL),
|
||
nTextTuningB = _GetFreeTypeProfileInt(_T("TextTuningB"), 0, NULL);
|
||
InitInitTuneTable();
|
||
InitTuneTable(nTextTuning, m_nTuneTable);
|
||
InitTuneTable(nTextTuningR, m_nTuneTableR);
|
||
InitTuneTable(nTextTuningG, m_nTuneTableG);
|
||
InitTuneTable(nTextTuningB, m_nTuneTableB);
|
||
RefreshAlphaTable();
|
||
|
||
//FontSubstitutes
|
||
CFontSubstitutesIniArray arrFontSubstitutes;
|
||
wstring names = _T("FontSubstitutes@") + wstring(m_szexeName);
|
||
if (_IsFreeTypeProfileSectionExists(names.c_str(), m_szFileName))
|
||
AddListFromSection(names.c_str(), m_szFileName, arrFontSubstitutes);
|
||
else
|
||
AddListFromSection(_T("FontSubstitutes"), m_szFileName, arrFontSubstitutes);
|
||
m_FontSubstitutesInfo.init(m_nFontSubstitutes, arrFontSubstitutes);
|
||
|
||
names = _T("Individual@") + wstring(m_szexeName);
|
||
if (_IsFreeTypeProfileSectionExists(names.c_str(), NULL))
|
||
AddIndividualFromSection(names.c_str(), NULL, m_arrIndividual);
|
||
else
|
||
AddIndividualFromSection(_T("Individual"), NULL, m_arrIndividual);
|
||
|
||
AddExcludeListFromSection(_T("Exclude"), NULL, m_arrExcludeFont);
|
||
AddExcludeListFromSection(_T("Include"), NULL, m_arrIncludeFont); //I know it's include not exclude, but they share the same logic.
|
||
//WritePrivateProfileString(NULL, NULL, NULL, m_szFileName);
|
||
|
||
//m_bDelayedInit = true;
|
||
|
||
//FontLink
|
||
if (FontLink()) {
|
||
m_fontlinkinfo.init();
|
||
}
|
||
|
||
// Init LCD settings
|
||
this->m_bHarmonyLCDRendering = FT_Library_SetLcdFilter(NULL, FT_LCD_FILTER_NONE) == FT_Err_Unimplemented_Feature;
|
||
if (this->m_bHarmonyLCDRendering) {
|
||
// Harmony LCD rendering
|
||
if (m_bUseCustomPixelLayout) {
|
||
FT_Vector sub[3] = { { m_arrPixelLayout[0], m_arrPixelLayout[1]},
|
||
{m_arrPixelLayout[2], m_arrPixelLayout[3]},
|
||
{m_arrPixelLayout[4], m_arrPixelLayout[5]}}; // custom layout
|
||
FT_Library_SetLcdGeometry(freetype_library, sub);
|
||
}
|
||
else {
|
||
switch (this->m_FontSettings.GetAntiAliasMode()) {
|
||
case 0:
|
||
case 1: {
|
||
FT_Vector sub[3] = { { 0, 0 }, { 0, 0 }, { 0, 0 } }; // gray scale
|
||
FT_Library_SetLcdGeometry(freetype_library, sub);
|
||
break;
|
||
}
|
||
case 2: //RGB
|
||
case 4: {
|
||
FT_Vector sub[3] = { { -21, 0 }, { 0, 0 }, { 21, 0 } };
|
||
FT_Library_SetLcdGeometry(freetype_library, sub);
|
||
break;
|
||
}
|
||
case 3: //BGR
|
||
case 5: {
|
||
FT_Vector sub[3] = { { 21, 0 }, { 0, 0 }, { -21, 0 } };
|
||
FT_Library_SetLcdGeometry(freetype_library, sub);
|
||
break;
|
||
}
|
||
case 6: {
|
||
//Pentile
|
||
FT_Vector sub[3] = { {-11, 16}, {-11, -16}, {22, 0} };
|
||
FT_Library_SetLcdGeometry(freetype_library, sub);
|
||
break;
|
||
}
|
||
}
|
||
if (m_FontSettings.GetAntiAliasMode() > 2)
|
||
m_FontSettings.SetAntiAliasMode(2); // all non-grayscale panel should use DrawLCD routine as its output.
|
||
}
|
||
}
|
||
else {
|
||
int nLcdFilter = LcdFilter();
|
||
if ((int)FT_LCD_FILTER_NONE < nLcdFilter && nLcdFilter < (int)FT_LCD_FILTER_MAX) {
|
||
switch (GetFontSettings().GetAntiAliasMode()) {
|
||
case 1:
|
||
case 4:
|
||
case 5:
|
||
nLcdFilter = FT_LCD_FILTER_LIGHT; // now we apply a light filter to lcd based on AA mode automatically, unless a custom lcd filter is defined.
|
||
}
|
||
FT_Library_SetLcdFilter(freetype_library, (FT_LcdFilter)nLcdFilter);
|
||
if (UseCustomLcdFilter())
|
||
{
|
||
unsigned char buff[5];
|
||
memcpy(buff, LcdFilterWeights(), sizeof(buff));
|
||
FT_Library_SetLcdFilterWeights(freetype_library, buff);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
//強制フォント
|
||
/*
|
||
LPCTSTR lpszFace = GetForceFontName();
|
||
if (lpszFace)
|
||
g_pFTEngine->AddFont(lpszFace, FW_NORMAL, false);*/
|
||
|
||
|
||
/* DWORD dwVersion = GetVersion();
|
||
|
||
if (m_bDirectWrite && (DWORD)(LOBYTE(LOWORD(dwVersion)))>5) //vista or later
|
||
{
|
||
if (GetModuleHandle(_T("d2d1.dll"))) //directwrite support
|
||
HookD2D1();
|
||
}*/
|
||
}
|
||
|
||
bool CGdippSettings::LoadSettings(HINSTANCE hModule)
|
||
{
|
||
CCriticalSectionLock __lock(CCriticalSectionLock::CS_SETTING);
|
||
int nSize = ::GetModuleFileName(hModule, m_szFileName, MAX_PATH - sizeof(".ini") + 1);
|
||
if (!nSize) {
|
||
return false;
|
||
}
|
||
ChangeFileName(m_szFileName, nSize, L"MacType.ini");
|
||
|
||
return LoadAppSettings(m_szFileName);
|
||
}
|
||
|
||
int CGdippSettings::_GetFreeTypeProfileIntFromSection(LPCTSTR lpszSection, LPCTSTR lpszKey, int nDefault, LPCTSTR lpszFile)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToInt();
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
return m_Config[lpszSection][lpszKey].ToInt();
|
||
else
|
||
return nDefault;
|
||
}
|
||
|
||
bool CGdippSettings::_GetFreeTypeProfileBoolFromSection(LPCTSTR lpszSection, LPCTSTR lpszKey, bool nDefault, LPCTSTR lpszFile)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToBool();
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
return m_Config[lpszSection][lpszKey].ToBool();
|
||
else
|
||
return nDefault;
|
||
}
|
||
|
||
wstring CGdippSettings::_GetFreeTypeProfileStrFromSection(LPCTSTR lpszSection, LPCTSTR lpszKey, const TCHAR* nDefault, LPCTSTR lpszFile)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToString();
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
return m_Config[lpszSection][lpszKey].ToString();
|
||
else
|
||
return nDefault;
|
||
}
|
||
|
||
int CGdippSettings::_GetFreeTypeProfileInt(LPCTSTR lpszKey, int nDefault, LPCTSTR lpszFile)
|
||
{
|
||
int ret = _GetFreeTypeProfileIntFromSection(c_szFreeType, lpszKey, nDefault, lpszFile);
|
||
if (ret == nDefault)
|
||
return _GetFreeTypeProfileIntFromSection(c_szGeneral, lpszKey, nDefault, lpszFile);
|
||
else
|
||
return ret;
|
||
}
|
||
|
||
int CGdippSettings::_GetFreeTypeProfileBoundInt(LPCTSTR lpszKey, int nDefault, int nMin, int nMax, LPCTSTR lpszFile)
|
||
{
|
||
const int ret = _GetFreeTypeProfileInt(lpszKey, nDefault, lpszFile);
|
||
return Bound(ret, nMin, nMax);
|
||
}
|
||
|
||
bool CGdippSettings::_IsFreeTypeProfileSectionExists(LPCTSTR lpszKey, LPCTSTR lpszFile)
|
||
{
|
||
return m_Config.IsPartExists(lpszKey);
|
||
}
|
||
|
||
float CGdippSettings::FastGetProfileFloat(LPCTSTR lpszSection, LPCTSTR lpszKey, float fDefault)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToDouble();
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
return m_Config[lpszSection][lpszKey].ToDouble();
|
||
else
|
||
return fDefault;
|
||
}
|
||
|
||
int CGdippSettings::FastGetProfileInt(LPCTSTR lpszSection, LPCTSTR lpszKey, int nDefault)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToInt();
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
return m_Config[lpszSection][lpszKey].ToInt();
|
||
else
|
||
return nDefault;
|
||
}
|
||
|
||
float CGdippSettings::_GetFreeTypeProfileFloat(LPCTSTR lpszKey, float fDefault, LPCTSTR lpszFile)
|
||
{
|
||
wstring names = wstring((LPTSTR)c_szFreeType) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToInt();
|
||
else
|
||
{
|
||
names = wstring((LPTSTR)c_szGeneral) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
return m_Config[names.c_str()][lpszKey].ToDouble();
|
||
else
|
||
if (m_Config[c_szFreeType].IsValueExists(lpszKey))
|
||
return m_Config[c_szFreeType][lpszKey].ToDouble();
|
||
if (m_Config[c_szGeneral].IsValueExists(lpszKey))
|
||
return m_Config[c_szGeneral][lpszKey].ToDouble();
|
||
else
|
||
return fDefault;
|
||
}
|
||
}
|
||
|
||
float CGdippSettings::_GetFreeTypeProfileBoundFloat(LPCTSTR lpszKey, float fDefault, float fMin, float fMax, LPCTSTR lpszFile)
|
||
{
|
||
const float ret = _GetFreeTypeProfileFloat(lpszKey, fDefault, lpszFile);
|
||
return Bound(ret, fMin, fMax);
|
||
}
|
||
|
||
DWORD CGdippSettings::FastGetProfileString(LPCTSTR lpszSection, LPCTSTR lpszKey, LPCTSTR lpszDefault, LPTSTR lpszRet, DWORD cch)
|
||
{
|
||
wstring names = wstring((LPTSTR)lpszSection) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[names.c_str()][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
if (m_Config[lpszSection].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[lpszSection][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
{
|
||
if (lpszDefault) {
|
||
StringCchCopy(lpszRet, cch, lpszDefault);
|
||
return wcslen(lpszDefault);
|
||
}
|
||
else {
|
||
lpszRet = NULL;
|
||
return 0;
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
DWORD CGdippSettings::_GetFreeTypeProfileString(LPCTSTR lpszKey, LPCTSTR lpszDefault, LPTSTR lpszRet, DWORD cch, LPCTSTR lpszFile)
|
||
{
|
||
wstring names = wstring((LPTSTR)c_szFreeType) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[names.c_str()][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
{
|
||
names = wstring((LPTSTR)c_szGeneral) + _T("@") + wstring((LPTSTR)m_szexeName);
|
||
if (m_Config.IsPartExists(names.c_str()) && m_Config[names.c_str()].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[names.c_str()][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
if (m_Config[c_szFreeType].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[c_szFreeType][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
if (m_Config[c_szGeneral].IsValueExists(lpszKey))
|
||
{
|
||
LPCTSTR p = m_Config[c_szGeneral][lpszKey];
|
||
StringCchCopy(lpszRet, cch, p);
|
||
return wcslen(p);
|
||
}
|
||
else
|
||
{
|
||
StringCchCopy(lpszRet, cch, lpszDefault);
|
||
return wcslen(lpszDefault);
|
||
}
|
||
}
|
||
}
|
||
|
||
void CGdippSettings::GetOSVersion() {
|
||
OSVERSIONINFO info;
|
||
memset(&info, 0, sizeof(OSVERSIONINFO));
|
||
info.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
||
|
||
GetVersionEx(&info);
|
||
m_dwOSMajorVer = info.dwMajorVersion;
|
||
m_dwOSMinorVer = info.dwMinorVersion;
|
||
}
|
||
|
||
bool CGdippSettings::LoadAppSettings(LPCTSTR lpszFile)
|
||
{
|
||
// 各種設定読み込み
|
||
// INIファイルの例:
|
||
// [General]
|
||
// HookChildProcesses=0
|
||
// HintingMode=0
|
||
// AntiAliasMode=0
|
||
// NormalWeight=0
|
||
// BoldWeight=0
|
||
// ItalicSlant=0
|
||
// EnableKerning=0
|
||
// MaxHeight=0
|
||
// ForceChangeFont=MS Pゴシック
|
||
// TextTuning=0
|
||
// TextTuningR=0
|
||
// TextTuningG=0
|
||
// TextTuningB=0
|
||
// CacheMaxFaces=0
|
||
// CacheMaxSizes=0
|
||
// CacheMaxBytes=0
|
||
// AlternativeFile=
|
||
// LoadOnDemand=0
|
||
// UseMapping=0
|
||
// LcdFilter=0
|
||
// Shadow=1,1,4
|
||
// [Individual]
|
||
// MS Pゴシック=0,1,2,3,4,5
|
||
GetOSVersion();
|
||
WritePrivateProfileString(NULL, NULL, NULL, lpszFile);
|
||
|
||
m_Config.Clear();
|
||
m_Config.LoadFromFile(lpszFile);
|
||
|
||
TCHAR szAlternative[MAX_PATH], szMainFile[MAX_PATH];
|
||
if (FastGetProfileString(c_szGeneral, _T("AlternativeFile"), _T(""), szAlternative, MAX_PATH)) {
|
||
if (PathIsRelative(szAlternative)) {
|
||
TCHAR szDir[MAX_PATH];
|
||
StringCchCopy(szDir, MAX_PATH, lpszFile);
|
||
PathRemoveFileSpec(szDir);
|
||
PathCombine(szAlternative, szDir, szAlternative);
|
||
}
|
||
StringCchCopy(szMainFile, MAX_PATH, lpszFile); //把原始文件名保存下来
|
||
StringCchCopy(m_szFileName, MAX_PATH, szAlternative);
|
||
lpszFile = m_szFileName;
|
||
m_Config.Clear();
|
||
m_Config.LoadFromFile(lpszFile);
|
||
}
|
||
|
||
_GetAlternativeProfileName(m_szexeName, lpszFile);
|
||
CFontSettings& fs = m_FontSettings;
|
||
fs.Clear();
|
||
fs.SetHintingMode(_GetFreeTypeProfileBoundInt(_T("HintingMode"), 0, HINTING_MIN, HINTING_MAX, lpszFile));
|
||
fs.SetAntiAliasMode(_GetFreeTypeProfileBoundInt(_T("AntiAliasMode"), 0, AAMODE_MIN, AAMODE_MAX, lpszFile));
|
||
fs.SetNormalWeight(_GetFreeTypeProfileBoundInt(_T("NormalWeight"), 0, NWEIGHT_MIN, NWEIGHT_MAX, lpszFile));
|
||
fs.SetBoldWeight(_GetFreeTypeProfileBoundInt(_T("BoldWeight"), 0, BWEIGHT_MIN, BWEIGHT_MAX, lpszFile));
|
||
fs.SetItalicSlant(_GetFreeTypeProfileBoundInt(_T("ItalicSlant"), 0, SLANT_MIN, SLANT_MAX, lpszFile));
|
||
fs.SetKerning(!!_GetFreeTypeProfileInt(_T("EnableKerning"), 0, lpszFile));
|
||
m_nAntiAliasModeForDW = fs.GetAntiAliasMode(); // DirectWrite always use the user defined AA mode.
|
||
{
|
||
TCHAR szShadow[256];
|
||
CStringTokenizer token;
|
||
m_bEnableShadow = false;
|
||
if (!_GetFreeTypeProfileString(_T("Shadow"), _T(""), szShadow, countof(szShadow), lpszFile)
|
||
|| token.Parse(szShadow) < 3) {
|
||
goto SKIP;
|
||
}
|
||
for (int i=0; i<3; i++) {
|
||
m_nShadow[i] = _StrToInt(token.GetArgument(i), 0);
|
||
/*if (m_nShadow[i] <= 0) {
|
||
goto SKIP;
|
||
}*/
|
||
}
|
||
m_bEnableShadow = true;
|
||
if (token.GetCount()>=4) //如果指定了浅色阴影
|
||
m_nShadowDarkColor = _httoi(token.GetArgument(3)); //读取阴影
|
||
else
|
||
m_nShadowDarkColor = 0; //否则为黑色
|
||
if (token.GetCount()>=6) //如果指定了甥瀚阴影
|
||
{
|
||
m_nShadowLightColor = _httoi(token.GetArgument(5)); //读取阴影
|
||
m_nShadow[3] = _StrToInt(token.GetArgument(4), m_nShadow[2]); //读取甥胰
|
||
}
|
||
else
|
||
{
|
||
m_nShadowLightColor = m_nShadowDarkColor; //否则和浅色阴影相同
|
||
m_nShadow[3] = m_nShadow[2]; //甥胰也相同
|
||
}
|
||
SKIP:
|
||
;
|
||
}
|
||
|
||
m_bHookChildProcesses = !!_GetFreeTypeProfileInt(_T("HookChildProcesses"), false, lpszFile);
|
||
m_bUseMapping = !!_GetFreeTypeProfileInt(_T("UseMapping"), false, lpszFile);
|
||
m_nBolderMode = _GetFreeTypeProfileInt(_T("BolderMode"), 0, lpszFile);
|
||
m_nGammaMode = _GetFreeTypeProfileInt(_T("GammaMode"), -1, lpszFile);
|
||
m_fGammaValue = _GetFreeTypeProfileBoundFloat(_T("GammaValue"), 1.0f, GAMMAVALUE_MIN, GAMMAVALUE_MAX, lpszFile);
|
||
m_fRenderWeight = _GetFreeTypeProfileBoundFloat(_T("RenderWeight"), 1.0f, RENDERWEIGHT_MIN, RENDERWEIGHT_MAX, lpszFile);
|
||
m_fContrast = _GetFreeTypeProfileBoundFloat(_T("Contrast"), 1.0f, CONTRAST_MIN, CONTRAST_MAX, lpszFile);
|
||
|
||
//DirectWrite/Direct2D exclusive settings
|
||
float fCalculatedDWGamma = m_fGammaValue*m_fGammaValue > 1.3 ? m_fGammaValue * m_fGammaValue / 2 : 0.7f;
|
||
// if not set, use calculated gamma as DW gamma
|
||
m_fGammaValueForDW = Bound(FastGetProfileFloat(c_szDirectWrite, _T("GammaValue"), fCalculatedDWGamma), 0.0f, GAMMAVALUE_MAX);
|
||
m_fContrastForDW = Bound(FastGetProfileFloat(c_szDirectWrite, _T("Contrast"), 1.0f), CONTRAST_MIN, CONTRAST_MAX);
|
||
m_nRenderingModeForDW = Bound(FastGetProfileInt(c_szDirectWrite, _T("RenderingMode"), 5), 0, 6);
|
||
m_fClearTypeLevelForDW = Bound(FastGetProfileFloat(c_szDirectWrite, _T("ClearTypeLevel"), 1.0f), 0.0f, 1.0f);
|
||
|
||
#ifdef _DEBUG
|
||
// GammaValue検証用
|
||
//CHAR GammaValueTest[1025];
|
||
//sprintf(GammaValueTest, "GammaValue=%.6f\nContrast=%.6f\n", m_fGammaValue, m_fContrast);
|
||
//MessageBoxA(NULL, GammaValueTest, "GammaValueテスト", 0);
|
||
#endif
|
||
m_bLoadOnDemand = !!_GetFreeTypeProfileInt(_T("LoadOnDemand"), false, lpszFile);
|
||
m_bFontLink = _GetFreeTypeProfileInt(_T("FontLink"), 0, lpszFile);
|
||
|
||
m_bIsInclude = !!_GetFreeTypeProfileInt(_T("UseInclude"), false, lpszFile);
|
||
m_nMaxHeight = _GetFreeTypeProfileBoundInt(_T("MaxHeight"), 0, 0, 0xfff, lpszFile); //赃只能到65535,cache的限制,而且大字体无实际价值
|
||
m_nMinHeight = _GetFreeTypeProfileBoundInt(_T("MinHeight"), 0, 0,
|
||
(m_nMaxHeight) ? m_nMaxHeight : 0xfff, // shouldn't be greater than MaxHeight unless it is undefined
|
||
lpszFile); //Minimum size of rendered font. DPI aware alternative.
|
||
//patched by krrr https://github.com/krrr/mactype/commit/146a213e2304208cb3c1a3e6fa941a386d908761
|
||
m_nBitmapHeight = _GetFreeTypeProfileBoundInt(_T("MaxBitmap"), 0, 0, 255, lpszFile);
|
||
m_bHintSmallFont = _GetFreeTypeProfileInt(_T("HintSmallFont"), 0, lpszFile);
|
||
m_bDirectWrite = _GetFreeTypeProfileInt(_T("DirectWrite"), 0, lpszFile);
|
||
m_nLcdFilter = _GetFreeTypeProfileInt(_T("LcdFilter"), 0, lpszFile);
|
||
m_nFontSubstitutes = _GetFreeTypeProfileBoundInt(_T("FontSubstitutes"),
|
||
SETTING_FONTSUBSTITUTE_DISABLE,
|
||
SETTING_FONTSUBSTITUTE_DISABLE,
|
||
SETTING_FONTSUBSTITUTE_ALL,
|
||
lpszFile);
|
||
m_nWidthMode = SETTING_WIDTHMODE_GDI32;
|
||
/*
|
||
_GetFreeTypeProfileBoundInt(_T("WidthMode"),
|
||
SETTING_WIDTHMODE_GDI32,
|
||
SETTING_WIDTHMODE_GDI32,
|
||
SETTING_WIDTHMODE_FREETYPE,
|
||
lpszFile);*/
|
||
|
||
m_nFontLoader = _GetFreeTypeProfileBoundInt(_T("FontLoader"),
|
||
SETTING_FONTLOADER_FREETYPE,
|
||
SETTING_FONTLOADER_FREETYPE,
|
||
SETTING_FONTLOADER_WIN32,
|
||
lpszFile);
|
||
m_nCacheMaxFaces = _GetFreeTypeProfileInt(_T("CacheMaxFaces"), 64, lpszFile);
|
||
m_nCacheMaxFaces = m_nCacheMaxFaces > 64 ? m_nCacheMaxFaces : 64;
|
||
m_nCacheMaxSizes = _GetFreeTypeProfileInt(_T("CacheMaxSizes"), 1200, lpszFile);
|
||
m_nCacheMaxBytes = _GetFreeTypeProfileInt(_T("CacheMaxBytes"), 10485760, lpszFile);
|
||
|
||
//experimental settings:
|
||
m_bEnableClipBoxFix = !!_GetFreeTypeProfileIntFromSection(_T("Experimental"), _T("ClipBoxFix"), 1, lpszFile);
|
||
m_bColorFont = !!_GetFreeTypeProfileIntFromSection(_T("Experimental"), _T("ColorFont"), 0, lpszFile);
|
||
m_bInvertColor = !!_GetFreeTypeProfileIntFromSection(_T("Experimental"), _T("InvertColor"), 0, lpszFile);
|
||
#ifdef INFINALITY
|
||
// define some macros
|
||
#define INF_INT_ENV(y, def) \
|
||
nTemp = _GetFreeTypeProfileIntFromSection(_T("Infinality"), _T(y), def, lpszFile); \
|
||
FT_PutEnv(y, _ltoa(nTemp, buff, 10));
|
||
#define INF_BOOL_ENV(y, def) \
|
||
bTemp = _GetFreeTypeProfileBoolFromSection(_T("Infinality"), _T(y), def, lpszFile); \
|
||
FT_PutEnv(y, bTemp?"true":"false");
|
||
#define INF_STR_ENV(y, def) \
|
||
sTemp = _GetFreeTypeProfileStrFromSection(_T("Infinality"), _T(y), def, lpszFile); \
|
||
FT_PutEnv(y, WstringToString(sTemp).c_str());
|
||
|
||
char* buff = (char*)malloc(256);
|
||
int nTemp; bool bTemp; wstring sTemp;
|
||
|
||
// INFINALITY settings:
|
||
INF_INT_ENV( "INFINALITY_FT_CHROMEOS_STYLE_SHARPENING_STRENGTH", 0);
|
||
INF_INT_ENV( "INFINALITY_FT_CONTRAST", 0);
|
||
INF_INT_ENV( "INFINALITY_FT_STEM_FITTING_STRENGTH", 25);
|
||
INF_INT_ENV( "INFINALITY_FT_AUTOHINT_SNAP_STEM_HEIGHT", 100);
|
||
INF_INT_ENV( "INFINALITY_FT_GRAYSCALE_FILTER_STRENGTH", 0);
|
||
INF_INT_ENV( "INFINALITY_FT_WINDOWS_STYLE_SHARPENING_STRENGTH", 20);
|
||
INF_INT_ENV( "INFINALITY_FT_BRIGHTNESS", 0);
|
||
INF_INT_ENV( "INFINALITY_FT_AUTOHINT_HORIZONTAL_STEM_DARKEN_STRENGTH", 10);
|
||
INF_INT_ENV( "INFINALITY_FT_STEM_ALIGNMENT_STRENGTH", 25);
|
||
INF_INT_ENV( "INFINALITY_FT_AUTOHINT_VERTICAL_STEM_DARKEN_STRENGTH", 25);
|
||
INF_INT_ENV( "INFINALITY_FT_FRINGE_FILTER_STRENGTH", 0);
|
||
INF_INT_ENV("INFINALITY_FT_GLOBAL_EMBOLDEN_X_VALUE", 0);
|
||
INF_INT_ENV("INFINALITY_FT_GLOBAL_EMBOLDEN_Y_VALUE", 0);
|
||
INF_INT_ENV("INFINALITY_FT_BOLD_EMBOLDEN_X_VALUE", 0);
|
||
INF_INT_ENV("INFINALITY_FT_BOLD_EMBOLDEN_Y_VALUE", 0);
|
||
INF_INT_ENV("INFINALITY_FT_STEM_SNAPPING_SLIDING_SCALE", 0);
|
||
|
||
INF_BOOL_ENV("INFINALITY_FT_USE_KNOWN_SETTINGS_ON_SELECTED_FONTS", true);
|
||
INF_BOOL_ENV( "INFINALITY_FT_AUTOFIT_ADJUST_HEIGHTS", true);
|
||
INF_BOOL_ENV( "INFINALITY_FT_USE_VARIOUS_TWEAKS", true);
|
||
INF_BOOL_ENV( "INFINALITY_FT_AUTOHINT_INCREASE_GLYPH_HEIGHTS", true);
|
||
INF_BOOL_ENV( "INFINALITY_FT_STEM_DARKENING_CFF", true);
|
||
INF_BOOL_ENV( "INFINALITY_FT_STEM_DARKENING_AUTOFIT", true);
|
||
|
||
INF_STR_ENV( "INFINALITY_FT_GAMMA_CORRECTION", _T("0 100"));
|
||
INF_STR_ENV( "INFINALITY_FT_FILTER_PARAMS", _T("11 22 38 22 11"));
|
||
|
||
free(buff);
|
||
#endif
|
||
|
||
if (m_nFontLoader == SETTING_FONTLOADER_WIN32) {
|
||
// APIが処理してくれるはずなので自前処理は無効化
|
||
if (m_nFontSubstitutes == SETTING_FONTSUBSTITUTE_ALL) {
|
||
m_nFontSubstitutes = SETTING_FONTSUBSTITUTE_DISABLE;
|
||
}
|
||
m_bFontLink = 0;
|
||
}
|
||
|
||
// フォント指定
|
||
ZeroMemory(&m_lfForceFont, sizeof(LOGFONT));
|
||
m_szForceChangeFont[0] = _T('\0');
|
||
//_GetFreeTypeProfileString(_T("ForceChangeFont"), _T(""), m_szForceChangeFont, LF_FACESIZE, lpszFile);
|
||
|
||
// OSのバージョンがXP以降かどうか
|
||
//OSVERSIONINFO osvi = { sizeof(OSVERSIONINFO) };
|
||
//GetVersionEx(&osvi);
|
||
m_bIsWinXPorLater = IsWindowsXPOrGreater();
|
||
|
||
STARTUPINFO si = { sizeof(STARTUPINFO) };
|
||
GetStartupInfo(&si);
|
||
m_bRunFromGdiExe = IsGdiPPStartupInfo(si);
|
||
// if (!m_bRunFromGdiExe) {
|
||
// m_bHookChildProcesses = false;
|
||
// }
|
||
/*
|
||
const int nTextTuning = _GetFreeTypeProfileInt(_T("TextTuning"), 0, lpszFile),
|
||
nTextTuningR = _GetFreeTypeProfileInt(_T("TextTuningR"), 0, lpszFile),
|
||
nTextTuningG = _GetFreeTypeProfileInt(_T("TextTuningG"), 0, lpszFile),
|
||
nTextTuningB = _GetFreeTypeProfileInt(_T("TextTuningB"), 0, lpszFile);
|
||
InitInitTuneTable();
|
||
InitTuneTable(nTextTuning, m_nTuneTable);
|
||
InitTuneTable(nTextTuningR, m_nTuneTableR);
|
||
InitTuneTable(nTextTuningG, m_nTuneTableG);
|
||
InitTuneTable(nTextTuningB, m_nTuneTableB);*/
|
||
// m_bIsHDBench = (GetModuleHandle(_T("HDBENCH.EXE")) == GetModuleHandle(NULL));
|
||
|
||
m_arrExcludeFont.clear();
|
||
m_arrIncludeFont.clear();
|
||
m_arrExcludeModule.clear();
|
||
m_arrIncludeModule.clear();
|
||
m_arrUnloadModule.clear();
|
||
m_arrUnFontSubModule.clear();
|
||
|
||
// [Exclude]セクションから除外フォントリストを読み込む
|
||
// [ExcludeModule]セクションから除外モジュールリストを読み込む
|
||
AddListFromSection(_T("ExcludeModule"), lpszFile, m_arrExcludeModule);
|
||
//AddListFromSection(_T("ExcludeModule"), szMainFile, m_arrExcludeModule);
|
||
// [IncludeModule]セクションから対象モジュールリストを読み込む
|
||
AddListFromSection(_T("IncludeModule"), lpszFile, m_arrIncludeModule);
|
||
//AddListFromSection(_T("IncludeModule"), szMainFile, m_arrIncludeModule);
|
||
// [UnloadDLL]蛠E患釉氐哪?丒
|
||
AddListFromSection(_T("UnloadDLL"), lpszFile, m_arrUnloadModule);
|
||
//AddListFromSection(_T("UnloadDLL"), szMainFile, m_arrUnloadModule);
|
||
// [ExcludeSub]不进行字体替换的模縼E
|
||
AddListFromSection(L"ExcludeSub", lpszFile, m_arrUnFontSubModule);
|
||
//AddListFromSection(L"ExcludeSub", szMainFile, m_arrUnFontSubModule);
|
||
//如果是排除的模块,则关闭字体替换
|
||
if (m_nFontSubstitutes)
|
||
{
|
||
ModuleHashMap::const_iterator it=m_arrUnFontSubModule.begin();
|
||
while (it!=m_arrUnFontSubModule.end())
|
||
{
|
||
if (GetModuleHandle(it->c_str()))
|
||
{
|
||
m_nFontSubstitutes = 0; //关闭替换
|
||
break;
|
||
}
|
||
++it;
|
||
}
|
||
}
|
||
|
||
// [Individual]セクションからフォント別設定を読み込む
|
||
wstring names = _T("LcdFilterWeight@") + wstring(m_szexeName);
|
||
if (_IsFreeTypeProfileSectionExists(names.c_str(), lpszFile))
|
||
m_bUseCustomLcdFilter = AddLcdFilterFromSection(names.c_str(), lpszFile, m_arrLcdFilterWeights);
|
||
else
|
||
m_bUseCustomLcdFilter = AddLcdFilterFromSection(_T("LcdFilterWeight"), lpszFile, m_arrLcdFilterWeights);
|
||
|
||
m_bUseCustomPixelLayout = AddPixelModeFromSection(_T("PixelLayout"), lpszFile, m_arrPixelLayout);
|
||
|
||
return true;
|
||
}
|
||
|
||
bool CGdippSettings::AddExcludeListFromSection(LPCTSTR lpszSection, LPCTSTR lpszFile, set<wstring> & arr)
|
||
{
|
||
LPTSTR buffer = _GetPrivateProfileSection(lpszSection, lpszFile);
|
||
if (buffer == NULL) {
|
||
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
||
return false;
|
||
}
|
||
|
||
LPTSTR p = buffer;
|
||
TCHAR buff[LF_FACESIZE+1];
|
||
LOGFONT truefont={0};
|
||
while (*p) {
|
||
bool b = false;
|
||
GetFontLocalName(p, buff);//转换字体脕E
|
||
set<wstring>::const_iterator it = arr.find(buff);
|
||
if (it==arr.end())
|
||
arr.insert(buff);
|
||
for (; *p; p++); //来到下一行
|
||
p++;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
//template <typename T>
|
||
bool CGdippSettings::AddListFromSection(LPCTSTR lpszSection, LPCTSTR lpszFile, set<wstring> & arr)
|
||
{
|
||
LPTSTR buffer = _GetPrivateProfileSection(lpszSection, lpszFile);
|
||
if (buffer == NULL) {
|
||
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
||
return false;
|
||
}
|
||
|
||
LPTSTR p = buffer;
|
||
while (*p) {
|
||
bool b = false;
|
||
set<wstring>::const_iterator it = arr.find(p);
|
||
if (it==arr.end())
|
||
arr.insert(p);
|
||
for (; *p; p++); //来到下一行
|
||
p++;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
bool CGdippSettings::AddLcdFilterFromSection(LPCTSTR lpszKey, LPCTSTR lpszFile, unsigned char* arr)
|
||
{
|
||
TCHAR buffer[100];
|
||
_GetFreeTypeProfileString(lpszKey, _T("\0"), buffer, sizeof(buffer), lpszFile);
|
||
if (buffer[0] == '\0') {
|
||
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
||
return false;
|
||
}
|
||
|
||
LPTSTR p = buffer;
|
||
CStringTokenizer token;
|
||
int argc = 0;
|
||
argc = token.Parse(buffer);
|
||
|
||
for (int i = 0; i < 5; i++) {
|
||
LPCTSTR arg = token.GetArgument(i);
|
||
if (!arg)
|
||
return false; //参数少于5个则视为不使用此参数
|
||
arr[i] = _StrToInt(arg, arr[i]);
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
bool CGdippSettings::AddPixelModeFromSection(LPCTSTR lpszKey, LPCTSTR lpszFile, char* arr)
|
||
{
|
||
TCHAR buffer[100];
|
||
_GetFreeTypeProfileString(lpszKey, _T("\0"), buffer, sizeof(buffer), lpszFile);
|
||
if (buffer[0] == '\0') {
|
||
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
||
return false;
|
||
}
|
||
|
||
LPTSTR p = buffer;
|
||
CStringTokenizer token;
|
||
int argc = 0;
|
||
argc = token.Parse(buffer);
|
||
|
||
for (int i = 0; i < 6; i++) {
|
||
LPCTSTR arg = token.GetArgument(i);
|
||
if (!arg)
|
||
return false;
|
||
arr[i] = _StrToInt(arg, arr[i]);
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
bool CGdippSettings::AddIndividualFromSection(LPCTSTR lpszSection, LPCTSTR lpszFile, IndividualArray& arr)
|
||
{
|
||
LPTSTR buffer = _GetPrivateProfileSection(lpszSection, lpszFile);
|
||
if (buffer == NULL) {
|
||
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
||
return false;
|
||
}
|
||
|
||
LPTSTR p = buffer;
|
||
TCHAR buff[LF_FACESIZE+1];
|
||
LOGFONT truefont={0};
|
||
while (*p) {
|
||
bool b = false;
|
||
|
||
LPTSTR pnext = p;
|
||
for (; *pnext; pnext++);
|
||
|
||
//"MS Pゴシック=0,0" みたいな文字列を分割
|
||
LPTSTR value = _tcschr(p, _T('='));
|
||
CStringTokenizer token;
|
||
int argc = 0;
|
||
if (value) {
|
||
*value++ = _T('\0');
|
||
argc = token.Parse(value);
|
||
}
|
||
|
||
GetFontLocalName(p, buff);//转换字体脕E
|
||
|
||
CFontIndividual fi(buff);
|
||
const CFontSettings& fsCommon = m_FontSettings;
|
||
CFontSettings& fs = fi.GetIndividual();
|
||
//Individualが無ければ共通設定を使う
|
||
fs = fsCommon;
|
||
for (int i = 0; i < MAX_FONT_SETTINGS; i++) {
|
||
LPCTSTR arg = token.GetArgument(i);
|
||
if (!arg)
|
||
break;
|
||
const int n = _StrToInt(arg, fsCommon.GetParam(i));
|
||
fs.SetParam(i, n);
|
||
}
|
||
|
||
for (int i = 0 ; i < arr.GetSize(); i++) {
|
||
if (arr[i] == fi) {
|
||
b = true;
|
||
break;
|
||
}
|
||
}
|
||
if (!b) {
|
||
arr.Add(fi);
|
||
#ifdef _DEBUG
|
||
TRACE(_T("Individual: %s, %d, %d, %d, %d, %d, %d\n"), fi.GetName(),
|
||
fs.GetParam(0), fs.GetParam(1), fs.GetParam(2), fs.GetParam(3), fs.GetParam(4), fs.GetParam(5));
|
||
#endif
|
||
}
|
||
p = pnext;
|
||
p++;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
LPTSTR CGdippSettings::_GetPrivateProfileSection(LPCTSTR lpszSection, LPCTSTR lpszFile)
|
||
{
|
||
return const_cast<LPTSTR>((LPCTSTR)m_Config[lpszSection]);
|
||
}
|
||
|
||
//atolにデフォルト値を返せるようにしたような物
|
||
int CGdippSettings::_StrToInt(LPCTSTR pStr, int nDefault)
|
||
{
|
||
#define isspace(ch) (ch == _T('\t') || ch == _T(' '))
|
||
#define isdigit(ch) ((_TUCHAR)(ch - _T('0')) <= 9)
|
||
|
||
int ret;
|
||
bool neg = false;
|
||
LPCTSTR pStart;
|
||
|
||
for (; isspace(*pStr); pStr++);
|
||
switch (*pStr) {
|
||
case _T('-'):
|
||
neg = true;
|
||
case _T('+'):
|
||
pStr++;
|
||
break;
|
||
}
|
||
|
||
pStart = pStr;
|
||
ret = 0;
|
||
for (; isdigit(*pStr); pStr++) {
|
||
ret = 10 * ret + (*pStr - _T('0'));
|
||
}
|
||
|
||
if (pStr == pStart) {
|
||
return nDefault;
|
||
}
|
||
return neg ? -ret : ret;
|
||
|
||
#undef isspace
|
||
#undef isdigit
|
||
}
|
||
|
||
int CGdippSettings::_httoi(const TCHAR *value)
|
||
{
|
||
struct CHexMap
|
||
{
|
||
TCHAR chr;
|
||
int value;
|
||
};
|
||
const int HexMapL = 16;
|
||
CHexMap HexMap[HexMapL] =
|
||
{
|
||
{'0', 0}, {'1', 1},
|
||
{'2', 2}, {'3', 3},
|
||
{'4', 4}, {'5', 5},
|
||
{'6', 6}, {'7', 7},
|
||
{'8', 8}, {'9', 9},
|
||
{'A', 10}, {'B', 11},
|
||
{'C', 12}, {'D', 13},
|
||
{'E', 14}, {'F', 15}
|
||
};
|
||
TCHAR *mstr = _tcsupr(_tcsdup(value));
|
||
TCHAR *s = mstr;
|
||
int result = 0;
|
||
if (*s == '0' && *(s + 1) == 'X') s += 2;
|
||
bool firsttime = true;
|
||
while (*s != '\0')
|
||
{
|
||
bool found = false;
|
||
for (int i = 0; i < HexMapL; i++)
|
||
{
|
||
if (*s == HexMap[i].chr)
|
||
{
|
||
if (!firsttime) result <<= 4;
|
||
result |= HexMap[i].value;
|
||
found = true;
|
||
break;
|
||
}
|
||
}
|
||
if (!found) break;
|
||
s++;
|
||
firsttime = false;
|
||
}
|
||
free(mstr);
|
||
return result;
|
||
}
|
||
|
||
//atofにデフォルト値を返せるようにしたような物
|
||
float CGdippSettings::_StrToFloat(LPCTSTR pStr, float fDefault)
|
||
{
|
||
#define isspace(ch) (ch == _T('\t') || ch == _T(' '))
|
||
#define isdigit(ch) ((_TUCHAR)(ch - _T('0')) <= 9)
|
||
|
||
int ret_i;
|
||
int ret_d;
|
||
float ret;
|
||
bool neg = false;
|
||
LPCTSTR pStart;
|
||
|
||
for (; isspace(*pStr); pStr++);
|
||
switch (*pStr) {
|
||
case _T('-'):
|
||
neg = true;
|
||
case _T('+'):
|
||
pStr++;
|
||
break;
|
||
}
|
||
|
||
pStart = pStr;
|
||
ret = 0;
|
||
ret_i = 0;
|
||
ret_d = 1;
|
||
for (; isdigit(*pStr); pStr++) {
|
||
ret_i = 10 * ret_i + (*pStr - _T('0'));
|
||
}
|
||
if (*pStr == _T('.')) {
|
||
pStr++;
|
||
for (; isdigit(*pStr); pStr++) {
|
||
ret_i = 10 * ret_i + (*pStr - _T('0'));
|
||
ret_d *= 10;
|
||
}
|
||
}
|
||
ret = (float)ret_i / (float)ret_d;
|
||
|
||
if (pStr == pStart) {
|
||
return fDefault;
|
||
}
|
||
return neg ? -ret : ret;
|
||
|
||
#undef isspace
|
||
#undef isdigit
|
||
}
|
||
|
||
bool CGdippSettings::IsFontExcluded(LPCSTR lpFaceName) const
|
||
{
|
||
WCHAR szStack[LF_FACESIZE];
|
||
LPWSTR lpUnicode = _StrDupExAtoW(lpFaceName, -1, szStack, LF_FACESIZE, NULL);
|
||
if (!lpUnicode) {
|
||
return false;
|
||
}
|
||
|
||
bool b = IsFontExcluded(lpUnicode);
|
||
if (lpUnicode != szStack)
|
||
free(lpUnicode);
|
||
return b;
|
||
}
|
||
|
||
bool CGdippSettings::IsFontExcluded(LPCWSTR lpFaceName) const
|
||
{
|
||
FontHashMap::const_iterator it = m_arrExcludeFont.find(lpFaceName);
|
||
bool bExcluded = it != m_arrExcludeFont.end(); // if it's excluded, true
|
||
if (!bExcluded && m_arrIncludeFont.size() != 0) { // if it's not excluded, and includefont enabled
|
||
FontHashMap::const_iterator it = m_arrIncludeFont.find(lpFaceName);
|
||
bExcluded = it == m_arrIncludeFont.end(); // check if it's included
|
||
}
|
||
return bExcluded;
|
||
}
|
||
|
||
void CGdippSettings::AddFontExclude(LPCWSTR lpFaceName)
|
||
{
|
||
if (!IsFontExcluded(lpFaceName))
|
||
m_arrExcludeFont.insert(lpFaceName);
|
||
}
|
||
|
||
bool CGdippSettings::IsProcessUnload() const
|
||
{
|
||
if (m_bRunFromGdiExe) {
|
||
return false;
|
||
}
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_LOAD", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return false;
|
||
ModuleHashMap::const_iterator it = m_arrUnloadModule.begin();
|
||
for(; it != m_arrUnloadModule.end(); ++it) {
|
||
if (IsFolder(it->c_str())) {
|
||
// if the user is trying to include a folder instead of a single executable.
|
||
if (GetAppDir() == LowerCase(*it)) {
|
||
return true;
|
||
}
|
||
}
|
||
else
|
||
if (GetModuleHandleW(it->c_str())) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
bool CGdippSettings::IsExeUnload(LPCTSTR lpApp) const //紒E槭欠裨诤诿チ斜?
|
||
{
|
||
if (m_bRunFromGdiExe) {
|
||
return false;
|
||
}
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_LOAD", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return false;
|
||
ModuleHashMap::const_iterator it = m_arrUnloadModule.begin();
|
||
for(; it != m_arrUnloadModule.end(); ++it) {
|
||
if (!lstrcmpi(lpApp, it->c_str())) { //匹配排除蟻E
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
bool CGdippSettings::IsExeInclude(LPCTSTR lpApp) const //紒E槭欠裨诎酌チ斜?
|
||
{
|
||
if (m_bRunFromGdiExe) {
|
||
return false;
|
||
}
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_LOAD", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return false;
|
||
ModuleHashMap::const_iterator it = m_arrIncludeModule.begin();
|
||
for(; it != m_arrIncludeModule.end(); ++it) {
|
||
if (!lstrcmpi(lpApp, it->c_str())) { //匹配排除蟻E
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
bool CGdippSettings::IsProcessExcluded() const
|
||
{
|
||
if (m_bRunFromGdiExe) {
|
||
return false;
|
||
}
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_EXCLUDE", NULL, 0);
|
||
if (GetLastError() != ERROR_ENVVAR_NOT_FOUND)
|
||
return true;
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_LOAD", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return false;
|
||
ModuleHashMap::const_iterator it = m_arrExcludeModule.begin();
|
||
for(; it != m_arrExcludeModule.end(); ++it) {
|
||
if (IsFolder(it->c_str())) {
|
||
// if the user is trying to exclude a folder instead of a single executable.
|
||
if (GetAppDir().find(LowerCase(*it)) == 0) {
|
||
return true;
|
||
}
|
||
}
|
||
else
|
||
if (GetModuleHandleW(it->c_str())) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
bool CGdippSettings::IsProcessIncluded() const
|
||
{
|
||
if (m_bRunFromGdiExe) {
|
||
return true;
|
||
}
|
||
GetEnvironmentVariableW(L"MACTYPE_FORCE_LOAD", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return true;
|
||
ModuleHashMap::const_iterator it = m_arrIncludeModule.begin();
|
||
for(; it != m_arrIncludeModule.end(); ++it) {
|
||
if (IsFolder(it->c_str())) {
|
||
// if the user is trying to include a folder instead of a single executable.
|
||
if (GetAppDir() == LowerCase(*it)) {
|
||
return true;
|
||
}
|
||
}
|
||
else
|
||
if (GetModuleHandleW(it->c_str())) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
void CGdippSettings::InitInitTuneTable()
|
||
{
|
||
int i, *table;
|
||
#define init_table(name) \
|
||
for (i=0,table=name; i<256; i++) table[i] = i
|
||
init_table(m_nTuneTable);
|
||
init_table(m_nTuneTableR);
|
||
init_table(m_nTuneTableG);
|
||
init_table(m_nTuneTableB);
|
||
#undef init_table
|
||
}
|
||
|
||
// テーブル初期化関数 0 - 12まで
|
||
// LCD用テーブル初期化関数 各0 - 12まで
|
||
void CGdippSettings::InitTuneTable(int v, int* table)
|
||
{
|
||
int i;
|
||
int col;
|
||
double tmp, p;
|
||
|
||
if (v < 0) {
|
||
return;
|
||
}
|
||
v = Min(v, 12);
|
||
p = (double)v;
|
||
p = 1 - (p / (p + 10.0));
|
||
for(i = 0;i < 256;i++){
|
||
tmp = (double)i / 255.0;
|
||
tmp = pow(tmp, p);
|
||
col = 255 - (int)(tmp * 255.0 + 0.5);
|
||
table[255 - i] = col;
|
||
}
|
||
}
|
||
|
||
//見つからない場合は共通設定を返す
|
||
extern BOOL g_ccbIndividual;
|
||
const CFontSettings& CGdippSettings::FindIndividual(LPCTSTR lpFaceName) const
|
||
{
|
||
CFontIndividual* p = m_arrIndividual.Begin();
|
||
CFontIndividual* end = m_arrIndividual.End();
|
||
if (lpFaceName && *lpFaceName==L'@')
|
||
++lpFaceName; //纵向字体使用横向的设定
|
||
StringHashFont hash(lpFaceName);
|
||
|
||
for(; p != end; ++p) {
|
||
if (p->GetHash() == hash) {
|
||
CFontSettings& result = p->GetIndividual();
|
||
if (result.GetAntiAliasMode() > 2 && HarmonyLCD())
|
||
result.SetAntiAliasMode(2);
|
||
}
|
||
}
|
||
return GetFontSettings();
|
||
}
|
||
|
||
int CGdippSettings::_GetAlternativeProfileName(LPTSTR lpszName, LPCTSTR lpszFile)
|
||
{
|
||
TCHAR szexe[MAX_PATH + 1];
|
||
TCHAR* pexe = szexe + GetModuleFileName(NULL, szexe, MAX_PATH);
|
||
while (pexe >= szexe && *pexe != '\\')
|
||
pexe--;
|
||
pexe++;
|
||
wstring exename = _T("General@") + wstring((LPTSTR)pexe);
|
||
if (FastGetProfileString(exename.c_str(), _T("Alternative"), NULL, lpszName, MAX_PATH))
|
||
{
|
||
return true;
|
||
}
|
||
else
|
||
{
|
||
//StringCchCopy(lpszName, MAX_PATH + 1, pexe);
|
||
return false;
|
||
}
|
||
}
|
||
|
||
bool CGdippSettings::CopyForceFont(LOGFONT& lf, const LOGFONT& lfOrg) const
|
||
{
|
||
_ASSERTE(m_bDelayedInit);
|
||
//__asm{ int 3 }
|
||
GetEnvironmentVariableW(L"MACTYPE_FONTSUBSTITUTES_ENV", NULL, 0);
|
||
if (GetLastError()!=ERROR_ENVVAR_NOT_FOUND)
|
||
return false;
|
||
//&lf == &lfOrgも可
|
||
bool bForceFont = false;
|
||
BOOL bFontExist = true;
|
||
const LOGFONT *lplf;
|
||
lplf = GetFontSubstitutesInfo().lookup((LOGFONT&)lfOrg);
|
||
if (lplf) bForceFont = true;
|
||
|
||
if (bForceFont) {
|
||
memcpy(&lf, &lfOrg, sizeof(LOGFONT)-sizeof(lf.lfFaceName));
|
||
StringCchCopy(lf.lfFaceName, LF_FACESIZE, lplf->lfFaceName);
|
||
}
|
||
return bForceFont;
|
||
}
|
||
|
||
//値的にchar(-128~127)で十分
|
||
const char CFontSettings::m_bound[MAX_FONT_SETTINGS][2] = {
|
||
{ HINTING_MIN, HINTING_MAX }, //Hinting
|
||
{ AAMODE_MIN, AAMODE_MAX }, //AAMode
|
||
{ NWEIGHT_MIN, NWEIGHT_MAX }, //NormalWeight
|
||
{ BWEIGHT_MIN, BWEIGHT_MAX }, //BoldWeight
|
||
{ SLANT_MIN, SLANT_MAX }, //ItalicSlant
|
||
{ 0, 1 }, //Kerning
|
||
};
|
||
|
||
CFontLinkInfo::CFontLinkInfo()
|
||
{
|
||
memset(&info, 0, sizeof info);
|
||
memset(AllowDefaultLink, 1, sizeof(AllowDefaultLink)); //默认允喧笾体链接
|
||
}
|
||
|
||
CFontLinkInfo::~CFontLinkInfo()
|
||
{
|
||
clear();
|
||
}
|
||
|
||
/*
|
||
static int CALLBACK EnumFontCallBack(const LOGFONT *lplf, const TEXTMETRIC *lptm, DWORD / *FontType* /, LPARAM lParam)
|
||
{
|
||
LOGFONT * lf=(LOGFONT *)lParam;
|
||
StringCchCopy(lf->lfFaceName, LF_FACESIZE, lplf->lfFaceName);
|
||
return 0;
|
||
}
|
||
|
||
static void GetFontLocalName(LOGFONT& lf) //获得字体的本地化名称
|
||
{
|
||
HDC dc=GetDC(NULL);
|
||
EnumFontFamiliesEx(dc, &lf, &EnumFontCallBack, (LPARAM)&lf, 0);
|
||
ReleaseDC(NULL, dc);
|
||
}*/
|
||
|
||
|
||
void CFontLinkInfo::init()
|
||
{
|
||
const TCHAR REGKEY1[] = _T("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\FontLink\\SystemLink");
|
||
const TCHAR REGKEY2[] = _T("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Fonts");
|
||
const TCHAR REGKEY3[] = _T("SYSTEM\\CurrentControlSet\\Control\\FontAssoc\\Associated DefaultFonts");
|
||
const TCHAR REGKEY4[] = _T("SYSTEM\\CurrentControlSet\\Control\\FontAssoc\\Associated Charset");
|
||
|
||
HKEY h1;
|
||
HKEY h2;
|
||
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY1, 0, KEY_QUERY_VALUE, &h1)) return;
|
||
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY2, 0, KEY_QUERY_VALUE, &h2)) {
|
||
RegCloseKey(h1);
|
||
return;
|
||
}
|
||
//OSVERSIONINFO sOsVinfo={sizeof(OSVERSIONINFO),0,0,0,0,{0}};
|
||
//GetVersionEx(&sOsVinfo); //获得操作系统版本号
|
||
//const CGdippSettings* pSettings = CGdippSettings::GetInstance();
|
||
|
||
WCHAR* name = new WCHAR[0x2000];
|
||
DWORD namesz;
|
||
DWORD valuesz;
|
||
WCHAR* value = new WCHAR[0x2000];
|
||
WCHAR* buf = new WCHAR[0x2000];
|
||
const DWORD nBufSize = 0x2000 * sizeof(WCHAR);
|
||
LONG rc;
|
||
DWORD regtype;
|
||
|
||
for (int k = 0; ; ++k) { //获得字体柄蛐的所有字虂E
|
||
namesz = nBufSize;
|
||
valuesz = nBufSize;
|
||
rc = RegEnumValue(h2, k, name, &namesz, 0, ®type, (LPBYTE)value, &valuesz); //从字体柄蛐寻找
|
||
if (rc == ERROR_NO_MORE_ITEMS) break;
|
||
if (rc != ERROR_SUCCESS) break;
|
||
if (regtype != REG_SZ) continue;
|
||
StringCchCopy(buf, nBufSize / sizeof(buf[0]), name);
|
||
if (buf[wcslen(buf) - 1] == L')') { //去掉括号
|
||
LPWSTR p;
|
||
if ((p = wcsrchr(buf, L'(')) != NULL) {
|
||
*p = 0;
|
||
}
|
||
}
|
||
while (buf[wcslen(buf)-1] == L' ')
|
||
buf[wcslen(buf)-1] = 0;
|
||
//获得的对应的字体脕E
|
||
FontNameCache.Add(value, buf);
|
||
}
|
||
|
||
int row = 0;
|
||
LOGFONT truefont;
|
||
memset(&truefont, 0, sizeof(truefont));
|
||
for (int i = 0; row < INFOMAX; ++i) {
|
||
int col = 0;
|
||
|
||
namesz = nBufSize;
|
||
valuesz = nBufSize;
|
||
rc = RegEnumValue(h1, i, name, &namesz, 0, ®type, (LPBYTE)value, &valuesz); //获得一个字体的字体链接
|
||
if (rc == ERROR_NO_MORE_ITEMS) break;
|
||
if (rc != ERROR_SUCCESS) break;
|
||
if (regtype != REG_MULTI_SZ) continue; //有效的字体链接
|
||
//获得字体的真实名字
|
||
|
||
TCHAR buff[LF_FACESIZE];
|
||
GetFontLocalName(name, buff);
|
||
|
||
info[row][col] = _wcsdup(buff); //第一消戟字体脕E
|
||
++col;
|
||
|
||
for (LPCWSTR linep = value; col < FONTMAX && *linep; linep += wcslen(linep) + 1) {
|
||
LPCWSTR valp = NULL;
|
||
for (LPCWSTR p = linep; *p; ++p) {
|
||
if (*p == L',' && ((char)*(p+1)<0x30 || (char)*(p+1)>0x39)) //尝试寻找字体链接中“,”后提供的字体名称
|
||
{
|
||
LPWSTR lp;
|
||
StringCchCopy(buf, nBufSize / sizeof(buf[0]), p + 1);
|
||
if (lp=wcschr(buf, L','))
|
||
*lp = 0;
|
||
valp = buf;
|
||
break;
|
||
}
|
||
}
|
||
if (!valp) { //没找到字体链接中提供的名称
|
||
/*for (int k = 0; ; ++k) {
|
||
namesz = sizeof name;
|
||
value2sz = sizeof value2;
|
||
rc = RegEnumValue(h2, k, name, &namesz, 0, ®type, (LPBYTE)value2, &value2sz); //从字体柄蛐寻找
|
||
if (rc == ERROR_NO_MORE_ITEMS) break;
|
||
if (rc != ERROR_SUCCESS) break;
|
||
if (regtype != REG_SZ) continue;
|
||
if (lstrcmpi(value2, linep) != 0) continue; //寻找字体链接中字体文件对应的字体脕E
|
||
|
||
StringCchCopyW(buf, sizeof(buf)/sizeof(buf[0]), name);
|
||
if (buf[wcslen(buf) - 1] == L')') { //去掉括号
|
||
LPWSTR p;
|
||
if ((p = wcsrchr(buf, L'(')) != NULL) {
|
||
*p = 0;
|
||
}
|
||
}
|
||
while (buf[wcslen(buf)-1] == L' ')
|
||
buf[wcslen(buf)-1] = 0;
|
||
valp = buf;
|
||
break;
|
||
}*/
|
||
LPWSTR lp;
|
||
StringCchCopy(buf, nBufSize / sizeof(buf[0]), linep);
|
||
if (lp=wcschr(buf, L','))
|
||
*lp = 0;
|
||
|
||
valp = FontNameCache.Find((TCHAR*)buf);
|
||
}
|
||
if (valp) {
|
||
GetFontLocalName((TCHAR*)valp, buff);;
|
||
//StringCchCopy(truefont.lfFaceName, LF_FACESIZE, buff); //复制到结构中
|
||
//pSettings->CopyForceFont(truefont, truefont); //获得替换字虂E
|
||
info[row][col] = _wcsdup(buff);//truefont.lfFaceName); //复制到链接柄蛐
|
||
++col;
|
||
}
|
||
}
|
||
if (col == 1) { //只有一消楷即没有链接,删掉。
|
||
free(info[row][0]);
|
||
info[row][0] = NULL;
|
||
} else {
|
||
/*if (sOsVinfo.dwMajorVersion>=6 && sOsVinfo.dwMinorVersion>=1) //版本号>=6.1,是Win7系列
|
||
{
|
||
//对字体链接柄篥逆向处纴E
|
||
LPWSTR swapbuff[32];
|
||
memcpy(swapbuff, info[row], 32*sizeof(LPWSTR)); //整个柄源制过来
|
||
for (int i=1; i<col; i++)
|
||
info[row][i]=swapbuff[col-i]; //逆序字体链接眮E
|
||
}*/
|
||
++row;
|
||
}
|
||
}
|
||
RegCloseKey(h1);
|
||
RegCloseKey(h2);
|
||
LOGFONT syslf = {0};
|
||
HGDIOBJ h = ORIG_GetStockObject(DEFAULT_GUI_FONT);
|
||
if (h) {
|
||
ORIG_GetObjectW(h, sizeof syslf, &syslf);
|
||
GetFontLocalName(syslf.lfFaceName, syslf.lfFaceName);
|
||
}
|
||
|
||
extern HFONT g_alterGUIFont;
|
||
const CGdippSettings* pSettings = CGdippSettings::GetInstance();
|
||
if (pSettings->FontSubstitutes()>=SETTING_FONTSUBSTITUTE_SAFE && pSettings->CopyForceFont(truefont, syslf)) //使用蛠E婊荒J绞保婊坏粝低匙痔丒
|
||
{
|
||
WCHAR envname[30] = L"MT_SYSFONT";
|
||
WCHAR envvalue[30] = { 0 };
|
||
HFONT tempfont;
|
||
if (GetEnvironmentVariable(L"MT_SYSFONT", envvalue, 29) && GetObjectType(tempfont = (HFONT)wcstoull(envvalue, 0 ,10)) == OBJ_FONT)//已经有字体存在
|
||
{
|
||
g_alterGUIFont = tempfont; //直接使用先前的字虂E
|
||
}
|
||
else
|
||
{
|
||
g_alterGUIFont = CreateFontIndirectW(&truefont); //创建一个新的替换字虂E
|
||
_ui64tow((ULONG_PTR)g_alterGUIFont, envvalue, 10); //转换为字符串
|
||
SetEnvironmentVariable(envname, envvalue); //写葋E肪潮淞?
|
||
}
|
||
}
|
||
|
||
//现在获取对应字体类型的默认字体链接
|
||
memset(DefaultFontLink, 0, sizeof(TCHAR)*(FF_DECORATIVE+1)*(LF_FACESIZE+1)); //初始化为0
|
||
HKEY h3;
|
||
DWORD len;
|
||
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY3, 0, KEY_QUERY_VALUE, &h3)) return;
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackage"), 0, ®type, (LPBYTE)DefaultFontLink[1], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageDecorative"), 0, ®type, (LPBYTE)DefaultFontLink[FF_DECORATIVE], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageDontCare"), 0, ®type, (LPBYTE)DefaultFontLink[FF_DONTCARE], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageModern"), 0, ®type, (LPBYTE)DefaultFontLink[FF_MODERN], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageRoman"), 0, ®type, (LPBYTE)DefaultFontLink[FF_ROMAN], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageScript"), 0, ®type, (LPBYTE)DefaultFontLink[FF_SCRIPT], &len);
|
||
len = (LF_FACESIZE+1)*sizeof(TCHAR);
|
||
RegQueryValueEx(h3, _T("FontPackageSwiss"), 0, ®type, (LPBYTE)DefaultFontLink[FF_SWISS], &len);
|
||
RegCloseKey(h3);
|
||
|
||
for (int i=0; i<FF_DECORATIVE+1; ++i) //转换字体名称
|
||
{
|
||
if (!*DefaultFontLink[i])
|
||
GetFontLocalName(DefaultFontLink[i], DefaultFontLink[i]);
|
||
}
|
||
|
||
//现在获取对应的CodePage是否需要进行fontlink。默认都需要进行链接。
|
||
HKEY h4;
|
||
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY4, 0, KEY_QUERY_VALUE, &h4)) return;
|
||
|
||
for (int i=0; i<0xff; ++i)
|
||
{
|
||
namesz = nBufSize;
|
||
valuesz = nBufSize;
|
||
rc = RegEnumValue(h4, i, name, &namesz, 0, ®type, (LPBYTE)value, &valuesz); //获得一个charset的值
|
||
if (rc == ERROR_NO_MORE_ITEMS) break;
|
||
if (rc != ERROR_SUCCESS) break;
|
||
if (regtype != REG_SZ) continue;
|
||
if (_tcsicmp(value, _T("YES")))
|
||
{
|
||
TCHAR* p = name;
|
||
while (*p!=_T('(') && p-name<(int)namesz) ++p;
|
||
++p;
|
||
AllowDefaultLink[_tcstol(p,NULL,16)]=false;
|
||
}
|
||
}
|
||
RegCloseKey(h4);
|
||
delete[]name;
|
||
delete[]value;
|
||
delete[]buf;
|
||
}
|
||
|
||
void CFontLinkInfo::clear()
|
||
{
|
||
for (int i = 0; i < INFOMAX; ++i) {
|
||
for (int j = 0; j < FONTMAX; ++j) {
|
||
free(info[i][j]);
|
||
info[i][j] = NULL;
|
||
}
|
||
}
|
||
}
|
||
|
||
const LPCWSTR * CFontLinkInfo::lookup(LPCWSTR fontname) const
|
||
{
|
||
for (int i = 0; i < INFOMAX && info[i][0]; ++i) {
|
||
if (_wcsicmp(fontname, info[i][0]) == 0) {
|
||
return &info[i][1];
|
||
}
|
||
}
|
||
return NULL;
|
||
}
|
||
|
||
LPCWSTR CFontLinkInfo::get(int row, int col) const
|
||
{
|
||
if ((unsigned int)row >= (unsigned int)INFOMAX || (unsigned int)col >= (unsigned int)FONTMAX) {
|
||
return NULL;
|
||
}
|
||
return info[row][col];
|
||
}
|
||
|
||
CFontSubstituteData::CFontSubstituteData()
|
||
{
|
||
memset(this, 0, sizeof *this);
|
||
}
|
||
|
||
int CALLBACK
|
||
CFontSubstituteData::EnumFontFamProc(const LOGFONT *lplf, const TEXTMETRIC *lptm, DWORD /*FontType*/, LPARAM lParam)
|
||
{
|
||
CFontSubstituteData& self = *(CFontSubstituteData *)lParam;
|
||
self.m_lf = *lplf;
|
||
return 0;
|
||
}
|
||
|
||
bool CFontSubstituteData::initnocheck(LPCTSTR config) {
|
||
memset(this, 0, sizeof *this);
|
||
|
||
TCHAR buf[LF_FACESIZE + 20];
|
||
StringCchCopy(buf, countof(buf), config);
|
||
|
||
memset(&m_lf, 0, sizeof m_lf);
|
||
|
||
LPTSTR p;
|
||
for (p = buf + lstrlen(buf) - 1; p >= buf; --p) {
|
||
if (*p == _T(',')) {
|
||
*p++ = 0;
|
||
break;
|
||
}
|
||
}
|
||
if (p >= buf) {
|
||
StringCchCopy(m_lf.lfFaceName, countof(m_lf.lfFaceName), buf);
|
||
m_lf.lfCharSet = (BYTE)CGdippSettings::_StrToInt(p + 1, 0);
|
||
m_bCharSet = true;
|
||
}
|
||
else {
|
||
StringCchCopy(m_lf.lfFaceName, LF_FACESIZE, buf);
|
||
m_lf.lfCharSet = DEFAULT_CHARSET;
|
||
m_bCharSet = false;
|
||
}
|
||
return m_lf.lfFaceName[0] != 0;
|
||
}
|
||
|
||
bool CFontSubstituteData::init(LPCTSTR config)
|
||
{
|
||
memset(this, 0, sizeof *this);
|
||
|
||
TCHAR buf[LF_FACESIZE + 20];
|
||
StringCchCopy(buf, countof(buf), config);
|
||
|
||
LOGFONT lf;
|
||
memset(&lf, 0, sizeof lf);
|
||
|
||
LPTSTR p;
|
||
for (p = buf + lstrlen(buf) - 1; p >= buf; --p ) {
|
||
if (*p == _T(',')) {
|
||
*p++ = 0;
|
||
break;
|
||
}
|
||
}
|
||
if (p >= buf) {
|
||
StringCchCopy(lf.lfFaceName, countof(lf.lfFaceName), buf);
|
||
lf.lfCharSet = (BYTE)CGdippSettings::_StrToInt(p + 1, 0);
|
||
m_bCharSet = true;
|
||
} else {
|
||
StringCchCopy(lf.lfFaceName, LF_FACESIZE, buf);
|
||
lf.lfCharSet = DEFAULT_CHARSET;
|
||
m_bCharSet = false;
|
||
}
|
||
|
||
HDC hdc = GetDC(NULL);
|
||
EnumFontFamiliesEx(hdc, &lf, &CFontSubstituteData::EnumFontFamProc, (LPARAM)this, 0);
|
||
ReleaseDC(NULL, hdc);
|
||
|
||
return m_lf.lfFaceName[0] != 0;
|
||
}
|
||
|
||
bool
|
||
CFontSubstituteData::operator == (const CFontSubstituteData& o) const
|
||
{
|
||
if (m_bCharSet != o.m_bCharSet) return false;
|
||
if (m_bCharSet) {
|
||
if (m_lf.lfCharSet != o.m_lf.lfCharSet) return false;
|
||
}
|
||
if (_wcsicmp(m_lf.lfFaceName, o.m_lf.lfFaceName) == 0) return true;
|
||
return false;
|
||
}
|
||
|
||
// We scan the registry and see if there is any font mapping whose mapping target is also one of our substitution source,
|
||
// and we add them as our rules.
|
||
void CFontSubstitutesInfo::initreg()
|
||
{
|
||
const LPCTSTR REGKEY = _T("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\FontSubstitutes");
|
||
HKEY h;
|
||
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY, 0, KEY_QUERY_VALUE, &h)) return;
|
||
CFontSubstituteData k;
|
||
CFontSubstituteData v;
|
||
|
||
std::vector<WCHAR> name(0x2000);
|
||
std::vector<WCHAR> value(0x2000);
|
||
DWORD namesz, valuesz;
|
||
DWORD regtype;
|
||
|
||
for (int i = 0; ; ++i) {
|
||
namesz = name.size();
|
||
valuesz = value.size();
|
||
LONG rc = RegEnumValue(h, i, name.data(), &namesz, 0, ®type, (LPBYTE)value.data(), &valuesz);
|
||
if (rc == ERROR_NO_MORE_ITEMS) break;
|
||
if (rc != ERROR_SUCCESS) break;
|
||
if (regtype != REG_SZ) continue;
|
||
|
||
if (k.initnocheck(name.data()) && v.init(value.data())) { // init k and v (k is a virtual font)
|
||
int pos = FindKey(v);
|
||
if ( pos >= 0) { // check if v is substituted to another font x
|
||
Add(k, GetValueAt(pos)); // add k=>x as well
|
||
}
|
||
}
|
||
}
|
||
RegCloseKey(h);
|
||
}
|
||
|
||
void
|
||
CFontSubstitutesInfo::initini(const CFontSubstitutesIniArray& iniarray)
|
||
{
|
||
CFontSubstitutesIniArray::const_iterator it=iniarray.begin();
|
||
// LOGFONT truefont={0}, truefont2={0};
|
||
// TCHAR* buff, *buff2;
|
||
for (; it!=iniarray.end(); ++it) {
|
||
LPCTSTR inistr = it->c_str();
|
||
LPTSTR buf = _tcsdup(inistr);
|
||
for (LPTSTR vp = buf; *vp; ++vp) {
|
||
if (*vp == _T('=')) {
|
||
*vp++ = 0;
|
||
CFontSubstituteData k;
|
||
CFontSubstituteData v;
|
||
if (k.init(buf) && v.init(vp)) {
|
||
if (FindKey(k) < 0 && k.m_bCharSet == v.m_bCharSet) Add(k, v);
|
||
}
|
||
/* StringCchCopy(truefont.lfFaceName, LF_FACESIZE, buf);
|
||
truefont.lfCharSet=DEFAULT_CHARSET;
|
||
if (!GetFontLocalName(truefont))
|
||
continue; //没有此字虂E
|
||
buff = truefont.lfFaceName;
|
||
|
||
StringCchCopy(truefont2.lfFaceName, LF_FACESIZE, vp);
|
||
truefont2.lfCharSet=DEFAULT_CHARSET;
|
||
if (!GetFontLocalName(truefont2))
|
||
continue; //没有此字虂E
|
||
buff2 = truefont2.lfFaceName;
|
||
|
||
if (m_mfontsub.find(buff)==m_mfontsub.end())
|
||
m_mfontsub[buff]=wstring(buff2);
|
||
*/
|
||
}
|
||
}
|
||
|
||
free(buf);
|
||
}
|
||
}
|
||
|
||
void
|
||
CFontSubstitutesInfo::init(int nFontSubstitutes, const CFontSubstitutesIniArray& iniarray)
|
||
{
|
||
if (nFontSubstitutes >= SETTING_FONTSUBSTITUTE_SAFE) {
|
||
initini(iniarray); // init substitution from ini array
|
||
initreg(); // add more substitutions from registry
|
||
}
|
||
}
|
||
|
||
void GetMacTypeInternalFontName(LOGFONT* lf, LPTSTR fn)
|
||
{
|
||
StringCchCopy(fn, 9, _T("MACTYPE_"));
|
||
fn+=8;
|
||
TCHAR* lfbyte=(TCHAR*)lf;
|
||
for (int i=0;i<(sizeof(LOGFONT)-sizeof(TCHAR)*LF_FACESIZE+1)/sizeof(TCHAR);i++)
|
||
*fn++=_T('0') + *lfbyte++;
|
||
*fn=0;
|
||
}
|
||
|
||
const LOGFONT *
|
||
CFontSubstitutesInfo::lookup(LOGFONT& lf) const
|
||
{
|
||
if (GetSize() <= 0) return NULL;
|
||
//bFontExist = true;
|
||
CFontSubstituteData v;
|
||
CFontSubstituteData k;
|
||
|
||
k.m_bCharSet = true;
|
||
k.m_lf = lf;
|
||
|
||
TCHAR * buff; //縼E倩竦米痔宓恼媸得?
|
||
LOGFONT mylf(lf);
|
||
if (!(buff = FontNameCache.Find((TCHAR*)lf.lfFaceName)))
|
||
{
|
||
TCHAR localname[LF_FACESIZE+1];
|
||
if (GetFontLocalName(mylf.lfFaceName, localname)) {
|
||
FontNameCache.Add((TCHAR*)lf.lfFaceName, localname);
|
||
StringCchCopy(mylf.lfFaceName, LF_FACESIZE, localname);
|
||
}
|
||
else
|
||
{
|
||
TCHAR inName[LF_FACESIZE];
|
||
GetMacTypeInternalFontName(&mylf, inName);
|
||
if (!(buff = FontNameCache.Find(inName)))
|
||
{
|
||
mylf.lfClipPrecision = FONT_MAGIC_NUMBER;
|
||
HFONT tempfont = CreateFontIndirect(&mylf);
|
||
HDC dc=CreateCompatibleDC(NULL);
|
||
HFONT oldfont = SelectFont(dc, tempfont);
|
||
ORIG_GetTextFaceW(dc, LF_FACESIZE, mylf.lfFaceName);
|
||
SelectFont(dc, oldfont);
|
||
DeleteFont(tempfont);
|
||
DeleteDC(dc);
|
||
FontNameCache.Add(inName, mylf.lfFaceName);
|
||
}
|
||
}
|
||
buff = mylf.lfFaceName;
|
||
}
|
||
StringCchCopy(k.m_lf.lfFaceName, LF_FACESIZE, buff);
|
||
StringCchCopy(lf.lfFaceName, LF_FACESIZE, buff);
|
||
|
||
int pos = FindKey(k);
|
||
if (pos < 0) {
|
||
k.m_bCharSet = false;
|
||
pos = FindKey(k);
|
||
}
|
||
if (pos >= 0) {
|
||
return (const LOGFONT *)&GetValueAt(pos);
|
||
}
|
||
return NULL;
|
||
}
|
||
|
||
CFontFaceNamesEnumerator::CFontFaceNamesEnumerator(LPCWSTR facename, int nFontFamily) : m_pos(0)
|
||
{
|
||
//CCriticalSectionLock __lock;
|
||
const CGdippSettings* pSettings = CGdippSettings::GetInstance();
|
||
TCHAR buff[LF_FACESIZE+1];
|
||
GetFontLocalName((TCHAR*)facename, buff);
|
||
LPCWSTR srcfacenames[] = {
|
||
buff, NULL, NULL
|
||
};
|
||
|
||
int destpos = 0;
|
||
for (const LPCWSTR *p = srcfacenames; *p && destpos < MAXFACENAMES; ++p) {
|
||
m_facenames[destpos++] = *p;
|
||
if (pSettings->FontLink()) {
|
||
const LPCWSTR *facenamep = pSettings->GetFontLinkInfo().lookup(*p);
|
||
if (facenamep) {
|
||
for ( ; *facenamep && **facenamep && destpos < MAXFACENAMES; ++facenamep) {
|
||
m_facenames[destpos++] = *facenamep;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
m_facenames[0] = facename;
|
||
if (pSettings->IsWinXPorLater() && pSettings->FontLink() &&
|
||
pSettings->FontLoader() == SETTING_FONTLOADER_FREETYPE) {
|
||
m_facenames[destpos++] = pSettings->GetFontLinkInfo().sysfn(nFontFamily);
|
||
}
|
||
m_endpos = destpos;
|
||
}
|