{------------------------------------------------------------------------------- The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/ Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is: SynEditMiscProcs.pas, released 2000-04-07. The Original Code is based on the mwSupportProcs.pas file from the mwEdit component suite by Martin Waldenburg and other developers, the Initial Author of this file is Michael Hieke. Unicode translation by Maël Hörz. All Rights Reserved. Contributors to the SynEdit and mwEdit projects are listed in the Contributors.txt file. Alternatively, the contents of this file may be used under the terms of the GNU General Public License Version 2 or later (the "GPL"), in which case the provisions of the GPL are applicable instead of those above. If you wish to allow use of your version of this file only under the terms of the GPL and not to allow others to use your version of this file under the MPL, indicate your decision by deleting the provisions above and replace them with the notice and other provisions required by the GPL. If you do not delete the provisions above, a recipient may use your version of this file under either the MPL or the GPL. $Id: SynEditMiscProcs.pas,v 1.35.2.8 2009/09/28 17:54:20 maelh Exp $ You may retrieve the latest version of this file at the SynEdit home page, located at http://SynEdit.SourceForge.net Known Issues: -------------------------------------------------------------------------------} {$IFNDEF QSYNEDITMISCPROCS} unit SynEditMiscProcs; {$ENDIF} {$I SynEdit.inc} interface uses Windows, Graphics, SynEditTypes, SynEditHighlighter, SynUnicode, {$IFDEF SYN_COMPILER_4_UP} Math, {$ENDIF} Classes; const MaxIntArraySize = MaxInt div 16; type PIntArray = ^TIntArray; TIntArray = array[0..MaxIntArraySize - 1] of Integer; {$IFNDEF SYN_COMPILER_4_UP} function Max(x, y: Integer): Integer; function Min(x, y: Integer): Integer; {$ENDIF} function MinMax(x, mi, ma: Integer): Integer; procedure SwapInt(var l, r: Integer); function MaxPoint(const P1, P2: TPoint): TPoint; function MinPoint(const P1, P2: TPoint): TPoint; function GetIntArray(Count: Cardinal; InitialValue: Integer): PIntArray; procedure InternalFillRect(dc: HDC; const rcPaint: TRect); // Converting tabs to spaces: To use the function several times it's better // to use a function pointer that is set to the fastest conversion function. type TConvertTabsProc = function(const Line: UnicodeString; TabWidth: Integer): UnicodeString; function GetBestConvertTabsProc(TabWidth: Integer): TConvertTabsProc; // This is the slowest conversion function which can handle TabWidth <> 2^n. function ConvertTabs(const Line: UnicodeString; TabWidth: Integer): UnicodeString; type TConvertTabsProcEx = function(const Line: UnicodeString; TabWidth: Integer; var HasTabs: Boolean): UnicodeString; function GetBestConvertTabsProcEx(TabWidth: Integer): TConvertTabsProcEx; // This is the slowest conversion function which can handle TabWidth <> 2^n. function ConvertTabsEx(const Line: UnicodeString; TabWidth: Integer; var HasTabs: Boolean): UnicodeString; function GetExpandedLength(const aStr: UnicodeString; aTabWidth: Integer): Integer; function CharIndex2CaretPos(Index, TabWidth: Integer; const Line: UnicodeString): Integer; function CaretPos2CharIndex(Position, TabWidth: Integer; const Line: UnicodeString; var InsideTabChar: Boolean): Integer; // search for the first char of set AChars in Line, starting at index Start function StrScanForCharInCategory(const Line: UnicodeString; Start: Integer; IsOfCategory: TCategoryMethod): Integer; // the same, but searching backwards function StrRScanForCharInCategory(const Line: UnicodeString; Start: Integer; IsOfCategory: TCategoryMethod): Integer; function GetEOL(Line: PWideChar): PWideChar; // Remove all '/' characters from string by changing them into '\.'. // Change all '\' characters into '\\' to allow for unique decoding. function EncodeString(s: UnicodeString): UnicodeString; // Decodes string, encoded with EncodeString. function DecodeString(s: UnicodeString): UnicodeString; {$IFNDEF SYN_COMPILER_5_UP} procedure FreeAndNil(var Obj); {$ENDIF} {$IFNDEF SYN_COMPILER_3_UP} procedure Assert(Expr: Boolean); { stub for Delphi 2 } {$ENDIF} {$IFNDEF SYN_COMPILER_3_UP} function LastDelimiter(const Delimiters, S: UnicodeString): Integer; {$ENDIF} {$IFNDEF SYN_COMPILER_4_UP} type TReplaceFlags = set of (rfReplaceAll, rfIgnoreCase); function StringReplace(const S, OldPattern, NewPattern: UnicodeString; Flags: TReplaceFlags): UnicodeString; {$ENDIF} type THighlighterAttriProc = function (Highlighter: TSynCustomHighlighter; Attri: TSynHighlighterAttributes; UniqueAttriName: string; Params: array of Pointer): Boolean of object; // Enums all child highlighters and their attributes of a TSynMultiSyn through a // callback function. // This function also handles nested TSynMultiSyns including their MarkerAttri. function EnumHighlighterAttris(Highlighter: TSynCustomHighlighter; SkipDuplicates: Boolean; HighlighterAttriProc: THighlighterAttriProc; Params: array of Pointer): Boolean; {$IFDEF SYN_HEREDOC} // Calculates Frame Check Sequence (FCS) 16-bit Checksum (as defined in RFC 1171) function CalcFCS(const ABuf; ABufSize: Cardinal): Word; {$ENDIF} procedure SynDrawGradient(const ACanvas: TCanvas; const AStartColor, AEndColor: TColor; ASteps: Integer; const ARect: TRect; const AHorizontal: Boolean); overload; function DeleteTypePrefixAndSynSuffix(S: string): string; implementation uses SysUtils, SynHighlighterMulti; {$IFNDEF SYN_COMPILER_4_UP} function Max(x, y: Integer): Integer; begin if x > y then Result := x else Result := y; end; function Min(x, y: Integer): Integer; begin if x < y then Result := x else Result := y; end; {$ENDIF} function MinMax(x, mi, ma: Integer): Integer; begin x := Min(x, ma); Result := Max(x, mi); end; procedure SwapInt(var l, r: Integer); var tmp: Integer; begin tmp := r; r := l; l := tmp; end; function MaxPoint(const P1, P2: TPoint): TPoint; begin if (P2.y > P1.y) or ((P2.y = P1.y) and (P2.x > P1.x)) then Result := P2 else Result := P1; end; function MinPoint(const P1, P2: TPoint): TPoint; begin if (P2.y < P1.y) or ((P2.y = P1.y) and (P2.x < P1.x)) then Result := P2 else Result := P1; end; function GetIntArray(Count: Cardinal; InitialValue: Integer): PIntArray; var p: PInteger; begin Result := AllocMem(Count * SizeOf(Integer)); if Assigned(Result) and (InitialValue <> 0) then begin p := PInteger(Result); while (Count > 0) do begin p^ := InitialValue; Inc(p); Dec(Count); end; end; end; procedure InternalFillRect(dc: HDC; const rcPaint: TRect); begin ExtTextOut(dc, 0, 0, ETO_OPAQUE, @rcPaint, nil, 0, nil); end; // Please don't change this function; no stack frame and efficient register use. function GetHasTabs(pLine: PWideChar; var CharsBefore: Integer): Boolean; begin CharsBefore := 0; if Assigned(pLine) then begin while pLine^ <> #0 do begin if pLine^ = #9 then Break; Inc(CharsBefore); Inc(pLine); end; Result := pLine^ = #9; end else Result := False; end; function ConvertTabs1Ex(const Line: UnicodeString; TabWidth: Integer; var HasTabs: Boolean): UnicodeString; var pDest: PWideChar; nBeforeTab: Integer; begin Result := Line; // increment reference count only if GetHasTabs(Pointer(Line), nBeforeTab) then begin HasTabs := True; pDest := @Result[nBeforeTab + 1]; // this will make a copy of Line // We have at least one tab in the string, and the tab width is 1. // pDest points to the first tab char. We overwrite all tabs with spaces. repeat if (pDest^ = #9) then pDest^ := ' '; Inc(pDest); until (pDest^ = #0); end else HasTabs := False; end; function ConvertTabs1(const Line: UnicodeString; TabWidth: Integer): UnicodeString; var HasTabs: Boolean; begin Result := ConvertTabs1Ex(Line, TabWidth, HasTabs); end; function ConvertTabs2nEx(const Line: UnicodeString; TabWidth: Integer; var HasTabs: Boolean): UnicodeString; var i, DestLen, TabCount, TabMask: Integer; pSrc, pDest: PWideChar; begin Result := Line; // increment reference count only if GetHasTabs(Pointer(Line), DestLen) then begin HasTabs := True; pSrc := @Line[1 + DestLen]; // We have at least one tab in the string, and the tab width equals 2^n. // pSrc points to the first tab char in Line. We get the number of tabs // and the length of the expanded string now. TabCount := 0; TabMask := (TabWidth - 1) xor $7FFFFFFF; repeat if pSrc^ = #9 then begin DestLen := (DestLen + TabWidth) and TabMask; Inc(TabCount); end else Inc(DestLen); Inc(pSrc); until (pSrc^ = #0); // Set the length of the expanded string. SetLength(Result, DestLen); DestLen := 0; pSrc := PWideChar(Line); pDest := PWideChar(Result); // We use another TabMask here to get the difference to 2^n. TabMask := TabWidth - 1; repeat if pSrc^ = #9 then begin i := TabWidth - (DestLen and TabMask); Inc(DestLen, i); //This is used for both drawing and other stuff and is meant to be #9 and not #32 repeat pDest^ := #9; Inc(pDest); Dec(i); until (i = 0); Dec(TabCount); if TabCount = 0 then begin repeat Inc(pSrc); pDest^ := pSrc^; Inc(pDest); until (pSrc^ = #0); Exit; end; end else begin pDest^ := pSrc^; Inc(pDest); Inc(DestLen); end; Inc(pSrc); until (pSrc^ = #0); end else HasTabs := False; end; function ConvertTabs2n(const Line: UnicodeString; TabWidth: Integer): UnicodeString; var HasTabs: Boolean; begin Result := ConvertTabs2nEx(Line, TabWidth, HasTabs); end; function ConvertTabsEx(const Line: UnicodeString; TabWidth: Integer; var HasTabs: Boolean): UnicodeString; var i, DestLen, TabCount: Integer; pSrc, pDest: PWideChar; begin Result := Line; // increment reference count only if GetHasTabs(Pointer(Line), DestLen) then begin HasTabs := True; pSrc := @Line[1 + DestLen]; // We have at least one tab in the string, and the tab width is greater // than 1. pSrc points to the first tab char in Line. We get the number // of tabs and the length of the expanded string now. TabCount := 0; repeat if pSrc^ = #9 then begin DestLen := DestLen + TabWidth - DestLen mod TabWidth; Inc(TabCount); end else Inc(DestLen); Inc(pSrc); until (pSrc^ = #0); // Set the length of the expanded string. SetLength(Result, DestLen); DestLen := 0; pSrc := PWideChar(Line); pDest := PWideChar(Result); repeat if pSrc^ = #9 then begin i := TabWidth - (DestLen mod TabWidth); Inc(DestLen, i); repeat pDest^ := #9; Inc(pDest); Dec(i); until (i = 0); Dec(TabCount); if TabCount = 0 then begin repeat Inc(pSrc); pDest^ := pSrc^; Inc(pDest); until (pSrc^ = #0); Exit; end; end else begin pDest^ := pSrc^; Inc(pDest); Inc(DestLen); end; Inc(pSrc); until (pSrc^ = #0); end else HasTabs := False; end; function ConvertTabs(const Line: UnicodeString; TabWidth: Integer): UnicodeString; var HasTabs: Boolean; begin Result := ConvertTabsEx(Line, TabWidth, HasTabs); end; function IsPowerOfTwo(TabWidth: Integer): Boolean; var nW: Integer; begin nW := 2; repeat if (nW >= TabWidth) then Break; Inc(nW, nW); until (nW >= $10000); // we don't want 64 kByte spaces... Result := (nW = TabWidth); end; function GetBestConvertTabsProc(TabWidth: Integer): TConvertTabsProc; begin if (TabWidth < 2) then Result := TConvertTabsProc(@ConvertTabs1) else if IsPowerOfTwo(TabWidth) then Result := TConvertTabsProc(@ConvertTabs2n) else Result := TConvertTabsProc(@ConvertTabs); end; function GetBestConvertTabsProcEx(TabWidth: Integer): TConvertTabsProcEx; begin if (TabWidth < 2) then Result := ConvertTabs1Ex else if IsPowerOfTwo(TabWidth) then Result := ConvertTabs2nEx else Result := ConvertTabsEx; end; function GetExpandedLength(const aStr: UnicodeString; aTabWidth: Integer): Integer; var iRun: PWideChar; begin Result := 0; iRun := PWideChar(aStr); while iRun^ <> #0 do begin if iRun^ = #9 then Inc(Result, aTabWidth - (Result mod aTabWidth)) else Inc(Result); Inc(iRun); end; end; function CharIndex2CaretPos(Index, TabWidth: Integer; const Line: UnicodeString): Integer; var iChar: Integer; pNext: PWideChar; begin // possible sanity check here: Index := Max(Index, Length(Line)); if Index > 1 then begin if (TabWidth <= 1) or not GetHasTabs(Pointer(Line), iChar) then Result := Index else begin if iChar + 1 >= Index then Result := Index else begin // iChar is number of chars before first #9 Result := iChar; // Index is *not* zero-based Inc(iChar); Dec(Index, iChar); pNext := @Line[iChar]; while Index > 0 do begin case pNext^ of #0: begin Inc(Result, Index); Break; end; #9: begin // Result is still zero-based Inc(Result, TabWidth); Dec(Result, Result mod TabWidth); end; else Inc(Result); end; Dec(Index); Inc(pNext); end; // done with zero-based computation Inc(Result); end; end; end else Result := 1; end; function CaretPos2CharIndex(Position, TabWidth: Integer; const Line: UnicodeString; var InsideTabChar: Boolean): Integer; var iPos: Integer; pNext: PWideChar; begin InsideTabChar := False; if Position > 1 then begin if (TabWidth <= 1) or not GetHasTabs(Pointer(Line), iPos) then Result := Position else begin if iPos + 1 >= Position then Result := Position else begin // iPos is number of chars before first #9 Result := iPos + 1; pNext := @Line[Result]; // for easier computation go zero-based (mod-operation) Dec(Position); while iPos < Position do begin case pNext^ of #0: Break; #9: begin Inc(iPos, TabWidth); Dec(iPos, iPos mod TabWidth); if iPos > Position then begin InsideTabChar := True; Break; end; end; else Inc(iPos); end; Inc(Result); Inc(pNext); end; end; end; end else Result := Position; end; function StrScanForCharInCategory(const Line: UnicodeString; Start: Integer; IsOfCategory: TCategoryMethod): Integer; var p: PWideChar; begin if (Start > 0) and (Start <= Length(Line)) then begin p := PWideChar(@Line[Start]); repeat if IsOfCategory(p^) then begin Result := Start; Exit; end; Inc(p); Inc(Start); until p^ = #0; end; Result := 0; end; function StrRScanForCharInCategory(const Line: UnicodeString; Start: Integer; IsOfCategory: TCategoryMethod): Integer; var I: Integer; begin Result := 0; if (Start > 0) and (Start <= Length(Line)) then begin for I := Start downto 1 do if IsOfCategory(Line[I]) then begin Result := I; Exit; end; end; end; function GetEOL(Line: PWideChar): PWideChar; begin Result := Line; if Assigned(Result) then while (Result^ <> #0) and (Result^ <> #10) and (Result^ <> #13) do Inc(Result); end; {$IFOPT R+}{$DEFINE RestoreRangeChecking}{$ELSE}{$UNDEF RestoreRangeChecking}{$ENDIF} {$R-} function EncodeString(s: UnicodeString): UnicodeString; var i, j: Integer; begin SetLength(Result, 2 * Length(s)); // worst case j := 0; for i := 1 to Length(s) do begin Inc(j); if s[i] = '\' then begin Result[j] := '\'; Result[j + 1] := '\'; Inc(j); end else if s[i] = '/' then begin Result[j] := '\'; Result[j + 1] := '.'; Inc(j); end else Result[j] := s[i]; end; //for SetLength(Result, j); end; { EncodeString } function DecodeString(s: UnicodeString): UnicodeString; var i, j: Integer; begin SetLength(Result, Length(s)); // worst case j := 0; i := 1; while i <= Length(s) do begin Inc(j); if s[i] = '\' then begin Inc(i); if s[i] = '\' then Result[j] := '\' else Result[j] := '/'; end else Result[j] := s[i]; Inc(i); end; //for SetLength(Result,j); end; { DecodeString } {$IFDEF RestoreRangeChecking}{$R+}{$ENDIF} {$IFNDEF SYN_COMPILER_5_UP} procedure FreeAndNil(var Obj); var P: TObject; begin P := TObject(Obj); TObject(Obj) := nil; P.Free; end; {$ENDIF} {$IFNDEF SYN_COMPILER_3_UP} procedure Assert(Expr: Boolean); { stub for Delphi 2 } begin end; {$ENDIF} {$IFNDEF SYN_COMPILER_3_UP} function LastDelimiter(const Delimiters, S: UnicodeString): Integer; var P: PWideChar; begin Result := Length(S); P := PWideChar(Delimiters); while Result > 0 do begin if (S[Result] <> #0) and (StrScan(P, S[Result]) <> nil) then Exit; Dec(Result); end; end; {$ENDIF} {$IFNDEF SYN_COMPILER_4_UP} function StringReplace(const S, OldPattern, NewPattern: UnicodeString; Flags: TReplaceFlags): UnicodeString; var SearchStr, Patt, NewStr: UnicodeString; Offset: Integer; begin if rfIgnoreCase in Flags then begin SearchStr := SynWideUpperCase(S); Patt := SynWideUpperCase(OldPattern); end else begin SearchStr := S; Patt := OldPattern; end; NewStr := S; Result := ''; while SearchStr <> '' do begin Offset := Pos(Patt, SearchStr); if Offset = 0 then begin Result := Result + NewStr; Break; end; Result := Result + Copy(NewStr, 1, Offset - 1) + NewPattern; NewStr := Copy(NewStr, Offset + Length(OldPattern), MaxInt); if not (rfReplaceAll in Flags) then begin Result := Result + NewStr; Break; end; SearchStr := Copy(SearchStr, Offset + Length(Patt), MaxInt); end; end; {$ENDIF} function DeleteTypePrefixAndSynSuffix(S: string): string; begin Result := S; if CharInSet(Result[1], ['T', 't']) then //ClassName is never empty so no AV possible if Pos('tsyn', LowerCase(Result)) = 1 then Delete(Result, 1, 4) else Delete(Result, 1, 1); if Copy(LowerCase(Result), Length(Result) - 2, 3) = 'syn' then SetLength(Result, Length(Result) - 3); end; function GetHighlighterIndex(Highlighter: TSynCustomHighlighter; HighlighterList: TList): Integer; var i: Integer; begin Result := 1; for i := 0 to HighlighterList.Count - 1 do if HighlighterList[i] = Highlighter then Exit else if Assigned(HighlighterList[i]) and (TObject(HighlighterList[i]).ClassType = Highlighter.ClassType) then Inc(Result); end; function InternalEnumHighlighterAttris(Highlighter: TSynCustomHighlighter; SkipDuplicates: Boolean; HighlighterAttriProc: THighlighterAttriProc; Params: array of Pointer; HighlighterList: TList): Boolean; var i: Integer; UniqueAttriName: string; begin Result := True; if (HighlighterList.IndexOf(Highlighter) >= 0) then begin if SkipDuplicates then Exit; end else HighlighterList.Add(Highlighter); if Highlighter is TSynMultiSyn then with TSynMultiSyn(Highlighter) do begin Result := InternalEnumHighlighterAttris(DefaultHighlighter, SkipDuplicates, HighlighterAttriProc, Params, HighlighterList); if not Result then Exit; for i := 0 to Schemes.Count - 1 do begin UniqueAttriName := Highlighter.ExportName + IntToStr(GetHighlighterIndex(Highlighter, HighlighterList)) + '.' + Schemes[i].MarkerAttri.Name + IntToStr(i + 1); Result := HighlighterAttriProc(Highlighter, Schemes[i].MarkerAttri, UniqueAttriName, Params); if not Result then Exit; Result := InternalEnumHighlighterAttris(Schemes[i].Highlighter, SkipDuplicates, HighlighterAttriProc, Params, HighlighterList); if not Result then Exit end end else if Assigned(Highlighter) then for i := 0 to Highlighter.AttrCount - 1 do begin UniqueAttriName := Highlighter.ExportName + IntToStr(GetHighlighterIndex(Highlighter, HighlighterList)) + '.' + Highlighter.Attribute[i].Name; Result := HighlighterAttriProc(Highlighter, Highlighter.Attribute[i], UniqueAttriName, Params); if not Result then Exit end end; function EnumHighlighterAttris(Highlighter: TSynCustomHighlighter; SkipDuplicates: Boolean; HighlighterAttriProc: THighlighterAttriProc; Params: array of Pointer): Boolean; var HighlighterList: TList; begin if not Assigned(Highlighter) or not Assigned(HighlighterAttriProc) then begin Result := False; Exit; end; HighlighterList := TList.Create; try Result := InternalEnumHighlighterAttris(Highlighter, SkipDuplicates, HighlighterAttriProc, Params, HighlighterList) finally HighlighterList.Free end end; {$IFDEF SYN_HEREDOC} // Fast Frame Check Sequence (FCS) Implementation // Translated from sample code given with RFC 1171 by Marko Njezic const fcstab : array[Byte] of Word = ( $0000, $1189, $2312, $329b, $4624, $57ad, $6536, $74bf, $8c48, $9dc1, $af5a, $bed3, $ca6c, $dbe5, $e97e, $f8f7, $1081, $0108, $3393, $221a, $56a5, $472c, $75b7, $643e, $9cc9, $8d40, $bfdb, $ae52, $daed, $cb64, $f9ff, $e876, $2102, $308b, $0210, $1399, $6726, $76af, $4434, $55bd, $ad4a, $bcc3, $8e58, $9fd1, $eb6e, $fae7, $c87c, $d9f5, $3183, $200a, $1291, $0318, $77a7, $662e, $54b5, $453c, $bdcb, $ac42, $9ed9, $8f50, $fbef, $ea66, $d8fd, $c974, $4204, $538d, $6116, $709f, $0420, $15a9, $2732, $36bb, $ce4c, $dfc5, $ed5e, $fcd7, $8868, $99e1, $ab7a, $baf3, $5285, $430c, $7197, $601e, $14a1, $0528, $37b3, $263a, $decd, $cf44, $fddf, $ec56, $98e9, $8960, $bbfb, $aa72, $6306, $728f, $4014, $519d, $2522, $34ab, $0630, $17b9, $ef4e, $fec7, $cc5c, $ddd5, $a96a, $b8e3, $8a78, $9bf1, $7387, $620e, $5095, $411c, $35a3, $242a, $16b1, $0738, $ffcf, $ee46, $dcdd, $cd54, $b9eb, $a862, $9af9, $8b70, $8408, $9581, $a71a, $b693, $c22c, $d3a5, $e13e, $f0b7, $0840, $19c9, $2b52, $3adb, $4e64, $5fed, $6d76, $7cff, $9489, $8500, $b79b, $a612, $d2ad, $c324, $f1bf, $e036, $18c1, $0948, $3bd3, $2a5a, $5ee5, $4f6c, $7df7, $6c7e, $a50a, $b483, $8618, $9791, $e32e, $f2a7, $c03c, $d1b5, $2942, $38cb, $0a50, $1bd9, $6f66, $7eef, $4c74, $5dfd, $b58b, $a402, $9699, $8710, $f3af, $e226, $d0bd, $c134, $39c3, $284a, $1ad1, $0b58, $7fe7, $6e6e, $5cf5, $4d7c, $c60c, $d785, $e51e, $f497, $8028, $91a1, $a33a, $b2b3, $4a44, $5bcd, $6956, $78df, $0c60, $1de9, $2f72, $3efb, $d68d, $c704, $f59f, $e416, $90a9, $8120, $b3bb, $a232, $5ac5, $4b4c, $79d7, $685e, $1ce1, $0d68, $3ff3, $2e7a, $e70e, $f687, $c41c, $d595, $a12a, $b0a3, $8238, $93b1, $6b46, $7acf, $4854, $59dd, $2d62, $3ceb, $0e70, $1ff9, $f78f, $e606, $d49d, $c514, $b1ab, $a022, $92b9, $8330, $7bc7, $6a4e, $58d5, $495c, $3de3, $2c6a, $1ef1, $0f78 ); function CalcFCS(const ABuf; ABufSize: Cardinal): Word; var CurFCS: Word; P: ^Byte; begin CurFCS := $ffff; P := @ABuf; while ABufSize <> 0 do begin CurFCS := (CurFCS shr 8) xor fcstab[(CurFCS xor P^) and $ff]; Dec(ABufSize); Inc(P); end; Result := CurFCS; end; {$ENDIF} procedure SynDrawGradient(const ACanvas: TCanvas; const AStartColor, AEndColor: TColor; ASteps: Integer; const ARect: TRect; const AHorizontal: Boolean); var StartColorR, StartColorG, StartColorB: Byte; DiffColorR, DiffColorG, DiffColorB: Integer; i, Size: Integer; PaintRect: TRect; begin StartColorR := GetRValue(ColorToRGB(AStartColor)); StartColorG := GetGValue(ColorToRGB(AStartColor)); StartColorB := GetBValue(ColorToRGB(AStartColor)); DiffColorR := GetRValue(ColorToRGB(AEndColor)) - StartColorR; DiffColorG := GetGValue(ColorToRGB(AEndColor)) - StartColorG; DiffColorB := GetBValue(ColorToRGB(AEndColor)) - StartColorB; ASteps := MinMax(ASteps, 2, 256); if AHorizontal then begin Size := ARect.Right - ARect.Left; PaintRect.Top := ARect.Top; PaintRect.Bottom := ARect.Bottom; for i := 0 to ASteps - 1 do begin PaintRect.Left := ARect.Left + MulDiv(i, Size, ASteps); PaintRect.Right := ARect.Left + MulDiv(i + 1, Size, ASteps); ACanvas.Brush.Color := RGB(StartColorR + MulDiv(i, DiffColorR, ASteps - 1), StartColorG + MulDiv(i, DiffColorG, ASteps - 1), StartColorB + MulDiv(i, DiffColorB, ASteps - 1)); ACanvas.FillRect(PaintRect); end; end else begin Size := ARect.Bottom - ARect.Top; PaintRect.Left := ARect.Left; PaintRect.Right := ARect.Right; for i := 0 to ASteps - 1 do begin PaintRect.Top := ARect.Top + MulDiv(i, Size, ASteps); PaintRect.Bottom := ARect.Top + MulDiv(i + 1, Size, ASteps); ACanvas.Brush.Color := RGB(StartColorR + MulDiv(i, DiffColorR, ASteps - 1), StartColorG + MulDiv(i, DiffColorG, ASteps - 1), StartColorB + MulDiv(i, DiffColorB, ASteps - 1)); ACanvas.FillRect(PaintRect); end; end; end; end.