2017-01-27 16:37:51 +01:00
|
|
|
unit uCEFResponseFilter;
|
|
|
|
|
2018-05-12 14:50:54 +02:00
|
|
|
{$IFDEF FPC}
|
|
|
|
{$MODE OBJFPC}{$H+}
|
|
|
|
{$ENDIF}
|
|
|
|
|
2017-02-05 20:56:46 +01:00
|
|
|
{$I cef.inc}
|
|
|
|
|
2022-02-19 18:56:41 +01:00
|
|
|
{$IFNDEF TARGET_64BITS}{$ALIGN ON}{$ENDIF}
|
|
|
|
{$MINENUMSIZE 4}
|
|
|
|
|
2017-01-27 16:37:51 +01:00
|
|
|
interface
|
|
|
|
|
|
|
|
uses
|
2017-03-16 19:09:42 +01:00
|
|
|
uCEFBaseRefCounted, uCEFInterfaces, uCEFTypes;
|
2017-01-27 16:37:51 +01:00
|
|
|
|
|
|
|
type
|
2018-02-03 17:52:48 +01:00
|
|
|
TOnFilterEvent = procedure(Sender: TObject; data_in: Pointer; data_in_size: NativeUInt; var data_in_read: NativeUInt; data_out: Pointer; data_out_size : NativeUInt; var data_out_written: NativeUInt; var aResult : TCefResponseFilterStatus) of object;
|
|
|
|
TOnInitFilterEvent = procedure(Sender: TObject; var aResult : boolean) of object;
|
|
|
|
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Reference class implementing a resource filter.
|
|
|
|
/// </summary>
|
|
|
|
/// <remarks>
|
|
|
|
/// <para>The functions of this interface will be called on the browser process IO thread.</para>
|
|
|
|
/// <para><see href="https://bitbucket.org/chromiumembedded/cef/src/master/include/capi/cef_response_filter_capi.h">CEF source file: /include/capi/cef_response_filter_capi.h (cef_response_filter_t)</see></para>
|
|
|
|
/// </remarks>
|
2019-06-16 10:31:13 +02:00
|
|
|
TCefResponseFilterRef = class(TCefBaseRefCountedRef, ICefResponseFilter)
|
|
|
|
protected
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Initialize the response filter. Will only be called a single time. The
|
|
|
|
/// filter will not be installed if this function returns false (0).
|
|
|
|
/// </summary>
|
2019-06-16 10:31:13 +02:00
|
|
|
function InitFilter: Boolean; virtual;
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// <para>Called to filter a chunk of data. Expected usage is as follows:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. Read input data from |data_in| and set |data_in_read| to the number of
|
|
|
|
/// bytes that were read up to a maximum of |data_in_size|. |data_in| will
|
|
|
|
/// be NULL if |data_in_size| is zero.
|
|
|
|
/// 2. Write filtered output data to |data_out| and set |data_out_written| to
|
|
|
|
/// the number of bytes that were written up to a maximum of
|
|
|
|
/// |data_out_size|. If no output data was written then all data must be
|
|
|
|
/// read from |data_in| (user must set |data_in_read| = |data_in_size|).
|
|
|
|
/// 3. Return RESPONSE_FILTER_DONE if all output data was written or
|
|
|
|
/// RESPONSE_FILTER_NEED_MORE_DATA if output data is still pending.
|
|
|
|
/// </code>
|
|
|
|
/// <para>This function will be called repeatedly until the input buffer has been
|
|
|
|
/// fully read (user sets |data_in_read| = |data_in_size|) and there is no
|
|
|
|
/// more input data to filter (the resource response is complete). This
|
|
|
|
/// function may then be called an additional time with an NULL input buffer
|
|
|
|
/// if the user filled the output buffer (set |data_out_written| =
|
|
|
|
/// |data_out_size|) and returned RESPONSE_FILTER_NEED_MORE_DATA to indicate
|
|
|
|
/// that output data is still pending.</para>
|
|
|
|
/// <para>Calls to this function will stop when one of the following conditions is
|
|
|
|
/// met:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. There is no more input data to filter (the resource response is
|
|
|
|
/// complete) and the user sets |data_out_written| = 0 or returns
|
|
|
|
/// RESPONSE_FILTER_DONE to indicate that all data has been written, or;
|
|
|
|
/// 2. The user returns RESPONSE_FILTER_ERROR to indicate an error.
|
|
|
|
/// </code>
|
|
|
|
/// <para>Do not keep a reference to the buffers passed to this function.</para>
|
|
|
|
/// </summary>
|
2019-06-16 10:31:13 +02:00
|
|
|
function Filter(data_in: Pointer; data_in_size: NativeUInt; var data_in_read: NativeUInt; data_out: Pointer; data_out_size : NativeUInt; var data_out_written: NativeUInt): TCefResponseFilterStatus; virtual;
|
|
|
|
|
|
|
|
public
|
|
|
|
class function UnWrap(data: Pointer): ICefResponseFilter;
|
|
|
|
end;
|
2018-02-03 17:52:48 +01:00
|
|
|
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Owned class implementing a resource filter.
|
|
|
|
/// </summary>
|
|
|
|
/// <remarks>
|
|
|
|
/// <para>The functions of this interface will be called on the browser process IO thread.</para>
|
|
|
|
/// <para><see href="https://bitbucket.org/chromiumembedded/cef/src/master/include/capi/cef_response_filter_capi.h">CEF source file: /include/capi/cef_response_filter_capi.h (cef_response_filter_t)</see></para>
|
|
|
|
/// </remarks>
|
2017-03-16 19:09:42 +01:00
|
|
|
TCefResponseFilterOwn = class(TCefBaseRefCountedOwn, ICefResponseFilter)
|
2018-02-03 17:52:48 +01:00
|
|
|
protected
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Initialize the response filter. Will only be called a single time. The
|
|
|
|
/// filter will not be installed if this function returns false (0).
|
|
|
|
/// </summary>
|
2018-02-03 17:52:48 +01:00
|
|
|
function InitFilter: Boolean; virtual; abstract;
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// <para>Called to filter a chunk of data. Expected usage is as follows:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. Read input data from |data_in| and set |data_in_read| to the number of
|
|
|
|
/// bytes that were read up to a maximum of |data_in_size|. |data_in| will
|
|
|
|
/// be NULL if |data_in_size| is zero.
|
|
|
|
/// 2. Write filtered output data to |data_out| and set |data_out_written| to
|
|
|
|
/// the number of bytes that were written up to a maximum of
|
|
|
|
/// |data_out_size|. If no output data was written then all data must be
|
|
|
|
/// read from |data_in| (user must set |data_in_read| = |data_in_size|).
|
|
|
|
/// 3. Return RESPONSE_FILTER_DONE if all output data was written or
|
|
|
|
/// RESPONSE_FILTER_NEED_MORE_DATA if output data is still pending.
|
|
|
|
/// </code>
|
|
|
|
/// <para>This function will be called repeatedly until the input buffer has been
|
|
|
|
/// fully read (user sets |data_in_read| = |data_in_size|) and there is no
|
|
|
|
/// more input data to filter (the resource response is complete). This
|
|
|
|
/// function may then be called an additional time with an NULL input buffer
|
|
|
|
/// if the user filled the output buffer (set |data_out_written| =
|
|
|
|
/// |data_out_size|) and returned RESPONSE_FILTER_NEED_MORE_DATA to indicate
|
|
|
|
/// that output data is still pending.</para>
|
|
|
|
/// <para>Calls to this function will stop when one of the following conditions is
|
|
|
|
/// met:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. There is no more input data to filter (the resource response is
|
|
|
|
/// complete) and the user sets |data_out_written| = 0 or returns
|
|
|
|
/// RESPONSE_FILTER_DONE to indicate that all data has been written, or;
|
|
|
|
/// 2. The user returns RESPONSE_FILTER_ERROR to indicate an error.
|
|
|
|
/// </code>
|
|
|
|
/// <para>Do not keep a reference to the buffers passed to this function.</para>
|
|
|
|
/// </summary>
|
2018-02-03 17:52:48 +01:00
|
|
|
function Filter(data_in: Pointer; data_in_size: NativeUInt; var data_in_read: NativeUInt; data_out: Pointer; data_out_size : NativeUInt; var data_out_written: NativeUInt): TCefResponseFilterStatus; virtual; abstract;
|
|
|
|
|
|
|
|
public
|
|
|
|
constructor Create; virtual;
|
|
|
|
end;
|
|
|
|
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Custom class implementing a resource filter with events.
|
|
|
|
/// </summary>
|
|
|
|
/// <remarks>
|
|
|
|
/// <para>The functions and events of this interface will be called on the browser process IO thread.</para>
|
|
|
|
/// <para><see href="https://bitbucket.org/chromiumembedded/cef/src/master/include/capi/cef_response_filter_capi.h">CEF source file: /include/capi/cef_response_filter_capi.h (cef_response_filter_t)</see></para>
|
|
|
|
/// </remarks>
|
2018-02-03 17:52:48 +01:00
|
|
|
TCustomResponseFilter = class(TCefResponseFilterOwn)
|
|
|
|
protected
|
|
|
|
FOnFilter : TOnFilterEvent;
|
|
|
|
FOnInitFilter : TOnInitFilterEvent;
|
|
|
|
|
|
|
|
function InitFilter: Boolean; override;
|
|
|
|
function Filter(data_in: Pointer; data_in_size: NativeUInt; var data_in_read: NativeUInt; data_out: Pointer; data_out_size : NativeUInt; var data_out_written: NativeUInt): TCefResponseFilterStatus; override;
|
|
|
|
|
|
|
|
public
|
|
|
|
constructor Create; override;
|
|
|
|
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// <para>OnFilter is triggered when ICefResponseFilter.Filter is executed to filter a chunk of data.</para>
|
|
|
|
/// <para>Expected usage is as follows:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. Read input data from |data_in| and set |data_in_read| to the number of
|
|
|
|
/// bytes that were read up to a maximum of |data_in_size|. |data_in| will
|
|
|
|
/// be NULL if |data_in_size| is zero.
|
|
|
|
/// 2. Write filtered output data to |data_out| and set |data_out_written| to
|
|
|
|
/// the number of bytes that were written up to a maximum of
|
|
|
|
/// |data_out_size|. If no output data was written then all data must be
|
|
|
|
/// read from |data_in| (user must set |data_in_read| = |data_in_size|).
|
|
|
|
/// 3. Return RESPONSE_FILTER_DONE if all output data was written or
|
|
|
|
/// RESPONSE_FILTER_NEED_MORE_DATA if output data is still pending.
|
|
|
|
/// </code>
|
|
|
|
/// <para>This function will be called repeatedly until the input buffer has been
|
|
|
|
/// fully read (user sets |data_in_read| = |data_in_size|) and there is no
|
|
|
|
/// more input data to filter (the resource response is complete). This
|
|
|
|
/// function may then be called an additional time with an NULL input buffer
|
|
|
|
/// if the user filled the output buffer (set |data_out_written| =
|
|
|
|
/// |data_out_size|) and returned RESPONSE_FILTER_NEED_MORE_DATA to indicate
|
|
|
|
/// that output data is still pending.</para>
|
|
|
|
/// <para>Calls to this function will stop when one of the following conditions is
|
|
|
|
/// met:</para>
|
|
|
|
/// <code>
|
|
|
|
/// 1. There is no more input data to filter (the resource response is
|
|
|
|
/// complete) and the user sets |data_out_written| = 0 or returns
|
|
|
|
/// RESPONSE_FILTER_DONE to indicate that all data has been written, or;
|
|
|
|
/// 2. The user returns RESPONSE_FILTER_ERROR to indicate an error.
|
|
|
|
/// </code>
|
|
|
|
/// <para>Do not keep a reference to the buffers passed to this function.</para>
|
|
|
|
/// </summary>
|
2018-02-03 17:52:48 +01:00
|
|
|
property OnFilter : TOnFilterEvent read FOnFilter write FOnFilter;
|
2024-04-20 17:46:13 +02:00
|
|
|
/// <summary>
|
|
|
|
/// OnInitFilter is triggered when ICefResponseFilter.InitFilter is executed.
|
|
|
|
/// Set the aResult parameter to install the filter.
|
|
|
|
/// </summary>
|
2018-02-03 17:52:48 +01:00
|
|
|
property OnInitFilter : TOnInitFilterEvent read FOnInitFilter write FOnInitFilter;
|
2017-01-27 16:37:51 +01:00
|
|
|
end;
|
|
|
|
|
|
|
|
implementation
|
|
|
|
|
|
|
|
uses
|
|
|
|
uCEFMiscFunctions, uCEFLibFunctions;
|
|
|
|
|
2018-02-03 17:52:48 +01:00
|
|
|
// TCefResponseFilterOwn
|
|
|
|
|
2017-01-27 16:37:51 +01:00
|
|
|
function cef_response_filter_init_filter(self: PCefResponseFilter): Integer; stdcall;
|
2018-03-29 20:02:04 +02:00
|
|
|
var
|
|
|
|
TempObject : TObject;
|
2017-01-27 16:37:51 +01:00
|
|
|
begin
|
2018-03-29 20:02:04 +02:00
|
|
|
Result := Ord(True);
|
|
|
|
TempObject := CefGetObject(self);
|
|
|
|
|
|
|
|
if (TempObject <> nil) and (TempObject is TCefResponseFilterOwn) then
|
|
|
|
Result := Ord(TCefResponseFilterOwn(TempObject).InitFilter());
|
2017-01-27 16:37:51 +01:00
|
|
|
end;
|
|
|
|
|
2018-03-29 20:02:04 +02:00
|
|
|
function cef_response_filter_filter( self : PCefResponseFilter;
|
|
|
|
data_in : Pointer;
|
|
|
|
data_in_size : NativeUInt;
|
|
|
|
var data_in_read : NativeUInt;
|
|
|
|
data_out : Pointer;
|
|
|
|
data_out_size : NativeUInt;
|
|
|
|
var data_out_written : NativeUInt): TCefResponseFilterStatus; stdcall;
|
|
|
|
var
|
|
|
|
TempObject : TObject;
|
2017-01-27 16:37:51 +01:00
|
|
|
begin
|
2018-03-29 20:02:04 +02:00
|
|
|
Result := RESPONSE_FILTER_DONE;
|
|
|
|
TempObject := CefGetObject(self);
|
|
|
|
|
|
|
|
if (TempObject <> nil) and (TempObject is TCefResponseFilterOwn) then
|
|
|
|
Result := TCefResponseFilterOwn(TempObject).Filter(data_in, data_in_size, data_in_read,
|
|
|
|
data_out, data_out_size, data_out_written);
|
2017-01-27 16:37:51 +01:00
|
|
|
end;
|
|
|
|
|
|
|
|
constructor TCefResponseFilterOwn.Create;
|
|
|
|
begin
|
2018-03-29 20:02:04 +02:00
|
|
|
inherited CreateData(SizeOf(TCefResponseFilter));
|
2018-02-03 17:52:48 +01:00
|
|
|
|
2017-01-27 16:37:51 +01:00
|
|
|
with PCefResponseFilter(FData)^ do
|
2018-02-03 17:52:48 +01:00
|
|
|
begin
|
2018-05-12 14:50:54 +02:00
|
|
|
init_filter := {$IFDEF FPC}@{$ENDIF}cef_response_filter_init_filter;
|
|
|
|
filter := {$IFDEF FPC}@{$ENDIF}cef_response_filter_filter;
|
2018-02-03 17:52:48 +01:00
|
|
|
end;
|
2017-01-27 16:37:51 +01:00
|
|
|
end;
|
|
|
|
|
2018-02-03 17:52:48 +01:00
|
|
|
|
|
|
|
// TCustomResponseFilter
|
|
|
|
|
|
|
|
|
|
|
|
constructor TCustomResponseFilter.Create;
|
|
|
|
begin
|
|
|
|
inherited Create;
|
|
|
|
|
|
|
|
FOnFilter := nil;
|
|
|
|
FOnInitFilter := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TCustomResponseFilter.InitFilter: Boolean;
|
|
|
|
begin
|
|
|
|
Result := True;
|
|
|
|
if assigned(FOnInitFilter) then FOnInitFilter(self, Result);
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TCustomResponseFilter.Filter( data_in : Pointer;
|
|
|
|
data_in_size : NativeUInt;
|
|
|
|
var data_in_read : NativeUInt;
|
|
|
|
data_out : Pointer;
|
|
|
|
data_out_size : NativeUInt;
|
|
|
|
var data_out_written : NativeUInt) : TCefResponseFilterStatus;
|
|
|
|
begin
|
|
|
|
Result := RESPONSE_FILTER_DONE;
|
|
|
|
|
|
|
|
if assigned(FOnFilter) then
|
|
|
|
FOnFilter(self,
|
|
|
|
data_in, data_in_size, data_in_read,
|
|
|
|
data_out, data_out_size, data_out_written,
|
|
|
|
Result);
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
2019-06-16 10:31:13 +02:00
|
|
|
// TCefResponseFilterRef
|
|
|
|
|
|
|
|
class function TCefResponseFilterRef.UnWrap(data: Pointer): ICefResponseFilter;
|
|
|
|
begin
|
|
|
|
if (data <> nil) then
|
|
|
|
Result := Create(data) as ICefResponseFilter
|
|
|
|
else
|
|
|
|
Result := nil;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TCefResponseFilterRef.InitFilter: Boolean;
|
|
|
|
begin
|
|
|
|
Result := PCefResponseFilter(FData)^.init_filter(PCefResponseFilter(FData)) <> 0;
|
|
|
|
end;
|
|
|
|
|
|
|
|
function TCefResponseFilterRef.Filter( data_in : Pointer;
|
|
|
|
data_in_size : NativeUInt;
|
|
|
|
var data_in_read : NativeUInt;
|
|
|
|
data_out : Pointer;
|
|
|
|
data_out_size : NativeUInt;
|
|
|
|
var data_out_written : NativeUInt) : TCefResponseFilterStatus;
|
|
|
|
begin
|
|
|
|
Result := PCefResponseFilter(FData)^.filter(PCefResponseFilter(FData),
|
|
|
|
data_in,
|
|
|
|
data_in_size,
|
|
|
|
data_in_read,
|
|
|
|
data_out,
|
|
|
|
data_out_size,
|
|
|
|
data_out_written);
|
|
|
|
end;
|
|
|
|
|
2017-01-27 16:37:51 +01:00
|
|
|
end.
|