CEF4Delphi/source/uCEFJson.pas
2023-11-16 16:58:47 +01:00

413 lines
13 KiB
ObjectPascal

unit uCEFJson;
{$IFDEF FPC}
{$MODE OBJFPC}{$H+}
{$ENDIF}
{$I cef.inc}
{$IFNDEF TARGET_64BITS}{$ALIGN ON}{$ENDIF}
{$MINENUMSIZE 4}
interface
uses
{$IFDEF DELPHI16_UP}
System.Classes, System.SysUtils,
{$ELSE}
Classes, SysUtils,
{$ENDIF}
uCEFInterfaces, uCEFTypes, uCEFConstants;
type
TCEFJson = class
public
/// <summary>
/// Returns the ICefValue value at the specified key.
/// </summary>
class function ReadValue(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefValue) : boolean;
/// <summary>
/// Returns the boolean value at the specified key.
/// </summary>
class function ReadBoolean(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : boolean) : boolean;
/// <summary>
/// Returns the integer value at the specified key.
/// </summary>
class function ReadInteger(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : integer) : boolean;
/// <summary>
/// Returns the double value at the specified key.
/// </summary>
class function ReadDouble(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : double) : boolean;
/// <summary>
/// Returns the ustring value at the specified key.
/// </summary>
class function ReadString(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ustring) : boolean;
/// <summary>
/// Returns the ICefBinaryValue value at the specified key.
/// </summary>
class function ReadBinary(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefBinaryValue) : boolean;
/// <summary>
/// Returns the ICefDictionaryValue value at the specified key.
/// </summary>
class function ReadDictionary(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefDictionaryValue) : boolean;
/// <summary>
/// Returns the ICefListValue value at the specified key.
/// </summary>
class function ReadList(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefListValue) : boolean;
/// <summary>
/// Parses the specified |json_string| and returns a dictionary or list
/// representation. If JSON parsing fails this function returns NULL.
/// </summary>
class function Parse(const jsonString: ustring; options: TCefJsonParserOptions = JSON_PARSER_RFC): ICefValue; overload;
/// <summary>
/// Parses the specified UTF8-encoded |json| buffer of size |json_size| and
/// returns a dictionary or list representation. If JSON parsing fails this
/// function returns NULL.
/// </summary>
class function Parse(const json: Pointer; json_size: NativeUInt; options: TCefJsonParserOptions = JSON_PARSER_RFC): ICefValue; overload;
/// <summary>
/// Parses the specified |json_string| and returns a dictionary or list
/// representation. If JSON parsing fails this function returns NULL and
/// populates |error_msg_out| with a formatted error message.
/// </summary>
class function ParseAndReturnError(const jsonString: ustring; options: TCefJsonParserOptions; out errorMsgOut: ustring): ICefValue;
/// <summary>
/// Generates a JSON string from the specified root |node|.
/// Returns an NULL string on failure. This function
/// requires exclusive access to |node| including any underlying data.
/// </summary>
class function Write(const node: ICefValue; options: TCefJsonWriterOptions = JSON_WRITER_DEFAULT): ustring; overload;
/// <summary>
/// Generates a JSON string from the specified root |node|.
/// Returns an NULL string on failure. This function
/// requires exclusive access to |node| including any underlying data.
/// </summary>
class function Write(const node: ICefDictionaryValue; options: TCefJsonWriterOptions = JSON_WRITER_DEFAULT): ustring; overload;
/// <summary>
/// Generates a JSON string from the specified root |node|.
/// Returns an NULL string on failure. This function
/// requires exclusive access to |node| including any underlying data.
/// </summary>
class function Write(const node: ICefValue; var aRsltStrings: TStringList): boolean; overload;
/// <summary>
/// Generates a JSON string from the specified root |node|.
/// Returns an NULL string on failure. This function
/// requires exclusive access to |node| including any underlying data.
/// </summary>
class function Write(const node: ICefDictionaryValue; var aRsltStrings: TStringList): boolean; overload;
/// <summary>
/// Saves the JSON data in |node| to a file in aFileName.
/// </summary>
class function SaveToFile(const node: ICefValue; const aFileName: ustring): boolean; overload;
/// <summary>
/// Saves the JSON data in |node| to a file in aFileName.
/// </summary>
class function SaveToFile(const node: ICefDictionaryValue; const aFileName: ustring): boolean; overload;
/// <summary>
/// Loads the JSON data in |aFileName| using the |encoding| and returns an ICefValue node in |aRsltNode|.
/// </summary>
class function LoadFromFile(const aFileName: ustring; var aRsltNode: ICefValue; {$IFDEF DELPHI12_UP}encoding: TEncoding = nil;{$ENDIF} options: TCefJsonParserOptions = JSON_PARSER_RFC): boolean;
end;
implementation
uses
uCEFLibFunctions, uCEFApplicationCore, uCEFMiscFunctions, uCEFValue;
class function TCEFJson.ReadValue(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefValue) : boolean;
begin
Result := False;
aValue := nil;
if (aDictionary <> nil) then
begin
aValue := aDictionary.GetValue(aKey);
Result := (aValue <> nil);
end;
end;
class function TCEFJson.ReadBoolean(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : boolean) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := False;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_BOOL) then
begin
aValue := TempValue.GetBool;
Result := True;
end;
end;
class function TCEFJson.ReadInteger(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : integer) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := 0;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_INT) then
begin
aValue := TempValue.GetInt;
Result := True;
end;
end;
class function TCEFJson.ReadDouble(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : double) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := 0;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_DOUBLE) then
begin
aValue := TempValue.GetDouble;
Result := True;
end;
end;
class function TCEFJson.ReadString(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ustring) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := '';
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_STRING) then
begin
aValue := TempValue.GetString;
Result := True;
end;
end;
class function TCEFJson.ReadBinary(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefBinaryValue) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := nil;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_BINARY) then
begin
aValue := TempValue.GetBinary;
Result := True;
end;
end;
class function TCEFJson.ReadDictionary(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefDictionaryValue) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := nil;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_DICTIONARY) then
begin
aValue := TempValue.GetDictionary;
Result := True;
end;
end;
class function TCEFJson.ReadList(const aDictionary : ICefDictionaryValue; const aKey : string; var aValue : ICefListValue) : boolean;
var
TempValue : ICefValue;
begin
Result := False;
aValue := nil;
if ReadValue(aDictionary, aKey, TempValue) and
(TempValue.GetType = VTYPE_LIST) then
begin
aValue := TempValue.GetList;
Result := True;
end;
end;
class function TCEFJson.Parse(const jsonString: ustring; options: TCefJsonParserOptions): ICefValue;
var
TempJSON : TCefString;
begin
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
begin
TempJSON := CefString(jsonString);
Result := TCefValueRef.UnWrap(cef_parse_json(@TempJSON, options));
end
else
Result := nil;
end;
// json must be a pointer to a UTF8 string
class function TCEFJson.Parse(const json: Pointer; json_size: NativeUInt; options: TCefJsonParserOptions): ICefValue;
begin
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded and (json <> nil) and (json_size > 0) then
Result := TCefValueRef.UnWrap(cef_parse_json_buffer(json, json_size, options))
else
Result := nil;
end;
class function TCEFJson.ParseAndReturnError(const jsonString : ustring;
options : TCefJsonParserOptions;
out errorMsgOut : ustring): ICefValue;
var
TempJSON, TempError : TCefString;
begin
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded then
begin
CefStringInitialize(@TempError);
TempJSON := CefString(jsonString);
Result := TCefValueRef.UnWrap(cef_parse_jsonand_return_error(@TempJSON, options, @TempError));
errorMsgOut := CefStringClearAndGet(@TempError);
end
else
begin
Result := nil;
errorMsgOut := '';
end;
end;
class function TCEFJson.Write(const node: ICefValue; options: TCefJsonWriterOptions): ustring;
begin
if (GlobalCEFApp <> nil) and GlobalCEFApp.LibLoaded and (node <> nil) then
Result := CefStringFreeAndGet(cef_write_json(CefGetData(node), options))
else
Result := '';
end;
class function TCEFJson.Write(const node: ICefDictionaryValue; options: TCefJsonWriterOptions): ustring;
var
TempValue : ICefValue;
begin
Result := '';
if (node = nil) then exit;
try
TempValue := TCefValueRef.New;
TempValue.SetDictionary(node);
Result := Write(TempValue, options);
finally
TempValue := nil;
end;
end;
class function TCEFJson.Write(const node: ICefValue; var aRsltStrings: TStringList): boolean;
var
TempJSON : ustring;
begin
Result := False;
if (aRsltStrings <> nil) then
begin
TempJSON := Write(node, JSON_WRITER_PRETTY_PRINT);
if (length(TempJSON) > 0) then
begin
aRsltStrings.SetText(@TempJSON[1]);
Result := True;
end;
end;
end;
class function TCEFJson.Write(const node: ICefDictionaryValue; var aRsltStrings: TStringList): boolean;
var
TempJSON : ustring;
begin
Result := False;
if (aRsltStrings <> nil) then
begin
TempJSON := Write(node, JSON_WRITER_PRETTY_PRINT);
if (length(TempJSON) > 0) then
begin
aRsltStrings.SetText(@TempJSON[1]);
Result := True;
end;
end;
end;
class function TCEFJson.SaveToFile(const node: ICefValue; const aFileName: ustring): boolean;
var
TempJSON : TStringList;
begin
Result := False;
TempJSON := nil;
try
try
TempJSON := TStringList.Create;
if Write(node, TempJSON) then
begin
TempJSON.SaveToFile(aFileName);
Result := True;
end;
except
on e : exception do
if CustomExceptionHandler('TCEFJson.SaveToFile', e) then raise;
end;
finally
if (TempJSON <> nil) then FreeAndNil(TempJSON);
end;
end;
class function TCEFJson.SaveToFile(const node: ICefDictionaryValue; const aFileName: ustring): boolean;
var
TempJSON : TStringList;
begin
Result := False;
TempJSON := nil;
try
try
TempJSON := TStringList.Create;
if Write(node, TempJSON) then
begin
TempJSON.SaveToFile(aFileName);
Result := True;
end;
except
on e : exception do
if CustomExceptionHandler('TCEFJson.SaveToFile', e) then raise;
end;
finally
if (TempJSON <> nil) then FreeAndNil(TempJSON);
end;
end;
class function TCEFJson.LoadFromFile(const aFileName: ustring; var aRsltNode: ICefValue; {$IFDEF DELPHI12_UP}encoding: TEncoding;{$ENDIF} options: TCefJsonParserOptions): boolean;
var
TempJSON : TStringList;
begin
Result := False;
TempJSON := nil;
try
try
if (length(aFileName) > 0) and FileExists(aFileName) then
begin
TempJSON := TStringList.Create;
TempJSON.LoadFromFile(aFileName{$IFDEF DELPHI12_UP}, encoding{$ENDIF});
aRsltNode := Parse(TempJSON.Text, options);
Result := True;
end;
except
on e : exception do
if CustomExceptionHandler('TCEFJson.LoadFromFile', e) then raise;
end;
finally
if (TempJSON <> nil) then FreeAndNil(TempJSON);
end;
end;
end.