2018-10-28 12:52:14 +01:00
// ************************************************************************
// ***************************** CEF4Delphi *******************************
// ************************************************************************
//
2019-10-19 10:58:34 +02:00
// CEF4Delphi is based on DCEF3 which uses CEF to embed a chromium-based
2018-10-28 12:52:14 +01: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
//
2020-01-01 12:48:10 +01:00
// Copyright <20> 2020 Salvador Diaz Fau. All rights reserved.
2018-10-28 12:52:14 +01: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 uPostInspectorBrowser;
{$I cef.inc}
interface
uses
{$IFDEF DELPHI16_UP}
Winapi . Windows, Winapi . Messages, System. SysUtils, System. Variants, System. Classes, Vcl. Graphics,
Vcl. Controls, Vcl. Forms, Vcl. Dialogs, Vcl. StdCtrls, Vcl. ExtCtrls, System. SyncObjs,
{$ELSE}
Windows, Messages, SysUtils, Variants, Classes, Graphics,
Controls, Forms, Dialogs, StdCtrls, ExtCtrls, SyncObjs,
{$ENDIF}
uCEFChromium, uCEFWindowParent, uCEFInterfaces, uCEFConstants, uCEFTypes,
2019-10-14 15:39:27 +02:00
uCEFWinControl, uCEFSentinel;
2018-10-28 12:52:14 +01:00
const
CEF_SHOWDATA = WM_APP + $B00 ;
type
2019-10-14 15:39:27 +02:00
{ TForm1 }
2018-10-28 12:52:14 +01:00
TForm1 = class( TForm)
AddressPnl: TPanel;
2019-10-14 15:39:27 +02:00
CEFSentinel1: TCEFSentinel;
2018-10-28 12:52:14 +01:00
GoBtn: TButton;
Timer1: TTimer;
Chromium1: TChromium;
CEFWindowParent1: TCEFWindowParent;
Memo1: TMemo;
AddressCb: TComboBox;
Splitter1: TSplitter;
2019-10-14 15:39:27 +02:00
procedure CEFSentinel1Close( Sender: TObject) ;
2018-10-28 12:52:14 +01:00
procedure GoBtnClick( Sender: TObject) ;
procedure FormShow( Sender: TObject) ;
procedure Timer1Timer( Sender: TObject) ;
procedure Chromium1AfterCreated( Sender: TObject; const browser: ICefBrowser) ;
procedure Chromium1BeforePopup( Sender: TObject;
const browser: ICefBrowser; const frame: ICefFrame; const targetUrl,
targetFrameName: ustring;
targetDisposition: TCefWindowOpenDisposition; userGesture: Boolean ;
const popupFeatures: TCefPopupFeatures; var windowInfo: TCefWindowInfo;
var client: ICefClient; var settings: TCefBrowserSettings;
2019-06-16 10:31:13 +02:00
var extra_info: ICefDictionaryValue;
2018-10-28 12:52:14 +01:00
var noJavascriptAccess: Boolean ; var Result : Boolean ) ;
procedure FormCreate( Sender: TObject) ;
procedure FormCloseQuery( Sender: TObject; var CanClose: Boolean ) ;
procedure Chromium1Close( Sender: TObject; const browser: ICefBrowser;
2019-03-28 10:40:36 +01:00
var aAction : TCefCloseBrowserAction) ;
2018-10-28 12:52:14 +01:00
procedure Chromium1BeforeClose( Sender: TObject;
const browser: ICefBrowser) ;
procedure FormDestroy( Sender: TObject) ;
procedure Chromium1BeforeResourceLoad( Sender: TObject;
const browser: ICefBrowser; const frame: ICefFrame;
const request: ICefRequest; const callback: ICefRequestCallback;
out Result : TCefReturnValue) ;
protected
// Variables to control when can we destroy the form safely
FCanClose : boolean ; // Set to True in TChromium.OnBeforeClose
FClosing : boolean ; // Set to True in the CloseQuery event.
FRequestSL : TStringList;
FRequestCS : TCriticalSection;
// You have to handle this two messages to call NotifyMoveOrResizeStarted or some page elements will be misaligned.
procedure WMMove( var aMessage : TWMMove) ; message WM_MOVE;
procedure WMMoving( var aMessage : TMessage) ; message WM_MOVING;
// You also have to handle these two messages to set GlobalCEFApp.OsmodalLoop
procedure WMEnterMenuLoop( var aMessage: TMessage) ; message WM_ENTERMENULOOP;
procedure WMExitMenuLoop( var aMessage: TMessage) ; message WM_EXITMENULOOP;
procedure BrowserCreatedMsg( var aMessage : TMessage) ; message CEF_AFTERCREATED;
procedure BrowserDestroyMsg( var aMessage : TMessage) ; message CEF_DESTROY;
procedure ShowDataMsg( var aMessage : TMessage) ; message CEF_SHOWDATA;
procedure HandleRequest( const request : ICefRequest; aIsMain : boolean ) ;
procedure HandleHeaderMap( const request : ICefRequest) ;
procedure HandlePostData( const request : ICefRequest) ;
procedure HandlePostDataBytes( const aElement : ICefPostDataElement) ;
public
{ Public declarations }
end ;
var
Form1: TForm1;
2019-06-19 16:53:26 +02:00
procedure CreateGlobalCEFApp;
2018-10-28 12:52:14 +01:00
implementation
2019-05-19 16:08:15 +02:00
{$R *.lfm}
2018-10-28 12:52:14 +01:00
uses
uCEFApplication, uCefMiscFunctions, uCEFStringMultimap;
// This demo shows how to inspect the data sent in all requests for all the resources.
// We use the TChromium.OnBeforeResourceLoad event to handle the request but this event
// is executed in a different thread. This means that we have to protect the data with a
// critical section when we handle it.
// After the request has been handled we send a custom message to the form (CEF_SHOWDATA)
// to add the information to the TMemo safely in the main thread.
// Destruction steps
// =================
// 1. FormCloseQuery sets CanClose to FALSE calls TChromium.CloseBrowser which triggers the TChromium.OnClose event.
// 2. TChromium.OnClose sends a CEFBROWSER_DESTROY message to destroy CEFWindowParent1 in the main thread, which triggers the TChromium.OnBeforeClose event.
2019-10-14 15:39:27 +02:00
// 3. TChromium.OnBeforeClose calls TCEFSentinel.Start, which will trigger TCEFSentinel.OnClose when the renderer processes are closed.
// 4. TCEFSentinel.OnClose sets FCanClose := True and sends WM_CLOSE to the form.
2019-06-19 16:53:26 +02:00
procedure CreateGlobalCEFApp;
begin
2019-10-01 18:47:24 +02:00
GlobalCEFApp : = TCefApplication. Create;
//GlobalCEFApp.LogFile := 'cef.log';
//GlobalCEFApp.LogSeverity := LOGSEVERITY_VERBOSE;
2019-06-19 16:53:26 +02:00
end ;
2018-10-28 12:52:14 +01:00
procedure TForm1. FormCloseQuery( Sender: TObject; var CanClose: Boolean ) ;
begin
CanClose : = FCanClose;
if not( FClosing) then
begin
FClosing : = True ;
Visible : = False ;
Chromium1. CloseBrowser( True ) ;
end ;
end ;
procedure TForm1. FormCreate( Sender: TObject) ;
begin
FCanClose : = False ;
FClosing : = False ;
Chromium1. DefaultURL : = AddressCb. Text ;
FRequestSL : = TStringList. Create;
FRequestCS : = TCriticalSection. Create;
end ;
procedure TForm1. FormDestroy( Sender: TObject) ;
begin
FRequestSL. Free;
FRequestCS. Free;
end ;
procedure TForm1. FormShow( Sender: TObject) ;
begin
// You *MUST* call CreateBrowser to create and initialize the browser.
// This will trigger the AfterCreated event when the browser is fully
// initialized and ready to receive commands.
// GlobalCEFApp.GlobalContextInitialized has to be TRUE before creating any browser
// If it's not initialized yet, we use a simple timer to create the browser later.
if not( Chromium1. CreateBrowser( CEFWindowParent1) ) then Timer1. Enabled : = True ;
end ;
procedure TForm1. Chromium1AfterCreated( Sender: TObject; const browser: ICefBrowser) ;
begin
// Now the browser is fully initialized we can send a message to the main form to load the initial web page.
PostMessage( Handle, CEF_AFTERCREATED, 0 , 0 ) ;
end ;
procedure TForm1. HandleRequest( const request : ICefRequest; aIsMain : boolean ) ;
begin
try
if ( FRequestCS < > nil ) then FRequestCS. Acquire;
if ( FRequestSL < > nil ) then
begin
FRequestSL. Add( '--------------------' ) ;
FRequestSL. Add( 'URL : ' + request. url) ;
FRequestSL. Add( 'Method : ' + request. Method) ;
FRequestSL. Add( 'Main frame : ' + BoolToStr( aIsMain, true ) ) ;
HandleHeaderMap( request) ;
HandlePostData( request) ;
PostMessage( Handle, CEF_SHOWDATA, 0 , 0 ) ;
end ;
finally
if ( FRequestCS < > nil ) then FRequestCS. Release;
end ;
end ;
procedure TForm1. HandleHeaderMap( const request : ICefRequest) ;
var
TempHeaderMap : ICefStringMultimap;
i : NativeUInt ;
begin
try
TempHeaderMap : = TCefStringMultimapOwn. Create;
request. GetHeaderMap( TempHeaderMap) ;
if ( TempHeaderMap < > nil ) and ( TempHeaderMap. Size > 0 ) then
begin
FRequestSL. Add( '--------------------' ) ;
FRequestSL. Add( 'Headers :' ) ;
i : = 0 ;
while ( i < TempHeaderMap. Size) do
begin
FRequestSL. Add( TempHeaderMap. Key[ i] + '=' + TempHeaderMap. Value[ i] ) ;
inc( i) ;
end ;
end ;
except
on e : exception do
begin
if CustomExceptionHandler( 'TForm1.HandleHeaderMap' , e) then raise ;
end ;
end ;
end ;
procedure TForm1. HandlePostData( const request : ICefRequest) ;
var
TempPostData : ICefPostData;
2019-09-25 17:23:16 +02:00
TempArray : TCefPostDataElementArray;
2018-10-28 12:52:14 +01:00
i : integer ;
begin
2019-09-25 17:23:16 +02:00
TempArray : = nil ;
2018-10-28 12:52:14 +01:00
try
2019-09-25 17:23:16 +02:00
try
TempPostData : = request. PostData;
2018-10-28 12:52:14 +01:00
2019-09-25 17:23:16 +02:00
if ( TempPostData < > nil ) and ( TempPostData. GetElementCount > 0 ) then
begin
FRequestSL. Add( '--------------------' ) ;
FRequestSL. Add( 'POST data :' ) ;
if TempPostData. HasExcludedElements then
FRequestSL. Add( 'Has excluded elements! (For example, multi-part file upload data.)' ) ;
2018-10-28 12:52:14 +01:00
2019-09-25 17:23:16 +02:00
TempPostData. GetElements( TempPostData. GetElementCount, TempArray) ;
2018-10-28 12:52:14 +01:00
2019-09-25 17:23:16 +02:00
i : = 0 ;
while ( i < length( TempArray) ) do
begin
FRequestSL. Add( 'Element : ' + inttostr( i) ) ;
FRequestSL. Add( 'Size : ' + inttostr( TempArray[ i] . GetBytesCount) ) ;
case TempArray[ i] . GetType of
PDE_TYPE_BYTES :
begin
FRequestSL. Add( 'Type : Bytes' ) ;
HandlePostDataBytes( TempArray[ i] ) ;
end ;
PDE_TYPE_FILE :
begin
FRequestSL. Add( 'Type : File' ) ;
// This element type can be read using a TBuffer like we do in HandlePostDataBytes
end
else
FRequestSL. Add( 'Type : Empty' ) ;
end ;
inc( i) ;
2018-10-28 12:52:14 +01:00
end ;
2019-09-25 17:23:16 +02:00
// Set interfaces to nil to release them
i : = 0 ;
while ( i < length( TempArray) ) do
begin
TempArray[ i] : = nil ;
inc( i) ;
end ;
end ;
except
on e : exception do
if CustomExceptionHandler( 'TForm1.HandlePostData' , e) then raise ;
end ;
finally
if ( TempArray < > nil ) then
begin
Finalize( TempArray) ;
TempArray : = nil ;
2018-10-28 12:52:14 +01:00
end ;
end ;
end ;
procedure TForm1. HandlePostDataBytes( const aElement : ICefPostDataElement) ;
var
TempStream : TStringStream;
2019-09-25 17:23:16 +02:00
TempBuffer : Pointer ;
2018-10-28 12:52:14 +01:00
TempSize : NativeUInt ;
begin
TempStream : = nil ;
TempBuffer : = nil ;
try
try
2019-09-25 17:23:16 +02:00
if ( aElement < > nil ) then
2018-10-28 12:52:14 +01:00
begin
2019-09-25 17:23:16 +02:00
TempSize : = aElement. GetBytesCount;
2018-10-28 12:52:14 +01:00
if ( TempSize > 0 ) then
begin
2019-09-25 17:23:16 +02:00
GetMem( TempBuffer, TempSize) ;
TempSize : = aElement. GetBytes( TempSize, TempBuffer) ;
if ( TempSize > 0 ) then
begin
TempStream : = TStringStream. Create( '' ) ;
TempStream. WriteBuffer( TempBuffer^ , TempSize) ;
TempStream. Seek( 0 , soBeginning) ;
FRequestSL. Add( TempStream. ReadString( TempSize) ) ;
end ;
2018-10-28 12:52:14 +01:00
end ;
end ;
except
on e : exception do
if CustomExceptionHandler( 'TForm1.HandlePostDataBytes' , e) then raise ;
end ;
finally
2019-09-25 17:23:16 +02:00
if ( TempStream < > nil ) then FreeAndNil( TempStream) ;
if ( TempBuffer < > nil ) then
begin
FreeMem( TempBuffer) ;
TempBuffer : = nil ;
end ;
2018-10-28 12:52:14 +01:00
end ;
end ;
procedure TForm1. Chromium1BeforeResourceLoad( Sender: TObject;
const browser: ICefBrowser; const frame: ICefFrame;
const request: ICefRequest; const callback: ICefRequestCallback;
out Result : TCefReturnValue) ;
begin
// This event is called before a resource request is loaded.
// The request object may be modified.
2019-10-13 18:50:23 +02:00
if ( frame < > nil ) and frame. IsValid then
HandleRequest( request, frame. IsMain) ;
2018-10-28 12:52:14 +01:00
Result : = RV_CONTINUE;
end ;
procedure TForm1. ShowDataMsg( var aMessage : TMessage) ;
begin
try
if ( FRequestCS < > nil ) then FRequestCS. Acquire;
if ( FRequestSL < > nil ) then
begin
Memo1. lines. AddStrings( FRequestSL) ;
FRequestSL. Clear;
end ;
finally
if ( FRequestCS < > nil ) then FRequestCS. Release;
// Scroll to the last memo line
SendMessage( Memo1. Handle, EM_LINESCROLL, 0 , Memo1. Lines. Count) ;
end ;
end ;
procedure TForm1. Chromium1BeforeClose( Sender: TObject; const browser: ICefBrowser) ;
begin
2019-10-14 15:39:27 +02:00
CEFSentinel1. Start;
2018-10-28 12:52:14 +01:00
end ;
procedure TForm1. Chromium1BeforePopup( Sender: TObject;
const browser: ICefBrowser; const frame: ICefFrame; const targetUrl,
targetFrameName: ustring; targetDisposition: TCefWindowOpenDisposition;
userGesture: Boolean ; const popupFeatures: TCefPopupFeatures;
var windowInfo: TCefWindowInfo; var client: ICefClient;
2019-06-16 10:31:13 +02:00
var settings: TCefBrowserSettings;
var extra_info: ICefDictionaryValue;
var noJavascriptAccess: Boolean ;
2018-10-28 12:52:14 +01:00
var Result : Boolean ) ;
begin
// For simplicity, this demo blocks all popup windows and new tabs
Result : = ( targetDisposition in [ WOD_NEW_FOREGROUND_TAB, WOD_NEW_BACKGROUND_TAB, WOD_NEW_POPUP, WOD_NEW_WINDOW] ) ;
end ;
procedure TForm1. Chromium1Close( Sender: TObject;
2019-03-28 10:40:36 +01:00
const browser: ICefBrowser; var aAction : TCefCloseBrowserAction) ;
2018-10-28 12:52:14 +01:00
begin
PostMessage( Handle, CEF_DESTROY, 0 , 0 ) ;
2019-03-28 10:40:36 +01:00
aAction : = cbaDelay;
2018-10-28 12:52:14 +01:00
end ;
procedure TForm1. BrowserCreatedMsg( var aMessage : TMessage) ;
begin
Caption : = 'POST Inspector Browser' ;
AddressPnl. Enabled : = True ;
end ;
procedure TForm1. BrowserDestroyMsg( var aMessage : TMessage) ;
begin
CEFWindowParent1. Free;
end ;
procedure TForm1. GoBtnClick( Sender: TObject) ;
begin
// This will load the URL in the edit box
Chromium1. LoadURL( AddressCb. Text ) ;
end ;
2019-10-14 15:39:27 +02:00
procedure TForm1. CEFSentinel1Close( Sender: TObject) ;
begin
FCanClose : = True ;
PostMessage( Handle, WM_CLOSE, 0 , 0 ) ;
end ;
2018-10-28 12:52:14 +01:00
procedure TForm1. Timer1Timer( Sender: TObject) ;
begin
Timer1. Enabled : = False ;
if not( Chromium1. CreateBrowser( CEFWindowParent1) ) and not( Chromium1. Initialized) then
Timer1. Enabled : = True ;
end ;
procedure TForm1. WMMove( var aMessage : TWMMove) ;
begin
inherited ;
if ( Chromium1 < > nil ) then Chromium1. NotifyMoveOrResizeStarted;
end ;
procedure TForm1. WMMoving( var aMessage : TMessage) ;
begin
inherited ;
if ( Chromium1 < > nil ) then Chromium1. NotifyMoveOrResizeStarted;
end ;
procedure TForm1. WMEnterMenuLoop( var aMessage: TMessage) ;
begin
inherited ;
if ( aMessage. wParam = 0 ) and ( GlobalCEFApp < > nil ) then GlobalCEFApp. OsmodalLoop : = True ;
end ;
procedure TForm1. WMExitMenuLoop( var aMessage: TMessage) ;
begin
inherited ;
if ( aMessage. wParam = 0 ) and ( GlobalCEFApp < > nil ) then GlobalCEFApp. OsmodalLoop : = False ;
end ;
end .