mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-14 03:19:32 +02:00
1156 lines
33 KiB
PHP
1156 lines
33 KiB
PHP
{%MainUnit ../graphics.pp}
|
|
{******************************************************************************
|
|
TFONT
|
|
******************************************************************************
|
|
|
|
*****************************************************************************
|
|
* *
|
|
* This file is part of the Lazarus Component Library (LCL) *
|
|
* *
|
|
* See the file COPYING.modifiedLGPL, included in this distribution, *
|
|
* for details about the copyright. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. *
|
|
* *
|
|
*****************************************************************************
|
|
}
|
|
|
|
|
|
{ TFontHandleCache }
|
|
|
|
type
|
|
TLogFontAndName = record
|
|
LogFont: TLogFont;
|
|
LongFontName: string;
|
|
end;
|
|
PLogFontAndName = ^TLogFontAndName;
|
|
|
|
function CompareLogFontAndNameWithResDesc(Key: PLogFontAndName; Desc: TFontHandleCacheDescriptor): integer;
|
|
begin
|
|
Result:=CompareStr(Key^.LongFontName,Desc.LongFontName);
|
|
//writeln('CompareLogFontAndNameWithResDesc A ',Key^.LongFontName,' ',Desc.LongFontName,' ',DbgS(Desc),' Result=',Result);
|
|
if Result=0 then
|
|
Result:=CompareMemRange(@Key^.LogFont,@Desc.LogFont,SizeOf(Desc.LogFont));
|
|
//writeln('CompareLogFontAndNameWithResDesc END Result=',Result);
|
|
end;
|
|
|
|
procedure TFontHandleCache.RemoveItem(Item: TResourceCacheItem);
|
|
begin
|
|
DeleteObject(Item.Handle);
|
|
inherited RemoveItem(Item);
|
|
end;
|
|
|
|
constructor TFontHandleCache.Create;
|
|
begin
|
|
inherited Create;
|
|
FResourceCacheDescriptorClass:=TFontHandleCacheDescriptor;
|
|
end;
|
|
|
|
function TFontHandleCache.CompareDescriptors(Tree: TAvgLvlTree; Desc1,
|
|
Desc2: Pointer): integer;
|
|
var
|
|
Descriptor1: TFontHandleCacheDescriptor;
|
|
Descriptor2: TFontHandleCacheDescriptor;
|
|
begin
|
|
Descriptor1:=TFontHandleCacheDescriptor(Desc1);
|
|
Descriptor2:=TFontHandleCacheDescriptor(Desc2);
|
|
Result:=CompareStr(Descriptor1.LongFontName,Descriptor2.LongFontName);
|
|
if Result<>0 then exit;
|
|
Result:=CompareMemRange(@Descriptor1.LogFont,@Descriptor2.LogFont,
|
|
SizeOf(Descriptor1.LogFont));
|
|
end;
|
|
|
|
function TFontHandleCache.FindFont(TheFont: HFONT): TResourceCacheItem;
|
|
var
|
|
ANode: TAvgLvlTreeNode;
|
|
begin
|
|
ANode:=FItems.FindKey(@THandle(TheFont),
|
|
TListSortCompare(@ComparePHandleWithResourceCacheItem));
|
|
if ANode<>nil then
|
|
Result:=TResourceCacheItem(ANode.Data)
|
|
else
|
|
Result:=nil;
|
|
end;
|
|
|
|
function TFontHandleCache.FindFontDesc(const LogFont: TLogFont;
|
|
const LongFontName: string): TFontHandleCacheDescriptor;
|
|
var
|
|
LogFontAndName: TLogFontAndName;
|
|
ANode: TAvgLvlTreeNode;
|
|
begin
|
|
LogFontAndName.LogFont:=LogFont;
|
|
LogFontAndName.LongFontName:=LongFontName;
|
|
ANode:=FDescriptors.Findkey(@LogFontAndName,
|
|
TListSortCompare(@CompareLogFontAndNameWithResDesc));
|
|
if ANode<>nil then
|
|
Result:=TFontHandleCacheDescriptor(ANode.Data)
|
|
else
|
|
Result:=nil;
|
|
end;
|
|
|
|
function TFontHandleCache.Add(TheFont: HFONT; const LogFont: TLogFont;
|
|
const LongFontName: string): TFontHandleCacheDescriptor;
|
|
var
|
|
Item: TResourceCacheItem;
|
|
begin
|
|
if FindFontDesc(LogFont,LongFontName)<>nil then
|
|
RaiseGDBException('TFontHandleCache.Add font desc added twice');
|
|
|
|
// find cache item with TheFont
|
|
Item:=FindFont(TheFont);
|
|
if Item=nil then begin
|
|
// create new item
|
|
Item:=TResourceCacheItem.Create(Self,TheFont);
|
|
FItems.Add(Item);
|
|
end;
|
|
|
|
// create descriptor
|
|
Result:=TFontHandleCacheDescriptor.Create(Self,Item);
|
|
Result.LongFontName:=LongFontName;
|
|
Result.LogFont:=LogFont;
|
|
FDescriptors.Add(Result);
|
|
if FindFontDesc(LogFont,LongFontName)=nil then begin
|
|
DebugLn('TFontHandleCache.Add Added: %p LongFontName=%s',[Pointer(Result), Result.LongFontName]);
|
|
RaiseGDBException('');
|
|
end;
|
|
end;
|
|
|
|
|
|
{ TFont }
|
|
|
|
procedure GetCharsetValues(Proc: TGetStrProc);
|
|
var
|
|
I: Integer;
|
|
begin
|
|
for I:=Low(FontCharsets) to High(FontCharsets) do
|
|
Proc(FontCharsets[I].Name);
|
|
end;
|
|
|
|
function CharsetToIdent(Charset: Longint; var Ident: string): Boolean;
|
|
begin
|
|
Result:=IntToIdent(Charset, Ident, FontCharsets);
|
|
end;
|
|
|
|
function IdentToCharset(const Ident: string; var Charset: Longint): Boolean;
|
|
begin
|
|
Result:=IdentToInt(Ident, CharSet, FontCharsets);
|
|
end;
|
|
|
|
function GetFontData(Font: HFont): TFontData;
|
|
var
|
|
ALogFont: TLogFont;
|
|
begin
|
|
Result := DefFontData;
|
|
if Font <> 0 then
|
|
begin
|
|
if GetObject(Font, SizeOf(ALogFont), @ALogFont) <> 0 then
|
|
with Result, ALogFont do
|
|
begin
|
|
Height := lfHeight;
|
|
if lfWeight >= FW_BOLD then
|
|
Include(Style, fsBold);
|
|
if lfItalic = 1 then
|
|
Include(Style, fsItalic);
|
|
if lfUnderline = 1 then
|
|
Include(Style, fsUnderline);
|
|
if lfStrikeOut = 1 then
|
|
Include(Style, fsStrikeOut);
|
|
Charset := TFontCharset(lfCharSet);
|
|
Name := lfFaceName;
|
|
case lfPitchAndFamily and $F of
|
|
VARIABLE_PITCH: Pitch := fpVariable;
|
|
FIXED_PITCH: Pitch := fpFixed;
|
|
else
|
|
Pitch := fpDefault;
|
|
end;
|
|
Handle := Font;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function GetDefFontCharSet: TFontCharSet;
|
|
//var
|
|
// DisplayDC: HDC;
|
|
// TxtMetric: TTEXTMETRIC;
|
|
begin
|
|
Result := DEFAULT_CHARSET;
|
|
{DisplayDC := GetDC(0);
|
|
if (DisplayDC <> 0) then begin
|
|
if (SelectObject(DisplayDC, StockFont) <> 0) then
|
|
if (GetTextMetrics(DisplayDC, TxtMetric)) then
|
|
Result := TxtMetric.tmCharSet;
|
|
ReleaseDC(0, DisplayDC);
|
|
end;}
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: FindXLFDItem
|
|
Params: const XLFDName: string; Index: integer;
|
|
var ItemStart, ItemEnd: integer
|
|
Returns: boolean
|
|
|
|
Searches the XLFD item on position Index. Index starts from 0.
|
|
Returns true on sucess.
|
|
ItemStart will be on the first character and ItemEnd after the last character.
|
|
------------------------------------------------------------------------------}
|
|
function FindXLFDItem(const XLFDName: string; Index: integer;
|
|
var ItemStart, ItemEnd: integer): boolean;
|
|
begin
|
|
if Index<0 then begin
|
|
Result:=false;
|
|
exit;
|
|
end;
|
|
ItemStart:=1;
|
|
ItemEnd:=ItemStart;
|
|
while true do begin
|
|
if (ItemEnd>length(XLFDName)) then begin
|
|
dec(Index);
|
|
break;
|
|
end;
|
|
if XLFDName[ItemEnd]='-' then begin
|
|
dec(Index);
|
|
if Index<0 then break;
|
|
ItemStart:=ItemEnd+1;
|
|
end;
|
|
inc(ItemEnd);
|
|
end;
|
|
Result:=(Index=-1);
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ExtractXLFDItem
|
|
Params: const XLFDName: string; Index: integer
|
|
Returns: string
|
|
|
|
Parses a font name in XLFD format and extracts one item.
|
|
(see http://wwwinfo.cern.ch/umtf/working-groups/X11/fonts/hp_xlfd.html)
|
|
|
|
An XLFD name is
|
|
FontNameRegistry-Foundry-FamilyName-WeightName-Slant-SetwidthName
|
|
-AddStyleName-PixelSize-PointSize-ResolutionX-ResolutionY-Spacing
|
|
-AverageWidth-CharSetRegistry-CharSetCoding
|
|
|
|
------------------------------------------------------------------------------}
|
|
function ExtractXLFDItem(const XLFDName: string; Index: integer): string;
|
|
var StartPos, EndPos: integer;
|
|
begin
|
|
if FindXLFDItem(XLFDName,Index,StartPos,EndPos) then
|
|
Result:=copy(XLFDName,StartPos,EndPos-StartPos)
|
|
else
|
|
Result:='';
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ExtractFamilyFromXLFDName
|
|
Params: const XLFDName: string
|
|
Returns: string
|
|
|
|
Parses a font name in XLFD format and extracts the FamilyName.
|
|
(see http://wwwinfo.cern.ch/umtf/working-groups/X11/fonts/hp_xlfd.html)
|
|
|
|
An XLFD name is
|
|
FontNameRegistry-Foundry-FamilyName-WeightName-Slant-SetwidthName
|
|
-AddStyleName-PixelSize-PointSize-ResolutionX-ResolutionY-Spacing
|
|
-AverageWidth-CharSetRegistry-CharSetCoding
|
|
|
|
------------------------------------------------------------------------------}
|
|
function ExtractFamilyFromXLFDName(const XLFDName: string): string;
|
|
var StartPos, EndPos: integer;
|
|
begin
|
|
if FindXLFDItem(XLFDName,2,StartPos,EndPos) then
|
|
Result:=copy(XLFDName,StartPos,EndPos-StartPos)
|
|
else
|
|
Result:='';
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: XLFDNameToLogFont
|
|
Params: const XLFDName: string
|
|
Returns: TLogFont
|
|
|
|
Parses a font name in XLFD format and creates a TLogFont record from it.
|
|
(see http://wwwinfo.cern.ch/umtf/working-groups/X11/fonts/hp_xlfd.html)
|
|
|
|
An XLFD name is
|
|
FontNameRegistry-Foundry-FamilyName-WeightName-Slant-SetwidthName
|
|
-AddStyleName-PixelSize-PointSize-ResolutionX-ResolutionY-Spacing
|
|
-AverageWidth-CharSetRegistry-CharSetCoding
|
|
|
|
------------------------------------------------------------------------------}
|
|
function XLFDNameToLogFont(const XLFDName: string): TLogFont;
|
|
type
|
|
TWeightMapEntry = record
|
|
Name: string;
|
|
Weight: integer;
|
|
end;
|
|
const
|
|
WeightMap: array[1..15] of TWeightMapEntry = (
|
|
(Name: 'DONTCARE'; Weight: FW_DONTCARE),
|
|
(Name: 'THIN'; Weight: FW_THIN),
|
|
(Name: 'EXTRALIGHT'; Weight: FW_EXTRALIGHT),
|
|
(Name: 'LIGHT'; Weight: FW_LIGHT),
|
|
(Name: 'NORMAL'; Weight: FW_NORMAL),
|
|
(Name: 'MEDIUM'; Weight: FW_MEDIUM),
|
|
(Name: 'SEMIBOLD'; Weight: FW_SEMIBOLD),
|
|
(Name: 'BOLD'; Weight: FW_BOLD),
|
|
(Name: 'EXTRABOLD'; Weight: FW_EXTRABOLD),
|
|
(Name: 'HEAVY'; Weight: FW_HEAVY),
|
|
(Name: 'ULTRALIGHT'; Weight: FW_ULTRALIGHT),
|
|
(Name: 'REGULAR'; Weight: FW_REGULAR),
|
|
(Name: 'DEMIBOLD'; Weight: FW_DEMIBOLD),
|
|
(Name: 'ULTRABOLD'; Weight: FW_ULTRABOLD),
|
|
(Name: 'BLACK'; Weight: FW_BLACK)
|
|
);
|
|
var
|
|
ItemStart, ItemEnd: integer;
|
|
Item: string;
|
|
|
|
procedure GetNextItem;
|
|
begin
|
|
ItemStart:=ItemEnd+1;
|
|
ItemEnd:=ItemStart;
|
|
while (ItemEnd<=length(XLFDName)) and (XLFDName[ItemEnd]<>'-') do
|
|
inc(ItemEnd);
|
|
Item:=copy(XLFDName,ItemStart,ItemEnd-ItemStart);
|
|
end;
|
|
|
|
function WeightNameToWeightID(const WeightName: string): integer;
|
|
var i: integer;
|
|
begin
|
|
for i:=Low(WeightMap) to High(WeightMap) do begin
|
|
if AnsiCompareText(WeightMap[i].Name,WeightName)=0 then begin
|
|
Result:=WeightMap[i].Weight;
|
|
exit;
|
|
end;
|
|
end;
|
|
Result:=FW_DONTCARE;
|
|
end;
|
|
|
|
var l, FaceNameMax, PixelSize, PointSize, Resolution, AverageWidth: integer;
|
|
begin
|
|
FillChar(Result,SizeOf(TLogFont),0);
|
|
ItemEnd:=0;
|
|
GetNextItem; // 1. read FontNameRegistry
|
|
// ToDo
|
|
|
|
GetNextItem; // 2. read Foundry
|
|
// ToDo
|
|
|
|
GetNextItem; // 3. read FamilyName
|
|
l:=length(Item);
|
|
FaceNameMax:=High(Result.lfFaceName)-Low(Result.lfFaceName); // max without #0
|
|
if l>FaceNameMax then l:=FaceNameMax;
|
|
if l>0 then Move(Item[1],Result.lfFaceName[Low(Result.lfFaceName)],l);
|
|
Result.lfFaceName[Low(Result.lfFaceName)+l]:=#0;
|
|
|
|
GetNextItem; // 4. read WeightName
|
|
Result.lfWeight:=WeightNameToWeightID(Item);
|
|
|
|
GetNextItem; // 5. read Slant
|
|
if (AnsiCompareText(Item,'I')=0) or (AnsiCompareText(Item,'RI')=0)
|
|
or (AnsiCompareText(Item,'O')=0) then
|
|
// I = italic, RI = reverse italic, O = oblique
|
|
Result.lfItalic:=1
|
|
else
|
|
Result.lfItalic:=0;
|
|
|
|
GetNextItem; // 6. read SetwidthName
|
|
// ToDO: NORMAL, CONDENSED, NARROW, WIDE, EXPANDED
|
|
|
|
GetNextItem; // 7. read AddStyleName
|
|
// calculate Style name extentions (=rotation)
|
|
// API XLFD
|
|
// --------------------- --------------
|
|
// Orientation 1/10 deg 1/64 deg
|
|
Result.lfOrientation:=(StrToIntDef(Item,0)*10) div 64;
|
|
|
|
GetNextItem; // 8. read PixelSize
|
|
PixelSize:=StrToIntDef(Item,0);
|
|
GetNextItem; // 9. read PointSize
|
|
PointSize:=StrToIntDef(Item,0) div 10;
|
|
GetNextItem; // 10. read ResolutionX
|
|
Resolution:=StrToIntDef(Item,0);
|
|
if Resolution<=0 then Resolution:=72;
|
|
|
|
if PixelSize=0 then begin
|
|
if PointSize<=0 then
|
|
Result.lfHeight:=(12*Resolution) div 72
|
|
else
|
|
Result.lfHeight:=(PointSize*Resolution) div 72;
|
|
end else begin
|
|
Result.lfHeight:=PixelSize;
|
|
end;
|
|
|
|
GetNextItem; // 11. read ResolutionY
|
|
Resolution:=StrToIntDef(Item,0);
|
|
if Resolution<=0 then Resolution:=72;
|
|
|
|
GetNextItem; // 12. read Spacing
|
|
{M Monospaced (fixed pitch)
|
|
P Proportional spaced (variable pitch)
|
|
C Character cell. The glyphs of the font can be thought of as
|
|
"boxes" of the same width and height that are stacked side by
|
|
side or top to bottom.}
|
|
if AnsiCompareText(Item,'M')=0 then
|
|
Result.lfPitchAndFamily:=FIXED_PITCH
|
|
else if AnsiCompareText(Item,'P')=0 then
|
|
Result.lfPitchAndFamily:=VARIABLE_PITCH
|
|
else if AnsiCompareText(Item,'C')=0 then
|
|
Result.lfPitchAndFamily:=VARIABLE_PITCH;
|
|
|
|
GetNextItem; // 13. read AverageWidth
|
|
AverageWidth := StrToIntDef(Item,0);
|
|
Result.lfWidth := AverageWidth div 10;
|
|
|
|
GetNextItem; // 14. read CharSetRegistry
|
|
// ToDo
|
|
|
|
GetNextItem; // 15. read CharSetCoding
|
|
// ToDo
|
|
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ClearXLFDItem
|
|
Params: const LongFontName: string; Index: integer
|
|
Returns: string
|
|
|
|
Replaces an item of a font name in XLFD format with a '*'.
|
|
------------------------------------------------------------------------------}
|
|
function ClearXLFDItem(const LongFontName: string; Index: integer): string;
|
|
var ItemStart, ItemEnd: integer;
|
|
begin
|
|
if FindXLFDItem(LongFontName,Index,ItemStart,ItemEnd)
|
|
and ((ItemEnd-ItemStart<>1) or (LongFontName[ItemStart]<>'*')) then
|
|
Result:=LeftStr(LongFontName,ItemStart-1)+'*'
|
|
+RightStr(LongFontName,length(LongFontName)-ItemEnd+1)
|
|
else
|
|
Result:=LongFontName;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ClearXLFDHeight
|
|
Params: const LongFontName: string
|
|
Returns: string
|
|
|
|
Replaces the PixelSize, PointSize, ResolutionX, ResolutionY and AverageWidth
|
|
of a font name in XLFD format with '*'.
|
|
|
|
An XLFD name is
|
|
FontNameRegistry-Foundry-FamilyName-WeightName-Slant-SetwidthName
|
|
-AddStyleName-PixelSize-PointSize-ResolutionX-ResolutionY-Spacing
|
|
-AverageWidth-CharSetRegistry-CharSetCoding
|
|
------------------------------------------------------------------------------}
|
|
function ClearXLFDHeight(const LongFontName: string): string;
|
|
begin
|
|
Result:=ClearXLFDItem(LongFontName,7); // PixelSize
|
|
Result:=ClearXLFDItem(Result,8); // PointSize
|
|
Result:=ClearXLFDItem(Result,9); // ResolutionX
|
|
Result:=ClearXLFDItem(Result,10); // ResolutionY
|
|
Result:=ClearXLFDItem(Result,12); // AverageWidth
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ClearXLFDPitch
|
|
Params: const LongFontName: string
|
|
Returns: string
|
|
|
|
Replaces the spacing a font name in XLFD format with a '*'.
|
|
------------------------------------------------------------------------------}
|
|
function ClearXLFDPitch(const LongFontName: string): string;
|
|
begin
|
|
Result:=ClearXLFDItem(LongFontName,11);
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: ClearXLFDStyle
|
|
Params: const LongFontName: string
|
|
Returns: string
|
|
|
|
Replaces the WeightName, Slant and SetwidthName of a font name in XLFD format
|
|
with '*'.
|
|
------------------------------------------------------------------------------}
|
|
function ClearXLFDStyle(const LongFontName: string): string;
|
|
begin
|
|
Result:=ClearXLFDItem(ClearXLFDItem(ClearXLFDItem(LongFontName,3),4),5);
|
|
end;
|
|
|
|
function XLFDHeightIsSet(const LongFontName: string): boolean;
|
|
begin
|
|
Result:=(ExtractXLFDItem(LongFontName,7)<>'')
|
|
or (ExtractXLFDItem(LongFontName,8)<>'')
|
|
or (ExtractXLFDItem(LongFontName,9)<>'')
|
|
or (ExtractXLFDItem(LongFontName,10)<>'');
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function: IsFontNameXLogicalFontDesc
|
|
Params: const LongFontName: string
|
|
Returns: boolean
|
|
|
|
Checks if font name is in X Logical Font Description format.
|
|
(see http://wwwinfo.cern.ch/umtf/working-groups/X11/fonts/hp_xlfd.html)
|
|
|
|
An XLFD name is
|
|
FontNameRegistry-Foundry-FamilyName-WeightName-Slant-SetwidthName
|
|
-AddStyleName-PixelSize-PointSize-ResolutionX-ResolutionY-Spacing
|
|
-AverageWidth-CharSetRegistry-CharSetCoding
|
|
------------------------------------------------------------------------------}
|
|
function IsFontNameXLogicalFontDesc(const LongFontName: string): boolean;
|
|
// Quick test: check if LongFontName contains 14 times the char '-'
|
|
var MinusCnt, p: integer;
|
|
begin
|
|
MinusCnt:=0;
|
|
for p:=1 to length(LongFontName) do
|
|
if LongFontName[p]='-' then inc(MinusCnt);
|
|
Result:=(MinusCnt=14);
|
|
end;
|
|
|
|
// split a given fontName into Pango Font description components
|
|
// font name is supposed to follow this layout:
|
|
// [FAMILY-LIST][STYLE-LIST][SIZE]
|
|
// where:
|
|
// [FAMILY-LIST] is a comma separated list of families optionally
|
|
// ended by a comma
|
|
// [STYLE-LIST] is white space separated list of words where each word
|
|
// describe one of style, variant, slant, weight or stretch
|
|
// [SIZE] is a decimal number (size in points) (... and points in PANGO_UNITS)
|
|
// any of these options may be absent.
|
|
procedure FontNameToPangoFontDescStr(const LongFontName: string;
|
|
var aFamily,aStyle: string; var aSize: Integer);
|
|
|
|
var
|
|
i: Integer;
|
|
|
|
procedure addStyle(const s: string);
|
|
begin
|
|
if (s<>'') and (s<>'*') and (s<>'r') then begin
|
|
// 'r' is regular
|
|
if aStyle<>'' then
|
|
aStyle := aStyle + ' ' + s
|
|
else
|
|
aStyle := s;
|
|
end;
|
|
end;
|
|
|
|
function GetSize: string;
|
|
var
|
|
c: char;
|
|
validblank: boolean;
|
|
|
|
function IsBlank: boolean;
|
|
begin
|
|
result := c in [#0..' '];
|
|
end;
|
|
|
|
function IsDigit: boolean;
|
|
begin
|
|
result := c in ['0'..'9'];
|
|
end;
|
|
|
|
begin
|
|
Result := '';
|
|
validblank := true;
|
|
i := Length(LongFontname);
|
|
while i>0 do begin
|
|
c := longFontName[i];
|
|
if IsBlank then
|
|
if ValidBlank then begin
|
|
dec(i);
|
|
continue
|
|
end else
|
|
break;
|
|
ValidBlank := false;
|
|
if IsDigit then begin
|
|
Result := C + Result;
|
|
dec(i);
|
|
end else
|
|
break;
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
aStyle := '';
|
|
if IsFontNameXLogicalFontDesc(LongFontName) then begin
|
|
aFamily := ExtractXLFDItem(LongFontName, XLFD_FAMILY);
|
|
if aFamily='*' then
|
|
aFamily:='';
|
|
aSize := StrToIntDef(ExtractXLFDItem(LongFontName, XLFD_POINTSIZE),0) div 10;
|
|
addStyle( ExtractXLFDItem(LongFontName, XLFD_STYLENAME ));
|
|
addStyle( ExtractXLFDItem(LongFontname, XLFD_WEIGHTNAME));
|
|
addStyle( ExtractXLFDItem(LongFontname, XLFD_SLANT));
|
|
addStyle( ExtractXLFDItem(LongFontname, XLFD_WidthName));
|
|
end else begin
|
|
// this could go through, but we want to know at least the pointSize from
|
|
// the fontname
|
|
aSize := StrToIntDef(GetSize,0);
|
|
aFamily := Copy(LongFontName, 1, i);
|
|
// todo: parse aFamily to separate Family and Style
|
|
end;
|
|
end;
|
|
|
|
{ TFont }
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.Create
|
|
Params: none
|
|
Returns: Nothing
|
|
|
|
Constructor for the class.
|
|
------------------------------------------------------------------------------}
|
|
constructor TFont.Create;
|
|
begin
|
|
inherited Create;
|
|
FColor:=clWindowText;
|
|
FPixelsPerInch:=ScreenInfo.PixelsPerInchX;
|
|
FPitch:=DefFontData.Pitch;
|
|
FCharSet:=DefFontData.CharSet;
|
|
DelayAllocate:=true;
|
|
inherited SetName(DefFontData.Name);
|
|
inherited SetFPColor(colBlack);
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.Assign
|
|
Params: Source: Another font
|
|
Returns: nothing
|
|
|
|
Copies the Source font to itself
|
|
------------------------------------------------------------------------------}
|
|
Procedure TFont.Assign(Source: TPersistent);
|
|
begin
|
|
if Source is TFont then begin
|
|
//TODO:lock;
|
|
try
|
|
//TODO: TFont(Source).Lock;
|
|
try
|
|
BeginUpdate;
|
|
try
|
|
CharSet:= TFont(Source).CharSet;
|
|
SetColor(TFont(Source).Color,TFPCanvasHelper(Source).FPColor);
|
|
if TFont(Source).PixelsPerInch <> FPixelsPerInch then
|
|
// use size to convert source height pixels to current resolution
|
|
Size := TFont(Source).Size
|
|
else
|
|
// use height which users could have changed directly
|
|
Height := TFont(Source).Height;
|
|
Name := TFont(Source).Name;
|
|
Pitch := TFont(Source).Pitch;
|
|
Style := TFont(Source).Style;
|
|
finally
|
|
EndUpdate;
|
|
end;
|
|
finally
|
|
//TODO: TFont(Source).UnLock;
|
|
end;
|
|
finally
|
|
//TODO: UnLock;
|
|
end;
|
|
Exit;
|
|
end;
|
|
|
|
inherited Assign(Source);
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.Assign
|
|
Params: ALogFont: TLogFont
|
|
Returns: nothing
|
|
|
|
Copies the logfont settings to itself
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.Assign(const ALogFont: TLogFont);
|
|
var
|
|
AStyle: TFontStyles;
|
|
begin
|
|
BeginUpdate;
|
|
try
|
|
with ALogFont do
|
|
begin
|
|
Height := ALogFont.lfHeight;
|
|
Charset := TFontCharset(ALogFont.lfCharSet);
|
|
AStyle := [];
|
|
with ALogFont do
|
|
begin
|
|
if lfWeight >= FW_SEMIBOLD then Include(AStyle, fsBold);
|
|
if lfItalic <> 0 then Include(AStyle, fsItalic);
|
|
if lfUnderline <> 0 then Include(AStyle, fsUnderline);
|
|
if lfStrikeOut <> 0 then Include(AStyle, fsStrikeOut);
|
|
end;
|
|
if (FIXED_PITCH and lfPitchAndFamily) <> 0 then
|
|
Pitch := fpFixed
|
|
else if (VARIABLE_PITCH and lfPitchAndFamily) <> 0 then
|
|
Pitch := fpVariable
|
|
else
|
|
Pitch := fpDefault;
|
|
Style := AStyle;
|
|
Name := ALogFont.lfFaceName;
|
|
end;
|
|
finally
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
function TFont.IsEqual(AFont: TFont): boolean;
|
|
begin
|
|
if (AFont=Self) then exit(true);
|
|
if (AFont=nil)
|
|
or (CharSet<>AFont.CharSet)
|
|
or (Color<>AFont.Color)
|
|
or (PixelsPerInch<>AFont.PixelsPerInch)
|
|
or (Size<>AFont.Size)
|
|
or (Height<>AFont.Height)
|
|
or (Name<>AFont.Name)
|
|
or (Pitch<>AFont.Pitch)
|
|
or (Style<>AFont.Style) then
|
|
Result:=false
|
|
else
|
|
Result:=true;
|
|
end;
|
|
|
|
procedure TFont.BeginUpdate;
|
|
begin
|
|
inc(FUpdateCount);
|
|
end;
|
|
|
|
procedure TFont.EndUpdate;
|
|
begin
|
|
if FUpdateCount=0 then exit;
|
|
dec(FUpdateCount);
|
|
if (FUpdateCount=0) and FChanged then Changed;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.HandleAllocated
|
|
Params: none
|
|
Returns: boolean
|
|
|
|
Resturns True on handle allocated.
|
|
------------------------------------------------------------------------------}
|
|
function TFont.HandleAllocated: boolean;
|
|
begin
|
|
Result:=FHandle<>0;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
function TFont.IsDefault: boolean;
|
|
------------------------------------------------------------------------------}
|
|
function TFont.IsDefault: boolean;
|
|
begin
|
|
Result:=(CharSet=DEFAULT_CHARSET)
|
|
and (Color=clWindowText)
|
|
and (Height=0)
|
|
and (not IsNameStored)
|
|
and (Pitch=fpDefault)
|
|
and (Size=0)
|
|
and (Style=[]);
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetSize
|
|
Params: AValue: the new value
|
|
Returns: nothing
|
|
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetSize(AValue: Integer);
|
|
begin
|
|
if Size <> AValue then begin
|
|
BeginUpdate;
|
|
try
|
|
FreeHandle;
|
|
inherited SetSize(AValue);
|
|
FHeight := - (AValue * FPixelsPerInch) div 72;
|
|
if IsFontNameXLogicalFontDesc(Name) then
|
|
Name:=ClearXLFDHeight(Name);
|
|
Changed;
|
|
finally
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Function: TFont.GetSize
|
|
Params: none
|
|
Returns: The font size
|
|
|
|
Calculates the size based on height
|
|
------------------------------------------------------------------------------}
|
|
function TFont.GetSize: Integer;
|
|
begin
|
|
Result := inherited Size;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetPitch
|
|
Params: Value: the new value
|
|
Returns: nothing
|
|
|
|
Sets the pitch of a font
|
|
------------------------------------------------------------------------------}
|
|
Procedure TFont.SetPitch(Value : TFontPitch);
|
|
Begin
|
|
if FPitch <> Value then begin
|
|
BeginUpdate;
|
|
FreeHandle;
|
|
FPitch := Value;
|
|
if IsFontNameXLogicalFontDesc(Name) then
|
|
Name:=ClearXLFDPitch(Name);
|
|
Changed;
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetHeight
|
|
Params: Value: the new value
|
|
Returns: nothing
|
|
|
|
Sets the height of a font
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetHeight(Value: Integer);
|
|
begin
|
|
if FHeight=Value then exit;
|
|
// set Size first. This will set FHeight to a rounded equivalent
|
|
Size := -(Value * 72) div FPixelsPerInch;
|
|
// store the real FHeight
|
|
FHeight:=Value;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetStyle
|
|
Params: Value: the new value
|
|
Returns: nothing
|
|
|
|
Sets the style of a font
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetStyle(value : TFontStyles);
|
|
begin
|
|
if FStyle <> Value then begin
|
|
BeginUpdate;
|
|
FreeHandle;
|
|
FStyle := Value;
|
|
if IsFontNameXLogicalFontDesc(Name) then
|
|
Name:=ClearXLFDStyle(Name);
|
|
Changed;
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetColor
|
|
Params: Value: the new value
|
|
Returns: nothing
|
|
|
|
Sets the pencolor of a font
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetColor(Value : TColor);
|
|
begin
|
|
if FColor <> Value then SetColor(Value,TColorToFPColor(Value));
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Function: TFont.GetName
|
|
Params: none
|
|
Returns: The font name
|
|
|
|
Returns the name of the font
|
|
------------------------------------------------------------------------------}
|
|
function TFont.GetName: string;
|
|
begin
|
|
Result:=inherited Name;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetName
|
|
Params: Value: the new value
|
|
Returns: nothing
|
|
|
|
Sets the name of a font
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetName(AValue: string);
|
|
begin
|
|
if Name <> AValue then begin
|
|
FreeHandle;
|
|
inherited SetName(AValue);
|
|
Changed;
|
|
end;
|
|
end;
|
|
|
|
procedure TFont.DoAllocateResources;
|
|
begin
|
|
inherited DoAllocateResources;
|
|
GetHandle;
|
|
end;
|
|
|
|
procedure TFont.DoDeAllocateResources;
|
|
begin
|
|
FreeHandle;
|
|
inherited DoDeAllocateResources;
|
|
end;
|
|
|
|
procedure TFont.DoCopyProps(From: TFPCanvasHelper);
|
|
var
|
|
SrcFont: TFont;
|
|
begin
|
|
BeginUpdate;
|
|
try
|
|
inherited DoCopyProps(From);
|
|
if From is TFont then begin
|
|
SrcFont:=TFont(From);
|
|
Pitch:=SrcFont.Pitch;
|
|
CharSet:=SrcFont.CharSet;
|
|
Style:=SrcFont.Style;
|
|
end;
|
|
finally
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
procedure TFont.SetFlags(Index: integer; AValue: boolean);
|
|
|
|
procedure SetStyleFlag(Flag: TFontStyle; NewValue: boolean);
|
|
begin
|
|
BeginUpdate;
|
|
FreeHandle;
|
|
if NewValue then Include(FStyle,Flag) else Exclude(FStyle,Flag);
|
|
if IsFontNameXLogicalFontDesc(Name) then
|
|
Name:=ClearXLFDStyle(Name);
|
|
Changed;
|
|
EndUpdate;
|
|
end;
|
|
|
|
begin
|
|
if GetFlags(Index)=AValue then exit;
|
|
inherited SetFlags(Index, AValue);
|
|
case Index of
|
|
5: SetStyleFlag(fsBold,AValue);
|
|
6: SetStyleFlag(fsItalic,AValue);
|
|
7: SetStyleFlag(fsUnderline,AValue);
|
|
8: SetStyleFlag(fsStrikeOut,AValue);
|
|
end;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
procedure TFont.SetFPColor(const AValue: TFPColor);
|
|
|
|
Set FPColor and Color
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetFPColor(const AValue: TFPColor);
|
|
begin
|
|
if FPColor=AValue then exit;
|
|
SetColor(FPColorToTColor(AValue),AValue);
|
|
end;
|
|
|
|
procedure TFont.SetColor(const NewColor: TColor; const NewFPColor: TFPColor);
|
|
begin
|
|
if (NewColor=Color) and (NewFPColor=FPColor) then exit;
|
|
FColor:=NewColor;
|
|
inherited SetFPColor(NewFPColor);
|
|
Changed;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.Destroy
|
|
Params: None
|
|
Returns: Nothing
|
|
|
|
Destructor for the class.
|
|
------------------------------------------------------------------------------}
|
|
destructor TFont.Destroy;
|
|
begin
|
|
FreeHandle;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.SetHandle
|
|
Params: a font handle
|
|
Returns: nothing
|
|
|
|
sets the font to an external created font
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.SetHandle(const Value: HFONT);
|
|
begin
|
|
SetData(GetFontData(Value));
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Function: TFont.GetHandle
|
|
Params: none
|
|
Returns: a handle to a font gdiobject
|
|
|
|
Creates a font if needed
|
|
------------------------------------------------------------------------------}
|
|
function TFont.GetHandle: HFONT;
|
|
const
|
|
LF_BOOL: array[Boolean] of Byte = (0, 255);
|
|
LF_WEIGHT: array[Boolean] of Integer = (FW_NORMAL, FW_BOLD);
|
|
var
|
|
ALogFont: TLogFont;
|
|
CachedFont: TFontHandleCacheDescriptor;
|
|
|
|
procedure SetLogFontName(const NewName: string);
|
|
var l: integer;
|
|
aName: string;
|
|
begin
|
|
if IsFontNameXLogicalFontDesc(NewName) then
|
|
aName:=ExtractFamilyFromXLFDName(NewName)
|
|
else
|
|
aName:=NewName;
|
|
l:=High(ALogFont.lfFaceName)-Low(ALogFont.lfFaceName);
|
|
if l>length(aName) then l:=length(aName);
|
|
if l>0 then
|
|
Move(aName[1],ALogFont.lfFaceName[Low(ALogFont.lfFaceName)],l);
|
|
ALogFont.lfFaceName[Low(ALogFont.lfFaceName)+l]:=#0;
|
|
end;
|
|
|
|
begin
|
|
if FHandle = 0 then with ALogFont do
|
|
begin
|
|
FillChar(ALogFont,SizeOf(ALogFont),0);
|
|
lfHeight := Height;
|
|
lfWidth := 0;
|
|
lfEscapement := 0;
|
|
lfOrientation := 0;
|
|
lfWeight := LF_WEIGHT[fsBold in Style];
|
|
lfItalic := LF_BOOL[fsItalic in Style];
|
|
lfUnderline := LF_BOOL[fsUnderline in Style];
|
|
lfStrikeOut := LF_BOOL[fsStrikeOut in Style];
|
|
lfCharSet := Byte(FCharset);
|
|
SetLogFontName(Name);
|
|
|
|
lfQuality := DEFAULT_QUALITY;
|
|
lfOutPrecision := OUT_DEFAULT_PRECIS;
|
|
lfClipPrecision := CLIP_DEFAULT_PRECIS;
|
|
case Pitch of
|
|
fpVariable: lfPitchAndFamily := VARIABLE_PITCH;
|
|
fpFixed: lfPitchAndFamily := FIXED_PITCH;
|
|
else
|
|
lfPitchAndFamily := DEFAULT_PITCH;
|
|
end;
|
|
|
|
// ask the interface for the nearest font
|
|
CachedFont:=FontResourceCache.FindFontDesc(ALogFont,Name);
|
|
if CachedFont<>nil then begin
|
|
CachedFont.Item.IncreaseRefCount;
|
|
FHandle := CachedFont.Item.Handle;
|
|
end else begin
|
|
FHandle := CreateFontIndirectEx(ALogFont,Name);
|
|
FontResourceCache.Add(FHandle,ALogFont,Name);
|
|
end;
|
|
FFontHandleCached:=true;
|
|
FCanUTF8:=FontCanUTF8(FHandle);
|
|
end;
|
|
|
|
Result := FHandle;
|
|
end;
|
|
|
|
{------------------------------------------------------------------------------
|
|
Method: TFont.FreeHandle
|
|
Params: none
|
|
Returns: Nothing
|
|
|
|
Frees a fonthandle if needed
|
|
------------------------------------------------------------------------------}
|
|
procedure TFont.FreeHandle;
|
|
begin
|
|
if FHandle <> 0
|
|
then begin
|
|
if FFontHandleCached then begin
|
|
FontResourceCache.FindFont(FHandle).DecreaseRefCount;
|
|
FFontHandleCached:=false;
|
|
end else
|
|
DeleteObject(FHandle);
|
|
FHandle := 0;
|
|
end;
|
|
end;
|
|
|
|
function TFont.GetCharSet: TFontCharSet;
|
|
begin
|
|
Result:=FCharSet;
|
|
end;
|
|
|
|
procedure TFont.SetCharSet(const AValue: TFontCharSet);
|
|
begin
|
|
if FCharSet <> AValue then begin
|
|
FreeHandle;
|
|
FCharSet := AValue;
|
|
Changed;
|
|
end;
|
|
end;
|
|
|
|
procedure TFont.GetData(var FontData: TFontData);
|
|
begin
|
|
FontData := DefFontData;
|
|
FontData.Handle:=0;
|
|
FontData.Height:=Height;
|
|
FontData.Pitch:=Pitch;
|
|
FontData.Style:=Style;
|
|
FontData.CharSet:=CharSet;
|
|
FontData.Name:=LeftStr(Name,SizeOf(FontData.Name)-1);
|
|
end;
|
|
|
|
function TFont.IsNameStored: boolean;
|
|
begin
|
|
Result:=DefFontData.Name<>Name;
|
|
end;
|
|
|
|
procedure TFont.SetData(const FontData: TFontData);
|
|
var
|
|
OldStyle: TFontStylesbase;
|
|
begin
|
|
if (FHandle <> FontData.Handle) or (FHandle=0) then begin
|
|
OldStyle:=FStyle;
|
|
FreeHandle;
|
|
FHandle := FontData.Handle;
|
|
inherited SetSize(-(FontData.Height * 72) div FPixelsPerInch);
|
|
FHeight := FontData.Height;
|
|
FPitch:=FontData.Pitch;
|
|
FStyle:=FontData.Style;
|
|
FCharSet:=FontData.CharSet;
|
|
inherited SetName(FontData.Name);
|
|
bold;
|
|
if (fsBold in OldStyle)<>(fsBold in FStyle) then
|
|
inherited SetFlags(5,fsBold in FStyle);
|
|
if (fsItalic in OldStyle)<>(fsItalic in FStyle) then
|
|
inherited SetFlags(6,fsItalic in FStyle);
|
|
if (fsUnderline in OldStyle)<>(fsUnderline in FStyle) then
|
|
inherited SetFlags(7,fsUnderline in FStyle);
|
|
if (fsStrikeOut in OldStyle)<>(fsStrikeOut in FStyle) then
|
|
inherited SetFlags(8,fsStrikeOut in FStyle);
|
|
Changed;
|
|
end;
|
|
end;
|
|
|
|
function TFont.GetHeight: Integer;
|
|
begin
|
|
Result := FHeight;
|
|
end;
|
|
|
|
function TFont.GetPitch: TFontPitch;
|
|
begin
|
|
Result:=FPitch;
|
|
end;
|
|
|
|
function TFont.GetStyle: TFontStyles;
|
|
begin
|
|
Result:=FStyle;
|
|
end;
|
|
|
|
procedure TFont.Changed;
|
|
begin
|
|
if FUpdateCount>0 then begin
|
|
FChanged:=true;
|
|
exit;
|
|
end;
|
|
FChanged:=false;
|
|
inherited Changed;
|
|
// ToDo: we need interfaces:
|
|
// if FNotify <> nil then FNotify.Changed;
|
|
end;
|
|
|
|
// included by graphics.pp
|