2018-04-18 10:25:38 +02:00
|
|
|
// ************************************************************************
|
|
|
|
// ***************************** CEF4Delphi *******************************
|
|
|
|
// ************************************************************************
|
|
|
|
//
|
2019-10-19 10:58:34 +02:00
|
|
|
// CEF4Delphi is based on DCEF3 which uses CEF to embed a chromium-based
|
2018-04-18 10:25:38 +02:00
|
|
|
// browser in Delphi applications.
|
|
|
|
//
|
|
|
|
// The original license of DCEF3 still applies to CEF4Delphi.
|
|
|
|
//
|
|
|
|
// For more information about CEF4Delphi visit :
|
|
|
|
// https://www.briskbard.com/index.php?lang=en&pageid=cef
|
|
|
|
//
|
2021-01-01 11:11:30 +01:00
|
|
|
// Copyright � 2021 Salvador Diaz Fau. All rights reserved.
|
2018-04-18 10:25:38 +02:00
|
|
|
//
|
|
|
|
// ************************************************************************
|
|
|
|
// ************ vvvv Original license and comments below vvvv *************
|
|
|
|
// ************************************************************************
|
|
|
|
(*
|
|
|
|
* Delphi Chromium Embedded 3
|
|
|
|
*
|
|
|
|
* Usage allowed under the restrictions of the Lesser GNU General Public License
|
|
|
|
* or alternatively the restrictions of the Mozilla Public License 1.1
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Unit owner : Henri Gourvest <hgourvest@gmail.com>
|
|
|
|
* Web site : http://www.progdigy.com
|
|
|
|
* Repository : http://code.google.com/p/delphichromiumembedded/
|
|
|
|
* Group : http://groups.google.com/group/delphichromiumembedded
|
|
|
|
*
|
|
|
|
* Embarcadero Technologies, Inc is not permitted to use or redistribute
|
|
|
|
* this source code without explicit permission.
|
|
|
|
*
|
|
|
|
*)
|
|
|
|
|
|
|
|
unit uCEFMiscFunctions;
|
|
|
|
|
2018-05-12 14:50:54 +02:00
|
|
|
{$IFDEF FPC}
|
|
|
|
{$MODE OBJFPC}{$H+}
|
|
|
|
{$ENDIF}
|
|
|
|
|
2018-11-22 15:02:21 +01:00
|
|
|
{$IFNDEF CPUX64}{$ALIGN ON}{$ENDIF}
|
|
|
|
{$MINENUMSIZE 4}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
{$I cef.inc}
|
|
|
|
|
2021-01-05 11:44:41 +01:00
|
|
|
{$IFNDEF FPC}{$IFNDEF DELPHI12_UP}
|
|
|
|
// Workaround for "Internal error" in old Delphi versions caused by uint64 handling
|
|
|
|
{$R-}
|
|
|
|
{$ENDIF}{$ENDIF}
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
interface
|
|
|
|
|
|
|
|
uses
|
|
|
|
{$IFDEF DELPHI16_UP}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
WinApi.Windows, WinApi.ActiveX, {$IFDEF FMX}FMX.Types,{$ENDIF}
|
|
|
|
{$ELSE}
|
2020-04-03 17:57:52 +02:00
|
|
|
{$IFDEF FMX}FMX.Types,{$ENDIF} {$IFDEF MACOS}Macapi.Foundation, FMX.Helpers.Mac,{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ENDIF}
|
2020-04-03 17:57:52 +02:00
|
|
|
System.Types, System.IOUtils, System.Classes, System.SysUtils, System.UITypes, System.Math,
|
2018-04-18 10:25:38 +02:00
|
|
|
{$ELSE}
|
2020-04-03 17:57:52 +02:00
|
|
|
{$IFDEF MSWINDOWS}Windows, ActiveX,{$ENDIF}
|
|
|
|
{$IFDEF DELPHI14_UP}Types, IOUtils,{$ENDIF} Classes, SysUtils, Math,
|
2019-11-10 18:23:39 +01:00
|
|
|
{$IFDEF FPC}LCLType,{$IFNDEF MSWINDOWS}InterfaceBase, Forms,{$ENDIF}{$ENDIF}
|
2020-12-31 12:15:10 +01:00
|
|
|
{$IFDEF LINUX}{$IFDEF FPC}
|
|
|
|
ctypes, keysym, xf86keysym, x, xlib,
|
|
|
|
{$IFDEF LCLGTK2}gtk2, glib2, gdk2, gtk2proc, gtk2int, Gtk2Def, gdk2x, Gtk2Extra,{$ENDIF}
|
|
|
|
{$ENDIF}{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
{$ENDIF}
|
|
|
|
uCEFTypes, uCEFInterfaces, uCEFLibFunctions, uCEFResourceHandler,
|
2018-10-12 12:21:43 +02:00
|
|
|
uCEFRegisterCDMCallback, uCEFConstants;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
const
|
|
|
|
Kernel32DLL = 'kernel32.dll';
|
|
|
|
SHLWAPIDLL = 'shlwapi.dll';
|
2021-01-08 16:01:06 +01:00
|
|
|
NTDLL = 'ntdll.dll';
|
|
|
|
User32DLL = 'User32.dll';
|
2020-01-13 15:39:44 +01:00
|
|
|
|
|
|
|
type
|
|
|
|
TOSVersionInfoEx = record
|
|
|
|
dwOSVersionInfoSize: DWORD;
|
|
|
|
dwMajorVersion: DWORD;
|
|
|
|
dwMinorVersion: DWORD;
|
|
|
|
dwBuildNumber: DWORD;
|
|
|
|
dwPlatformId: DWORD;
|
|
|
|
szCSDVersion: array[0..127] of WideChar;
|
|
|
|
wServicePackMajor: WORD;
|
|
|
|
wServicePackMinor: WORD;
|
|
|
|
wSuiteMask: WORD;
|
|
|
|
wProductType: BYTE;
|
|
|
|
wReserved:BYTE;
|
|
|
|
end;
|
2020-02-05 14:40:22 +01:00
|
|
|
{$IFDEF DELPHI14_UP}
|
|
|
|
TDigitizerStatus = record
|
|
|
|
IntegratedTouch : boolean;
|
|
|
|
ExternalTouch : boolean;
|
|
|
|
IntegratedPen : boolean;
|
|
|
|
ExternalPen : boolean;
|
|
|
|
MultiInput : boolean;
|
|
|
|
Ready : boolean;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function CefColorGetA(color: TCefColor): Byte;
|
|
|
|
function CefColorGetR(color: TCefColor): byte;
|
|
|
|
function CefColorGetG(color: TCefColor): Byte;
|
|
|
|
function CefColorGetB(color: TCefColor): Byte;
|
|
|
|
|
|
|
|
function CefColorSetARGB(a, r, g, b: Byte): TCefColor;
|
|
|
|
|
|
|
|
function CefInt64Set(int32_low, int32_high: Integer): Int64;
|
|
|
|
|
|
|
|
function CefInt64GetLow(const int64_val: Int64): Integer;
|
|
|
|
function CefInt64GetHigh(const int64_val: Int64): Integer;
|
|
|
|
|
2021-01-18 11:12:01 +01:00
|
|
|
function CefGetObject(ptr: Pointer): TObject; {$IFDEF SUPPORTS_INLINE}inline;{$ENDIF}
|
|
|
|
function CefGetData(const i: ICefBaseRefCounted): Pointer; {$IFDEF SUPPORTS_INLINE}inline;{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function CefStringAlloc(const str: ustring): TCefString;
|
2019-11-24 18:19:49 +01:00
|
|
|
function CefStringClearAndGet(str: PCefString): ustring;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function CefString(const str: ustring): TCefString; overload;
|
|
|
|
function CefString(const str: PCefString): ustring; overload;
|
|
|
|
function CefUserFreeString(const str: ustring): PCefStringUserFree;
|
|
|
|
procedure CefStringFree(const str: PCefString);
|
|
|
|
function CefStringFreeAndGet(const str: PCefStringUserFree): ustring;
|
|
|
|
procedure CefStringSet(const str: PCefString; const value: ustring);
|
2021-01-18 11:12:01 +01:00
|
|
|
procedure CefStringInitialize(const aCefString : PCefString); {$IFDEF SUPPORTS_INLINE}inline;{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2019-10-19 10:58:34 +02:00
|
|
|
function CefRegisterExtension(const name, code: ustring; const Handler: ICefv8Handler): Boolean;
|
|
|
|
|
|
|
|
function CefPostTask(aThreadId : TCefThreadId; const aTask: ICefTask) : boolean;
|
|
|
|
function CefPostDelayedTask(aThreadId : TCefThreadId; const aTask : ICefTask; aDelayMs : Int64) : boolean;
|
|
|
|
function CefCurrentlyOn(aThreadId : TCefThreadId) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2020-12-18 16:51:02 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefTimeToSystemTime(const dt: TCefTime): TSystemTime;
|
|
|
|
function SystemTimeToCefTime(const dt: TSystemTime): TCefTime;
|
2020-12-18 16:51:02 +01:00
|
|
|
{$ELSE}
|
|
|
|
{$IFDEF LINUX}
|
|
|
|
{$IFDEF FPC}
|
|
|
|
function CefTimeToSystemTime(const dt: TCefTime): TSystemTime;
|
|
|
|
function SystemTimeToCefTime(const dt: TSystemTime): TCefTime;
|
|
|
|
{$ENDIF}
|
|
|
|
{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ENDIF}
|
|
|
|
|
2021-01-18 11:12:01 +01:00
|
|
|
function FixCefTime(const dt : TCefTime): TCefTime;
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefTimeToDateTime(const dt: TCefTime): TDateTime;
|
|
|
|
function DateTimeToCefTime(dt: TDateTime): TCefTime;
|
|
|
|
|
|
|
|
function cef_string_wide_copy(const src: PWideChar; src_len: NativeUInt; output: PCefStringWide): Integer;
|
|
|
|
function cef_string_utf8_copy(const src: PAnsiChar; src_len: NativeUInt; output: PCefStringUtf8): Integer;
|
|
|
|
function cef_string_utf16_copy(const src: PChar16; src_len: NativeUInt; output: PCefStringUtf16): Integer;
|
|
|
|
function cef_string_copy(const src: PCefChar; src_len: NativeUInt; output: PCefString): Integer;
|
|
|
|
|
2018-05-24 19:15:41 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-09-10 11:21:39 +02:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring = ''; aExStyle : DWORD = 0);
|
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = ''; aExStyle : DWORD = 0);
|
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = ''; aExStyle : DWORD = 0);
|
2018-05-24 19:15:41 +02:00
|
|
|
{$ENDIF}
|
|
|
|
|
|
|
|
{$IFDEF MACOS}
|
2020-01-28 11:36:34 +01:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring = ''; aHidden : boolean = False);
|
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = ''; aHidden : boolean = False);
|
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = ''; aHidden : boolean = False);
|
2018-05-24 19:15:41 +02:00
|
|
|
{$ENDIF}
|
|
|
|
|
|
|
|
{$IFDEF LINUX}
|
2019-03-15 17:17:14 +01:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring = '');
|
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = '');
|
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = '');
|
2018-05-24 19:15:41 +02:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2018-06-03 17:18:54 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-01-18 15:15:41 +01:00
|
|
|
function ProcessUnderWow64(hProcess: THandle; var Wow64Process: BOOL): BOOL; stdcall; external Kernel32DLL name 'IsWow64Process';
|
2018-04-18 10:25:38 +02:00
|
|
|
function PathIsRelativeAnsi(pszPath: LPCSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsRelativeA';
|
|
|
|
function PathIsRelativeUnicode(pszPath: LPCWSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsRelativeW';
|
2018-10-25 12:50:01 +02:00
|
|
|
function GetGlobalMemoryStatusEx(var Buffer: TMyMemoryStatusEx): BOOL; stdcall; external Kernel32DLL name 'GlobalMemoryStatusEx';
|
2019-09-21 11:37:13 +02:00
|
|
|
function PathCanonicalizeAnsi(pszBuf: LPSTR; pszPath: LPCSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathCanonicalizeA';
|
|
|
|
function PathCanonicalizeUnicode(pszBuf: LPWSTR; pszPath: LPCWSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathCanonicalizeW';
|
2019-09-23 12:01:39 +02:00
|
|
|
function PathIsUNCAnsi(pszPath: LPCSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsUNCA';
|
2019-09-23 11:42:20 +02:00
|
|
|
function PathIsUNCUnicode(pszPath: LPCWSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsUNCW';
|
2019-09-23 12:01:39 +02:00
|
|
|
function PathIsURLAnsi(pszPath: LPCSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsURLA';
|
2019-09-23 11:42:20 +02:00
|
|
|
function PathIsURLUnicode(pszPath: LPCWSTR): BOOL; stdcall; external SHLWAPIDLL name 'PathIsURLW';
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
{$IFNDEF DELPHI12_UP}
|
2019-11-10 18:23:39 +01:00
|
|
|
const
|
|
|
|
GWLP_WNDPROC = GWL_WNDPROC;
|
2019-11-11 21:21:14 +01:00
|
|
|
GWLP_HWNDPARENT = GWL_HWNDPARENT;
|
2018-06-03 17:18:54 +02:00
|
|
|
{$IFDEF WIN64}
|
|
|
|
function SetWindowLongPtr(hWnd: HWND; nIndex: Integer; dwNewLong: int64): int64; stdcall; external user32 name 'SetWindowLongPtrW';
|
|
|
|
{$ELSE}
|
|
|
|
function SetWindowLongPtr(hWnd: HWND; nIndex: Integer; dwNewLong: LongInt): LongInt; stdcall; external user32 name 'SetWindowLongW';
|
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
{$ENDIF}
|
2018-06-03 17:18:54 +02:00
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
{$ENDIF}
|
|
|
|
|
|
|
|
function CustomPathIsRelative(const aPath : string) : boolean;
|
2019-09-21 11:37:13 +02:00
|
|
|
function CustomPathCanonicalize(const aOriginalPath : string; var aCanonicalPath : string) : boolean;
|
|
|
|
function CustomAbsolutePath(const aPath : string; aMustExist : boolean = False) : string;
|
2019-09-23 11:42:20 +02:00
|
|
|
function CustomPathIsURL(const aPath : string) : boolean;
|
|
|
|
function CustomPathIsUNC(const aPath : string) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
function GetModulePath : string;
|
|
|
|
|
|
|
|
function CefIsCertStatusError(Status : TCefCertStatus) : boolean;
|
|
|
|
|
|
|
|
function CefCrashReportingEnabled : boolean;
|
|
|
|
procedure CefSetCrashKeyValue(const aKey, aValue : ustring);
|
|
|
|
|
|
|
|
procedure CefLog(const aFile : string; aLine, aSeverity : integer; const aMessage : string);
|
2018-10-12 12:21:43 +02:00
|
|
|
procedure CefDebugLog(const aMessage : string; aSeverity : integer = CEF_LOG_SEVERITY_ERROR);
|
2020-04-23 11:11:53 +02:00
|
|
|
procedure CefKeyEventLog(const aEvent : TCefKeyEvent);
|
|
|
|
procedure CefMouseEventLog(const aEvent : TCefMouseEvent);
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure OutputDebugMessage(const aMessage : string);
|
|
|
|
function CustomExceptionHandler(const aFunctionName : string; const aException : exception) : boolean;
|
|
|
|
|
2018-06-11 18:38:51 +02:00
|
|
|
function CefRegisterSchemeHandlerFactory(const SchemeName, DomainName : ustring; const handler: TCefResourceHandlerClass = nil): Boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefClearSchemeHandlerFactories : boolean;
|
|
|
|
|
|
|
|
function CefAddCrossOriginWhitelistEntry(const SourceOrigin, TargetProtocol, TargetDomain: ustring; AllowTargetSubdomains: Boolean): Boolean;
|
|
|
|
function CefRemoveCrossOriginWhitelistEntry(const SourceOrigin, TargetProtocol, TargetDomain: ustring; AllowTargetSubdomains: Boolean): Boolean;
|
|
|
|
function CefClearCrossOriginWhitelist: Boolean;
|
|
|
|
|
|
|
|
procedure UInt64ToFileVersionInfo(const aVersion : uint64; var aVersionInfo : TFileVersionInfo);
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-06-13 17:24:22 +02:00
|
|
|
function GetExtendedFileVersion(const aFileName : ustring) : uint64;
|
2020-06-20 16:10:29 +02:00
|
|
|
function GetStringFileInfo(const aFileName, aField : ustring; var aValue : ustring) : boolean;
|
2020-06-13 17:24:22 +02:00
|
|
|
function GetDLLVersion(const aDLLFile : ustring; var aVersionInfo : TFileVersionInfo) : boolean;
|
|
|
|
procedure OutputLastErrorMessage;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function SplitLongString(aSrcString : string) : string;
|
|
|
|
function GetAbsoluteDirPath(const aSrcPath : string; var aRsltPath : string) : boolean;
|
2018-10-12 12:21:43 +02:00
|
|
|
function CheckSubprocessPath(const aSubprocessPath : string; var aMissingFiles : string) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
function CheckLocales(const aLocalesDirPath : string; var aMissingFiles : string; const aLocalesRequired : string = '') : boolean;
|
2018-10-03 10:58:51 +03:00
|
|
|
function CheckResources(const aResourcesDirPath : string; var aMissingFiles : string; aCheckDevResources: boolean = True; aCheckExtensions: boolean = True) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
function CheckDLLs(const aFrameworkDirPath : string; var aMissingFiles : string) : boolean;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-06-13 17:24:22 +02:00
|
|
|
function CheckDLLVersion(const aDLLFile : ustring; aMajor, aMinor, aRelease, aBuild : uint16) : boolean;
|
|
|
|
function GetDLLHeaderMachine(const aDLLFile : ustring; var aMachine : integer) : boolean;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
function FileVersionInfoToString(const aVersionInfo : TFileVersionInfo) : string;
|
|
|
|
function CheckFilesExist(var aList : TStringList; var aMissingFiles : string) : boolean;
|
2018-09-02 13:11:43 +02:00
|
|
|
function Is32BitProcess : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function CefParseUrl(const url: ustring; var parts: TUrlParts): Boolean;
|
|
|
|
function CefCreateUrl(var parts: TUrlParts): ustring;
|
|
|
|
function CefFormatUrlForSecurityDisplay(const originUrl: string): string;
|
|
|
|
function CefGetMimeType(const extension: ustring): ustring;
|
|
|
|
procedure CefGetExtensionsForMimeType(const mimeType: ustring; var extensions: TStringList);
|
|
|
|
|
|
|
|
function CefBase64Encode(const data: Pointer; dataSize: NativeUInt): ustring;
|
|
|
|
function CefBase64Decode(const data: ustring): ICefBinaryValue;
|
|
|
|
function CefUriEncode(const text: ustring; usePlus: Boolean): ustring;
|
|
|
|
function CefUriDecode(const text: ustring; convertToUtf8: Boolean; unescapeRule: TCefUriUnescapeRule): ustring;
|
|
|
|
|
2019-11-24 18:19:49 +01:00
|
|
|
function CefGetPath(const aPathKey : TCefPathKey) : ustring;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefCreateDirectory(const fullPath: ustring): Boolean;
|
|
|
|
function CefGetTempDirectory(out tempDir: ustring): Boolean;
|
|
|
|
function CefCreateNewTempDirectory(const prefix: ustring; out newTempPath: ustring): Boolean;
|
|
|
|
function CefCreateTempDirectoryInDirectory(const baseDir, prefix: ustring; out newDir: ustring): Boolean;
|
|
|
|
function CefDirectoryExists(const path: ustring): Boolean;
|
|
|
|
function CefDeleteFile(const path: ustring; recursive: Boolean): Boolean;
|
|
|
|
function CefZipDirectory(const srcDir, destFile: ustring; includeHiddenFiles: Boolean): Boolean;
|
|
|
|
procedure CefLoadCRLSetsFile(const path : ustring);
|
|
|
|
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-04-10 09:49:34 +02:00
|
|
|
function CefIsKeyDown(aWparam : WPARAM) : boolean;
|
|
|
|
function CefIsKeyToggled(aWparam : WPARAM) : boolean;
|
|
|
|
function GetCefMouseModifiers : TCefEventFlags; overload;
|
|
|
|
function GetCefMouseModifiers(awparam : WPARAM) : TCefEventFlags; overload;
|
|
|
|
function GetCefKeyboardModifiers(aWparam : WPARAM; aLparam : LPARAM) : TCefEventFlags;
|
|
|
|
procedure CefCheckAltGrPressed(aWparam : WPARAM; var aEvent : TCefKeyEvent);
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
procedure DropEffectToDragOperation(aEffect : Longint; var aAllowedOps : TCefDragOperations);
|
|
|
|
procedure DragOperationToDropEffect(const aDragOperations : TCefDragOperations; var aEffect: Longint);
|
2020-01-13 15:39:44 +01:00
|
|
|
|
|
|
|
function GetWindowsMajorMinorVersion(var wMajorVersion, wMinorVersion : DWORD) : boolean;
|
2020-10-31 14:23:06 +01:00
|
|
|
function RunningWindows10OrNewer : boolean;
|
|
|
|
function GetDPIForHandle(aHandle : HWND; var aDPI : UINT) : boolean;
|
2020-01-13 15:39:44 +01:00
|
|
|
function GetDefaultCEFUserAgent : string;
|
2020-02-04 11:50:38 +01:00
|
|
|
{$IFDEF DELPHI14_UP}
|
2020-02-03 12:02:30 +01:00
|
|
|
function TouchPointToPoint(aHandle : HWND; const TouchPoint: TTouchInput): TPoint;
|
2020-02-05 14:40:22 +01:00
|
|
|
function GetDigitizerStatus(var aDigitizerStatus : TDigitizerStatus; aDPI : cardinal = 0) : boolean;
|
|
|
|
function HasTouchOrPen(aDPI : cardinal = 0) : boolean;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2020-02-03 16:14:31 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function DeviceToLogical(aValue : integer; const aDeviceScaleFactor : double) : integer; overload;
|
2020-02-03 12:02:30 +01:00
|
|
|
function DeviceToLogical(aValue : single; const aDeviceScaleFactor : double) : single; overload;
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure DeviceToLogical(var aEvent : TCEFMouseEvent; const aDeviceScaleFactor : double); overload;
|
2020-02-03 12:02:30 +01:00
|
|
|
procedure DeviceToLogical(var aEvent : TCefTouchEvent; const aDeviceScaleFactor : double); overload;
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure DeviceToLogical(var aPoint : TPoint; const aDeviceScaleFactor : double); overload;
|
|
|
|
function LogicalToDevice(aValue : integer; const aDeviceScaleFactor : double) : integer; overload;
|
|
|
|
procedure LogicalToDevice(var aRect : TCEFRect; const aDeviceScaleFactor : double); overload;
|
|
|
|
|
|
|
|
function GetScreenDPI : integer;
|
|
|
|
function GetDeviceScaleFactor : single;
|
|
|
|
|
2019-06-19 16:53:26 +02:00
|
|
|
function DeleteDirContents(const aDirectory : string; const aExcludeFiles : TStringList = nil) : boolean;
|
|
|
|
function DeleteFileList(const aFileList : TStringList) : boolean;
|
|
|
|
function MoveFileList(const aFileList : TStringList; const aSrcDirectory, aDstDirectory : string) : boolean;
|
2018-09-18 15:19:44 +02:00
|
|
|
|
2019-10-13 18:50:23 +02:00
|
|
|
function CefGetDataURI(const aString, aMimeType : ustring) : ustring; overload;
|
|
|
|
function CefGetDataURI(aData : pointer; aSize : integer; const aMimeType : ustring; const aCharset : ustring = '') : ustring; overload;
|
|
|
|
|
2020-04-03 17:57:52 +02:00
|
|
|
function ValidCefWindowHandle(aHandle : TCefWindowHandle) : boolean;
|
2020-05-05 18:10:33 +02:00
|
|
|
procedure InitializeWindowHandle(var aHandle : TCefWindowHandle);
|
2020-01-02 20:02:47 +01:00
|
|
|
|
2020-12-28 18:11:27 +01:00
|
|
|
{$IFDEF LINUX}{$IFDEF FPC}{$IFDEF LCLGTK2}
|
|
|
|
procedure GdkEventKeyToCEFKeyEvent(GdkEvent: PGdkEventKey; var aCEFKeyEvent : TCEFKeyEvent);
|
|
|
|
function KeyboardCodeFromXKeysym(keysym : cuint) : integer;
|
|
|
|
function GetCefStateModifiers(state : cuint) : integer;
|
|
|
|
function GdkEventToWindowsKeyCode(Event: PGdkEventKey) : integer;
|
|
|
|
function GetWindowsKeyCodeWithoutLocation(key_code : integer) : integer;
|
|
|
|
function GetControlCharacter(windows_key_code : integer; shift : boolean) : integer;
|
2020-12-31 12:15:10 +01:00
|
|
|
{$ENDIF}
|
|
|
|
procedure ShowX11Message(const aMessage : string);
|
|
|
|
{$ENDIF}{$ENDIF}
|
2020-12-28 18:11:27 +01:00
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
implementation
|
|
|
|
|
|
|
|
uses
|
2021-01-14 16:03:04 +01:00
|
|
|
{$IFDEF LINUX}{$IFDEF FMX}Posix.Unistd, Posix.Stdio,{$ENDIF}{$ENDIF}
|
2019-11-08 23:15:53 +01:00
|
|
|
uCEFApplicationCore, uCEFSchemeHandlerFactory, uCEFValue,
|
2018-04-18 10:25:38 +02:00
|
|
|
uCEFBinaryValue, uCEFStringList;
|
|
|
|
|
|
|
|
function CefColorGetA(color: TCefColor): Byte;
|
|
|
|
begin
|
|
|
|
Result := (color shr 24) and $FF;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefColorGetR(color: TCefColor): byte;
|
|
|
|
begin
|
|
|
|
Result := (color shr 16) and $FF;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefColorGetG(color: TCefColor): Byte;
|
|
|
|
begin
|
|
|
|
Result := (color shr 8) and $FF;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefColorGetB(color: TCefColor): Byte;
|
|
|
|
begin
|
|
|
|
Result := color and $FF;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefColorSetARGB(a, r, g, b: Byte): TCefColor;
|
|
|
|
begin
|
|
|
|
Result := (a shl 24) or (r shl 16) or (g shl 8) or b;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefInt64Set(int32_low, int32_high: Integer): Int64;
|
|
|
|
begin
|
|
|
|
Result := int32_low or (int32_high shl 32);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefInt64GetLow(const int64_val: Int64): Integer;
|
|
|
|
begin
|
|
|
|
Result := Integer(int64_val);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefInt64GetHigh(const int64_val: Int64): Integer;
|
|
|
|
begin
|
|
|
|
Result := (int64_val shr 32) and $FFFFFFFF;
|
|
|
|
end;
|
|
|
|
|
2019-11-24 18:19:49 +01:00
|
|
|
function CefStringClearAndGet(str: PCefString): ustring;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
if (str <> nil) and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
2018-06-11 09:01:38 +02:00
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
Result := CefString(str);
|
|
|
|
cef_string_utf16_clear(str);
|
2018-06-11 09:01:38 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefGetObject(ptr: Pointer): TObject; {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
|
|
|
|
begin
|
|
|
|
if (ptr <> nil) then
|
|
|
|
begin
|
|
|
|
Dec(PByte(ptr), SizeOf(Pointer));
|
|
|
|
Result := TObject(PPointer(ptr)^);
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefGetData(const i: ICefBaseRefCounted): Pointer; {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
|
|
|
|
begin
|
|
|
|
if (i <> nil) then
|
|
|
|
Result := i.Wrap
|
|
|
|
else
|
|
|
|
Result := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefString(const str: PCefString): ustring;
|
|
|
|
begin
|
2020-07-14 15:50:23 +02:00
|
|
|
if (str <> nil) and (str^.str <> nil) and (str^.length > 0) and (str^.length < nativeuint(high(integer))) then
|
2018-05-12 14:50:54 +02:00
|
|
|
SetString(Result, str^.str, str^.length)
|
2018-04-18 10:25:38 +02:00
|
|
|
else
|
|
|
|
Result := '';
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefString(const str: ustring): TCefString;
|
|
|
|
begin
|
|
|
|
Result.str := PChar16(PWideChar(str));
|
|
|
|
Result.length := Length(str);
|
|
|
|
Result.dtor := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefStringFree(const str: PCefString);
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (str <> nil) and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
cef_string_utf16_clear(str);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefStringSet(const str: PCefString; const value: ustring);
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (str <> nil) and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
2019-11-24 18:19:49 +01:00
|
|
|
cef_string_utf16_set(PWideChar(value), Length(value), str, Ord(True));
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefStringInitialize(const aCefString : PCefString); {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
|
|
|
|
begin
|
|
|
|
if (aCefString <> nil) then
|
|
|
|
begin
|
|
|
|
aCefString^.str := nil;
|
|
|
|
aCefString^.length := 0;
|
|
|
|
aCefString^.dtor := nil;
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefStringFreeAndGet(const str: PCefStringUserFree): ustring;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (str <> nil) and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
|
|
|
Result := CefString(PCefString(str));
|
|
|
|
cef_string_userfree_utf16_free(str);
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefStringAlloc(const str: ustring): TCefString;
|
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
CefStringInitialize(@Result);
|
|
|
|
|
2018-06-11 09:01:38 +02:00
|
|
|
if (str <> '') and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
cef_string_wide_to_utf16(PWideChar(str), Length(str), @Result);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2020-06-21 21:27:55 +02:00
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure _free_string(str: PChar16); stdcall;
|
|
|
|
begin
|
|
|
|
if (str <> nil) then FreeMem(str);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefUserFreeString(const str: ustring): PCefStringUserFree;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
Result := cef_string_userfree_utf16_alloc();
|
|
|
|
Result^.length := Length(str);
|
|
|
|
GetMem(Result^.str, Result^.length * SizeOf(TCefChar));
|
|
|
|
Move(PCefChar(str)^, Result^.str^, Result^.length * SizeOf(TCefChar));
|
|
|
|
Result^.dtor := @_free_string;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := nil;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefRegisterExtension(const name, code: ustring; const Handler: ICefv8Handler): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempName, TempCode : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2019-09-12 11:40:52 +02:00
|
|
|
if (GlobalCEFApp <> nil) and
|
2018-07-24 18:18:54 +02:00
|
|
|
GlobalCEFApp.LibLoaded and
|
2020-01-05 15:15:48 +01:00
|
|
|
((GlobalCEFApp.ProcessType = ptRenderer) or GlobalCEFApp.SingleProcess) and
|
2019-09-12 11:40:52 +02:00
|
|
|
(length(name) > 0) and
|
|
|
|
(length(code) > 0) then
|
2018-06-11 09:01:38 +02:00
|
|
|
begin
|
|
|
|
TempName := CefString(name);
|
|
|
|
TempCode := CefString(code);
|
|
|
|
Result := cef_register_extension(@TempName, @TempCode, CefGetData(handler)) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2019-10-19 10:58:34 +02:00
|
|
|
function CefPostTask(aThreadId : TCefThreadId; const aTask : ICefTask) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2019-10-19 10:58:34 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded and (aTask <> nil) then
|
|
|
|
Result := cef_post_task(aThreadId, aTask.Wrap) <> 0
|
|
|
|
else
|
|
|
|
Result := False;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefPostDelayedTask(aThreadId : TCefThreadId; const aTask : ICefTask; aDelayMs : Int64) : boolean;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded and (aTask <> nil) then
|
|
|
|
Result := cef_post_delayed_task(aThreadId, aTask.Wrap, aDelayMs) <> 0
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2019-10-19 10:58:34 +02:00
|
|
|
function CefCurrentlyOn(aThreadId : TCefThreadId) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
2019-10-19 10:58:34 +02:00
|
|
|
Result := cef_currently_on(aThreadId) <> 0
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2020-12-18 16:51:02 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefTimeToSystemTime(const dt: TCefTime): TSystemTime;
|
|
|
|
begin
|
|
|
|
Result.wYear := dt.year;
|
|
|
|
Result.wMonth := dt.month;
|
|
|
|
Result.wDayOfWeek := dt.day_of_week;
|
|
|
|
Result.wDay := dt.day_of_month;
|
|
|
|
Result.wHour := dt.hour;
|
|
|
|
Result.wMinute := dt.minute;
|
|
|
|
Result.wSecond := dt.second;
|
|
|
|
Result.wMilliseconds := dt.millisecond;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function SystemTimeToCefTime(const dt: TSystemTime): TCefTime;
|
|
|
|
begin
|
|
|
|
Result.year := dt.wYear;
|
|
|
|
Result.month := dt.wMonth;
|
|
|
|
Result.day_of_week := dt.wDayOfWeek;
|
|
|
|
Result.day_of_month := dt.wDay;
|
|
|
|
Result.hour := dt.wHour;
|
|
|
|
Result.minute := dt.wMinute;
|
|
|
|
Result.second := dt.wSecond;
|
|
|
|
Result.millisecond := dt.wMilliseconds;
|
2020-12-18 16:51:02 +01:00
|
|
|
end;
|
|
|
|
{$ELSE}
|
|
|
|
{$IFDEF LINUX}
|
|
|
|
{$IFDEF FPC}
|
|
|
|
function CefTimeToSystemTime(const dt: TCefTime): TSystemTime;
|
|
|
|
begin
|
|
|
|
Result.Year := dt.year;
|
|
|
|
Result.Month := dt.month;
|
|
|
|
Result.DayOfWeek := dt.day_of_week;
|
|
|
|
Result.Day := dt.day_of_month;
|
|
|
|
Result.Hour := dt.hour;
|
|
|
|
Result.Minute := dt.minute;
|
|
|
|
Result.Second := dt.second;
|
|
|
|
Result.Millisecond := dt.millisecond;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function SystemTimeToCefTime(const dt: TSystemTime): TCefTime;
|
|
|
|
begin
|
2019-01-08 19:15:25 +01:00
|
|
|
Result.year := dt.Year;
|
|
|
|
Result.month := dt.Month;
|
|
|
|
Result.day_of_week := dt.DayOfWeek;
|
|
|
|
Result.day_of_month := dt.Day;
|
|
|
|
Result.hour := dt.Hour;
|
|
|
|
Result.minute := dt.Minute;
|
|
|
|
Result.second := dt.Second;
|
|
|
|
Result.millisecond := dt.Millisecond;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
2020-12-18 16:51:02 +01:00
|
|
|
{$ENDIF}
|
|
|
|
{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2021-01-18 11:12:01 +01:00
|
|
|
function FixCefTime(const dt : TCefTime): TCefTime;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
2021-01-18 11:12:01 +01:00
|
|
|
DayTable : PDayTable;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2021-01-18 11:12:01 +01:00
|
|
|
Result := dt;
|
2018-10-25 12:50:01 +02:00
|
|
|
|
2021-01-18 11:12:01 +01:00
|
|
|
Result.year := min(9999, max(1, Result.year));
|
|
|
|
Result.month := min(12, max(1, Result.month));
|
|
|
|
Result.hour := min(23, max(1, Result.hour));
|
|
|
|
Result.minute := min(59, max(1, Result.minute));
|
|
|
|
Result.second := min(59, max(1, Result.second));
|
|
|
|
Result.millisecond := min(999, max(1, Result.millisecond));
|
|
|
|
|
|
|
|
DayTable := @MonthDays[IsLeapYear(Result.year)];
|
|
|
|
Result.day_of_month := min(DayTable^[Result.month], max(1, Result.day_of_month));
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefTimeToDateTime(const dt: TCefTime): TDateTime;
|
|
|
|
var
|
|
|
|
TempFixedCefTime : TCefTime;
|
|
|
|
begin
|
|
|
|
TempFixedCefTime := FixCefTime(dt);
|
|
|
|
Result := EncodeDate(TempFixedCefTime.year, TempFixedCefTime.month, TempFixedCefTime.day_of_month) +
|
|
|
|
EncodeTime(TempFixedCefTime.hour, TempFixedCefTime.minute, TempFixedCefTime.second, TempFixedCefTime.millisecond);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function DateTimeToCefTime(dt: TDateTime): TCefTime;
|
|
|
|
var
|
2019-01-08 19:15:25 +01:00
|
|
|
TempYear, TempMonth, TempDay, TempHour, TempMin, TempSec, TempMSec : Word;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2019-01-08 19:15:25 +01:00
|
|
|
DecodeDate(dt, TempYear, TempMonth, TempDay);
|
|
|
|
DecodeTime(dt, TempHour, TempMin, TempSec, TempMSec);
|
|
|
|
|
|
|
|
Result.year := TempYear;
|
|
|
|
Result.month := TempMonth;
|
|
|
|
Result.day_of_week := DayOfWeek(dt);
|
|
|
|
Result.day_of_month := TempMonth;
|
|
|
|
Result.hour := TempHour;
|
|
|
|
Result.minute := TempMin;
|
|
|
|
Result.second := TempSec;
|
|
|
|
Result.millisecond := TempMSec;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function cef_string_wide_copy(const src: PWideChar; src_len: NativeUInt; output: PCefStringWide): Integer;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
Result := cef_string_wide_set(src, src_len, output, ord(True))
|
|
|
|
else
|
|
|
|
Result := 0;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function cef_string_utf8_copy(const src: PAnsiChar; src_len: NativeUInt; output: PCefStringUtf8): Integer;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
Result := cef_string_utf8_set(src, src_len, output, ord(True))
|
|
|
|
else
|
|
|
|
Result := 0;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function cef_string_utf16_copy(const src: PChar16; src_len: NativeUInt; output: PCefStringUtf16): Integer;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
Result := cef_string_utf16_set(src, src_len, output, ord(True))
|
|
|
|
else
|
|
|
|
Result := 0;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function cef_string_copy(const src: PCefChar; src_len: NativeUInt; output: PCefString): Integer;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
Result := cef_string_utf16_set(src, src_len, output, ord(True))
|
|
|
|
else
|
|
|
|
Result := 0;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2018-05-24 19:15:41 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-09-10 11:21:39 +02:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring; aExStyle : DWORD);
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-10-26 10:32:10 +02:00
|
|
|
aWindowInfo.ex_style := aExStyle;
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
|
|
|
aWindowInfo.style := WS_CHILD or WS_VISIBLE or WS_CLIPCHILDREN or WS_CLIPSIBLINGS or WS_TABSTOP;
|
|
|
|
aWindowInfo.x := aRect.left;
|
|
|
|
aWindowInfo.y := aRect.top;
|
|
|
|
aWindowInfo.width := aRect.right - aRect.left;
|
|
|
|
aWindowInfo.height := aRect.bottom - aRect.top;
|
|
|
|
aWindowInfo.parent_window := aParent;
|
|
|
|
aWindowInfo.menu := 0;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
|
|
|
|
2020-09-10 11:21:39 +02:00
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring; aExStyle : DWORD);
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-10-26 10:32:10 +02:00
|
|
|
aWindowInfo.ex_style := aExStyle;
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
|
|
|
aWindowInfo.style := WS_OVERLAPPEDWINDOW or WS_CLIPCHILDREN or WS_CLIPSIBLINGS or WS_VISIBLE;
|
|
|
|
aWindowInfo.x := integer(CW_USEDEFAULT);
|
|
|
|
aWindowInfo.y := integer(CW_USEDEFAULT);
|
|
|
|
aWindowInfo.width := integer(CW_USEDEFAULT);
|
|
|
|
aWindowInfo.height := integer(CW_USEDEFAULT);
|
|
|
|
aWindowInfo.parent_window := aParent;
|
|
|
|
aWindowInfo.menu := 0;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
|
|
|
|
2020-09-10 11:21:39 +02:00
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring; aExStyle : DWORD);
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-10-26 10:32:10 +02:00
|
|
|
aWindowInfo.ex_style := aExStyle;
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
|
|
|
aWindowInfo.style := 0;
|
|
|
|
aWindowInfo.x := 0;
|
|
|
|
aWindowInfo.y := 0;
|
|
|
|
aWindowInfo.width := 0;
|
|
|
|
aWindowInfo.height := 0;
|
|
|
|
aWindowInfo.parent_window := aParent;
|
|
|
|
aWindowInfo.menu := 0;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(True);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-04-18 10:25:38 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
2018-05-24 19:15:41 +02:00
|
|
|
{$ENDIF}
|
|
|
|
|
|
|
|
{$IFDEF MACOS}
|
2020-01-28 11:36:34 +01:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring; aHidden : boolean);
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
|
|
|
aWindowInfo.x := aRect.left;
|
|
|
|
aWindowInfo.y := aRect.top;
|
|
|
|
aWindowInfo.width := aRect.right - aRect.left;
|
|
|
|
aWindowInfo.height := aRect.bottom - aRect.top;
|
|
|
|
aWindowInfo.hidden := Ord(aHidden);
|
|
|
|
aWindowInfo.parent_view := aParent;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.view := 0;
|
|
|
|
end;
|
|
|
|
|
2020-01-28 11:36:34 +01:00
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring; aHidden : boolean);
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
2020-01-28 11:36:34 +01:00
|
|
|
aWindowInfo.x := 0;
|
|
|
|
aWindowInfo.y := 0;
|
|
|
|
aWindowInfo.width := 0;
|
|
|
|
aWindowInfo.height := 0;
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.hidden := Ord(aHidden);
|
|
|
|
aWindowInfo.parent_view := aParent;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.view := 0;
|
|
|
|
end;
|
|
|
|
|
2020-01-28 11:36:34 +01:00
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring; aHidden : boolean);
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
|
|
|
aWindowInfo.x := 0;
|
|
|
|
aWindowInfo.y := 0;
|
|
|
|
aWindowInfo.width := 0;
|
|
|
|
aWindowInfo.height := 0;
|
|
|
|
aWindowInfo.hidden := Ord(aHidden);
|
|
|
|
aWindowInfo.parent_view := aParent;
|
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(True);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.view := 0;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
|
|
|
|
|
|
|
{$IFDEF LINUX}
|
2021-01-08 16:01:06 +01:00
|
|
|
procedure WindowInfoAsChild(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; aRect : TRect; const aWindowName : ustring = '');
|
2021-01-06 12:18:14 +01:00
|
|
|
var
|
|
|
|
TempParent : TCefWindowHandle;
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
2021-01-08 16:01:06 +01:00
|
|
|
// TODO: Find a way to get the right "parent_window" in FMX
|
|
|
|
TempParent := aParent;
|
|
|
|
{$IFDEF FPC}
|
2021-01-06 12:18:14 +01:00
|
|
|
if ValidCefWindowHandle(aParent) and (PGtkWidget(aParent)^.window <> nil) then
|
2021-01-08 16:01:06 +01:00
|
|
|
TempParent := gdk_window_xwindow(PGtkWidget(aParent)^.window);
|
|
|
|
{$ENDIF}
|
2021-01-06 12:18:14 +01:00
|
|
|
|
2019-05-04 09:53:50 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.x := aRect.left;
|
|
|
|
aWindowInfo.y := aRect.top;
|
|
|
|
aWindowInfo.width := aRect.right - aRect.left;
|
|
|
|
aWindowInfo.height := aRect.bottom - aRect.top;
|
2021-01-06 12:18:14 +01:00
|
|
|
aWindowInfo.parent_window := TempParent;
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
|
|
|
|
2021-01-06 12:18:14 +01:00
|
|
|
// WindowInfoAsPopUp only exists for Windows. The Linux version of cefclient
|
|
|
|
// calls WindowInfoAsChild with aParent set to NULL to create a popup window.
|
2019-03-15 17:17:14 +01:00
|
|
|
procedure WindowInfoAsPopUp(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = '');
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
2019-05-04 09:53:50 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
2019-01-08 19:15:25 +01:00
|
|
|
aWindowInfo.x := 0;
|
|
|
|
aWindowInfo.y := 0;
|
|
|
|
aWindowInfo.width := 0;
|
|
|
|
aWindowInfo.height := 0;
|
2021-01-06 12:18:14 +01:00
|
|
|
aWindowInfo.parent_window := aParent;
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(False);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
|
|
|
|
2019-03-15 17:17:14 +01:00
|
|
|
procedure WindowInfoAsWindowless(var aWindowInfo : TCefWindowInfo; aParent : TCefWindowHandle; const aWindowName : ustring = '');
|
2018-05-24 19:15:41 +02:00
|
|
|
begin
|
2019-05-04 09:53:50 +02:00
|
|
|
aWindowInfo.window_name := CefString(aWindowName);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.x := 0;
|
|
|
|
aWindowInfo.y := 0;
|
|
|
|
aWindowInfo.width := 0;
|
|
|
|
aWindowInfo.height := 0;
|
|
|
|
aWindowInfo.parent_window := aParent;
|
2019-12-18 15:10:30 +01:00
|
|
|
aWindowInfo.windowless_rendering_enabled := ord(True);
|
2018-12-12 17:13:23 +01:00
|
|
|
aWindowInfo.shared_texture_enabled := ord(False);
|
|
|
|
aWindowInfo.external_begin_frame_enabled := ord(False);
|
2018-05-24 19:15:41 +02:00
|
|
|
aWindowInfo.window := 0;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function CefIsCertStatusError(Status : TCefCertStatus) : boolean;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
Result := (GlobalCEFApp <> nil) and
|
|
|
|
GlobalCEFApp.LibLoaded and
|
|
|
|
(cef_is_cert_status_error(Status) <> 0);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefCrashReportingEnabled : boolean;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
Result := (GlobalCEFApp <> nil) and
|
|
|
|
GlobalCEFApp.LibLoaded and
|
|
|
|
(cef_crash_reporting_enabled() <> 0);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefSetCrashKeyValue(const aKey, aValue : ustring);
|
|
|
|
var
|
|
|
|
TempKey, TempValue : TCefString;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempKey := CefString(aKey);
|
|
|
|
TempValue := CefString(aValue);
|
|
|
|
cef_set_crash_key_value(@TempKey, @TempValue);
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefLog(const aFile : string; aLine, aSeverity : integer; const aMessage : string);
|
|
|
|
var
|
|
|
|
TempFile, TempMessage : AnsiString;
|
|
|
|
begin
|
|
|
|
if (length(aFile) > 0) and (length(aMessage) > 0) then
|
|
|
|
begin
|
|
|
|
TempFile := AnsiString(aFile);
|
|
|
|
TempMessage := AnsiString(aMessage);
|
|
|
|
|
|
|
|
cef_log(@TempFile[1], aLine, aSeverity, @TempMessage[1]);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-10-12 12:21:43 +02:00
|
|
|
procedure CefDebugLog(const aMessage : string; aSeverity : integer);
|
2018-04-18 10:25:38 +02:00
|
|
|
const
|
|
|
|
DEFAULT_LINE = 1;
|
|
|
|
var
|
|
|
|
TempString : string;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-01-28 11:36:34 +01:00
|
|
|
TempString := 'PID: ' + IntToStr(GetCurrentProcessID) + ', TID: ' + IntToStr(GetCurrentThreadID);
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ELSE}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$IFDEF MACOS}
|
|
|
|
TempString := 'PID: ' + IntToStr(TNSProcessInfo.Wrap(TNSProcessInfo.OCClass.processInfo).processIdentifier) + ', TID: ' + IntToStr(TThread.Current.ThreadID);
|
|
|
|
{$ELSE}
|
2020-12-18 16:51:02 +01:00
|
|
|
{$IFDEF FPC}
|
|
|
|
TempString := 'PID: ' + IntToStr(GetProcessID()) + ', TID: ' + IntToStr(GetCurrentThreadID());
|
|
|
|
{$ELSE}
|
|
|
|
// TODO: Find the equivalent function to get the process ID in Delphi FMX for Linux
|
|
|
|
// TempString := 'PID: ' + IntToStr(GetProcessID()) + ', TID: ' + IntToStr(GetCurrentThreadID());
|
|
|
|
{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ENDIF}
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2018-03-29 20:02:04 +02:00
|
|
|
case GlobalCEFApp.ProcessType of
|
|
|
|
ptBrowser : TempString := TempString + ', PT: Browser';
|
|
|
|
ptRenderer : TempString := TempString + ', PT: Renderer';
|
|
|
|
ptZygote : TempString := TempString + ', PT: Zygote';
|
|
|
|
ptGPU : TempString := TempString + ', PT: GPU';
|
2020-04-25 16:47:16 +02:00
|
|
|
ptUtility : TempString := TempString + ', PT: Utility';
|
|
|
|
ptOther : TempString := TempString + ', PT: Other';
|
2018-03-29 20:02:04 +02:00
|
|
|
end;
|
|
|
|
|
2018-10-12 12:21:43 +02:00
|
|
|
CefLog('CEF4Delphi', DEFAULT_LINE, aSeverity, TempString + ' - ' + aMessage);
|
2018-03-29 20:02:04 +02:00
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2020-04-23 11:11:53 +02:00
|
|
|
procedure CefKeyEventLog(const aEvent : TCefKeyEvent);
|
|
|
|
const
|
|
|
|
DEFAULT_LINE = 1;
|
|
|
|
var
|
|
|
|
TempString : string;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
case aEvent.kind of
|
|
|
|
KEYEVENT_RAWKEYDOWN : TempString := 'kind: KEYEVENT_RAWKEYDOWN';
|
|
|
|
KEYEVENT_KEYDOWN : TempString := 'kind: KEYEVENT_KEYDOWN';
|
|
|
|
KEYEVENT_KEYUP : TempString := 'kind: KEYEVENT_KEYUP';
|
|
|
|
KEYEVENT_CHAR : TempString := 'kind: KEYEVENT_CHAR';
|
|
|
|
end;
|
|
|
|
|
|
|
|
TempString := TempString + ', modifiers: $' + inttohex(aEvent.modifiers, SizeOf(aEvent.modifiers) * 2);
|
|
|
|
TempString := TempString + ', windows_key_code: $' + inttohex(aEvent.windows_key_code, SizeOf(aEvent.windows_key_code) * 2);
|
|
|
|
TempString := TempString + ', native_key_code: $' + inttohex(aEvent.native_key_code, SizeOf(aEvent.native_key_code) * 2);
|
|
|
|
TempString := TempString + ', is_system_key: ' + BoolToStr((aEvent.is_system_key <> 0), true);
|
|
|
|
TempString := TempString + ', character: $' + inttohex(ord(aEvent.character), SizeOf(aEvent.character) * 2);
|
|
|
|
TempString := TempString + ', unmodified_character: $' + inttohex(ord(aEvent.unmodified_character), SizeOf(aEvent.unmodified_character) * 2);
|
|
|
|
TempString := TempString + ', focus_on_editable_field: ' + BoolToStr((aEvent.focus_on_editable_field <> 0), true);;
|
|
|
|
|
|
|
|
CefLog('CEF4Delphi', DEFAULT_LINE, CEF_LOG_SEVERITY_INFO, TempString);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefMouseEventLog(const aEvent : TCefMouseEvent);
|
|
|
|
const
|
|
|
|
DEFAULT_LINE = 1;
|
|
|
|
var
|
|
|
|
TempString : string;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempString := TempString + ', x: $' + inttohex(aEvent.x, SizeOf(aEvent.x) * 2);
|
|
|
|
TempString := TempString + ', y: $' + inttohex(aEvent.y, SizeOf(aEvent.y) * 2);
|
|
|
|
TempString := TempString + ', modifiers: $' + inttohex(aEvent.modifiers, SizeOf(aEvent.modifiers) * 2);
|
|
|
|
|
|
|
|
CefLog('CEF4Delphi', DEFAULT_LINE, CEF_LOG_SEVERITY_INFO, TempString);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure OutputDebugMessage(const aMessage : string);
|
|
|
|
const
|
|
|
|
DEFAULT_LINE = 1;
|
|
|
|
begin
|
|
|
|
{$IFDEF DEBUG}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$IFDEF FMX}
|
|
|
|
FMX.Types.Log.d(aMessage);
|
|
|
|
{$ELSE}
|
2020-06-13 17:24:22 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-01-28 11:36:34 +01:00
|
|
|
OutputDebugString({$IFDEF DELPHI12_UP}PWideChar{$ELSE}PAnsiChar{$ENDIF}(aMessage + chr(0)));
|
|
|
|
{$ENDIF}
|
2019-12-18 15:10:30 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
CefLog('CEF4Delphi', DEFAULT_LINE, CEF_LOG_SEVERITY_ERROR, aMessage);
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CustomExceptionHandler(const aFunctionName : string; const aException : exception) : boolean;
|
|
|
|
begin
|
|
|
|
OutputDebugMessage(aFunctionName + ' error : ' + aException.message);
|
|
|
|
|
|
|
|
Result := (GlobalCEFApp <> nil) and GlobalCEFApp.ReRaiseExceptions;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefRegisterSchemeHandlerFactory(const SchemeName : ustring;
|
2018-06-11 18:38:51 +02:00
|
|
|
const DomainName : ustring;
|
2018-04-18 10:25:38 +02:00
|
|
|
const handler : TCefResourceHandlerClass) : boolean;
|
|
|
|
var
|
2018-06-11 18:38:51 +02:00
|
|
|
TempScheme, TempDomainName : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
TempFactory : ICefSchemeHandlerFactory;
|
2018-06-11 18:38:51 +02:00
|
|
|
TempDomainNamePtr : PCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
2018-06-11 18:38:51 +02:00
|
|
|
if (GlobalCEFApp <> nil) and
|
|
|
|
GlobalCEFApp.LibLoaded and
|
|
|
|
(length(SchemeName) > 0) then
|
2018-06-11 09:01:38 +02:00
|
|
|
begin
|
2018-06-11 18:38:51 +02:00
|
|
|
if (length(DomainName) > 0) then
|
|
|
|
begin
|
|
|
|
TempDomainName := CefString(DomainName);
|
|
|
|
TempDomainNamePtr := @TempDomainName;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
TempDomainNamePtr := nil;
|
|
|
|
|
|
|
|
TempScheme := CefString(SchemeName);
|
|
|
|
TempFactory := TCefSchemeHandlerFactoryOwn.Create(handler);
|
|
|
|
Result := cef_register_scheme_handler_factory(@TempScheme, TempDomainNamePtr, TempFactory.Wrap) <> 0;
|
2018-06-11 09:01:38 +02:00
|
|
|
end;
|
|
|
|
finally
|
|
|
|
TempFactory := nil;
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefClearSchemeHandlerFactories : boolean;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
Result := (GlobalCEFApp <> nil) and
|
|
|
|
GlobalCEFApp.LibLoaded and
|
|
|
|
(cef_clear_scheme_handler_factories() <> 0);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefAddCrossOriginWhitelistEntry(const SourceOrigin : ustring;
|
|
|
|
const TargetProtocol : ustring;
|
|
|
|
const TargetDomain : ustring;
|
|
|
|
AllowTargetSubdomains : Boolean): Boolean;
|
|
|
|
var
|
|
|
|
TempSourceOrigin, TempTargetProtocol, TempTargetDomain : TCefString;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempSourceOrigin := CefString(SourceOrigin);
|
|
|
|
TempTargetProtocol := CefString(TargetProtocol);
|
|
|
|
TempTargetDomain := CefString(TargetDomain);
|
|
|
|
Result := cef_add_cross_origin_whitelist_entry(@TempSourceOrigin,
|
|
|
|
@TempTargetProtocol,
|
|
|
|
@TempTargetDomain,
|
|
|
|
Ord(AllowTargetSubdomains)) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefRemoveCrossOriginWhitelistEntry(const SourceOrigin : ustring;
|
|
|
|
const TargetProtocol : ustring;
|
|
|
|
const TargetDomain : ustring;
|
|
|
|
AllowTargetSubdomains : Boolean): Boolean;
|
|
|
|
var
|
|
|
|
TempSourceOrigin, TempTargetProtocol, TempTargetDomain : TCefString;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempSourceOrigin := CefString(SourceOrigin);
|
|
|
|
TempTargetProtocol := CefString(TargetProtocol);
|
|
|
|
TempTargetDomain := CefString(TargetDomain);
|
|
|
|
Result := cef_remove_cross_origin_whitelist_entry(@TempSourceOrigin,
|
|
|
|
@TempTargetProtocol,
|
|
|
|
@TempTargetDomain,
|
|
|
|
Ord(AllowTargetSubdomains)) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefClearCrossOriginWhitelist: Boolean;
|
|
|
|
begin
|
2018-05-12 14:50:54 +02:00
|
|
|
Result := cef_clear_cross_origin_whitelist() <> 0;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function SplitLongString(aSrcString : string) : string;
|
|
|
|
const
|
|
|
|
MAXLINELENGTH = 50;
|
|
|
|
begin
|
2019-11-10 18:23:39 +01:00
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
while (length(aSrcString) > 0) do
|
|
|
|
begin
|
2019-11-10 18:23:39 +01:00
|
|
|
if (Result <> '') then
|
2018-04-18 10:25:38 +02:00
|
|
|
Result := Result + CRLF + copy(aSrcString, 1, MAXLINELENGTH)
|
|
|
|
else
|
|
|
|
Result := Result + copy(aSrcString, 1, MAXLINELENGTH);
|
|
|
|
|
|
|
|
aSrcString := copy(aSrcString, succ(MAXLINELENGTH), length(aSrcString));
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetAbsoluteDirPath(const aSrcPath : string; var aRsltPath : string) : boolean;
|
|
|
|
begin
|
|
|
|
Result := True;
|
|
|
|
|
|
|
|
if (length(aSrcPath) > 0) then
|
|
|
|
begin
|
2019-09-21 11:37:13 +02:00
|
|
|
aRsltPath := IncludeTrailingPathDelimiter(CustomAbsolutePath(aSrcPath));
|
|
|
|
Result := DirectoryExists(aRsltPath);
|
2018-04-18 10:25:38 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
aRsltPath := '';
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CheckLocales(const aLocalesDirPath : string; var aMissingFiles : string; const aLocalesRequired : string) : boolean;
|
|
|
|
const
|
|
|
|
LOCALES_REQUIRED_DEFAULT =
|
|
|
|
'am,' +
|
|
|
|
'ar,' +
|
|
|
|
'bg,' +
|
|
|
|
'bn,' +
|
|
|
|
'ca,' +
|
|
|
|
'cs,' +
|
|
|
|
'da,' +
|
|
|
|
'de,' +
|
|
|
|
'el,' +
|
|
|
|
'en-GB,' +
|
|
|
|
'en-US,' +
|
|
|
|
'es,' +
|
|
|
|
'es-419,' +
|
|
|
|
'et,' +
|
|
|
|
'fa,' +
|
|
|
|
'fi,' +
|
|
|
|
'fil,' +
|
|
|
|
'fr,' +
|
|
|
|
'gu,' +
|
|
|
|
'he,' +
|
|
|
|
'hi,' +
|
|
|
|
'hr,' +
|
|
|
|
'hu,' +
|
|
|
|
'id,' +
|
|
|
|
'it,' +
|
|
|
|
'ja,' +
|
|
|
|
'kn,' +
|
|
|
|
'ko,' +
|
|
|
|
'lt,' +
|
|
|
|
'lv,' +
|
|
|
|
'ml,' +
|
|
|
|
'mr,' +
|
|
|
|
'ms,' +
|
|
|
|
'nb,' +
|
|
|
|
'nl,' +
|
|
|
|
'pl,' +
|
|
|
|
'pt-BR,' +
|
|
|
|
'pt-PT,' +
|
|
|
|
'ro,' +
|
|
|
|
'ru,' +
|
|
|
|
'sk,' +
|
|
|
|
'sl,' +
|
|
|
|
'sr,' +
|
|
|
|
'sv,' +
|
|
|
|
'sw,' +
|
|
|
|
'ta,' +
|
|
|
|
'te,' +
|
|
|
|
'th,' +
|
|
|
|
'tr,' +
|
|
|
|
'uk,' +
|
|
|
|
'vi,' +
|
|
|
|
'zh-CN,' +
|
|
|
|
'zh-TW';
|
|
|
|
var
|
|
|
|
i : integer;
|
|
|
|
TempDir : string;
|
|
|
|
TempList : TStringList;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
TempList := nil;
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
|
|
|
if (length(aLocalesDirPath) > 0) then
|
|
|
|
TempDir := IncludeTrailingPathDelimiter(aLocalesDirPath)
|
|
|
|
else
|
2019-11-10 18:23:39 +01:00
|
|
|
TempDir := 'locales' + PathDelim;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
TempList := TStringList.Create;
|
|
|
|
|
|
|
|
if (length(aLocalesRequired) > 0) then
|
|
|
|
TempList.CommaText := aLocalesRequired
|
|
|
|
else
|
|
|
|
TempList.CommaText := LOCALES_REQUIRED_DEFAULT;
|
|
|
|
|
|
|
|
i := 0;
|
2017-12-18 19:38:56 +01:00
|
|
|
while (i < TempList.Count) do
|
|
|
|
begin
|
|
|
|
TempList[i] := TempDir + TempList[i] + '.pak';
|
|
|
|
inc(i);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
if DirectoryExists(TempDir) then
|
|
|
|
Result := CheckFilesExist(TempList, aMissingFiles)
|
|
|
|
else
|
|
|
|
aMissingFiles := trim(aMissingFiles) + CRLF + TempList.Text;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('CheckLocales', e) then raise;
|
|
|
|
end;
|
2017-12-18 19:38:56 +01:00
|
|
|
finally
|
|
|
|
if (TempList <> nil) then FreeAndNil(TempList);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-10-03 10:58:51 +03:00
|
|
|
function CheckResources(const aResourcesDirPath : string; var aMissingFiles : string; aCheckDevResources, aCheckExtensions: boolean) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
|
|
|
TempDir : string;
|
|
|
|
TempList : TStringList;
|
|
|
|
TempExists : boolean;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
|
|
|
TempExists := GetAbsoluteDirPath(aResourcesDirPath, TempDir);
|
|
|
|
|
|
|
|
TempList := TStringList.Create;
|
|
|
|
TempList.Add(TempDir + 'snapshot_blob.bin');
|
|
|
|
TempList.Add(TempDir + 'v8_context_snapshot.bin');
|
|
|
|
TempList.Add(TempDir + 'cef.pak');
|
|
|
|
TempList.Add(TempDir + 'cef_100_percent.pak');
|
|
|
|
TempList.Add(TempDir + 'cef_200_percent.pak');
|
|
|
|
|
2019-09-21 11:37:13 +02:00
|
|
|
if aCheckExtensions then TempList.Add(TempDir + 'cef_extensions.pak');
|
2018-04-18 10:25:38 +02:00
|
|
|
if aCheckDevResources then TempList.Add(TempDir + 'devtools_resources.pak');
|
|
|
|
|
|
|
|
if TempExists then
|
|
|
|
Result := CheckFilesExist(TempList, aMissingFiles)
|
|
|
|
else
|
|
|
|
aMissingFiles := trim(aMissingFiles) + CRLF + TempList.Text;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('CheckResources', e) then raise;
|
|
|
|
end;
|
2017-12-18 19:38:56 +01:00
|
|
|
finally
|
|
|
|
if (TempList <> nil) then FreeAndNil(TempList);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-10-12 12:21:43 +02:00
|
|
|
function CheckSubprocessPath(const aSubprocessPath : string; var aMissingFiles : string) : boolean;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
|
|
|
if (length(aSubprocessPath) = 0) or FileExists(aSubprocessPath) then
|
|
|
|
Result := True
|
|
|
|
else
|
|
|
|
aMissingFiles := trim(aMissingFiles) + CRLF + ExtractFileName(aSubprocessPath);
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('CheckSubprocessPath', e) then raise;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
function CheckDLLs(const aFrameworkDirPath : string; var aMissingFiles : string) : boolean;
|
|
|
|
var
|
|
|
|
TempDir : string;
|
|
|
|
TempList : TStringList;
|
|
|
|
TempExists : boolean;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
TempList := nil;
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
|
|
|
TempExists := GetAbsoluteDirPath(aFrameworkDirPath, TempDir);
|
|
|
|
|
|
|
|
// The icudtl.dat file must be placed next to libcef.dll
|
|
|
|
// http://www.magpcss.org/ceforum/viewtopic.php?f=6&t=14503#p32263
|
|
|
|
|
|
|
|
TempList := TStringList.Create;
|
|
|
|
TempList.Add(TempDir + LIBCEF_DLL);
|
2019-11-10 18:23:39 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-07-21 12:41:03 +02:00
|
|
|
TempList.Add(TempDir + CHROMEELF_DLL);
|
2018-04-18 10:25:38 +02:00
|
|
|
TempList.Add(TempDir + 'd3dcompiler_47.dll');
|
|
|
|
TempList.Add(TempDir + 'libEGL.dll');
|
|
|
|
TempList.Add(TempDir + 'libGLESv2.dll');
|
|
|
|
TempList.Add(TempDir + 'swiftshader\libEGL.dll');
|
|
|
|
TempList.Add(TempDir + 'swiftshader\libGLESv2.dll');
|
2019-11-10 18:23:39 +01:00
|
|
|
{$ENDIF}
|
|
|
|
{$IFDEF LINUX}
|
|
|
|
TempList.Add(TempDir + 'libEGL.so');
|
|
|
|
TempList.Add(TempDir + 'libGLESv2.so');
|
|
|
|
TempList.Add(TempDir + 'swiftshader/libEGL.so');
|
|
|
|
TempList.Add(TempDir + 'swiftshader/libGLESv2.so');
|
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
TempList.Add(TempDir + 'icudtl.dat');
|
|
|
|
|
|
|
|
if TempExists then
|
|
|
|
Result := CheckFilesExist(TempList, aMissingFiles)
|
|
|
|
else
|
|
|
|
aMissingFiles := trim(aMissingFiles) + CRLF + TempList.Text;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('CheckDLLs', e) then raise;
|
|
|
|
end;
|
2017-12-18 19:38:56 +01:00
|
|
|
finally
|
|
|
|
if (TempList <> nil) then FreeAndNil(TempList);
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CheckFilesExist(var aList : TStringList; var aMissingFiles : string) : boolean;
|
|
|
|
var
|
|
|
|
i : integer;
|
|
|
|
begin
|
2017-12-18 19:38:56 +01:00
|
|
|
Result := True;
|
|
|
|
|
|
|
|
try
|
|
|
|
if (aList <> nil) then
|
|
|
|
begin
|
|
|
|
i := 0;
|
|
|
|
|
|
|
|
while (i < aList.Count) do
|
|
|
|
begin
|
|
|
|
if (length(aList[i]) > 0) and not(FileExists(aList[i])) then
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
aMissingFiles := aMissingFiles + aList[i] + CRLF;
|
|
|
|
end;
|
|
|
|
|
|
|
|
inc(i);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
except
|
2018-04-18 10:25:38 +02:00
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('CheckFilesExist', e) then raise;
|
2017-12-18 19:38:56 +01:00
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure UInt64ToFileVersionInfo(const aVersion : uint64; var aVersionInfo : TFileVersionInfo);
|
|
|
|
begin
|
|
|
|
aVersionInfo.MajorVer := uint16(aVersion shr 48);
|
|
|
|
aVersionInfo.MinorVer := uint16((aVersion shr 32) and $FFFF);
|
|
|
|
aVersionInfo.Release := uint16((aVersion shr 16) and $FFFF);
|
|
|
|
aVersionInfo.Build := uint16(aVersion and $FFFF);
|
|
|
|
end;
|
|
|
|
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-06-13 17:24:22 +02:00
|
|
|
function GetExtendedFileVersion(const aFileName : ustring) : uint64;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
|
|
|
TempSize : DWORD;
|
|
|
|
TempBuffer : pointer;
|
|
|
|
TempLen : UINT;
|
|
|
|
TempHandle : cardinal;
|
|
|
|
TempInfo : PVSFixedFileInfo;
|
|
|
|
begin
|
|
|
|
Result := 0;
|
|
|
|
TempBuffer := nil;
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
2020-06-13 17:24:22 +02:00
|
|
|
TempSize := GetFileVersionInfoSizeW(PWideChar(aFileName), TempHandle);
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
if (TempSize > 0) then
|
|
|
|
begin
|
|
|
|
GetMem(TempBuffer, TempSize);
|
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
if GetFileVersionInfoW(PWideChar(aFileName), TempHandle, TempSize, TempBuffer) and
|
2018-04-18 10:25:38 +02:00
|
|
|
VerQueryValue(TempBuffer, '\', Pointer(TempInfo), TempLen) then
|
|
|
|
begin
|
2018-05-12 14:50:54 +02:00
|
|
|
Result := TempInfo^.dwFileVersionMS;
|
2018-04-18 10:25:38 +02:00
|
|
|
Result := Result shl 32;
|
2018-05-12 14:50:54 +02:00
|
|
|
Result := Result or TempInfo^.dwFileVersionLS;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
2020-06-13 17:24:22 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
OutputLastErrorMessage;
|
2018-04-18 10:25:38 +02:00
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('GetExtendedFileVersion', e) then raise;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
if (TempBuffer <> nil) then FreeMem(TempBuffer);
|
|
|
|
end;
|
|
|
|
end;
|
2017-10-24 20:44:51 +02:00
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
procedure OutputLastErrorMessage;
|
|
|
|
begin
|
|
|
|
{$IFDEF DEBUG}
|
|
|
|
OutputDebugString({$IFDEF DELPHI12_UP}PWideChar{$ELSE}PAnsiChar{$ENDIF}(SysErrorMessage(GetLastError()) + chr(0)));
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2020-06-20 16:10:29 +02:00
|
|
|
function GetStringFileInfo(const aFileName, aField : ustring; var aValue : ustring) : boolean;
|
2017-10-24 20:44:51 +02:00
|
|
|
type
|
|
|
|
PLangAndCodepage = ^TLangAndCodepage;
|
|
|
|
TLangAndCodepage = record
|
|
|
|
wLanguage : word;
|
|
|
|
wCodePage : word;
|
|
|
|
end;
|
|
|
|
var
|
|
|
|
TempSize : DWORD;
|
|
|
|
TempBuffer : pointer;
|
|
|
|
TempHandle : cardinal;
|
|
|
|
TempPointer : pointer;
|
2020-06-20 16:10:29 +02:00
|
|
|
TempSubBlock : ustring;
|
2017-10-24 20:44:51 +02:00
|
|
|
TempLang : PLangAndCodepage;
|
|
|
|
TempArray : array of TLangAndCodepage;
|
|
|
|
i, j : DWORD;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
TempBuffer := nil;
|
|
|
|
TempArray := nil;
|
|
|
|
aValue := '';
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
2020-06-13 17:24:22 +02:00
|
|
|
TempSize := GetFileVersionInfoSizeW(PWideChar(aFileName), TempHandle);
|
2017-10-24 20:44:51 +02:00
|
|
|
|
|
|
|
if (TempSize > 0) then
|
|
|
|
begin
|
|
|
|
GetMem(TempBuffer, TempSize);
|
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
if GetFileVersionInfoW(PWideChar(aFileName), 0, TempSize, TempBuffer) then
|
2017-10-24 20:44:51 +02:00
|
|
|
begin
|
|
|
|
if VerQueryValue(TempBuffer, '\VarFileInfo\Translation\', Pointer(TempLang), TempSize) then
|
|
|
|
begin
|
|
|
|
i := 0;
|
|
|
|
j := TempSize div SizeOf(TLangAndCodepage);
|
|
|
|
|
|
|
|
SetLength(TempArray, j);
|
|
|
|
|
|
|
|
while (i < j) do
|
|
|
|
begin
|
2018-05-12 14:50:54 +02:00
|
|
|
TempArray[i].wLanguage := TempLang^.wLanguage;
|
|
|
|
TempArray[i].wCodePage := TempLang^.wCodePage;
|
2017-10-24 20:44:51 +02:00
|
|
|
inc(TempLang);
|
|
|
|
inc(i);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
i := 0;
|
|
|
|
j := Length(TempArray);
|
|
|
|
|
|
|
|
while (i < j) and not(Result) do
|
|
|
|
begin
|
|
|
|
TempSubBlock := '\StringFileInfo\' +
|
|
|
|
IntToHex(TempArray[i].wLanguage, 4) + IntToHex(TempArray[i].wCodePage, 4) +
|
|
|
|
'\' + aField;
|
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
if VerQueryValueW(TempBuffer, PWideChar(TempSubBlock), TempPointer, TempSize) then
|
2017-10-24 20:44:51 +02:00
|
|
|
begin
|
|
|
|
aValue := trim(PChar(TempPointer));
|
|
|
|
Result := (length(aValue) > 0);
|
|
|
|
end;
|
|
|
|
|
|
|
|
inc(i);
|
|
|
|
end;
|
|
|
|
|
|
|
|
// Adobe's flash player DLL uses a different codepage to store the StringFileInfo fields
|
|
|
|
if not(Result) and (j > 0) and (TempArray[0].wCodePage <> 1252) then
|
|
|
|
begin
|
|
|
|
TempSubBlock := '\StringFileInfo\' +
|
|
|
|
IntToHex(TempArray[0].wLanguage, 4) + IntToHex(1252, 4) +
|
|
|
|
'\' + aField;
|
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
if VerQueryValueW(TempBuffer, PWideChar(TempSubBlock), TempPointer, TempSize) then
|
2017-10-24 20:44:51 +02:00
|
|
|
begin
|
|
|
|
aValue := trim(PChar(TempPointer));
|
|
|
|
Result := (length(aValue) > 0);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('GetStringFileInfo', e) then raise;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
if (TempBuffer <> nil) then FreeMem(TempBuffer);
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
function GetDLLVersion(const aDLLFile : ustring; var aVersionInfo : TFileVersionInfo) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
|
|
|
TempVersion : uint64;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
|
|
|
if FileExists(aDLLFile) then
|
|
|
|
begin
|
|
|
|
TempVersion := GetExtendedFileVersion(aDLLFile);
|
2020-06-13 17:24:22 +02:00
|
|
|
if (TempVersion <> 0) then
|
|
|
|
begin
|
|
|
|
UInt64ToFileVersionInfo(TempVersion, aVersionInfo);
|
|
|
|
Result := True;
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('GetDLLVersion', e) then raise;
|
|
|
|
end;
|
2019-01-08 19:15:25 +01:00
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
function CheckDLLVersion(const aDLLFile : ustring; aMajor, aMinor, aRelease, aBuild : uint16) : boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
|
|
|
TempVersionInfo : TFileVersionInfo;
|
|
|
|
begin
|
|
|
|
Result := GetDLLVersion(aDLLFile, TempVersionInfo) and
|
|
|
|
(TempVersionInfo.MajorVer = aMajor) and
|
|
|
|
(TempVersionInfo.MinorVer = aMinor) and
|
|
|
|
(TempVersionInfo.Release = aRelease) and
|
|
|
|
(TempVersionInfo.Build = aBuild);
|
|
|
|
end;
|
|
|
|
|
2018-09-02 13:11:43 +02:00
|
|
|
// This function is based on the answer given by 'Alex' in StackOverflow
|
|
|
|
// https://stackoverflow.com/questions/2748474/how-to-determine-if-dll-file-was-compiled-as-x64-or-x86-bit-using-either-delphi
|
2020-06-13 17:24:22 +02:00
|
|
|
function GetDLLHeaderMachine(const aDLLFile : ustring; var aMachine : integer) : boolean;
|
2018-09-02 13:11:43 +02:00
|
|
|
var
|
|
|
|
TempHeader : TImageDosHeader;
|
|
|
|
TempImageNtHeaders : TImageNtHeaders;
|
|
|
|
TempStream : TFileStream;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
aMachine := IMAGE_FILE_MACHINE_UNKNOWN;
|
|
|
|
TempStream := nil;
|
|
|
|
|
|
|
|
try
|
|
|
|
try
|
|
|
|
if FileExists(aDLLFile) then
|
|
|
|
begin
|
2020-09-30 12:43:48 +02:00
|
|
|
TempStream := TFileStream.Create(aDLLFile, fmOpenRead or fmShareDenyWrite);
|
2018-09-02 13:11:43 +02:00
|
|
|
TempStream.seek(0, soFromBeginning);
|
|
|
|
TempStream.ReadBuffer(TempHeader, SizeOf(TempHeader));
|
|
|
|
|
|
|
|
if (TempHeader.e_magic = IMAGE_DOS_SIGNATURE) and
|
|
|
|
(TempHeader._lfanew <> 0) then
|
|
|
|
begin
|
|
|
|
TempStream.Position := TempHeader._lfanew;
|
|
|
|
TempStream.ReadBuffer(TempImageNtHeaders, SizeOf(TempImageNtHeaders));
|
|
|
|
|
|
|
|
if (TempImageNtHeaders.Signature = IMAGE_NT_SIGNATURE) then
|
|
|
|
begin
|
|
|
|
aMachine := TempImageNtHeaders.FileHeader.Machine;
|
|
|
|
Result := True;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
2018-09-02 15:40:56 +02:00
|
|
|
if CustomExceptionHandler('GetDLLHeaderMachine', e) then raise;
|
2018-09-02 13:11:43 +02:00
|
|
|
end;
|
|
|
|
finally
|
|
|
|
if (TempStream <> nil) then FreeAndNil(TempStream);
|
|
|
|
end;
|
2019-01-08 19:15:25 +01:00
|
|
|
end;
|
|
|
|
{$ENDIF}
|
2018-09-02 13:11:43 +02:00
|
|
|
|
2020-06-13 17:24:22 +02:00
|
|
|
function FileVersionInfoToString(const aVersionInfo : TFileVersionInfo) : string;
|
|
|
|
begin
|
|
|
|
Result := IntToStr(aVersionInfo.MajorVer) + '.' +
|
|
|
|
IntToStr(aVersionInfo.MinorVer) + '.' +
|
|
|
|
IntToStr(aVersionInfo.Release) + '.' +
|
|
|
|
IntToStr(aVersionInfo.Build);
|
|
|
|
end;
|
|
|
|
|
2018-09-02 13:11:43 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2020-07-28 20:04:25 +02:00
|
|
|
function Is32BitProcessRunningIn64BitOS : boolean;
|
2018-09-02 13:11:43 +02:00
|
|
|
var
|
|
|
|
TempResult : BOOL;
|
|
|
|
begin
|
2020-07-28 20:04:25 +02:00
|
|
|
Result := ProcessUnderWow64(GetCurrentProcess, TempResult) and TempResult;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
2018-09-02 13:11:43 +02:00
|
|
|
|
2020-07-28 20:04:25 +02:00
|
|
|
function Is32BitProcess : boolean;
|
|
|
|
begin
|
|
|
|
{$IFDEF TARGET_32BITS}
|
|
|
|
Result := True;
|
2018-09-02 13:11:43 +02:00
|
|
|
{$ELSE}
|
2020-07-28 20:04:25 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
Result := Is32BitProcessRunningIn64BitOS;
|
|
|
|
{$ELSE}
|
|
|
|
{$IFDEF DELPHI17_UP}
|
|
|
|
Result := TOSVersion.Architecture in [arIntelX86, arARM32];
|
|
|
|
{$ELSE}
|
|
|
|
Result := False;
|
|
|
|
{$ENDIF}
|
|
|
|
{$ENDIF}
|
2018-09-02 13:11:43 +02:00
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
function CustomPathIsRelative(const aPath : string) : boolean;
|
|
|
|
begin
|
2020-01-28 11:36:34 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
{$IFDEF DELPHI12_UP}
|
|
|
|
Result := PathIsRelativeUnicode(PChar(aPath));
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ELSE}
|
2020-01-28 11:36:34 +01:00
|
|
|
Result := PathIsRelativeAnsi(PChar(aPath));
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ELSE}
|
|
|
|
Result := (length(aPath) > 0) and (aPath[1] <> '/');
|
2018-04-18 10:25:38 +02:00
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2019-09-23 11:42:20 +02:00
|
|
|
function CustomPathIsURL(const aPath : string) : boolean;
|
|
|
|
begin
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
{$IFDEF DELPHI12_UP}
|
|
|
|
Result := PathIsURLUnicode(PChar(aPath + #0));
|
|
|
|
{$ELSE}
|
|
|
|
Result := PathIsURLAnsi(PChar(aPath + #0));
|
|
|
|
{$ENDIF}
|
|
|
|
{$ELSE}
|
|
|
|
Result := False;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CustomPathIsUNC(const aPath : string) : boolean;
|
|
|
|
begin
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
{$IFDEF DELPHI12_UP}
|
|
|
|
Result := PathIsUNCUnicode(PChar(aPath + #0));
|
|
|
|
{$ELSE}
|
|
|
|
Result := PathIsUNCAnsi(PChar(aPath + #0));
|
|
|
|
{$ENDIF}
|
|
|
|
{$ELSE}
|
|
|
|
Result := False;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2019-09-21 11:37:13 +02:00
|
|
|
function CustomPathCanonicalize(const aOriginalPath : string; var aCanonicalPath : string) : boolean;
|
|
|
|
var
|
|
|
|
TempBuffer: array [0..pred(MAX_PATH)] of Char;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
aCanonicalPath := '';
|
|
|
|
|
2019-09-23 11:42:20 +02:00
|
|
|
if (length(aOriginalPath) > MAX_PATH) or
|
|
|
|
(Copy(aOriginalPath, 1, 4) = '\\?\') or
|
|
|
|
CustomPathIsUNC(aOriginalPath) then
|
|
|
|
exit;
|
|
|
|
|
2019-09-21 11:37:13 +02:00
|
|
|
FillChar(TempBuffer, MAX_PATH * SizeOf(Char), 0);
|
|
|
|
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
{$IFDEF DELPHI12_UP}
|
2019-09-23 11:42:20 +02:00
|
|
|
if PathCanonicalizeUnicode(@TempBuffer[0], PChar(aOriginalPath + #0)) then
|
2019-09-21 11:37:13 +02:00
|
|
|
begin
|
|
|
|
aCanonicalPath := TempBuffer;
|
|
|
|
Result := True;
|
|
|
|
end;
|
|
|
|
{$ELSE}
|
2019-09-23 11:42:20 +02:00
|
|
|
if PathCanonicalizeAnsi(@TempBuffer[0], PChar(aOriginalPath + #0)) then
|
2019-09-21 11:37:13 +02:00
|
|
|
begin
|
|
|
|
aCanonicalPath := TempBuffer;
|
|
|
|
Result := True;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CustomAbsolutePath(const aPath : string; aMustExist : boolean) : string;
|
2019-11-10 18:23:39 +01:00
|
|
|
var
|
2019-09-25 17:23:16 +02:00
|
|
|
TempNewPath, TempOldPath : string;
|
2019-09-21 11:37:13 +02:00
|
|
|
begin
|
|
|
|
if (length(aPath) > 0) then
|
|
|
|
begin
|
2019-09-23 11:42:20 +02:00
|
|
|
if CustomPathIsRelative(aPath) then
|
2019-09-25 17:23:16 +02:00
|
|
|
TempOldPath := GetModulePath + aPath
|
2019-09-23 11:42:20 +02:00
|
|
|
else
|
2019-09-25 17:23:16 +02:00
|
|
|
TempOldPath := aPath;
|
|
|
|
|
|
|
|
if not(CustomPathCanonicalize(TempOldPath, TempNewPath)) then
|
|
|
|
TempNewPath := TempOldPath;
|
2019-09-23 11:42:20 +02:00
|
|
|
|
2019-09-25 17:23:16 +02:00
|
|
|
if aMustExist and not(DirectoryExists(TempNewPath)) then
|
2019-11-10 18:23:39 +01:00
|
|
|
Result := ''
|
|
|
|
else
|
2019-09-25 17:23:16 +02:00
|
|
|
Result := TempNewPath;
|
2019-09-21 11:37:13 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
function GetModulePath : string;
|
|
|
|
begin
|
2019-11-10 18:23:39 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-05-14 16:12:21 +02:00
|
|
|
Result := IncludeTrailingPathDelimiter(ExtractFileDir(GetModuleName(HINSTANCE{$IFDEF FPC}(){$ENDIF})));
|
2019-11-10 18:23:39 +01:00
|
|
|
{$ELSE}
|
|
|
|
// DLL filename not supported
|
|
|
|
Result := IncludeTrailingPathDelimiter(ExtractFileDir(ParamStr(0)));
|
|
|
|
{$ENDIF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefParseUrl(const url: ustring; var parts: TUrlParts): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempURL : TCefString;
|
|
|
|
TempParts : TCefUrlParts;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
FillChar(TempParts, sizeof(TempParts), 0);
|
|
|
|
TempURL := CefString(url);
|
|
|
|
Result := cef_parse_url(@TempURL, TempParts) <> 0;
|
|
|
|
|
|
|
|
if Result then
|
|
|
|
begin
|
2020-04-10 09:49:34 +02:00
|
|
|
parts.spec := CefString(@TempParts.spec);
|
2018-06-11 09:01:38 +02:00
|
|
|
parts.scheme := CefString(@TempParts.scheme);
|
|
|
|
parts.username := CefString(@TempParts.username);
|
|
|
|
parts.password := CefString(@TempParts.password);
|
|
|
|
parts.host := CefString(@TempParts.host);
|
|
|
|
parts.port := CefString(@TempParts.port);
|
|
|
|
parts.origin := CefString(@TempParts.origin);
|
|
|
|
parts.path := CefString(@TempParts.path);
|
|
|
|
parts.query := CefString(@TempParts.query);
|
2020-04-10 09:49:34 +02:00
|
|
|
parts.fragment := CefString(@TempParts.fragment);
|
2018-06-11 09:01:38 +02:00
|
|
|
end;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefCreateUrl(var parts: TUrlParts): ustring;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempURL : TCefString;
|
|
|
|
TempParts : TCefUrlParts;
|
|
|
|
begin
|
|
|
|
Result := '';
|
|
|
|
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempParts.spec := CefString(parts.spec);
|
|
|
|
TempParts.scheme := CefString(parts.scheme);
|
|
|
|
TempParts.username := CefString(parts.username);
|
|
|
|
TempParts.password := CefString(parts.password);
|
|
|
|
TempParts.host := CefString(parts.host);
|
|
|
|
TempParts.port := CefString(parts.port);
|
|
|
|
TempParts.origin := CefString(parts.origin);
|
|
|
|
TempParts.path := CefString(parts.path);
|
|
|
|
TempParts.query := CefString(parts.query);
|
2020-06-21 21:27:55 +02:00
|
|
|
TempParts.fragment := CefString(parts.fragment);
|
2018-06-11 09:01:38 +02:00
|
|
|
|
2019-11-24 18:19:49 +01:00
|
|
|
CefStringInitialize(@TempURL);
|
|
|
|
|
|
|
|
if (cef_create_url(@TempParts, @TempURL) <> 0) then
|
|
|
|
Result := CefStringClearAndGet(@TempURL);
|
2018-06-11 09:01:38 +02:00
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefFormatUrlForSecurityDisplay(const originUrl: string): string;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempOrigin : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempOrigin := CefString(originUrl);
|
|
|
|
Result := CefStringFreeAndGet(cef_format_url_for_security_display(@TempOrigin));
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefGetMimeType(const extension: ustring): ustring;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempExt : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
TempExt := CefString(extension);
|
|
|
|
Result := CefStringFreeAndGet(cef_get_mime_type(@TempExt));
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefGetExtensionsForMimeType(const mimeType: ustring; var extensions: TStringList);
|
|
|
|
var
|
|
|
|
TempSL : ICefStringList;
|
|
|
|
TempMimeType : TCefString;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (extensions <> nil) and (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
|
|
|
TempSL := TCefStringListOwn.Create;
|
|
|
|
TempMimeType := CefString(mimeType);
|
|
|
|
cef_get_extensions_for_mime_type(@TempMimeType, TempSL.Handle);
|
|
|
|
TempSL.CopyToStrings(extensions);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefBase64Encode(const data: Pointer; dataSize: NativeUInt): ustring;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
Result := CefStringFreeAndGet(cef_base64encode(data, dataSize))
|
|
|
|
else
|
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefBase64Decode(const data: ustring): ICefBinaryValue;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempData : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempData := CefString(data);
|
|
|
|
Result := TCefBinaryValueRef.UnWrap(cef_base64decode(@TempData));
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := nil;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefUriEncode(const text: ustring; usePlus: Boolean): ustring;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempText : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempText := CefString(text);
|
|
|
|
Result := CefStringFreeAndGet(cef_uriencode(@TempText, Ord(usePlus)));
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2018-06-11 09:01:38 +02:00
|
|
|
function CefUriDecode(const text: ustring; convertToUtf8: Boolean; unescapeRule: TCefUriUnescapeRule): ustring;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempText : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempText := CefString(text);
|
|
|
|
Result := CefStringFreeAndGet(cef_uridecode(@TempText, Ord(convertToUtf8), unescapeRule));
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2019-11-24 18:19:49 +01:00
|
|
|
function CefGetPath(const aPathKey : TCefPathKey) : ustring;
|
|
|
|
var
|
|
|
|
TempPath : TCefString;
|
|
|
|
begin
|
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
CefStringInitialize(@TempPath);
|
|
|
|
|
|
|
|
if (cef_get_path(aPathKey, @TempPath) <> 0) then
|
|
|
|
Result := CefStringClearAndGet(@TempPath);
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := '';
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefCreateDirectory(const fullPath: ustring): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPath : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempPath := CefString(fullPath);
|
|
|
|
Result := cef_create_directory(@TempPath) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefGetTempDirectory(out tempDir: ustring): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPath : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
CefStringInitialize(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
Result := cef_get_temp_directory(@TempPath) <> 0;
|
2019-11-24 18:19:49 +01:00
|
|
|
tempDir := CefStringClearAndGet(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
tempDir := '';
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefCreateNewTempDirectory(const prefix: ustring; out newTempPath: ustring): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPath, TempPref : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
CefStringInitialize(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPref := CefString(prefix);
|
|
|
|
Result := cef_create_new_temp_directory(@TempPref, @TempPath) <> 0;
|
2019-11-24 18:19:49 +01:00
|
|
|
newTempPath := CefStringClearAndGet(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
newTempPath := '';
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2018-06-11 09:01:38 +02:00
|
|
|
function CefCreateTempDirectoryInDirectory(const baseDir, prefix: ustring; out newDir: ustring): Boolean;
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempBase, TempPath, TempPref: TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
2019-11-24 18:19:49 +01:00
|
|
|
CefStringInitialize(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPref := CefString(prefix);
|
|
|
|
TempBase := CefString(baseDir);
|
|
|
|
Result := cef_create_temp_directory_in_directory(@TempBase, @TempPref, @TempPath) <> 0;
|
2019-11-24 18:19:49 +01:00
|
|
|
newDir := CefStringClearAndGet(@TempPath);
|
2018-06-11 09:01:38 +02:00
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
newDir := '';
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefDirectoryExists(const path: ustring): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPath : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempPath := CefString(path);
|
|
|
|
Result := cef_directory_exists(@TempPath) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefDeleteFile(const path: ustring; recursive: Boolean): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempPath : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempPath := CefString(path);
|
|
|
|
Result := cef_delete_file(@TempPath, Ord(recursive)) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function CefZipDirectory(const srcDir, destFile: ustring; includeHiddenFiles: Boolean): Boolean;
|
|
|
|
var
|
2018-06-11 09:01:38 +02:00
|
|
|
TempSrc, TempDst : TCefString;
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempSrc := CefString(srcDir);
|
|
|
|
TempDst := CefString(destFile);
|
|
|
|
Result := cef_zip_directory(@TempSrc, @TempDst, Ord(includeHiddenFiles)) <> 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := False;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure CefLoadCRLSetsFile(const path : ustring);
|
|
|
|
var
|
|
|
|
TempPath : TCefString;
|
|
|
|
begin
|
2018-06-11 09:01:38 +02:00
|
|
|
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
|
|
|
|
begin
|
|
|
|
TempPath := CefString(path);
|
|
|
|
cef_load_crlsets_file(@TempPath);
|
|
|
|
end;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
function CefIsKeyDown(aWparam : WPARAM) : boolean;
|
|
|
|
begin
|
|
|
|
Result := (GetKeyState(aWparam) < 0);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefIsKeyToggled(aWparam : WPARAM) : boolean;
|
|
|
|
begin
|
|
|
|
Result := (GetKeyState(aWparam) and $1) <> 0;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetCefMouseModifiers(aWparam : WPARAM) : TCefEventFlags;
|
|
|
|
begin
|
|
|
|
Result := EVENTFLAG_NONE;
|
|
|
|
|
|
|
|
if ((aWparam and MK_CONTROL) <> 0) then Result := Result or EVENTFLAG_CONTROL_DOWN;
|
|
|
|
if ((aWparam and MK_SHIFT) <> 0) then Result := Result or EVENTFLAG_SHIFT_DOWN;
|
|
|
|
if ((aWparam and MK_LBUTTON) <> 0) then Result := Result or EVENTFLAG_LEFT_MOUSE_BUTTON;
|
|
|
|
if ((aWparam and MK_MBUTTON) <> 0) then Result := Result or EVENTFLAG_MIDDLE_MOUSE_BUTTON;
|
|
|
|
if ((aWparam and MK_RBUTTON) <> 0) then Result := Result or EVENTFLAG_RIGHT_MOUSE_BUTTON;
|
|
|
|
if CefIsKeyDown(VK_MENU) then Result := Result or EVENTFLAG_ALT_DOWN;
|
|
|
|
if CefIsKeyToggled(VK_NUMLOCK) then Result := Result or EVENTFLAG_NUM_LOCK_ON;
|
|
|
|
if CefIsKeyToggled(VK_CAPITAL) then Result := Result or EVENTFLAG_CAPS_LOCK_ON;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetCefMouseModifiers : TCefEventFlags;
|
|
|
|
begin
|
|
|
|
Result := EVENTFLAG_NONE;
|
|
|
|
|
|
|
|
if CefIsKeyDown(MK_CONTROL) then Result := Result or EVENTFLAG_CONTROL_DOWN;
|
|
|
|
if CefIsKeyDown(MK_SHIFT) then Result := Result or EVENTFLAG_SHIFT_DOWN;
|
|
|
|
if CefIsKeyDown(MK_LBUTTON) then Result := Result or EVENTFLAG_LEFT_MOUSE_BUTTON;
|
|
|
|
if CefIsKeyDown(MK_MBUTTON) then Result := Result or EVENTFLAG_MIDDLE_MOUSE_BUTTON;
|
|
|
|
if CefIsKeyDown(MK_RBUTTON) then Result := Result or EVENTFLAG_RIGHT_MOUSE_BUTTON;
|
|
|
|
if CefIsKeyDown(VK_MENU) then Result := Result or EVENTFLAG_ALT_DOWN;
|
|
|
|
if CefIsKeyToggled(VK_NUMLOCK) then Result := Result or EVENTFLAG_NUM_LOCK_ON;
|
|
|
|
if CefIsKeyToggled(VK_CAPITAL) then Result := Result or EVENTFLAG_CAPS_LOCK_ON;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetCefKeyboardModifiers(aWparam : WPARAM; aLparam : LPARAM) : TCefEventFlags;
|
|
|
|
begin
|
|
|
|
Result := EVENTFLAG_NONE;
|
|
|
|
|
|
|
|
if CefIsKeyDown(VK_SHIFT) then Result := Result or EVENTFLAG_SHIFT_DOWN;
|
|
|
|
if CefIsKeyDown(VK_CONTROL) then Result := Result or EVENTFLAG_CONTROL_DOWN;
|
|
|
|
if CefIsKeyDown(VK_MENU) then Result := Result or EVENTFLAG_ALT_DOWN;
|
|
|
|
if CefIsKeyToggled(VK_NUMLOCK) then Result := Result or EVENTFLAG_NUM_LOCK_ON;
|
|
|
|
if CefIsKeyToggled(VK_CAPITAL) then Result := Result or EVENTFLAG_CAPS_LOCK_ON;
|
|
|
|
|
|
|
|
|
|
|
|
case aWparam of
|
|
|
|
VK_RETURN:
|
|
|
|
if (((aLparam shr 16) and KF_EXTENDED) <> 0) then Result := Result or EVENTFLAG_IS_KEY_PAD;
|
|
|
|
|
|
|
|
VK_INSERT,
|
|
|
|
VK_DELETE,
|
|
|
|
VK_HOME,
|
|
|
|
VK_END,
|
|
|
|
VK_PRIOR,
|
|
|
|
VK_NEXT,
|
|
|
|
VK_UP,
|
|
|
|
VK_DOWN,
|
|
|
|
VK_LEFT,
|
|
|
|
VK_RIGHT :
|
|
|
|
if (((aLparam shr 16) and KF_EXTENDED) = 0) then Result := Result or EVENTFLAG_IS_KEY_PAD;
|
|
|
|
|
|
|
|
VK_NUMLOCK,
|
|
|
|
VK_NUMPAD0,
|
|
|
|
VK_NUMPAD1,
|
|
|
|
VK_NUMPAD2,
|
|
|
|
VK_NUMPAD3,
|
|
|
|
VK_NUMPAD4,
|
|
|
|
VK_NUMPAD5,
|
|
|
|
VK_NUMPAD6,
|
|
|
|
VK_NUMPAD7,
|
|
|
|
VK_NUMPAD8,
|
|
|
|
VK_NUMPAD9,
|
|
|
|
VK_DIVIDE,
|
|
|
|
VK_MULTIPLY,
|
|
|
|
VK_SUBTRACT,
|
|
|
|
VK_ADD,
|
|
|
|
VK_DECIMAL,
|
|
|
|
VK_CLEAR :
|
|
|
|
Result := Result or EVENTFLAG_IS_KEY_PAD;
|
|
|
|
|
|
|
|
VK_SHIFT :
|
|
|
|
if CefIsKeyDown(VK_LSHIFT) then
|
|
|
|
Result := Result or EVENTFLAG_IS_LEFT
|
|
|
|
else
|
|
|
|
if CefIsKeyDown(VK_RSHIFT) then
|
|
|
|
Result := Result or EVENTFLAG_IS_RIGHT;
|
|
|
|
|
|
|
|
VK_CONTROL :
|
|
|
|
if CefIsKeyDown(VK_LCONTROL) then
|
|
|
|
Result := Result or EVENTFLAG_IS_LEFT
|
|
|
|
else
|
|
|
|
if CefIsKeyDown(VK_RCONTROL) then
|
|
|
|
Result := Result or EVENTFLAG_IS_RIGHT;
|
|
|
|
|
|
|
|
VK_MENU :
|
|
|
|
if CefIsKeyDown(VK_LMENU) then
|
|
|
|
Result := Result or EVENTFLAG_IS_LEFT
|
|
|
|
else
|
|
|
|
if CefIsKeyDown(VK_RMENU) then
|
|
|
|
Result := Result or EVENTFLAG_IS_RIGHT;
|
|
|
|
|
|
|
|
VK_LWIN :
|
|
|
|
Result := Result or EVENTFLAG_IS_LEFT;
|
|
|
|
|
|
|
|
VK_RWIN :
|
|
|
|
Result := Result or EVENTFLAG_IS_RIGHT;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2020-04-10 09:49:34 +02:00
|
|
|
procedure CefCheckAltGrPressed(aWparam : WPARAM; var aEvent : TCefKeyEvent);
|
|
|
|
const
|
|
|
|
EITHER_SHIFT_KEY_PRESSED = $01;
|
|
|
|
EITHER_CONTROL_KEY_PRESSED = $02;
|
|
|
|
EITHER_ALT_KEY_PRESSED = $04;
|
|
|
|
EITHER_HANKAKU_KEY_PRESSED = $08;
|
|
|
|
EITHER_RESERVED1_KEY_PRESSED = $10;
|
|
|
|
EITHER_RESERVED2_KEY_PRESSED = $20;
|
|
|
|
var
|
|
|
|
TempKBLayout : HKL;
|
|
|
|
TempTranslatedChar : SHORT;
|
|
|
|
TempShiftState : byte;
|
|
|
|
begin
|
|
|
|
if (aEvent.kind = KEYEVENT_CHAR) and CefIsKeyDown(VK_RMENU) then
|
|
|
|
begin
|
|
|
|
TempKBLayout := GetKeyboardLayout(0);
|
|
|
|
TempTranslatedChar := VkKeyScanEx(char(aWparam), TempKBLayout);
|
|
|
|
TempShiftState := byte(TempTranslatedChar shr 8);
|
|
|
|
|
|
|
|
if ((TempShiftState and EITHER_CONTROL_KEY_PRESSED) <> 0) and
|
|
|
|
((TempShiftState and EITHER_ALT_KEY_PRESSED) <> 0) then
|
|
|
|
begin
|
|
|
|
aEvent.modifiers := aEvent.modifiers and not(EVENTFLAG_CONTROL_DOWN or EVENTFLAG_ALT_DOWN);
|
|
|
|
aEvent.modifiers := aEvent.modifiers or EVENTFLAG_ALTGR_DOWN;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure DropEffectToDragOperation(aEffect: Longint; var aAllowedOps : TCefDragOperations);
|
|
|
|
begin
|
|
|
|
aAllowedOps := DRAG_OPERATION_NONE;
|
|
|
|
|
|
|
|
if ((aEffect and DROPEFFECT_COPY) <> 0) then aAllowedOps := aAllowedOps or DRAG_OPERATION_COPY;
|
|
|
|
if ((aEffect and DROPEFFECT_LINK) <> 0) then aAllowedOps := aAllowedOps or DRAG_OPERATION_LINK;
|
|
|
|
if ((aEffect and DROPEFFECT_MOVE) <> 0) then aAllowedOps := aAllowedOps or DRAG_OPERATION_MOVE;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure DragOperationToDropEffect(const aDragOperations : TCefDragOperations; var aEffect: Longint);
|
|
|
|
begin
|
|
|
|
aEffect := DROPEFFECT_NONE;
|
|
|
|
|
|
|
|
if ((aDragOperations and DRAG_OPERATION_COPY) <> 0) then aEffect := aEffect or DROPEFFECT_COPY;
|
|
|
|
if ((aDragOperations and DRAG_OPERATION_LINK) <> 0) then aEffect := aEffect or DROPEFFECT_LINK;
|
|
|
|
if ((aDragOperations and DRAG_OPERATION_MOVE) <> 0) then aEffect := aEffect or DROPEFFECT_MOVE;
|
2020-01-02 20:02:47 +01:00
|
|
|
end;
|
2020-01-13 15:39:44 +01:00
|
|
|
|
|
|
|
function GetWindowsMajorMinorVersion(var wMajorVersion, wMinorVersion : DWORD) : boolean;
|
2020-05-09 11:41:20 +02:00
|
|
|
type
|
|
|
|
TRtlGetVersionFunc = function(var lpVersionInformation : TOSVersionInfoEx): LongInt; stdcall;
|
2020-01-13 15:39:44 +01:00
|
|
|
var
|
2020-05-09 11:41:20 +02:00
|
|
|
TempHandle : THandle;
|
2020-01-13 15:39:44 +01:00
|
|
|
TempInfo : TOSVersionInfoEx;
|
2020-05-09 11:41:20 +02:00
|
|
|
TempRtlGetVersionFunc : TRtlGetVersionFunc;
|
2020-01-13 15:39:44 +01:00
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
wMajorVersion := 0;
|
|
|
|
wMinorVersion := 0;
|
|
|
|
|
2020-05-09 11:41:20 +02:00
|
|
|
try
|
|
|
|
TempHandle := LoadLibrary(NTDLL);
|
2020-01-13 15:39:44 +01:00
|
|
|
|
2020-05-09 11:41:20 +02:00
|
|
|
if (TempHandle <> 0) then
|
|
|
|
try
|
|
|
|
{$IFDEF FPC}Pointer({$ENDIF}TempRtlGetVersionFunc{$IFDEF FPC}){$ENDIF} := GetProcAddress(TempHandle, 'RtlGetVersion');
|
|
|
|
|
|
|
|
if assigned(TempRtlGetVersionFunc) then
|
|
|
|
begin
|
|
|
|
ZeroMemory(@TempInfo, SizeOf(TOSVersionInfoEx));
|
|
|
|
|
|
|
|
if (TempRtlGetVersionFunc(TempInfo) = 0) then
|
|
|
|
begin
|
|
|
|
Result := True;
|
|
|
|
wMajorVersion := TempInfo.dwMajorVersion;
|
|
|
|
wMinorVersion := TempInfo.dwMinorVersion;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
FreeLibrary(TempHandle);
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('GetWindowsMajorMinorVersion', e) then raise;
|
|
|
|
end;
|
2020-01-13 15:39:44 +01:00
|
|
|
end;
|
|
|
|
|
2020-10-31 14:23:06 +01:00
|
|
|
// GetDpiForWindow is only available in Windows 10 (version 1607) or newer
|
|
|
|
function GetDPIForHandle(aHandle : HWND; var aDPI : UINT) : boolean;
|
|
|
|
type
|
|
|
|
TGetDpiForWindow = function(hwnd: HWND): UINT; stdcall;
|
|
|
|
var
|
|
|
|
TempHandle : THandle;
|
|
|
|
TempGetDpiForWindowFunc : TGetDpiForWindow;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
aDPI := 0;
|
|
|
|
|
|
|
|
if (aHandle = 0) then exit;
|
|
|
|
|
|
|
|
try
|
|
|
|
TempHandle := LoadLibrary(User32DLL);
|
|
|
|
|
|
|
|
if (TempHandle <> 0) then
|
|
|
|
try
|
|
|
|
{$IFDEF FPC}Pointer({$ENDIF}TempGetDpiForWindowFunc{$IFDEF FPC}){$ENDIF} := GetProcAddress(TempHandle, 'GetDpiForWindow');
|
|
|
|
|
|
|
|
if assigned(TempGetDpiForWindowFunc) then
|
|
|
|
begin
|
|
|
|
aDPI := TempGetDpiForWindowFunc(aHandle);
|
|
|
|
Result := (aDPI <> 0);
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
FreeLibrary(TempHandle);
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('GetDPIForHandle', e) then raise;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function RunningWindows10OrNewer : boolean;
|
|
|
|
var
|
|
|
|
TempMajorVer, TempMinorVer : DWORD;
|
|
|
|
begin
|
|
|
|
Result := GetWindowsMajorMinorVersion(TempMajorVer, TempMinorVer) and (TempMajorVer >= 10);
|
|
|
|
end;
|
|
|
|
|
2020-01-13 15:39:44 +01:00
|
|
|
function GetDefaultCEFUserAgent : string;
|
|
|
|
var
|
|
|
|
TempOS, TempChromiumVersion : string;
|
|
|
|
TempMajorVer, TempMinorVer : DWORD;
|
|
|
|
Temp64bit : BOOL;
|
|
|
|
begin
|
|
|
|
if GetWindowsMajorMinorVersion(TempMajorVer, TempMinorVer) and
|
|
|
|
(TempMajorVer >= 4) then
|
|
|
|
TempOS := 'Windows NT'
|
|
|
|
else
|
|
|
|
TempOS := 'Windows';
|
|
|
|
|
|
|
|
TempOS := TempOS + ' ' + inttostr(TempMajorVer) + '.' + inttostr(TempMinorVer);
|
|
|
|
|
|
|
|
if ProcessUnderWow64(GetCurrentProcess(), Temp64bit) and Temp64bit then
|
|
|
|
TempOS := TempOS + '; WOW64';
|
|
|
|
|
|
|
|
if (GlobalCEFApp <> nil) then
|
|
|
|
TempChromiumVersion := GlobalCEFApp.ChromeVersion
|
|
|
|
else
|
|
|
|
TempChromiumVersion := inttostr(CEF_CHROMEELF_VERSION_MAJOR) + '.' +
|
|
|
|
inttostr(CEF_CHROMEELF_VERSION_MINOR) + '.' +
|
|
|
|
inttostr(CEF_CHROMEELF_VERSION_RELEASE) + '.' +
|
|
|
|
inttostr(CEF_CHROMEELF_VERSION_BUILD);
|
|
|
|
|
|
|
|
Result := 'Mozilla/5.0' + ' (' + TempOS + ') ' +
|
|
|
|
'AppleWebKit/537.36 (KHTML, like Gecko) ' +
|
|
|
|
'Chrome/' + TempChromiumVersion + ' Safari/537.36';
|
|
|
|
end;
|
2020-02-03 16:14:31 +01:00
|
|
|
|
2020-02-04 11:50:38 +01:00
|
|
|
{$IFDEF DELPHI14_UP}
|
2020-02-03 16:14:31 +01:00
|
|
|
function TouchPointToPoint(aHandle : HWND; const TouchPoint: TTouchInput): TPoint;
|
2020-02-03 12:02:30 +01:00
|
|
|
begin
|
|
|
|
Result := Point(TouchPoint.X div 100, TouchPoint.Y div 100);
|
|
|
|
PhysicalToLogicalPoint(aHandle, Result);
|
|
|
|
end;
|
2020-02-05 14:40:22 +01:00
|
|
|
|
|
|
|
function GetDigitizerStatus(var aDigitizerStatus : TDigitizerStatus; aDPI : cardinal) : boolean;
|
|
|
|
var
|
|
|
|
TempStatus : integer;
|
|
|
|
begin
|
2020-02-06 10:37:54 +01:00
|
|
|
{$IFDEF DELPHI26_UP}
|
2020-02-05 14:40:22 +01:00
|
|
|
if (aDPI > 0) then
|
|
|
|
TempStatus := GetSystemMetricsForDpi(SM_DIGITIZER, aDPI)
|
|
|
|
else
|
2020-02-06 10:37:54 +01:00
|
|
|
{$ENDIF}
|
2020-02-05 14:40:22 +01:00
|
|
|
TempStatus := GetSystemMetrics(SM_DIGITIZER);
|
|
|
|
|
|
|
|
aDigitizerStatus.IntegratedTouch := ((TempStatus and NID_INTEGRATED_TOUCH) <> 0);
|
|
|
|
aDigitizerStatus.ExternalTouch := ((TempStatus and NID_EXTERNAL_TOUCH) <> 0);
|
|
|
|
aDigitizerStatus.IntegratedPen := ((TempStatus and NID_INTEGRATED_PEN) <> 0);
|
|
|
|
aDigitizerStatus.ExternalPen := ((TempStatus and NID_EXTERNAL_PEN) <> 0);
|
|
|
|
aDigitizerStatus.MultiInput := ((TempStatus and NID_MULTI_INPUT) <> 0);
|
|
|
|
aDigitizerStatus.Ready := ((TempStatus and NID_READY) <> 0);
|
|
|
|
|
|
|
|
Result := (TempStatus <> 0);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function HasTouchOrPen(aDPI : cardinal) : boolean;
|
|
|
|
var
|
|
|
|
TempStatus : TDigitizerStatus;
|
|
|
|
begin
|
|
|
|
Result := GetDigitizerStatus(TempStatus, aDPI);
|
|
|
|
end;
|
2020-02-06 10:37:54 +01:00
|
|
|
{$ENDIF}
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
|
|
|
|
function DeviceToLogical(aValue : integer; const aDeviceScaleFactor : double) : integer;
|
|
|
|
begin
|
|
|
|
Result := floor(aValue / aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
2020-02-03 12:02:30 +01:00
|
|
|
function DeviceToLogical(aValue : single; const aDeviceScaleFactor : double) : single;
|
|
|
|
begin
|
|
|
|
Result := aValue / aDeviceScaleFactor;
|
|
|
|
end;
|
|
|
|
|
2018-04-18 10:25:38 +02:00
|
|
|
procedure DeviceToLogical(var aEvent : TCEFMouseEvent; const aDeviceScaleFactor : double);
|
2020-02-03 12:02:30 +01:00
|
|
|
begin
|
|
|
|
aEvent.x := DeviceToLogical(aEvent.x, aDeviceScaleFactor);
|
|
|
|
aEvent.y := DeviceToLogical(aEvent.y, aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure DeviceToLogical(var aEvent : TCefTouchEvent; const aDeviceScaleFactor : double);
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
|
|
|
aEvent.x := DeviceToLogical(aEvent.x, aDeviceScaleFactor);
|
|
|
|
aEvent.y := DeviceToLogical(aEvent.y, aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure DeviceToLogical(var aPoint : TPoint; const aDeviceScaleFactor : double);
|
|
|
|
begin
|
|
|
|
aPoint.x := DeviceToLogical(aPoint.x, aDeviceScaleFactor);
|
|
|
|
aPoint.y := DeviceToLogical(aPoint.y, aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function LogicalToDevice(aValue : integer; const aDeviceScaleFactor : double) : integer;
|
|
|
|
begin
|
|
|
|
Result := floor(aValue * aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure LogicalToDevice(var aRect : TCEFRect; const aDeviceScaleFactor : double);
|
|
|
|
begin
|
|
|
|
aRect.x := LogicalToDevice(aRect.x, aDeviceScaleFactor);
|
|
|
|
aRect.y := LogicalToDevice(aRect.y, aDeviceScaleFactor);
|
|
|
|
aRect.width := LogicalToDevice(aRect.width, aDeviceScaleFactor);
|
|
|
|
aRect.height := LogicalToDevice(aRect.height, aDeviceScaleFactor);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetScreenDPI : integer;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
var
|
|
|
|
TempDC : HDC;
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
begin
|
2019-01-08 19:15:25 +01:00
|
|
|
{$IFDEF MSWINDOWS}
|
2018-04-18 10:25:38 +02:00
|
|
|
TempDC := GetDC(0);
|
|
|
|
Result := GetDeviceCaps(TempDC, LOGPIXELSX);
|
|
|
|
ReleaseDC(0, TempDC);
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ELSE}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$IFDEF MACOS}
|
|
|
|
Result := trunc(MainScreen.backingScaleFactor);
|
|
|
|
{$ELSE}
|
2020-12-18 16:51:02 +01:00
|
|
|
{$IFDEF FPC}
|
2020-12-28 18:11:27 +01:00
|
|
|
if (Application <> nil) and
|
|
|
|
(Application.MainForm <> nil) and
|
|
|
|
(Application.MainForm.Monitor <> nil) then
|
|
|
|
Result := Application.MainForm.Monitor.PixelsPerInch
|
|
|
|
else
|
|
|
|
if (screen <> nil) then
|
|
|
|
begin
|
|
|
|
if (WidgetSet <> nil) and (screen.PrimaryMonitor <> nil) then
|
|
|
|
Result := screen.PrimaryMonitor.PixelsPerInch
|
|
|
|
else
|
|
|
|
Result := screen.PixelsPerInch;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := USER_DEFAULT_SCREEN_DPI;
|
2020-12-18 16:51:02 +01:00
|
|
|
{$ELSE}
|
|
|
|
// TODO: Find a way to get the screen scale in Delphi FMX for Linux
|
2020-12-28 18:11:27 +01:00
|
|
|
Result := USER_DEFAULT_SCREEN_DPI;
|
2020-12-18 16:51:02 +01:00
|
|
|
{$ENDIF}
|
2020-01-28 11:36:34 +01:00
|
|
|
{$ENDIF}
|
2019-01-08 19:15:25 +01:00
|
|
|
{$ENDIF}
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function GetDeviceScaleFactor : single;
|
|
|
|
begin
|
2020-10-31 14:23:06 +01:00
|
|
|
Result := GetScreenDPI / USER_DEFAULT_SCREEN_DPI;
|
2018-04-18 10:25:38 +02:00
|
|
|
end;
|
|
|
|
|
2019-06-19 16:53:26 +02:00
|
|
|
function DeleteDirContents(const aDirectory : string; const aExcludeFiles : TStringList) : boolean;
|
2018-09-18 16:19:21 +02:00
|
|
|
var
|
|
|
|
TempRec : TSearchRec;
|
|
|
|
TempPath : string;
|
2019-06-19 16:53:26 +02:00
|
|
|
TempIdx : integer;
|
2018-09-18 16:19:21 +02:00
|
|
|
begin
|
|
|
|
Result := True;
|
|
|
|
|
|
|
|
try
|
|
|
|
if (length(aDirectory) > 0) and
|
|
|
|
DirectoryExists(aDirectory) and
|
|
|
|
(FindFirst(aDirectory + '\*', faAnyFile, TempRec) = 0) then
|
|
|
|
try
|
|
|
|
repeat
|
|
|
|
TempPath := aDirectory + PathDelim + TempRec.Name;
|
|
|
|
|
|
|
|
if ((TempRec.Attr and faDirectory) <> 0) then
|
|
|
|
begin
|
|
|
|
if (TempRec.Name <> '.') and (TempRec.Name <> '..') then
|
|
|
|
begin
|
2019-06-19 16:53:26 +02:00
|
|
|
if DeleteDirContents(TempPath, aExcludeFiles) then
|
2018-09-18 16:19:21 +02:00
|
|
|
Result := RemoveDir(TempPath) and Result
|
|
|
|
else
|
|
|
|
Result := False;
|
|
|
|
end;
|
|
|
|
end
|
|
|
|
else
|
2019-06-19 16:53:26 +02:00
|
|
|
if (aExcludeFiles <> nil) then
|
|
|
|
begin
|
|
|
|
TempIdx := aExcludeFiles.IndexOf(TempRec.Name);
|
|
|
|
Result := ((TempIdx >= 0) or
|
|
|
|
((TempIdx < 0) and DeleteFile(TempPath))) and
|
|
|
|
Result;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
Result := DeleteFile(TempPath) and Result;
|
|
|
|
|
2018-09-18 16:19:21 +02:00
|
|
|
until (FindNext(TempRec) <> 0) or not(Result);
|
|
|
|
finally
|
|
|
|
FindClose(TempRec);
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('DeleteDirContents', e) then raise;
|
|
|
|
end;
|
2018-09-18 15:19:44 +02:00
|
|
|
end;
|
|
|
|
|
2019-06-19 16:53:26 +02:00
|
|
|
function DeleteFileList(const aFileList : TStringList) : boolean;
|
|
|
|
var
|
|
|
|
i, TempCount : integer;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
|
|
|
if (aFileList <> nil) then
|
|
|
|
begin
|
|
|
|
i := 0;
|
|
|
|
TempCount := 0;
|
|
|
|
|
|
|
|
while (i < aFileList.Count) do
|
|
|
|
begin
|
|
|
|
if FileExists(aFileList[i]) and DeleteFile(aFileList[i]) then inc(TempCount);
|
|
|
|
inc(i);
|
|
|
|
end;
|
|
|
|
|
|
|
|
Result := (aFileList.Count = TempCount);
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('DeleteFileList', e) then raise;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function MoveFileList(const aFileList : TStringList; const aSrcDirectory, aDstDirectory : string) : boolean;
|
|
|
|
var
|
|
|
|
i, TempCount : integer;
|
|
|
|
TempSrcPath, TempDstPath : string;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
try
|
|
|
|
if (aFileList <> nil) and
|
|
|
|
(length(aSrcDirectory) > 0) and
|
2019-11-10 18:23:39 +01:00
|
|
|
(length(aDstDirectory) > 0) and
|
|
|
|
DirectoryExists(aSrcDirectory) and
|
2019-06-19 16:53:26 +02:00
|
|
|
(DirectoryExists(aDstDirectory) or CreateDir(aDstDirectory)) then
|
|
|
|
begin
|
|
|
|
i := 0;
|
|
|
|
TempCount := 0;
|
|
|
|
|
|
|
|
while (i < aFileList.Count) do
|
|
|
|
begin
|
2019-11-10 18:23:39 +01:00
|
|
|
TempSrcPath := IncludeTrailingPathDelimiter(aSrcDirectory) + aFileList[i];
|
|
|
|
TempDstPath := IncludeTrailingPathDelimiter(aDstDirectory) + aFileList[i];
|
|
|
|
|
2019-06-19 16:53:26 +02:00
|
|
|
if FileExists(TempSrcPath) and RenameFile(TempSrcPath, TempDstPath) then inc(TempCount);
|
|
|
|
|
|
|
|
inc(i);
|
|
|
|
end;
|
|
|
|
|
|
|
|
Result := (aFileList.Count = TempCount);
|
|
|
|
end;
|
|
|
|
except
|
|
|
|
on e : exception do
|
|
|
|
if CustomExceptionHandler('MoveFileList', e) then raise;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2019-10-13 18:50:23 +02:00
|
|
|
function CefGetDataURI(const aString, aMimeType : ustring) : ustring;
|
|
|
|
var
|
|
|
|
TempUTF : AnsiString;
|
|
|
|
begin
|
|
|
|
TempUTF := UTF8Encode(aString);
|
|
|
|
|
|
|
|
if (length(TempUTF) > 0) then
|
|
|
|
Result := CefGetDataURI(@TempUTF[1], length(TempUTF), aMimeType, 'utf-8')
|
|
|
|
else
|
|
|
|
Result := '';
|
|
|
|
end;
|
|
|
|
|
|
|
|
function CefGetDataURI(aData : pointer; aSize : integer; const aMimeType, aCharset : ustring) : ustring;
|
|
|
|
begin
|
|
|
|
Result := 'data:' + aMimeType;
|
|
|
|
|
|
|
|
if (length(aCharset) > 0) then Result := Result + ';charset=' + aCharset;
|
|
|
|
|
|
|
|
Result := Result + ';base64,' + CefURIEncode(CefBase64Encode(aData, aSize), false);
|
|
|
|
end;
|
2020-02-03 16:14:31 +01:00
|
|
|
|
2020-04-03 17:57:52 +02:00
|
|
|
function ValidCefWindowHandle(aHandle : TCefWindowHandle) : boolean;
|
|
|
|
begin
|
|
|
|
{$IFDEF MACOS}
|
|
|
|
Result := (aHandle <> nil);
|
|
|
|
{$ELSE}
|
|
|
|
Result := (aHandle <> 0);
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2020-05-05 18:10:33 +02:00
|
|
|
procedure InitializeWindowHandle(var aHandle : TCefWindowHandle);
|
|
|
|
begin
|
|
|
|
{$IFDEF MACOS}
|
|
|
|
aHandle := nil;
|
|
|
|
{$ELSE}
|
|
|
|
aHandle := 0;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2020-12-28 18:11:27 +01:00
|
|
|
{$IFDEF LINUX}{$IFDEF FPC}{$IFDEF LCLGTK2}
|
|
|
|
function KeyboardCodeFromXKeysym(keysym : cuint) : integer;
|
|
|
|
begin
|
|
|
|
case keysym of
|
|
|
|
XK_BackSpace:
|
|
|
|
Result := VKEY_BACK;
|
|
|
|
XK_Delete,
|
|
|
|
XK_KP_Delete:
|
|
|
|
Result := VKEY_DELETE;
|
|
|
|
XK_Tab,
|
|
|
|
XK_KP_Tab,
|
|
|
|
XK_ISO_Left_Tab,
|
|
|
|
XK_3270_BackTab:
|
|
|
|
Result := VKEY_TAB;
|
|
|
|
XK_Linefeed,
|
|
|
|
XK_Return,
|
|
|
|
XK_KP_Enter,
|
|
|
|
XK_ISO_Enter:
|
|
|
|
Result := VKEY_Return;
|
|
|
|
XK_Clear,
|
|
|
|
XK_KP_Begin:
|
|
|
|
Result := VKEY_CLEAR;
|
|
|
|
XK_KP_Space,
|
|
|
|
XK_space:
|
|
|
|
Result := VKEY_SPACE;
|
|
|
|
XK_Home,
|
|
|
|
XK_KP_Home:
|
|
|
|
Result := VKEY_HOME;
|
|
|
|
XK_End,
|
|
|
|
XK_KP_End:
|
|
|
|
Result := VKEY_END;
|
|
|
|
XK_Page_Up,
|
|
|
|
XK_KP_Page_Up:
|
|
|
|
Result := VKEY_PRIOR;
|
|
|
|
XK_Page_Down,
|
|
|
|
XK_KP_Page_Down:
|
|
|
|
Result := VKEY_NEXT;
|
|
|
|
XK_Left,
|
|
|
|
XK_KP_Left:
|
|
|
|
Result := VKEY_LEFT;
|
|
|
|
XK_Right,
|
|
|
|
XK_KP_Right:
|
|
|
|
Result := VKEY_RIGHT;
|
|
|
|
XK_Down,
|
|
|
|
XK_KP_Down:
|
|
|
|
Result := VKEY_DOWN;
|
|
|
|
XK_Up,
|
|
|
|
XK_KP_Up:
|
|
|
|
Result := VKEY_UP;
|
|
|
|
XK_Escape:
|
|
|
|
Result := VKEY_ESCAPE;
|
|
|
|
XK_Kana_Lock,
|
|
|
|
XK_Kana_Shift:
|
|
|
|
Result := VKEY_KANA;
|
|
|
|
XK_Hangul:
|
|
|
|
Result := VKEY_HANGUL;
|
|
|
|
XK_Hangul_Hanja:
|
|
|
|
Result := VKEY_HANJA;
|
|
|
|
XK_Kanji:
|
|
|
|
Result := VKEY_KANJI;
|
|
|
|
XK_Henkan:
|
|
|
|
Result := VKEY_CONVERT;
|
|
|
|
XK_Muhenkan:
|
|
|
|
Result := VKEY_NONCONVERT;
|
|
|
|
XK_Zenkaku_Hankaku:
|
|
|
|
Result := VKEY_DBE_DBCSCHAR;
|
|
|
|
XKc_A,
|
|
|
|
XK_a:
|
|
|
|
Result := VKEY_A;
|
|
|
|
XKc_B,
|
|
|
|
XK_b:
|
|
|
|
Result := VKEY_B;
|
|
|
|
XKc_C,
|
|
|
|
XK_c:
|
|
|
|
Result := VKEY_C;
|
|
|
|
XKc_D,
|
|
|
|
XK_d:
|
|
|
|
Result := VKEY_D;
|
|
|
|
XKc_E,
|
|
|
|
XK_e:
|
|
|
|
Result := VKEY_E;
|
|
|
|
XKc_F,
|
|
|
|
XK_f:
|
|
|
|
Result := VKEY_F;
|
|
|
|
XKc_G,
|
|
|
|
XK_g:
|
|
|
|
Result := VKEY_G;
|
|
|
|
XKc_H,
|
|
|
|
XK_h:
|
|
|
|
Result := VKEY_H;
|
|
|
|
XKc_I,
|
|
|
|
XK_i:
|
|
|
|
Result := VKEY_I;
|
|
|
|
XKc_J,
|
|
|
|
XK_j:
|
|
|
|
Result := VKEY_J;
|
|
|
|
XKc_K,
|
|
|
|
XK_k:
|
|
|
|
Result := VKEY_K;
|
|
|
|
XKc_L,
|
|
|
|
XK_l:
|
|
|
|
Result := VKEY_L;
|
|
|
|
XKc_M,
|
|
|
|
XK_m:
|
|
|
|
Result := VKEY_M;
|
|
|
|
XKc_N,
|
|
|
|
XK_n:
|
|
|
|
Result := VKEY_N;
|
|
|
|
XKc_O,
|
|
|
|
XK_o:
|
|
|
|
Result := VKEY_O;
|
|
|
|
XKc_P,
|
|
|
|
XK_p:
|
|
|
|
Result := VKEY_P;
|
|
|
|
XKc_Q,
|
|
|
|
XK_q:
|
|
|
|
Result := VKEY_Q;
|
|
|
|
XKc_R,
|
|
|
|
XK_r:
|
|
|
|
Result := VKEY_R;
|
|
|
|
XKc_S,
|
|
|
|
XK_s:
|
|
|
|
Result := VKEY_S;
|
|
|
|
XKc_T,
|
|
|
|
XK_t:
|
|
|
|
Result := VKEY_T;
|
|
|
|
XKc_U,
|
|
|
|
XK_u:
|
|
|
|
Result := VKEY_U;
|
|
|
|
XKc_V,
|
|
|
|
XK_v:
|
|
|
|
Result := VKEY_V;
|
|
|
|
XKc_W,
|
|
|
|
XK_w:
|
|
|
|
Result := VKEY_W;
|
|
|
|
XKc_X,
|
|
|
|
XK_x:
|
|
|
|
Result := VKEY_X;
|
|
|
|
XKc_Y,
|
|
|
|
XK_y:
|
|
|
|
Result := VKEY_Y;
|
|
|
|
XKc_Z,
|
|
|
|
XK_z:
|
|
|
|
Result := VKEY_Z;
|
|
|
|
XK_0,
|
|
|
|
XK_1,
|
|
|
|
XK_2,
|
|
|
|
XK_3,
|
|
|
|
XK_4,
|
|
|
|
XK_5,
|
|
|
|
XK_6,
|
|
|
|
XK_7,
|
|
|
|
XK_8,
|
|
|
|
XK_9:
|
|
|
|
Result := VKEY_0 + (keysym - XK_0);
|
|
|
|
XK_parenright:
|
|
|
|
Result := VKEY_0;
|
|
|
|
XK_exclam:
|
|
|
|
Result := VKEY_1;
|
|
|
|
XK_at:
|
|
|
|
Result := VKEY_2;
|
|
|
|
XK_numbersign:
|
|
|
|
Result := VKEY_3;
|
|
|
|
XK_dollar:
|
|
|
|
Result := VKEY_4;
|
|
|
|
XK_percent:
|
|
|
|
Result := VKEY_5;
|
|
|
|
XK_asciicircum:
|
|
|
|
Result := VKEY_6;
|
|
|
|
XK_ampersand:
|
|
|
|
Result := VKEY_7;
|
|
|
|
XK_asterisk:
|
|
|
|
Result := VKEY_8;
|
|
|
|
XK_parenleft:
|
|
|
|
Result := VKEY_9;
|
|
|
|
XK_KP_0,
|
|
|
|
XK_KP_1,
|
|
|
|
XK_KP_2,
|
|
|
|
XK_KP_3,
|
|
|
|
XK_KP_4,
|
|
|
|
XK_KP_5,
|
|
|
|
XK_KP_6,
|
|
|
|
XK_KP_7,
|
|
|
|
XK_KP_8,
|
|
|
|
XK_KP_9:
|
|
|
|
Result := VKEY_NUMPAD0 + (keysym - XK_KP_0);
|
|
|
|
XK_multiply,
|
|
|
|
XK_KP_Multiply:
|
|
|
|
Result := VKEY_MULTIPLY;
|
|
|
|
XK_KP_Add:
|
|
|
|
Result := VKEY_ADD;
|
|
|
|
XK_KP_Separator:
|
|
|
|
Result := VKEY_SEPARATOR;
|
|
|
|
XK_KP_Subtract:
|
|
|
|
Result := VKEY_SUBTRACT;
|
|
|
|
XK_KP_Decimal:
|
|
|
|
Result := VKEY_DECIMAL;
|
|
|
|
XK_KP_Divide:
|
|
|
|
Result := VKEY_DIVIDE;
|
|
|
|
XK_KP_Equal,
|
|
|
|
XK_equal,
|
|
|
|
XK_plus:
|
|
|
|
Result := VKEY_OEM_PLUS;
|
|
|
|
XK_comma,
|
|
|
|
XK_less:
|
|
|
|
Result := VKEY_OEM_COMMA;
|
|
|
|
XK_minus,
|
|
|
|
XK_underscore:
|
|
|
|
Result := VKEY_OEM_MINUS;
|
|
|
|
XK_greater,
|
|
|
|
XK_period:
|
|
|
|
Result := VKEY_OEM_PERIOD;
|
|
|
|
XK_colon,
|
|
|
|
XK_semicolon:
|
|
|
|
Result := VKEY_OEM_1;
|
|
|
|
XK_question,
|
|
|
|
XK_slash:
|
|
|
|
Result := VKEY_OEM_2;
|
|
|
|
XK_asciitilde,
|
|
|
|
XK_quoteleft:
|
|
|
|
Result := VKEY_OEM_3;
|
|
|
|
XK_bracketleft,
|
|
|
|
XK_braceleft:
|
|
|
|
Result := VKEY_OEM_4;
|
|
|
|
XK_backslash,
|
|
|
|
XK_bar:
|
|
|
|
Result := VKEY_OEM_5;
|
|
|
|
XK_bracketright,
|
|
|
|
XK_braceright:
|
|
|
|
Result := VKEY_OEM_6;
|
|
|
|
XK_quoteright,
|
|
|
|
XK_quotedbl:
|
|
|
|
Result := VKEY_OEM_7;
|
|
|
|
XK_ISO_Level5_Shift:
|
|
|
|
Result := VKEY_OEM_8;
|
|
|
|
XK_Shift_L,
|
|
|
|
XK_Shift_R:
|
|
|
|
Result := VKEY_SHIFT;
|
|
|
|
XK_Control_L,
|
|
|
|
XK_Control_R:
|
|
|
|
Result := VKEY_CONTROL;
|
|
|
|
XK_Meta_L,
|
|
|
|
XK_Meta_R,
|
|
|
|
XK_Alt_L,
|
|
|
|
XK_Alt_R:
|
|
|
|
Result := VKEY_MENU;
|
|
|
|
XK_ISO_Level3_Shift:
|
|
|
|
Result := VKEY_ALTGR;
|
|
|
|
XK_Multi_key:
|
|
|
|
Result := VKEY_COMPOSE;
|
|
|
|
XK_Pause:
|
|
|
|
Result := VKEY_PAUSE;
|
|
|
|
XK_Caps_Lock:
|
|
|
|
Result := VKEY_CAPITAL;
|
|
|
|
XK_Num_Lock:
|
|
|
|
Result := VKEY_NUMLOCK;
|
|
|
|
XK_Scroll_Lock:
|
|
|
|
Result := VKEY_SCROLL;
|
|
|
|
XK_Select:
|
|
|
|
Result := VKEY_SELECT;
|
|
|
|
XK_Print:
|
|
|
|
Result := VKEY_PRINT;
|
|
|
|
XK_Execute:
|
|
|
|
Result := VKEY_EXECUTE;
|
|
|
|
XK_Insert,
|
|
|
|
XK_KP_Insert:
|
|
|
|
Result := VKEY_INSERT;
|
|
|
|
XK_Help:
|
|
|
|
Result := VKEY_HELP;
|
|
|
|
XK_Super_L:
|
|
|
|
Result := VKEY_LWIN;
|
|
|
|
XK_Super_R:
|
|
|
|
Result := VKEY_RWIN;
|
|
|
|
XK_Menu:
|
|
|
|
Result := VKEY_APPS;
|
|
|
|
XK_F1,
|
|
|
|
XK_F2,
|
|
|
|
XK_F3,
|
|
|
|
XK_F4,
|
|
|
|
XK_F5,
|
|
|
|
XK_F6,
|
|
|
|
XK_F7,
|
|
|
|
XK_F8,
|
|
|
|
XK_F9,
|
|
|
|
XK_F10,
|
|
|
|
XK_F11,
|
|
|
|
XK_F12,
|
|
|
|
XK_F13,
|
|
|
|
XK_F14,
|
|
|
|
XK_F15,
|
|
|
|
XK_F16,
|
|
|
|
XK_F17,
|
|
|
|
XK_F18,
|
|
|
|
XK_F19,
|
|
|
|
XK_F20,
|
|
|
|
XK_F21,
|
|
|
|
XK_F22,
|
|
|
|
XK_F23,
|
|
|
|
XK_F24:
|
|
|
|
Result := VKEY_F1 + (keysym - XK_F1);
|
|
|
|
XK_KP_F1,
|
|
|
|
XK_KP_F2,
|
|
|
|
XK_KP_F3,
|
|
|
|
XK_KP_F4:
|
|
|
|
Result := VKEY_F1 + (keysym - XK_KP_F1);
|
|
|
|
XK_guillemotleft,
|
|
|
|
XK_guillemotright,
|
|
|
|
XK_degree,
|
|
|
|
XK_ugrave,
|
|
|
|
XKc_Ugrave,
|
|
|
|
XK_brokenbar:
|
|
|
|
Result := VKEY_OEM_102;
|
|
|
|
XF86XK_Tools:
|
|
|
|
Result := VKEY_F13;
|
|
|
|
XF86XK_Launch5:
|
|
|
|
Result := VKEY_F14;
|
|
|
|
XF86XK_Launch6:
|
|
|
|
Result := VKEY_F15;
|
|
|
|
XF86XK_Launch7:
|
|
|
|
Result := VKEY_F16;
|
|
|
|
XF86XK_Launch8:
|
|
|
|
Result := VKEY_F17;
|
|
|
|
XF86XK_Launch9:
|
|
|
|
Result := VKEY_F18;
|
|
|
|
XF86XK_Refresh,
|
|
|
|
XF86XK_History,
|
|
|
|
XF86XK_OpenURL,
|
|
|
|
XF86XK_AddFavorite,
|
|
|
|
XF86XK_Go,
|
|
|
|
XF86XK_ZoomIn,
|
|
|
|
XF86XK_ZoomOut:
|
|
|
|
Result := VKEY_UNKNOWN;
|
|
|
|
XF86XK_Back:
|
|
|
|
Result := VKEY_BROWSER_BACK;
|
|
|
|
XF86XK_Forward:
|
|
|
|
Result := VKEY_BROWSER_FORWARD;
|
|
|
|
XF86XK_Reload:
|
|
|
|
Result := VKEY_BROWSER_REFRESH;
|
|
|
|
XF86XK_Stop:
|
|
|
|
Result := VKEY_BROWSER_STOP;
|
|
|
|
XF86XK_Search:
|
|
|
|
Result := VKEY_BROWSER_SEARCH;
|
|
|
|
XF86XK_Favorites:
|
|
|
|
Result := VKEY_BROWSER_FAVORITES;
|
|
|
|
XF86XK_HomePage:
|
|
|
|
Result := VKEY_BROWSER_HOME;
|
|
|
|
XF86XK_AudioMute:
|
|
|
|
Result := VKEY_VOLUME_MUTE;
|
|
|
|
XF86XK_AudioLowerVolume:
|
|
|
|
Result := VKEY_VOLUME_DOWN;
|
|
|
|
XF86XK_AudioRaiseVolume:
|
|
|
|
Result := VKEY_VOLUME_UP;
|
|
|
|
XF86XK_AudioNext:
|
|
|
|
Result := VKEY_MEDIA_NEXT_TRACK;
|
|
|
|
XF86XK_AudioPrev:
|
|
|
|
Result := VKEY_MEDIA_PREV_TRACK;
|
|
|
|
XF86XK_AudioStop:
|
|
|
|
Result := VKEY_MEDIA_STOP;
|
|
|
|
XF86XK_AudioPlay:
|
|
|
|
Result := VKEY_MEDIA_PLAY_PAUSE;
|
|
|
|
XF86XK_Mail:
|
|
|
|
Result := VKEY_MEDIA_LAUNCH_MAIL;
|
|
|
|
XF86XK_LaunchA:
|
|
|
|
Result := VKEY_MEDIA_LAUNCH_APP1;
|
|
|
|
XF86XK_LaunchB,
|
|
|
|
XF86XK_Calculator:
|
|
|
|
Result := VKEY_MEDIA_LAUNCH_APP2;
|
|
|
|
XF86XK_WLAN:
|
|
|
|
Result := VKEY_WLAN;
|
|
|
|
XF86XK_PowerOff:
|
|
|
|
Result := VKEY_POWER;
|
|
|
|
XF86XK_MonBrightnessDown:
|
|
|
|
Result := VKEY_BRIGHTNESS_DOWN;
|
|
|
|
XF86XK_MonBrightnessUp:
|
|
|
|
Result := VKEY_BRIGHTNESS_UP;
|
|
|
|
XF86XK_KbdBrightnessDown:
|
|
|
|
Result := VKEY_KBD_BRIGHTNESS_DOWN;
|
|
|
|
XF86XK_KbdBrightnessUp:
|
|
|
|
Result := VKEY_KBD_BRIGHTNESS_UP;
|
|
|
|
else Result := VKEY_UNKNOWN;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetCefStateModifiers(state : cuint) : integer;
|
|
|
|
begin
|
|
|
|
Result := EVENTFLAG_NONE;
|
|
|
|
|
|
|
|
if ((state and GDK_SHIFT_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_SHIFT_DOWN;
|
|
|
|
|
|
|
|
if ((state and GDK_LOCK_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_CAPS_LOCK_ON;
|
|
|
|
|
|
|
|
if ((state and GDK_CONTROL_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_CONTROL_DOWN;
|
|
|
|
|
|
|
|
if ((state and GDK_MOD1_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_ALT_DOWN;
|
|
|
|
|
|
|
|
if ((state and GDK_BUTTON1_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_LEFT_MOUSE_BUTTON;
|
|
|
|
|
|
|
|
if ((state and GDK_BUTTON2_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_MIDDLE_MOUSE_BUTTON;
|
|
|
|
|
|
|
|
if ((state and GDK_BUTTON3_MASK) <> 0) then
|
|
|
|
Result := Result or EVENTFLAG_RIGHT_MOUSE_BUTTON;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GdkEventToWindowsKeyCode(event: PGdkEventKey) : integer;
|
|
|
|
var
|
|
|
|
windows_key_code, keyval : integer;
|
|
|
|
begin
|
|
|
|
windows_key_code := KeyboardCodeFromXKeysym(event^.keyval);
|
|
|
|
if (windows_key_code <> 0) then
|
|
|
|
begin
|
|
|
|
Result := windows_key_code;
|
|
|
|
exit;
|
|
|
|
end;
|
|
|
|
|
|
|
|
if (event^.hardware_keycode < length(kHardwareCodeToGDKKeyval)) then
|
|
|
|
begin
|
|
|
|
keyval := kHardwareCodeToGDKKeyval[event^.hardware_keycode];
|
|
|
|
if (keyval <> 0) then
|
|
|
|
begin
|
|
|
|
Result := KeyboardCodeFromXKeysym(keyval);
|
|
|
|
exit;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
Result := KeyboardCodeFromXKeysym(event^.keyval);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetWindowsKeyCodeWithoutLocation(key_code : integer) : integer;
|
|
|
|
begin
|
|
|
|
case key_code of
|
|
|
|
VKEY_LCONTROL, VKEY_RCONTROL : Result := VKEY_CONTROL;
|
|
|
|
VKEY_LSHIFT, VKEY_RSHIFT : Result := VKEY_SHIFT;
|
|
|
|
VKEY_LMENU, VKEY_RMENU : Result := VKEY_MENU;
|
|
|
|
else Result := key_code;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function GetControlCharacter(windows_key_code : integer; shift : boolean) : integer;
|
|
|
|
begin
|
|
|
|
if (windows_key_code >= VKEY_A) and (windows_key_code <= VKEY_Z) then
|
|
|
|
Result := windows_key_code - VKEY_A + 1
|
|
|
|
else
|
|
|
|
if shift then
|
|
|
|
case windows_key_code of
|
|
|
|
VKEY_2 : Result := 0;
|
|
|
|
VKEY_6 : Result := $1E;
|
|
|
|
VKEY_OEM_MINUS : Result := $1F;
|
|
|
|
else Result := 0;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
case windows_key_code of
|
|
|
|
VKEY_OEM_4 : Result := $1B;
|
|
|
|
VKEY_OEM_5 : Result := $1C;
|
|
|
|
VKEY_OEM_6 : Result := $1D;
|
|
|
|
VKEY_RETURN : Result := $0A;
|
|
|
|
else Result := 0;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure GdkEventKeyToCEFKeyEvent(GdkEvent: PGdkEventKey; var aCEFKeyEvent : TCEFKeyEvent);
|
|
|
|
var
|
|
|
|
windows_key_code : integer;
|
|
|
|
begin
|
|
|
|
windows_key_code := GdkEventToWindowsKeyCode(GdkEvent);
|
|
|
|
aCEFKeyEvent.windows_key_code := GetWindowsKeyCodeWithoutLocation(windows_key_code);
|
|
|
|
aCEFKeyEvent.native_key_code := GdkEvent^.hardware_keycode;
|
|
|
|
aCEFKeyEvent.modifiers := GetCefStateModifiers(GdkEvent^.state);
|
|
|
|
aCEFKeyEvent.focus_on_editable_field := 0;
|
|
|
|
|
|
|
|
if (GdkEvent^.keyval >= GDK_KP_Space) and (GdkEvent^.keyval <= GDK_KP_9) then
|
|
|
|
aCEFKeyEvent.modifiers := aCEFKeyEvent.modifiers or EVENTFLAG_IS_KEY_PAD;
|
|
|
|
|
|
|
|
aCEFKeyEvent.is_system_key := ord((aCEFKeyEvent.modifiers and EVENTFLAG_ALT_DOWN) <> 0);
|
|
|
|
|
|
|
|
if (windows_key_code = VKEY_RETURN) then
|
|
|
|
aCEFKeyEvent.unmodified_character := #13
|
|
|
|
else
|
|
|
|
aCEFKeyEvent.unmodified_character := WideChar(gdk_keyval_to_unicode(GdkEvent^.keyval));
|
|
|
|
|
|
|
|
if ((aCEFKeyEvent.modifiers and EVENTFLAG_CONTROL_DOWN) <> 0) then
|
|
|
|
aCEFKeyEvent.character := WideChar(GetControlCharacter(windows_key_code, ((aCEFKeyEvent.modifiers and EVENTFLAG_SHIFT_DOWN) <> 0)))
|
|
|
|
else
|
|
|
|
aCEFKeyEvent.character := aCEFKeyEvent.unmodified_character;
|
|
|
|
end;
|
2020-12-31 12:15:10 +01:00
|
|
|
|
|
|
|
// This function is almost an identical copy of "ModalShowX11Window" available
|
|
|
|
// at https://wiki.lazarus.freepascal.org/X11
|
|
|
|
procedure ShowX11Message(const aMessage : string);
|
|
|
|
var
|
|
|
|
TempDisplay : PDisplay;
|
|
|
|
TempWindow : TWindow;
|
|
|
|
TempEvent : TXEvent;
|
|
|
|
TempMessage : PChar;
|
|
|
|
TempScreen : cint;
|
|
|
|
begin
|
|
|
|
TempMessage := PChar(trim(copy(aMessage, 1, pred(pos(#13, aMessage)))));
|
|
|
|
TempDisplay := XOpenDisplay(nil);
|
|
|
|
|
|
|
|
if (TempDisplay = nil) then
|
|
|
|
begin
|
|
|
|
WriteLn(aMessage);
|
|
|
|
exit;
|
|
|
|
end;
|
|
|
|
|
|
|
|
TempScreen := DefaultScreen(TempDisplay);
|
|
|
|
TempWindow := XCreateSimpleWindow(TempDisplay,
|
|
|
|
RootWindow(TempDisplay, TempScreen),
|
|
|
|
10, 10, 200, 100, 1,
|
|
|
|
BlackPixel(TempDisplay, TempScreen),
|
|
|
|
WhitePixel(TempDisplay, TempScreen));
|
|
|
|
|
|
|
|
XSelectInput(TempDisplay, TempWindow, ExposureMask or KeyPressMask);
|
|
|
|
|
|
|
|
XMapWindow(TempDisplay, TempWindow);
|
|
|
|
|
|
|
|
while (True) do
|
|
|
|
begin
|
|
|
|
XNextEvent(TempDisplay, @TempEvent);
|
|
|
|
|
|
|
|
if (TempEvent._type = Expose) then
|
|
|
|
XDrawString(TempDisplay,
|
|
|
|
TempWindow,
|
|
|
|
DefaultGC(TempDisplay, TempScreen),
|
|
|
|
40, 50,
|
|
|
|
TempMessage,
|
|
|
|
strlen(TempMessage));
|
|
|
|
|
|
|
|
if (TempEvent._type = KeyPress) then Break;
|
|
|
|
end;
|
|
|
|
|
|
|
|
XCloseDisplay(TempDisplay);
|
|
|
|
end;
|
|
|
|
|
2020-12-28 18:11:27 +01:00
|
|
|
{$ENDIF}{$ENDIF}{$ENDIF}
|
|
|
|
|
2020-02-03 16:14:31 +01:00
|
|
|
end.
|