{------------------------------------------------------------------------------- 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. 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$ You may retrieve the latest version of this file at the SynEdit home page, located at http://SynEdit.SourceForge.net Known Issues: -------------------------------------------------------------------------------} unit SynEditMiscProcs; {$I synedit.inc} interface uses {$IFDEF SYN_LAZARUS} LCLLinux, {$ELSE} Windows, {$ENDIF} Classes, SynEditTypes; type PIntArray = ^TIntArray; TIntArray = array[0..MaxListSize - 1] of integer; {$IFDEF FPC} function MulDiv(Factor1,Factor2,Divisor:integer):integer; {$ENDIF} function Max(x, y: integer): integer; function Min(x, y: integer): integer; function MinMax(x, mi, ma: integer): integer; procedure SwapInt(var l, r: integer); function maxPoint(P1, P2: TPoint): TPoint; function minPoint(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: AnsiString; TabWidth: integer): AnsiString; function GetBestConvertTabsProc(TabWidth: integer): TConvertTabsProc; // This is the slowest conversion function which can handle TabWidth <> 2^n. function ConvertTabs(const Line: AnsiString; TabWidth: integer): AnsiString; {begin} //mh 2000-10-19 type TConvertTabsProcEx = function(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): AnsiString; {$IFDEF SYN_LAZARUS} TSimulateConvertTabsProcEx = function(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): integer; // returns length of converted string {$ENDIF} function GetBestConvertTabsProcEx(TabWidth: integer): TConvertTabsProcEx; {$IFDEF SYN_LAZARUS} function GetBestSimulateConvertTabsProcEx( TabWidth:integer): TSimulateConvertTabsProcEx; {$ENDIF} // This is the slowest conversion function which can handle TabWidth <> 2^n. function ConvertTabsEx(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): AnsiString; {end} //mh 2000-10-19 function CharIndex2CaretPos(Index, TabWidth: integer; const Line: string): integer; function CaretPos2CharIndex(Position, TabWidth: integer; const Line: string; var InsideTabChar: boolean): integer; // search for the first char of set AChars in Line, starting at index Start function StrScanForCharInSet(const Line: string; Start: integer; AChars: TSynIdentChars): integer; // the same, but searching backwards function StrRScanForCharInSet(const Line: string; Start: integer; AChars: TSynIdentChars): integer; function GetEOL(Line: PChar): PChar; {begin} //gp 2000-06-24 // Remove all '/' characters from string by changing them into '\.'. // Change all '\' characters into '\\' to allow for unique decoding. function EncodeString(s: string): string; {$IFDEF SYN_LAZARUS} function EncodeStringLength(s: string): integer; {$ENDIF} // Decodes string, encoded with EncodeString. function DecodeString(s: string): string; {end} //gp 2000-06-24 implementation uses SysUtils; {***} {$IFDEF FPC} function MulDiv(Factor1,Factor2,Divisor:integer):integer; begin Result:=(int64(Factor1)*int64(Factor2)) div Divisor; end; {$ENDIF} 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; function MinMax(x, mi, ma: integer): integer; begin if (x < mi) then Result := mi else if (x > ma) then Result := ma else Result := x; end; procedure SwapInt(var l, r: integer); var tmp: integer; begin tmp := r; r := l; l := tmp; end; function maxPoint(P1, P2: TPoint): TPoint; begin Result := P1; if (P2.y > P1.y) or ((P2.y = P1.y) and (P2.x > P1.x)) then Result := P2; end; function minPoint(P1, P2: TPoint): TPoint; begin Result := P1; if (P2.y < P1.y) or ((P2.y = P1.y) and (P2.x < P1.x)) then Result := P2; 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; {***} // mh: Please don't change; no stack frame and efficient register use. function GetHasTabs(pLine: PChar; 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; {$IFDEF SYN_LAZARUS} function StringHasTabs(const Line: string; var CharsBefore: integer): boolean; var LineLen: integer; begin LineLen:=length(Line); CharsBefore := 1; while (CharsBefore<=LineLen) and (Line[CharsBefore]<>#9) do inc(CharsBefore); Result:=CharsBefore<=LineLen; dec(CharsBefore); end; {$ENDIF} {begin} //mh 2000-10-19 function ConvertTabs1Ex(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): AnsiString; var pDest: PChar; nBeforeTab: integer; begin Result := Line; // increment reference count only if GetHasTabs(pointer(Line), nBeforeTab) then begin HasTabs := TRUE; //mh 2000-11-08 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; {$IFDEF SYN_LAZARUS} function SimulateConvertTabs1Ex(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): integer; // TabWidth=1 var i: integer; begin Result:=length(Line); i:=1; while (i<=Result) and (Line[i]<>#9) do inc(i); HasTabs:=(i<=Result); end; {$ENDIF} function ConvertTabs1(const Line: AnsiString; TabWidth: integer): AnsiString; var HasTabs: boolean; begin Result := ConvertTabs1Ex(Line, TabWidth, HasTabs); end; function ConvertTabs2nEx(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): AnsiString; var i, DestLen, TabCount, TabMask: integer; pSrc, pDest: PChar; begin Result := Line; // increment reference count only if GetHasTabs(pointer(Line), DestLen) then begin HasTabs := TRUE; //mh 2000-11-08 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 := PChar(Line); pDest := PChar(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); repeat pDest^ := ' '; 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; {$IFDEF SYN_LAZARUS} function SimulateConvertTabs2nEx(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): integer; var LineLen, DestLen, SrcPos, TabMask: integer; begin LineLen:=length(Line); if StringHasTabs(Line, DestLen) then begin HasTabs := TRUE; SrcPos:=DestLen+1; // 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. TabMask := (TabWidth - 1) xor $7FFFFFFF; repeat if (Line[SrcPos] = #9) then DestLen := (DestLen + TabWidth) and TabMask else Inc(DestLen); Inc(SrcPos); until (SrcPos>LineLen); Result:=DestLen; end else begin Result := LineLen; HasTabs := FALSE; end; end; {$ENDIF} function ConvertTabs2n(const Line: AnsiString; TabWidth: integer): AnsiString; var HasTabs: boolean; begin Result := ConvertTabs2nEx(Line, TabWidth, HasTabs); end; function ConvertTabsEx(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): AnsiString; var i, DestLen, TabCount: integer; pSrc, pDest: PChar; begin Result := Line; // increment reference count only if GetHasTabs(pointer(Line), DestLen) then begin HasTabs := TRUE; //mh 2000-11-08 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 := PChar(Line); pDest := PChar(Result); repeat if (pSrc^ = #9) then begin i := TabWidth - (DestLen mod TabWidth); Inc(DestLen, i); repeat pDest^ := ' '; 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; {$IFDEF SYN_LAZARUS} function SimulateConvertTabsEx(const Line: AnsiString; TabWidth: integer; var HasTabs: boolean): integer; var LineLen, DestLen, SrcPos: integer; begin LineLen:=length(Line); if StringHasTabs(Line, DestLen) then begin HasTabs := TRUE; SrcPos := DestLen+1; // 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. repeat if (Line[SrcPos] = #9) then DestLen := DestLen + TabWidth - DestLen mod TabWidth else Inc(DestLen); Inc(SrcPos); until (SrcPos > LineLen); Result:=DestLen; end else begin Result:=LineLen; HasTabs := FALSE; end; end; {$ENDIF} function ConvertTabs(const Line: AnsiString; TabWidth: integer): AnsiString; 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 := TConvertTabsProcEx(@ConvertTabs1Ex) else if IsPowerOfTwo(TabWidth) then Result := TConvertTabsProcEx(@ConvertTabs2nEx) else Result := TConvertTabsProcEx(@ConvertTabsEx); end; {end} //mh 2000-10-19 {$IFDEF SYN_LAZARUS} function GetBestSimulateConvertTabsProcEx( TabWidth:integer): TSimulateConvertTabsProcEx; begin if (TabWidth < 2) then Result := TSimulateConvertTabsProcEx(@SimulateConvertTabs1Ex) else if IsPowerOfTwo(TabWidth) then Result := TSimulateConvertTabsProcEx(@SimulateConvertTabs2nEx) else Result := TSimulateConvertTabsProcEx(@SimulateConvertTabsEx); end; {$ENDIF} {***} function CharIndex2CaretPos(Index, TabWidth: integer; const Line: string): integer; var iChar: integer; pNext: PChar; 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: break; #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: string; var InsideTabChar: boolean): integer; var iPos: integer; pNext: PChar; 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 StrScanForCharInSet(const Line: string; Start: integer; AChars: TSynIdentChars): integer; var p: PChar; begin if (Start > 0) and (Start <= Length(Line)) then begin p := PChar(@Line[Start]); repeat if p^ in AChars then begin Result := Start; exit; end; Inc(p); Inc(Start); until p^ = #0; end; Result := 0; end; function StrRScanForCharInSet(const Line: string; Start: integer; AChars: TSynIdentChars): integer; var p: PChar; begin if (Start > 0) and (Start <= Length(Line)) then begin p := PChar(@Line[Start]); repeat if p^ in AChars then begin Result := Start; exit; end; Dec(p); Dec(Start); until Start < 1; end; Result := 0; end; function GetEOL(Line: PChar): PChar; begin Result := Line; if Assigned(Result) then while not (Result^ in [#0, #10, #13]) do Inc(Result); end; {begin} //gp 2000-06-24 {$IFOPT R+}{$DEFINE RestoreRangeChecking}{$ELSE}{$UNDEF RestoreRangeChecking}{$ENDIF} {$R-} function EncodeString(s: string): string; var i, j: integer; begin {$IFDEF SYN_LAZARUS} SetLength(Result, EncodeStringLength(s)); {$ELSE} SetLength(Result, 2 * Length(s)); // worst case {$ENDIF} j := 0; for i := 1 to Length(s) do begin Inc(j); if s[i] = '\' then begin Result[j] := '\'; {$IFDEF SYN_LAZARUS} Inc(j); Result[j] := '\'; {$ELSE} Result[j + 1] := '\'; Inc(j); {$ENDIF} end else if s[i] = '/' then begin Result[j] := '\'; {$IFDEF SYN_LAZARUS} Inc(j); Result[j] := '.'; {$ELSE} Result[j + 1] := '.'; Inc(j); {$ENDIF} end else Result[j] := s[i]; end; //for {$IFNDEF SYN_LAZARUS} SetLength(Result, j); {$ENDIF} end; { EncodeString } {$IFDEF SYN_LAZARUS} function EncodeStringLength(s: string): integer; var i, len: integer; begin len:=length(s); Result := len; for i := 1 to len do if (s[i] in ['\','/']) then Inc(Result); end; {$ENDIF} function DecodeString(s: string): string; 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} {end} //gp 2000-06-24 end.