[541] | 1 | //*************************************************************
|
---|
| 2 | // EwbCoreTools *
|
---|
| 3 | // *
|
---|
| 4 | // Freeware Unit *
|
---|
| 5 | // For Delphi *
|
---|
| 6 | // Developing Team: *
|
---|
| 7 | // Serge Voloshenyuk (SergeV@bsalsa.com) *
|
---|
| 8 | // Eran Bodankin (bsalsa) -(bsalsa@gmail.com) *
|
---|
| 9 | // *
|
---|
| 10 | // Documentation and updated versions: *
|
---|
| 11 | // *
|
---|
| 12 | // http://www.bsalsa.com *
|
---|
| 13 | //*************************************************************
|
---|
| 14 | {LICENSE:
|
---|
| 15 | THIS SOFTWARE IS PROVIDED TO YOU "AS IS" WITHOUT WARRANTY OF ANY KIND,
|
---|
| 16 | EITHER EXPRESSED OR IMPLIED INCLUDING BUT NOT LIMITED TO THE APPLIED
|
---|
| 17 | WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE.
|
---|
| 18 | YOU ASSUME THE ENTIRE RISK AS TO THE ACCURACY AND THE USE OF THE SOFTWARE
|
---|
| 19 | AND ALL OTHER RISK ARISING OUT OF THE USE OR PERFORMANCE OF THIS SOFTWARE
|
---|
| 20 | AND DOCUMENTATION. BSALSA PRODUCTIONS DOES NOT WARRANT THAT THE SOFTWARE IS ERROR-FREE
|
---|
| 21 | OR WILL OPERATE WITHOUT INTERRUPTION. THE SOFTWARE IS NOT DESIGNED, INTENDED
|
---|
| 22 | OR LICENSED FOR USE IN HAZARDOUS ENVIRONMENTS REQUIRING FAIL-SAFE CONTROLS,
|
---|
| 23 | INCLUDING WITHOUT LIMITATION, THE DESIGN, CONSTRUCTION, MAINTENANCE OR
|
---|
| 24 | OPERATION OF NUCLEAR FACILITIES, AIRCRAFT NAVIGATION OR COMMUNICATION SYSTEMS,
|
---|
| 25 | AIR TRAFFIC CONTROL, AND LIFE SUPPORT OR WEAPONS SYSTEMS. BSALSA PRODUCTIONS SPECIFICALLY
|
---|
| 26 | DISCLAIMS ANY EXPRESS OR IMPLIED WARRANTY OF FITNESS FOR SUCH PURPOSE.
|
---|
| 27 |
|
---|
| 28 | You may use/ change/ modify the component under 3 conditions:
|
---|
| 29 | 1. In your website, add a link to "http://www.bsalsa.com"
|
---|
| 30 | 2. In your application, add credits to "Embedded Web Browser"
|
---|
| 31 | 3. Mail me (bsalsa@gmail.com) any code change in the unit for the benefit
|
---|
| 32 | of the other users.
|
---|
| 33 | 4. Please, consider donation in our web site!
|
---|
| 34 | {*******************************************************************************}
|
---|
| 35 |
|
---|
| 36 |
|
---|
| 37 | unit EwbCoreTools;
|
---|
| 38 |
|
---|
| 39 | {$I EWB.inc}
|
---|
| 40 |
|
---|
| 41 | interface
|
---|
| 42 |
|
---|
| 43 | uses
|
---|
| 44 | Graphics, ActiveX, Mshtml_Ewb, Windows, SysUtils;
|
---|
| 45 |
|
---|
| 46 | function IsWinXPSP2OrLater(): Boolean;
|
---|
| 47 | function ColorToHTML(const Color: TColor): string;
|
---|
| 48 | function WideStringToLPOLESTR(const Source: Widestring): POleStr;
|
---|
| 49 | function XPath4Node(node: IHTMLElement): string;
|
---|
| 50 | function TaskAllocWideString(const S: string): PWChar;
|
---|
| 51 | function AnsiIndexStr(const AText: string; const AValues: array of string): Integer;
|
---|
| 52 | function GetPos(const SubSt, Text: string; StartPos: Integer = -1): Integer;
|
---|
| 53 | function _CharPos(const C: Char; const S: string): Integer;
|
---|
| 54 | function CutString(var Text: string; const Delimiter: string = ' ';
|
---|
| 55 | const Remove: Boolean = True): string;
|
---|
| 56 | procedure FormatPath(Path: string);
|
---|
| 57 | function GetWinText(WinHandle: THandle): string;
|
---|
| 58 | function GetWinClass(Handle: Hwnd): WideString;
|
---|
| 59 | function GetParentWinByClass(ChildHandle: HWND; const ClassName: string): HWND;
|
---|
| 60 | {$IFDEF DELPHI5}
|
---|
| 61 | function DirectoryExists(const Directory: string): Boolean;
|
---|
| 62 | function VarSupports(const V: Variant; const IID: TGUID; out Intf): Boolean;
|
---|
| 63 | {$ENDIF}
|
---|
| 64 | {$IFNDEF DELPHI12_UP}
|
---|
| 65 | function CharInSet(C: Char; const CharSet: TSysCharSet): Boolean;
|
---|
| 66 | {$ENDIF}
|
---|
| 67 | function AddBackSlash(const S: string): string;
|
---|
| 68 |
|
---|
| 69 | const
|
---|
| 70 | WM_SETWBFOCUS = $0400 {WM_USER} + $44;
|
---|
| 71 |
|
---|
| 72 | implementation
|
---|
| 73 |
|
---|
| 74 | uses
|
---|
| 75 | IeConst, EwbAcc;
|
---|
| 76 |
|
---|
| 77 | type
|
---|
| 78 | {VerifyVersion}
|
---|
| 79 | fn_VerifyVersionInfo = function(var VersionInformation: OSVERSIONINFOEX;
|
---|
| 80 | dwTypeMask: DWORD; dwlConditionMask: LONGLONG): BOOL; stdcall;
|
---|
| 81 | fn_VerSetConditionMask = function(ConditionMask: LONGLONG; TypeMask: DWORD;
|
---|
| 82 | Condition: Byte): LONGLONG; stdcall;
|
---|
| 83 |
|
---|
| 84 |
|
---|
| 85 | function IsWinXPSP2OrLater(): Boolean;
|
---|
| 86 | var
|
---|
| 87 | osvi: TOSVersionInfoEx;
|
---|
| 88 | dwlConditionMask: LONGLONG;
|
---|
| 89 | op: Integer;
|
---|
| 90 | hlib: THandle;
|
---|
| 91 | VerifyVersionInfo: fn_VerifyVersionInfo;
|
---|
| 92 | VerSetConditionMask: fn_VerSetConditionMask;
|
---|
| 93 | begin
|
---|
| 94 | Result := False;
|
---|
| 95 | hLib := GetModuleHandle('kernel32.dll');
|
---|
| 96 | if hLib = 0 then
|
---|
| 97 | hLib := LoadLibrary('kernel32.dll');
|
---|
| 98 | if (hLib <> 0) then
|
---|
| 99 | begin
|
---|
| 100 | @VerifyVersionInfo := GetProcAddress(hLib, 'VerifyVersionInfoA');
|
---|
| 101 | @VerSetConditionMask := GetProcAddress(hLib, 'VerSetConditionMask');
|
---|
| 102 | if ((@VerifyVersionInfo = nil) or (@VerSetConditionMask = nil)) then Exit;
|
---|
| 103 |
|
---|
| 104 | dwlConditionMask := 0;
|
---|
| 105 | op := VER_GREATER_EQUAL;
|
---|
| 106 |
|
---|
| 107 | // Initialize the OSVERSIONINFOEX structure.
|
---|
| 108 | ZeroMemory(@osvi, SizeOf(OSVERSIONINFOEX));
|
---|
| 109 | osvi.dwOSVersionInfoSize := SizeOf(OSVERSIONINFOEX);
|
---|
| 110 | osvi.dwMajorVersion := 5;
|
---|
| 111 | osvi.dwMinorVersion := 1;
|
---|
| 112 | osvi.wServicePackMajor := 2;
|
---|
| 113 | osvi.wServicePackMinor := 0;
|
---|
| 114 |
|
---|
| 115 | // Initialize the condition mask.
|
---|
| 116 | dwlConditionMask := VerSetConditionMask(dwlConditionMask, VER_MAJORVERSION, op);
|
---|
| 117 | dwlConditionMask := VerSetConditionMask(dwlConditionMask, VER_MINORVERSION, op);
|
---|
| 118 | dwlConditionMask := VerSetConditionMask(dwlConditionMask, VER_SERVICEPACKMAJOR, op);
|
---|
| 119 | dwlConditionMask := VerSetConditionMask(dwlConditionMask, VER_SERVICEPACKMINOR, op);
|
---|
| 120 |
|
---|
| 121 | // Perform the test.
|
---|
| 122 | Result := VerifyVersionInfo(osvi, VER_MAJORVERSION or VER_MINORVERSION or
|
---|
| 123 | VER_SERVICEPACKMAJOR or VER_SERVICEPACKMINOR, dwlConditionMask);
|
---|
| 124 | end;
|
---|
| 125 | end;
|
---|
| 126 |
|
---|
| 127 | function GetParentWinByClass(ChildHandle: HWND; const ClassName: string): HWND;
|
---|
| 128 | var
|
---|
| 129 | szClass: array[0..255] of Char;
|
---|
| 130 | begin
|
---|
| 131 | Result := GetParent(ChildHandle);
|
---|
| 132 | while IsWindow(Result) do
|
---|
| 133 | begin
|
---|
| 134 | if (GetClassName(Result, szClass, SizeOf(szClass)) > 0) and
|
---|
| 135 | (AnsiStrComp(PChar(ClassName), szClass) = 0) then Exit;
|
---|
| 136 | Result := GetParent(Result);
|
---|
| 137 | end;
|
---|
| 138 | end;
|
---|
| 139 |
|
---|
| 140 |
|
---|
| 141 |
|
---|
| 142 | {$IFNDEF DELPHI12_UP}
|
---|
| 143 |
|
---|
| 144 | function CharInSet(C: Char; const CharSet: TSysCharSet): Boolean;
|
---|
| 145 | begin
|
---|
| 146 | Result := C in CharSet;
|
---|
| 147 | end;
|
---|
| 148 | {$ENDIF}
|
---|
| 149 |
|
---|
| 150 | {$IFDEF DELPHI5}
|
---|
| 151 |
|
---|
| 152 | function DirectoryExists(const Directory: string): Boolean;
|
---|
| 153 | var
|
---|
| 154 | Code: Integer;
|
---|
| 155 | begin
|
---|
| 156 | {$RANGECHECKS OFF}
|
---|
| 157 | Code := GetFileAttributes(PChar(Directory));
|
---|
| 158 | Result := (Code <> -1) and (FILE_ATTRIBUTE_DIRECTORY and Code <> 0);
|
---|
| 159 | {$RANGECHECKS ON}
|
---|
| 160 | end;
|
---|
| 161 |
|
---|
| 162 | function VarSupports(const V: Variant; const IID: TGUID; out Intf): Boolean;
|
---|
| 163 | begin
|
---|
| 164 | Result := Supports(V, IID, Intf);
|
---|
| 165 | end;
|
---|
| 166 | {$ENDIF}
|
---|
| 167 |
|
---|
| 168 | function AddBackSlash(const S: string): string;
|
---|
| 169 | begin
|
---|
| 170 | {$IFDEF DELPHI5}
|
---|
| 171 | Result := IncludeTrailingBackslash(S);
|
---|
| 172 | {$ELSE}
|
---|
| 173 | {$IFDEF DELPHI6UP}
|
---|
| 174 | Result := IncludeTrailingPathDelimiter(S);
|
---|
| 175 | {$ELSE}
|
---|
| 176 | if Copy(S, Length(S), 1) = '\' then
|
---|
| 177 | Result := S
|
---|
| 178 | else
|
---|
| 179 | Result := S + '\';
|
---|
| 180 | {$ENDIF}
|
---|
| 181 | {$ENDIF}
|
---|
| 182 | end;
|
---|
| 183 |
|
---|
| 184 | function CutString(var Text: string; const Delimiter: string = ' ';
|
---|
| 185 | const Remove: Boolean = True): string;
|
---|
| 186 | var
|
---|
| 187 | IdxPos: Integer;
|
---|
| 188 | begin
|
---|
| 189 | if Delimiter = #0 then
|
---|
| 190 | IdxPos := Pos(Delimiter, Text)
|
---|
| 191 | else
|
---|
| 192 | IdxPos := AnsiPos(Delimiter, Text);
|
---|
| 193 |
|
---|
| 194 | if (IdxPos = 0) then
|
---|
| 195 | begin
|
---|
| 196 | Result := Text;
|
---|
| 197 | if Remove then
|
---|
| 198 | Text := '';
|
---|
| 199 | end
|
---|
| 200 | else
|
---|
| 201 | begin
|
---|
| 202 | Result := Copy(Text, 1, IdxPos - 1);
|
---|
| 203 | if Remove then
|
---|
| 204 | Delete(Text, 1, IdxPos + Length(Delimiter) - 1);
|
---|
| 205 | end;
|
---|
| 206 | end;
|
---|
| 207 |
|
---|
| 208 |
|
---|
| 209 | function GetPos(const SubSt, Text: string; StartPos: Integer = -1): Integer;
|
---|
| 210 | var
|
---|
| 211 | i: Integer;
|
---|
| 212 | LStartPos: Integer;
|
---|
| 213 | LTokenLen: Integer;
|
---|
| 214 | begin
|
---|
| 215 | result := 0;
|
---|
| 216 | LTokenLen := Length(SubSt);
|
---|
| 217 | if StartPos = -1 then
|
---|
| 218 | begin
|
---|
| 219 | StartPos := Length(Text);
|
---|
| 220 | end;
|
---|
| 221 | if StartPos < (Length(Text) - LTokenLen + 1) then
|
---|
| 222 | begin
|
---|
| 223 | LStartPos := StartPos;
|
---|
| 224 | end
|
---|
| 225 | else
|
---|
| 226 | begin
|
---|
| 227 | LStartPos := (Length(Text) - LTokenLen + 1);
|
---|
| 228 | end;
|
---|
| 229 | for i := LStartPos downto 1 do
|
---|
| 230 | begin
|
---|
| 231 | if AnsiSameText(Copy(Text, i, LTokenLen), SubSt) then
|
---|
| 232 | begin
|
---|
| 233 | Result := i;
|
---|
| 234 | Break;
|
---|
| 235 | end;
|
---|
| 236 | end;
|
---|
| 237 | end;
|
---|
| 238 |
|
---|
| 239 | function _CharPos(const C: Char; const S: string): Integer;
|
---|
| 240 | begin
|
---|
| 241 | for Result := 1 to Length(S) do
|
---|
| 242 | if S[Result] = C then Exit;
|
---|
| 243 | Result := 0;
|
---|
| 244 | end;
|
---|
| 245 |
|
---|
| 246 | procedure FormatPath(Path: string);
|
---|
| 247 | var
|
---|
| 248 | i: Integer;
|
---|
| 249 | begin
|
---|
| 250 | i := 1;
|
---|
| 251 | while i <= Length(Path) do
|
---|
| 252 | begin
|
---|
| 253 | if CharInSet(Path[i], LeadBytes) then
|
---|
| 254 | Inc(i, 2)
|
---|
| 255 | else
|
---|
| 256 | if Path[i] = '\' then
|
---|
| 257 | begin
|
---|
| 258 | Path[i] := '/';
|
---|
| 259 | Inc(i, 1);
|
---|
| 260 | end
|
---|
| 261 | else
|
---|
| 262 | Inc(i, 1);
|
---|
| 263 | end;
|
---|
| 264 | end;
|
---|
| 265 |
|
---|
| 266 | function AnsiIndexStr(const AText: string; const AValues: array of string): Integer;
|
---|
| 267 | var
|
---|
| 268 | I: Integer;
|
---|
| 269 | begin
|
---|
| 270 | Result := -1;
|
---|
| 271 | for I := Low(AValues) to High(AValues) do
|
---|
| 272 | if AnsiSameStr(AText, AValues[I]) then
|
---|
| 273 | begin
|
---|
| 274 | Result := I;
|
---|
| 275 | Break;
|
---|
| 276 | end;
|
---|
| 277 | end;
|
---|
| 278 |
|
---|
| 279 |
|
---|
| 280 | function TaskAllocWideString(const S: string): PWChar;
|
---|
| 281 | var
|
---|
| 282 | WideLength: integer;
|
---|
| 283 | Wide: PWideChar;
|
---|
| 284 | begin
|
---|
| 285 | WideLength := Length(S) + 1;
|
---|
| 286 | Wide := CoTaskMemAlloc(WideLength * SizeOf(WideChar));
|
---|
| 287 | StringToWideChar(S, Wide, WideLength);
|
---|
| 288 | Result := Wide;
|
---|
| 289 | end;
|
---|
| 290 |
|
---|
| 291 | {
|
---|
| 292 | function TaskAllocWideString(const S: string): PWChar;
|
---|
| 293 | var
|
---|
| 294 | Len: Integer;
|
---|
| 295 | begin
|
---|
| 296 | Len := Length(S) + 1;
|
---|
| 297 | Result := CoTaskMemAlloc(2 * Len);
|
---|
| 298 | StringToWideChar(S, Result, Len);
|
---|
| 299 | end;
|
---|
| 300 | }
|
---|
| 301 |
|
---|
| 302 | function WideStringToLPOLESTR(const Source: Widestring): POleStr;
|
---|
| 303 | var
|
---|
| 304 | Len: Integer;
|
---|
| 305 | begin
|
---|
| 306 | Len := Length(Source) * SizeOf(WideChar);
|
---|
| 307 | Result := CoTaskMemAlloc(Len + 2);
|
---|
| 308 | FillChar(Result^, Len + 2, 0);
|
---|
| 309 | Move(Result^, PWideString(Source)^, Len);
|
---|
| 310 | end;
|
---|
| 311 |
|
---|
| 312 | function ColorToHTML(const Color: TColor): string;
|
---|
| 313 | var
|
---|
| 314 | ColorRGB: LongWord;
|
---|
| 315 | begin
|
---|
| 316 | ColorRGB := ColorToRGB(Color);
|
---|
| 317 | FmtStr(Result, '#%0.2X%0.2X%0.2X',
|
---|
| 318 | [Byte(ColorRGB), Byte(ColorRGB shr 8), Byte(ColorRGB shr 16)]);
|
---|
| 319 | end;
|
---|
| 320 |
|
---|
| 321 | function GetWinText(WinHandle: THandle): string;
|
---|
| 322 | var
|
---|
| 323 | DlgName: string;
|
---|
| 324 | TxtLength: Integer;
|
---|
| 325 | begin
|
---|
| 326 | TxtLength := GetWindowTextLength(WinHandle);
|
---|
| 327 | SetLength(DlgName, TxtLength + 1);
|
---|
| 328 | GetWindowText(WinHandle, PChar(DlgName), TxtLength + 1);
|
---|
| 329 | Result := DlgName;
|
---|
| 330 | end;
|
---|
| 331 |
|
---|
| 332 |
|
---|
| 333 | function GetWinClass(Handle: Hwnd): WideString;
|
---|
| 334 | var
|
---|
| 335 | pwc: PWideChar;
|
---|
| 336 | const
|
---|
| 337 | maxbufsize = 32767 * SizeOf(WideChar);
|
---|
| 338 | begin
|
---|
| 339 | Result := '';
|
---|
| 340 | if IsWindow(Handle) then
|
---|
| 341 | begin
|
---|
| 342 | pwc := GetMemory(maxbufsize);
|
---|
| 343 | if Assigned(pwc) then
|
---|
| 344 | try
|
---|
| 345 | ZeroMemory(pwc, maxbufsize);
|
---|
| 346 | if GetClassnameW(Handle, pwc, maxbufsize) > 0 then
|
---|
| 347 | SetString(Result, pwc, lstrlenW(pwc));
|
---|
| 348 | finally
|
---|
| 349 | FreeMemory(pwc);
|
---|
| 350 | end;
|
---|
| 351 | end;
|
---|
| 352 | end;
|
---|
| 353 |
|
---|
| 354 | {
|
---|
| 355 | function GetWinClass(WinHandle: THANDLE): string;
|
---|
| 356 | begin
|
---|
| 357 | SetLength(Result, 80);
|
---|
| 358 | SetLength(Result, GetClassName(WinHandle, PChar(Result), Length(Result)));
|
---|
| 359 | end;
|
---|
| 360 | }
|
---|
| 361 |
|
---|
| 362 |
|
---|
| 363 | function XPath4Node(node: IHTMLElement): string;
|
---|
| 364 |
|
---|
| 365 | function NodePosition(elem: IHTMLElement): string;
|
---|
| 366 | var tag: Widestring;
|
---|
| 367 | Idx: Integer;
|
---|
| 368 | n: IHTMLElement;
|
---|
| 369 | cl: IHTMLElementCollection;
|
---|
| 370 | itm: IDispatch;
|
---|
| 371 | I, C, mI: Integer;
|
---|
| 372 | begin
|
---|
| 373 | Result := '';
|
---|
| 374 | if (elem.parentElement = nil) or
|
---|
| 375 | not Supports(elem.parentElement.children, IHTMLElementCollection, cl) then Exit;
|
---|
| 376 |
|
---|
| 377 | Tag := elem.tagName;
|
---|
| 378 | Idx := elem.sourceIndex;
|
---|
| 379 | C := 0;
|
---|
| 380 | mI := -1;
|
---|
| 381 |
|
---|
| 382 | for I := 0 to cl.length - 1 do
|
---|
| 383 | begin
|
---|
| 384 | itm := cl.item(I, I);
|
---|
| 385 | if Supports(itm, IHTMLElement, n) then
|
---|
| 386 | begin
|
---|
| 387 | if n.tagName = Tag then
|
---|
| 388 | begin
|
---|
| 389 | if n.sourceIndex = Idx then mI := C;
|
---|
| 390 | Inc(C);
|
---|
| 391 | end;
|
---|
| 392 | end;
|
---|
| 393 | end;
|
---|
| 394 | if (mI > 0) or (C > 1) then Result := Format('[%d]', [mI]);
|
---|
| 395 | end;
|
---|
| 396 |
|
---|
| 397 | var id: string;
|
---|
| 398 | begin
|
---|
| 399 | if node <> nil then
|
---|
| 400 | begin
|
---|
| 401 | id := node.id;
|
---|
| 402 | if id <> '' then
|
---|
| 403 | Result := Format('%s[@id="%s"]', [node.tagName, id])
|
---|
| 404 | else if node.parentElement = nil then
|
---|
| 405 | Result := '/' + node.tagName
|
---|
| 406 | else Result := Format('%s/%s%s',
|
---|
| 407 | [XPath4Node(node.parentElement), node.tagName, NodePosition(node)]);
|
---|
| 408 | end else Result := '';
|
---|
| 409 | end;
|
---|
| 410 |
|
---|
| 411 | end.
|
---|