{ Double commander ------------------------------------------------------------------------- This module contains classes with UTF8 file names support. Copyright (C) 2008-2011 Koblov Alexander (Alexx2000@mail.ru) This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. 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. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA } unit uClassesEx; {$mode objfpc}{$H+} interface uses Classes, RtlConsts, SysUtils, IniFiles, IniPropStorage; {$IF (FPC_VERSION = 2) and (FPC_RELEASE < 5)} const { TFileStream create mode } fmCreate = $FF00; {$ENDIF} type { TFileStreamEx class } TFileStreamEx = class(THandleStream) private FHandle: THandle; FFileName: UTF8String; public constructor Create(const AFileName: UTF8String; Mode: Word); destructor Destroy; override; {$IF (FPC_VERSION <= 2) and (FPC_RELEASE <= 4) and (FPC_PATCH <= 0)} function ReadQWord: QWord; procedure WriteQWord(q: QWord); {$ENDIF} property FileName: UTF8String read FFileName; end; { TStringListEx } TStringListEx = class(TStringList) public function IndexOfValue(const Value: String): Integer; procedure LoadFromFile(const FileName: String); override; procedure SaveToFile(const FileName: String); override; end; { TIniFileEx } THackIniFile = class private FFileName: UTF8String; end; TIniFileEx = class(TIniFile) private FIniFileStream: TFileStreamEx; function GetFileName: UTF8String; procedure SetFileName(const AValue: UTF8String); public constructor Create(const AFileName: String; Mode: Word); virtual; constructor Create(const AFileName: string; AEscapeLineFeeds : Boolean = False); override; destructor Destroy; override; procedure UpdateFile; override; property FileName: UTF8String read GetFileName write SetFileName; end; { TIniPropStorageEx } TIniPropStorageEx = class(TCustomIniPropStorage) private FPercentSize: Integer; function ChangeIdent(const Ident: String): String; protected function IniFileClass: TIniFileClass; override; public constructor Create(AOwner: TComponent); override; procedure Restore; override; function DoReadString(const Section, Ident, default: string): string; override; procedure DoWriteString(const Section, Ident, Value: string); override; property PercentSize: Integer read FPercentSize write FPercentSize; end; implementation uses Forms, uOSUtils, uDCUtils; { TFileStreamEx } constructor TFileStreamEx.Create(const AFileName: UTF8String; Mode: Word); begin if (Mode and fmCreate) <> 0 then begin FHandle:= mbFileCreate(AFileName, Mode); if FHandle = feInvalidHandle then raise EFCreateError.CreateFmt(SFCreateError, [AFileName]) else inherited Create(FHandle); end else begin FHandle:= mbFileOpen(AFileName, Mode); if FHandle = feInvalidHandle then raise EFOpenError.CreateFmt(SFOpenError, [AFilename]) else inherited Create(FHandle); end; FFileName:= AFileName; end; destructor TFileStreamEx.Destroy; begin inherited Destroy; // Close handle after destroying the base object, because it may use Handle in Destroy. if FHandle >= 0 then FileClose(FHandle); end; {$IF (FPC_VERSION <= 2) and (FPC_RELEASE <= 4) and (FPC_PATCH <= 0)} function TFileStreamEx.ReadQWord: QWord; var q: QWord; begin ReadBuffer(q, SizeOf(QWord)); ReadQWord:= q; end; procedure TFileStreamEx.WriteQWord(q: QWord); begin WriteBuffer(q, SizeOf(QWord)); end; {$ENDIF} { TStringListEx } function TStringListEx.IndexOfValue(const Value: String): Integer; var iStart: LongInt; sTemp: String; begin CheckSpecialChars; Result:= 0; while (Result < Count) do begin sTemp:= Strings[Result]; iStart:= Pos(NameValueSeparator, sTemp) + 1; if (iStart > 0) and (DoCompareText(Value, Copy(sTemp, iStart, MaxInt)) = 0) then Exit; Inc(result); end; Result:= -1; end; procedure TStringListEx.LoadFromFile(const FileName: String); var fsFileStream: TFileStreamEx; begin fsFileStream:= TFileStreamEx.Create(FileName, fmOpenRead or fmShareDenyNone); LoadFromStream(fsFileStream); fsFileStream.Free; end; procedure TStringListEx.SaveToFile(const FileName: String); var fsFileStream: TFileStreamEx; begin if mbFileExists(FileName) then begin fsFileStream:= TFileStreamEx.Create(FileName, fmOpenWrite or fmShareDenyNone); fsFileStream.Position:= 0; fsFileStream.Size:= 0; end else fsFileStream:= TFileStreamEx.Create(FileName, fmCreate); SaveToStream(fsFileStream); fsFileStream.Free; end; { TIniFileEx } function TIniFileEx.GetFileName: UTF8String; begin Result:= THackIniFile(Self).FFileName; end; procedure TIniFileEx.SetFileName(const AValue: UTF8String); begin THackIniFile(Self).FFileName:= AValue; end; constructor TIniFileEx.Create(const AFileName: String; Mode: Word); begin if mbFileExists(AFileName) then FIniFileStream:= TFileStreamEx.Create(AFileName, Mode or fmShareDenyNone) else FIniFileStream:= TFileStreamEx.Create(AFileName, fmCreate); inherited Create(FIniFileStream); FileName:= AFileName; end; constructor TIniFileEx.Create(const AFileName: string; AEscapeLineFeeds: Boolean); begin {$PUSH}{$HINTS OFF} Create(AFileName, fmOpenReadWrite); {$POP} end; procedure TIniFileEx.UpdateFile; begin Stream.Position:=0; Stream.Size:= 0; FileName:= EmptyStr; inherited UpdateFile; FileName:= FIniFileStream.FileName; end; destructor TIniFileEx.Destroy; begin inherited Destroy; // Destroy stream after destroying the base object, because it may use the stream in Destroy. FreeAndNil(FIniFileStream); end; { TIniPropStorageEx } function TIniPropStorageEx.IniFileClass: TIniFileClass; begin Result:= TIniFileEx; end; constructor TIniPropStorageEx.Create(AOwner: TComponent); begin inherited Create(AOwner); FPercentSize:= 5; end; procedure TIniPropStorageEx.Restore; var mLeft, mTop, // monitor left and top mWidth, mHeight, // monitor width and height pWidth, pHeight: Integer; begin inherited Restore; if Self.Owner is TCustomForm then with Self.Owner as TCustomForm do begin mLeft:= Monitor.Left; mTop:= Monitor.Top; mWidth:= Monitor.Width; mHeight:= Monitor.Height; pWidth:= (mWidth * FPercentSize) div 100; pHeight:= (mHeight * FPercentSize) div 100; if (mWidth < Width) or (mHeight < Height) then begin Width:= mWidth - pWidth; Height:= mHeight - (pHeight * 2); end; if (Top > (mTop + mHeight - pHeight)) or (Top < mTop) then Top:= mTop + pHeight; if (Left > (mLeft + mWidth - pWidth)) or ((Left + Width - pWidth) < mLeft) then Left:= mLeft + pWidth; end; end; function TIniPropStorageEx.DoReadString(const Section, Ident, default: string): string; begin Result := inherited DoReadString(Section, ChangeIdent(Ident), default); end; procedure TIniPropStorageEx.DoWriteString(const Section, Ident, Value: string); begin inherited DoWriteString(Section, ChangeIdent(Ident), Value); end; function TIniPropStorageEx.ChangeIdent(const Ident: String): String; begin // Change component name to class name. if StrBegins(Ident, Owner.Name) then Result := Owner.ClassName + Copy(Ident, 1 + Length(Owner.Name), MaxInt) else Result := Ident; end; end.