2019-04-04 10:26:44 +02:00
|
|
|
unit uCEFFMXChromium;
|
2018-01-25 22:34:04 +02:00
|
|
|
|
|
|
|
{$I cef.inc}
|
|
|
|
|
2022-02-19 19:56:41 +02:00
|
|
|
{$IFNDEF TARGET_64BITS}{$ALIGN ON}{$ENDIF}
|
|
|
|
{$MINENUMSIZE 4}
|
|
|
|
|
2018-01-25 22:34:04 +02:00
|
|
|
interface
|
|
|
|
|
|
|
|
uses
|
|
|
|
System.Classes, System.Types,
|
2018-03-31 18:08:18 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
2019-11-10 01:24:26 +02:00
|
|
|
WinApi.Windows, WinApi.Messages, FMX.Platform.Win,
|
2018-03-31 18:08:18 +02:00
|
|
|
{$ENDIF}
|
2021-12-08 18:45:03 +02:00
|
|
|
FMX.Types, FMX.Platform, FMX.Forms, FMX.Controls,
|
|
|
|
{$IFDEF DELPHI19_UP}
|
|
|
|
FMX.Graphics,
|
|
|
|
{$ENDIF}
|
2022-10-14 16:35:50 +02:00
|
|
|
uCEFTypes, uCEFInterfaces, uCEFConstants, uCEFChromiumCore;
|
2018-01-25 22:34:04 +02:00
|
|
|
|
|
|
|
type
|
2022-10-14 16:35:50 +02:00
|
|
|
{$IFNDEF FPC}{$IFDEF DELPHI16_UP}[ComponentPlatformsAttribute(pfidWindows or pfidOSX or pfidLinux)]{$ENDIF}{$ENDIF}
|
2023-07-19 11:59:20 +02:00
|
|
|
/// <summary>
|
|
|
|
/// FMX version of TChromiumCore that puts together all browser procedures, functions, properties and events in one place.
|
|
|
|
/// It has all you need to create, modify and destroy a web browser.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
TFMXChromium = class(TChromiumCore, IChromiumEvents)
|
2018-01-25 22:34:04 +02:00
|
|
|
protected
|
2020-01-28 12:36:34 +02:00
|
|
|
function GetParentFormHandle : TCefWindowHandle; override;
|
2019-11-10 01:24:26 +02:00
|
|
|
function GetParentForm : TCustomForm;
|
2020-04-09 18:43:03 +02:00
|
|
|
function GetScreenScale : Single;
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure InitializeDevToolsWindowInfo; virtual;
|
2020-04-09 18:43:03 +02:00
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
public
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Open developer tools (DevTools) in its own browser. If inspectElementAt has a valid point
|
|
|
|
/// with coordinates different than low(integer) then the element at the specified location
|
|
|
|
/// will be inspected. If the DevTools browser is already open then it will be focused.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure ShowDevTools(inspectElementAt: TPoint);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// close the developer tools.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure CloseDevTools;
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Move the parent form to the x and y coordinates.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure MoveFormTo(const x, y: Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Move the parent form adding x and y to the coordinates.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure MoveFormBy(const x, y: Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Add x to the parent form width.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure ResizeFormWidthTo(const x : Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Add y to the parent form height.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure ResizeFormHeightTo(const y : Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Set the parent form left property to x.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure SetFormLeftTo(const x : Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Set the parent form top property to y.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure SetFormTopTo(const y : Integer);
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Used to create the browser after the global request context has been
|
|
|
|
/// initialized. You need to set all properties and events before calling
|
|
|
|
/// this function because it will only create the internal handlers needed
|
|
|
|
/// for those events and the property values will be used in the browser
|
|
|
|
/// initialization.
|
|
|
|
/// The browser will be fully initialized when the TChromiumCore.OnAfterCreated
|
|
|
|
/// event is triggered.
|
|
|
|
/// </summary>
|
2019-11-10 01:24:26 +02:00
|
|
|
function CreateBrowser(const aWindowName : ustring = ''; const aContext : ICefRequestContext = nil; const aExtraInfo : ICefDictionaryValue = nil) : boolean; overload; virtual;
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Copy the DC to a bitmap stream. Only works on Windows with browsers without GPU acceleration.
|
|
|
|
/// It's recommended to use the "Page.captureScreenshot" DevTools method instead.
|
|
|
|
/// </summary>
|
2021-08-19 14:20:01 +02:00
|
|
|
function SaveAsBitmapStream(const aStream : TStream; const aRect : System.Types.TRect) : boolean;
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Takes a snapshot into a TBitmap using the SaveAsBitmapStream function.
|
|
|
|
/// </summary>
|
2020-01-02 21:02:47 +02:00
|
|
|
function TakeSnapshot(var aBitmap : TBitmap; const aRect : System.Types.TRect) : boolean;
|
2023-08-08 20:19:19 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Returns the screen scale of the monitor where the parent form is located.
|
|
|
|
/// </summary>
|
2020-04-09 18:43:03 +02:00
|
|
|
property ScreenScale : single read GetScreenScale;
|
2019-11-10 01:24:26 +02:00
|
|
|
end;
|
2018-01-25 22:34:04 +02:00
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
// *********************************************************
|
|
|
|
// ********************** ATTENTION ! **********************
|
|
|
|
// *********************************************************
|
|
|
|
// ** **
|
|
|
|
// ** MANY OF THE EVENTS IN CEF4DELPHI COMPONENTS LIKE **
|
|
|
|
// ** TCHROMIUM, TFMXCHROMIUM OR TCEFAPPLICATION ARE **
|
|
|
|
// ** EXECUTED IN A CEF THREAD BY DEFAULT. **
|
|
|
|
// ** **
|
|
|
|
// ** WINDOWS CONTROLS MUST BE CREATED AND DESTROYED IN **
|
|
|
|
// ** THE SAME THREAD TO AVOID ERRORS. **
|
|
|
|
// ** SOME OF THEM RECREATE THE HANDLERS IF THEY ARE **
|
|
|
|
// ** MODIFIED AND CAN CAUSE THE SAME ERRORS. **
|
|
|
|
// ** **
|
|
|
|
// ** DON'T CREATE, MODIFY OR DESTROY WINDOWS CONTROLS **
|
|
|
|
// ** INSIDE THE CEF4DELPHI EVENTS AND USE **
|
|
|
|
// ** SYNCHRONIZATION OBJECTS TO PROTECT VARIABLES AND **
|
|
|
|
// ** FIELDS IF THEY ARE ALSO USED IN THE MAIN THREAD. **
|
|
|
|
// ** **
|
|
|
|
// ** READ THIS FOR MORE INFORMATION : **
|
|
|
|
// ** https://www.briskbard.com/index.php?pageid=cef **
|
|
|
|
// ** **
|
|
|
|
// ** USE OUR FORUMS FOR MORE QUESTIONS : **
|
|
|
|
// ** https://www.briskbard.com/forum/ **
|
|
|
|
// ** **
|
|
|
|
// *********************************************************
|
|
|
|
// *********************************************************
|
2018-01-25 22:34:04 +02:00
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
implementation
|
2018-01-25 22:34:04 +02:00
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
uses
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF MSWINDOWS}{$IFDEF DELPHI24_UP}FMX.Helpers.Win,{$ENDIF}{$ENDIF}
|
2020-04-09 18:43:03 +02:00
|
|
|
System.SysUtils, System.Math,
|
|
|
|
uCEFApplicationCore;
|
2018-01-25 22:34:04 +02:00
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
function TFMXChromium.CreateBrowser(const aWindowName : ustring;
|
|
|
|
const aContext : ICefRequestContext;
|
|
|
|
const aExtraInfo : ICefDictionaryValue) : boolean;
|
2020-12-06 13:28:13 +02:00
|
|
|
var
|
|
|
|
TempHandle : TCefWindowHandle;
|
2018-01-25 22:34:04 +02:00
|
|
|
begin
|
2021-02-21 19:41:25 +02:00
|
|
|
{$IFDEF MACOS}
|
2020-12-06 13:28:13 +02:00
|
|
|
TempHandle := nil;
|
|
|
|
{$ELSE}
|
|
|
|
TempHandle := 0;
|
|
|
|
{$ENDIF}
|
|
|
|
Result := inherited CreateBrowser(TempHandle, Rect(0, 0, 0, 0), aWindowName, aContext, aExtraInfo);
|
2018-01-25 22:34:04 +02:00
|
|
|
end;
|
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure TFMXChromium.InitializeDevToolsWindowInfo;
|
2021-02-21 19:41:25 +02:00
|
|
|
var
|
|
|
|
TempHandle : TCefWindowHandle;
|
2018-01-25 22:34:04 +02:00
|
|
|
begin
|
2021-02-21 19:41:25 +02:00
|
|
|
{$IFDEF MACOS}
|
|
|
|
TempHandle := nil;
|
|
|
|
{$ELSE}
|
|
|
|
TempHandle := 0;
|
|
|
|
{$ENDIF}
|
|
|
|
DefaultInitializeDevToolsWindowInfo(TempHandle, Rect(0, 0, 0, 0), '');
|
2018-01-25 22:34:04 +02:00
|
|
|
end;
|
|
|
|
|
2019-11-10 19:23:39 +02:00
|
|
|
procedure TFMXChromium.ShowDevTools(inspectElementAt: TPoint);
|
2018-01-25 22:34:04 +02:00
|
|
|
begin
|
2019-11-10 01:24:26 +02:00
|
|
|
if Initialized then
|
2018-03-31 18:08:18 +02:00
|
|
|
begin
|
2019-11-10 01:24:26 +02:00
|
|
|
InitializeDevToolsWindowInfo;
|
|
|
|
inherited ShowDevTools(inspectElementAt, @FDevWindowInfo);
|
2018-03-31 18:08:18 +02:00
|
|
|
end;
|
2018-01-25 22:34:04 +02:00
|
|
|
end;
|
|
|
|
|
2019-11-10 01:24:26 +02:00
|
|
|
procedure TFMXChromium.CloseDevTools;
|
2018-01-25 22:34:04 +02:00
|
|
|
begin
|
2019-11-10 01:24:26 +02:00
|
|
|
inherited CloseDevTools;
|
2018-01-25 22:34:04 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
function TFMXChromium.GetParentForm : TCustomForm;
|
|
|
|
var
|
|
|
|
TempComp : TComponent;
|
|
|
|
begin
|
|
|
|
Result := nil;
|
|
|
|
TempComp := Owner;
|
|
|
|
|
|
|
|
while (TempComp <> nil) do
|
|
|
|
if (TempComp is TCustomForm) then
|
|
|
|
begin
|
|
|
|
Result := TCustomForm(TempComp);
|
|
|
|
exit;
|
|
|
|
end
|
|
|
|
else
|
|
|
|
TempComp := TempComp.owner;
|
|
|
|
end;
|
|
|
|
|
2020-04-09 18:43:03 +02:00
|
|
|
function TFMXChromium.GetScreenScale : Single;
|
2021-01-14 17:03:04 +02:00
|
|
|
{$IFDEF DELPHI24_UP}{$IFDEF MSWINDOWS}
|
2020-04-09 18:43:03 +02:00
|
|
|
var
|
|
|
|
TempHandle : TCefWindowHandle;
|
2021-01-14 17:03:04 +02:00
|
|
|
{$ENDIF}{$ENDIF}
|
2020-04-09 18:43:03 +02:00
|
|
|
begin
|
2021-01-14 17:03:04 +02:00
|
|
|
{$IFDEF DELPHI24_UP}{$IFDEF MSWINDOWS}
|
2020-04-09 18:43:03 +02:00
|
|
|
TempHandle := GetParentFormHandle;
|
|
|
|
|
|
|
|
if (TempHandle <> 0) then
|
|
|
|
Result := GetWndScale(TempHandle)
|
|
|
|
else
|
2021-01-14 17:03:04 +02:00
|
|
|
{$ENDIF}{$ENDIF}
|
2020-04-09 18:43:03 +02:00
|
|
|
if (GlobalCEFApp <> nil) then
|
|
|
|
Result := GlobalCEFApp.DeviceScaleFactor
|
|
|
|
else
|
|
|
|
Result := 1;
|
|
|
|
end;
|
|
|
|
|
2020-01-28 12:36:34 +02:00
|
|
|
function TFMXChromium.GetParentFormHandle : TCefWindowHandle;
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
|
|
|
{$ENDIF}
|
|
|
|
begin
|
|
|
|
Result := inherited GetParentFormHandle;
|
|
|
|
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
TempForm := GetParentForm;
|
|
|
|
|
|
|
|
if (TempForm <> nil) then
|
|
|
|
Result := FmxHandleToHWND(TempForm.Handle)
|
|
|
|
else
|
|
|
|
if (Application <> nil) and
|
|
|
|
(Application.MainForm <> nil) then
|
|
|
|
Result := FmxHandleToHWND(Application.MainForm.Handle);
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2018-01-25 22:34:04 +02:00
|
|
|
procedure TFMXChromium.MoveFormTo(const x, y: Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2018-01-25 22:34:04 +02:00
|
|
|
TempRect : TRect;
|
|
|
|
{$ENDIF}
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2018-01-25 22:34:04 +02:00
|
|
|
if (TempForm <> nil) then
|
|
|
|
begin
|
2021-09-14 12:45:14 +02:00
|
|
|
TempRect.Left := min(max(x, max(round(screen.DesktopLeft), 0)), round(screen.DesktopWidth) - TempForm.Width);
|
|
|
|
TempRect.Top := min(max(y, max(round(screen.DesktopTop), 0)), round(screen.DesktopHeight) - TempForm.Height);
|
2018-01-25 22:34:04 +02:00
|
|
|
TempRect.Right := TempRect.Left + TempForm.Width - 1;
|
|
|
|
TempRect.Bottom := TempRect.Top + TempForm.Height - 1;
|
|
|
|
|
|
|
|
TempForm.SetBounds(TempRect.Left, TempRect.Top, TempRect.Right - TempRect.Left + 1, TempRect.Bottom - TempRect.Top + 1);
|
|
|
|
end;
|
|
|
|
{$ELSE}
|
|
|
|
TempForm.SetBounds(x, y, TempForm.Width, TempForm.Height);
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TFMXChromium.MoveFormBy(const x, y: Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2018-01-25 22:34:04 +02:00
|
|
|
TempRect : TRect;
|
|
|
|
{$ENDIF}
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2018-01-25 22:34:04 +02:00
|
|
|
if (TempForm <> nil) then
|
|
|
|
begin
|
2021-09-14 12:45:14 +02:00
|
|
|
TempRect.Left := min(max(TempForm.Left + x, max(round(screen.DesktopLeft), 0)), round(screen.DesktopWidth) - TempForm.Width);
|
|
|
|
TempRect.Top := min(max(TempForm.Top + y, max(round(screen.DesktopTop), 0)), round(screen.DesktopHeight) - TempForm.Height);
|
2018-01-25 22:34:04 +02:00
|
|
|
TempRect.Right := TempRect.Left + TempForm.Width - 1;
|
|
|
|
TempRect.Bottom := TempRect.Top + TempForm.Height - 1;
|
|
|
|
|
|
|
|
TempForm.SetBounds(TempRect.Left, TempRect.Top, TempRect.Right - TempRect.Left + 1, TempRect.Bottom - TempRect.Top + 1);
|
|
|
|
end;
|
|
|
|
{$ELSE}
|
|
|
|
TempForm.SetBounds(TempForm.Left + x, TempForm.Top + y, TempForm.Width, TempForm.Height);
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TFMXChromium.ResizeFormWidthTo(const x : Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
|
|
|
TempX, TempDeltaX : integer;
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
|
|
|
|
|
|
|
if (TempForm <> nil) then
|
|
|
|
begin
|
|
|
|
TempX := max(x, 100);
|
|
|
|
TempDeltaX := TempForm.Width - TempForm.ClientWidth;
|
|
|
|
TempForm.Width := TempX + TempDeltaX;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TFMXChromium.ResizeFormHeightTo(const y : Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
|
|
|
TempY, TempDeltaY : integer;
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
|
|
|
|
|
|
|
if (TempForm <> nil) then
|
|
|
|
begin
|
|
|
|
TempY := max(y, 100);
|
|
|
|
TempDeltaY := TempForm.Height - TempForm.ClientHeight;
|
|
|
|
TempForm.Height := TempY + TempDeltaY;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TFMXChromium.SetFormLeftTo(const x : Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
|
|
|
|
|
|
|
if (TempForm <> nil) then
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2021-09-14 12:45:14 +02:00
|
|
|
TempForm.Left := min(max(x, max(round(screen.DesktopLeft), 0)), round(screen.DesktopWidth) - TempForm.Width);
|
2018-01-25 22:34:04 +02:00
|
|
|
{$ELSE}
|
|
|
|
TempForm.Left := x;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TFMXChromium.SetFormTopTo(const y : Integer);
|
|
|
|
var
|
|
|
|
TempForm : TCustomForm;
|
|
|
|
begin
|
|
|
|
TempForm := GetParentForm;
|
|
|
|
|
|
|
|
if (TempForm <> nil) then
|
2021-12-08 18:45:03 +02:00
|
|
|
{$IFDEF DELPHI21_UP}
|
2021-09-14 12:45:14 +02:00
|
|
|
TempForm.Top := min(max(y, max(round(screen.DesktopTop), 0)), round(screen.DesktopHeight) - TempForm.Height);
|
2018-01-25 22:34:04 +02:00
|
|
|
{$ELSE}
|
|
|
|
TempForm.Top := y;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
2021-08-19 14:20:01 +02:00
|
|
|
function TFMXChromium.SaveAsBitmapStream(const aStream : TStream; const aRect : System.Types.TRect) : boolean;
|
2020-01-02 21:02:47 +02:00
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
var
|
|
|
|
TempDC : HDC;
|
|
|
|
TempRect : System.Types.TRect;
|
|
|
|
{$ENDIF}
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
|
|
if not(FIsOSR) and (FRenderCompHWND <> 0) and (aStream <> nil) then
|
|
|
|
begin
|
|
|
|
TempDC := GetDC(FRenderCompHWND);
|
|
|
|
|
|
|
|
if (TempDC <> 0) then
|
|
|
|
try
|
|
|
|
TempRect := aRect;
|
|
|
|
Result := OffsetRect(TempRect, - TempRect.Left, - TempRect.Top) and
|
|
|
|
CopyDCToBitmapStream(TempDC, TempRect, aStream);
|
|
|
|
finally
|
|
|
|
ReleaseDC(FRenderCompHWND, TempDC);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
{$ENDIF}
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TFMXChromium.TakeSnapshot(var aBitmap : TBitmap; const aRect : System.Types.TRect) : boolean;
|
|
|
|
var
|
|
|
|
TempStream : TMemoryStream;
|
|
|
|
begin
|
|
|
|
Result := False;
|
|
|
|
TempStream := nil;
|
|
|
|
|
|
|
|
if FIsOSR or (aBitmap = nil) then exit;
|
|
|
|
|
|
|
|
try
|
|
|
|
TempStream := TMemoryStream.Create;
|
|
|
|
|
2021-08-19 14:20:01 +02:00
|
|
|
if SaveAsBitmapStream(TempStream, aRect) then
|
2020-01-02 21:02:47 +02:00
|
|
|
begin
|
|
|
|
aBitmap.LoadFromStream(TempStream);
|
|
|
|
Result := True;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
FreeAndNil(TempStream);
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
2018-01-25 22:34:04 +02:00
|
|
|
end.
|