📄 syntextdrawer.pas
字号:
pFontsInfo: PheSharedFontsInfo);
var
i: Integer;
begin
with pFontsInfo^ do
for i := Low(TheStockFontPatterns) to High(TheStockFontPatterns) do
with FontsData[i] do
if Handle <> 0 then
begin
DeleteObject(Handle);
Handle := 0;
end;
end;
function TheFontsInfoManager.FindFontsInfo(
const LF: TLogFont): PheSharedFontsInfo;
var
i: Integer;
begin
for i := 0 to FFontsInfo.Count - 1 do
begin
Result := PheSharedFontsInfo(FFontsInfo[i]);
if CompareMem(@(Result^.BaseLF), @LF, SizeOf(TLogFont)) then
Exit;
end;
Result := nil;
end;
function TheFontsInfoManager.GetFontsInfo(ABaseFont: TFont): PheSharedFontsInfo;
var
LF: TLogFont;
begin
ASSERT(Assigned(ABaseFont));
RetrieveLogFontForComparison(ABaseFont, LF);
Result := FindFontsInfo(LF);
if not Assigned(Result) then
begin
Result := CreateFontsInfo(ABaseFont, LF);
FFontsInfo.Add(Result);
end;
if Assigned(Result) then
Inc(Result^.RefCount);
end;
procedure TheFontsInfoManager.ReleaseFontsInfo(pFontsInfo: PheSharedFontsInfo);
begin
ASSERT(Assigned(pFontsInfo));
with pFontsInfo^ do
begin
{$IFDEF HE_ASSERT}
ASSERT(LockCount < RefCount,
'Call DeactivateFontsInfo before calling this.');
{$ELSE}
ASSERT(LockCount < RefCount);
{$ENDIF}
if RefCount > 1 then
Dec(RefCount)
else
begin
FFontsInfo.Remove(pFontsInfo);
// free all objects
BaseFont.Free;
Dispose(pFontsInfo);
end;
end;
end;
procedure TheFontsInfoManager.RetrieveLogFontForComparison(ABaseFont: TFont;
var LF: TLogFont);
var
pEnd: PChar;
begin
GetObject(ABaseFont.Handle, SizeOf(TLogFont), @LF);
with LF do
begin
lfItalic := 0;
lfUnderline := 0;
lfStrikeOut := 0;
pEnd := StrEnd(lfFaceName);
FillChar(pEnd[1], @lfFaceName[High(lfFaceName)] - pEnd, 0);
end;
end;
{ TheFontStock }
// CalcFontAdvance : Calculation a advance of a character of a font.
// [*]hCalcFont will be selected as FDC's font if FDC wouldn't be zero.
function TheFontStock.CalcFontAdvance(DC: HDC;
pCharHeight, pDBCharAdvance: PInteger): Integer;
var
TM: TTextMetric;
ABC: TABC;
ABC2: TABC;
w: Integer;
HasABC: Boolean;
begin
// Calculate advance of a character.
// The following code uses ABC widths instead TextMetric.tmAveCharWidth
// because ABC widths always tells truth but tmAveCharWidth does not.
// A true-type font will have ABC widths but others like raster type will not
// so if the function fails then use TextMetric.tmAveCharWidth.
GetTextMetrics(DC, TM);
HasABC := GetCharABCWidths(DC, Ord('M'), Ord('M'), ABC);
if not HasABC then
begin
with ABC do
begin
abcA := 0;
abcB := TM.tmAveCharWidth;
abcC := 0;
end;
TM.tmOverhang := 0;
end;
// Result(CharWidth)
with ABC do
Result := abcA + Integer(abcB) + abcC + TM.tmOverhang;
// pCharHeight
if Assigned(pCharHeight) then
pCharHeight^ := Abs(TM.tmHeight) {+ TM.tmInternalLeading};
// pDBCharAdvance
if Assigned(pDBCharAdvance) then
begin
pDBCharAdvance^ := DBCHAR_CALCULATION_FALED;
if IsDBCSFont then
begin
case TM.tmCharSet of
SHIFTJIS_CHARSET:
if HasABC and
GetCharABCWidths(DC, $8201, $8201, ABC) and // max width(maybe)
GetCharABCWidths(DC, $82A0, $82A0, ABC2) then // HIRAGANA 'a'
begin
with ABC do
w := abcA + Integer(abcB) + abcC;
if w > (1.5 * Result) then // it should be over 150% wider than SBChar(I think)
with ABC2 do
if w = (abcA + Integer(abcB) + abcC) then
pDBCharAdvance^ := w;
end;
// About the following character sets,
// I don't know with what character should be calculated.
{
ANSI_CHARSET:
DEFAULT_CHARSET:
SYMBOL_CHARSET:
HANGUL_CHARSET:
GB2312_CHARSET:
CHINESEBIG5_CHARSET:
OEM_CHARSET:
JOHAB_CHARSET:
HEBREW_CHARSET:
ARABIC_CHARSET:
GREEK_CHARSET:
TURKISH_CHARSET:
VIETNAMESE_CHARSET:
THAI_CHARSET:
EASTEUROPE_CHARSET:
RUSSIAN_CHARSET:
MAC_CHARSET:
BALTIC_CHARSET:
}
end;
end;
end;
end;
constructor TheFontStock.Create(InitialFont: TFont);
begin
inherited Create;
SetBaseFont(InitialFont);
end;
destructor TheFontStock.Destroy;
begin
ReleaseFontsInfo;
ASSERT(FDCRefCount = 0);
inherited;
end;
function TheFontStock.GetBaseFont: TFont;
begin
Result := FpInfo^.BaseFont;
end;
function TheFontStock.GetCharAdvance: Integer;
begin
Result := FpCrntFontData^.CharAdv;
end;
function TheFontStock.GetCharHeight: Integer;
begin
Result := FpCrntFontData^.CharHeight;
end;
function TheFontStock.GetDBCharAdvance: Integer;
begin
Result := FpCrntFontData^.DBCharAdv;
end;
function TheFontStock.GetFontData(idx: Integer): PheFontData;
begin
Result := @FpInfo^.FontsData[idx];
end;
function TheFontStock.GetIsDBCSFont: Boolean;
begin
Result := FpInfo^.IsDBCSFont;
end;
function TheFontStock.GetIsTrueType: Boolean;
begin
Result := FpInfo^.IsTrueType
end;
function TheFontStock.InternalCreateFont(Style: TFontStyles): HFONT;
const
Bolds: array[Boolean] of Integer = (400, 700);
begin
with FBaseLF do
begin
lfWeight := Bolds[fsBold in Style];
lfItalic := Ord(BOOL(fsItalic in Style));
lfUnderline := Ord(BOOL(fsUnderline in Style));
lfStrikeOut := Ord(BOOL(fsStrikeOut in Style));
end;
Result := CreateFontIndirect(FBaseLF);
end;
function TheFontStock.InternalGetDC: HDC;
begin
if FDCRefCount = 0 then
begin
ASSERT(FDC = 0);
FDC := GetDC(0);
end;
Inc(FDCRefCount);
Result := FDC;
end;
procedure TheFontStock.InternalReleaseDC(Value: HDC);
begin
Dec(FDCRefCount);
if FDCRefCount <= 0 then
begin
ASSERT((FDC <> 0) and (FDC = Value));
ReleaseDC(0, FDC);
FDC := 0;
ASSERT(FDCRefCount = 0);
end;
end;
procedure TheFontStock.ReleaseFontHandles;
begin
if FUsingFontHandles then
with GetFontsInfoManager do
begin
UnlockFontsInfo(FpInfo);
FUsingFontHandles := False;
end;
end;
procedure TheFontStock.ReleaseFontsInfo;
begin
if Assigned(FpInfo) then
with GetFontsInfoManager do
begin
if FUsingFontHandles then
begin
UnlockFontsInfo(FpInfo);
FUsingFontHandles := False;
end;
ReleaseFontsInfo(FpInfo);
FpInfo := nil;
end;
end;
procedure TheFontStock.SetBaseFont(Value: TFont);
var
pInfo: PheSharedFontsInfo;
begin
if Assigned(Value) then
begin
pInfo := GetFontsInfoManager.GetFontsInfo(Value);
if pInfo = FpInfo then
GetFontsInfoManager.ReleaseFontsInfo(pInfo)
else
begin
ReleaseFontsInfo;
FpInfo := pInfo;
FBaseLF := FpInfo^.BaseLF;
SetStyle(Value.Style);
end;
end
else
raise EheFontStockException.Create('SetBaseFont: ''Value'' must be specified.');
end;
procedure TheFontStock.SetStyle(Value: TFontStyles);
var
idx: Integer;
DC: HDC;
hOldFont: HFONT;
p: PheFontData;
begin
{$IFDEF HE_ASSERT}
ASSERT(SizeOf(TFontStyles) = 1,
'TheTextDrawer.SetStyle: There''s more than four font styles but the current '+
'code expects only four styles.');
{$ELSE}
ASSERT(SizeOf(TFontStyles) = 1);
{$ENDIF}
idx := Byte(Value);
ASSERT(idx <= High(TheStockFontPatterns));
UseFontHandles;
p := FontData[idx];
if FpCrntFontData = p then
Exit;
FpCrntFontData := p;
with p^ do
if Handle <> 0 then
begin
FCrntFont := Handle;
FCrntStyle := Style;
Exit;
end;
// create font
FCrntFont := InternalCreateFont(Value);
DC := InternalGetDC;
hOldFont := SelectObject(DC, FCrntFont);
// retrieve height and advances of new font
with FpCrntFontData^ do
begin
Handle := FCrntFont;
if IsDBCSFont then
CharAdv := CalcFontAdvance(DC, @CharHeight, @DBCharAdv)
else
CharAdv := CalcFontAdvance(DC, @CharHeight, nil);
end;
SelectObject(DC, hOldFont);
InternalReleaseDC(DC);
end;
procedure TheFontStock.UseFontHandles;
begin
if not FUsingFontHandles then
with GetFontsInfoManager do
begin
LockFontsInfo(FpInfo);
FUsingFontHandles := True;
end;
end;
{ TheTextDrawer }
constructor TheTextDrawer.Create(CalcExtentBaseStyle: TFontStyles; BaseFont: TFont);
begin
inherited Create;
// Modified by Administrator 2007-11-28 上午 09:40:54
FUnicodeFont := TFont.Create;
FUnicodeFontStock := TheFontStock.Create(FUnicodeFont);;
FFontStock := TheFontStock.Create(BaseFont);
FCalcExtentBaseStyle := CalcExtentBaseStyle;
SetBaseFont(BaseFont);
FColor := clWindowText;
FBkColor := clWindow;
end;
destructor TheTextDrawer.Destroy;
begin
FFontStock.Free;
ReleaseETODist;
FUnicodeFontStock.Free;
FUnicodeFont.Free;
inherited;
end;
procedure TheTextDrawer.ReleaseETODist;
begin
if Assigned(FETODist) then
begin
FETOSizeInChar := 0;
FreeMem(FETODist);
FETODist := nil;
end;
end;
procedure TheTextDrawer.BeginDrawing(DC: HDC);
begin
if (FDC = DC) then
ASSERT(FDC <> 0)
else
begin
ASSERT((FDC = 0) and (DC <> 0) and (FDrawingCount = 0));
FDC := DC;
FSaveDC := SaveDC(DC);
SelectObject(DC, FCrntFont);
Windows.SetTextColor(DC, ColorToRGB(FColor));
Windows.SetBkColor(DC, ColorToRGB(FBkColor));
DoSetCharExtra(FCharExtra);
end;
Inc(FDrawingCount);
end;
procedure TheTextDrawer.EndDrawing;
begin
ASSERT(FDrawingCount >= 1);
Dec(FDrawingCount);
if FDrawingCount <= 0 then
begin
if FDC <> 0 then
RestoreDC(FDC, FSaveDC);
FSaveDC := 0;
FDC := 0;
FDrawingCount := 0;
end;
end;
function TheTextDrawer.GetCharWidth: Integer;
begin
Result := FBaseCharWidth;// + FCharExtra;
end;
function TheTextDrawer.GetUnicodeFontName: TFontName;
begin
result := FUnicodeFont.Name;
end;
function TheTextDrawer.GetCharHeight: Integer;
begin
Result := FBaseCharHeight;
end;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -