delphimvcframework/sources/MVCFramework.View.Renderers.Mustache.pas

199 lines
5.4 KiB
ObjectPascal
Raw Normal View History

// ***************************************************************************
//
// Delphi MVC Framework
//
// Copyright (c) 2010-2023 Daniele Teti and the DMVCFramework Team
//
// https://github.com/danieleteti/delphimvcframework
//
// ***************************************************************************
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// ***************************************************************************
unit MVCFramework.View.Renderers.Mustache;
{$IFDEF LINUX}
This unit is not compatible with Linux
{$ENDIF}
interface
uses
MVCFramework, System.SysUtils, System.Generics.Collections,
MVCFramework.Commons, System.IOUtils, System.Classes, Data.DB, SynMustache;
type
{ This class implements the mustache view engine for server side views }
TMVCMustacheViewEngine = class(TMVCBaseViewEngine)
2017-05-25 16:57:49 +02:00
strict private
procedure PrepareModels;
private
class var fPartials: TSynMustachePartials;
var FJSONModel: string;
procedure LoadPartials;
public
procedure Execute(const ViewName: string; const OutputStream: TStream); override;
constructor Create(const AEngine: TMVCEngine; const AWebContext: TWebContext;
const AViewModel: TMVCViewDataObject;
const AViewDataSets: TObjectDictionary<string, TDataSet>;
const AContentType: string); override;
class destructor Destroy;
end;
implementation
uses
2017-05-25 16:57:49 +02:00
SynCommons,
JsonDataObjects,
2017-05-25 16:57:49 +02:00
MVCFramework.Serializer.Defaults,
MVCFramework.Serializer.Intf,
MVCFramework.DuckTyping,
MVCFramework.Serializer.JsonDataObjects.OptionalCustomTypes,
MVCFramework.Serializer.JsonDataObjects;
{$WARNINGS OFF}
2020-11-05 17:14:39 +01:00
type
TSynMustacheAccess = class(TSynMustache)
end;
var
gPartialsLoaded : Boolean = False;
constructor TMVCMustacheViewEngine.Create(const AEngine: TMVCEngine;
const AWebContext: TWebContext; const AViewModel: TMVCViewDataObject;
const AViewDataSets: TObjectDictionary<string, TDataSet>;
const AContentType: string);
begin
inherited;
LoadPartials;
end;
class destructor TMVCMustacheViewEngine.Destroy;
begin
fPartials.Free;
end;
2018-01-29 17:30:53 +01:00
procedure TMVCMustacheViewEngine.Execute(const ViewName: string; const OutputStream: TStream);
var
2018-02-20 22:36:18 +01:00
lViewFileName: string;
lViewTemplate: RawUTF8;
lViewEngine: TSynMustache;
2018-01-29 17:30:53 +01:00
lSW: TStreamWriter;
begin
2017-05-25 16:57:49 +02:00
PrepareModels;
2018-02-20 22:36:18 +01:00
lViewFileName := GetRealFileName(ViewName);
if not FileExists(lViewFileName) then
raise EMVCFrameworkViewException.CreateFmt('View [%s] not found', [ViewName]);
2018-02-20 22:36:18 +01:00
lViewTemplate := StringToUTF8(TFile.ReadAllText(lViewFileName, TEncoding.UTF8));
lViewEngine := TSynMustache.Parse(lViewTemplate);
2018-01-29 17:30:53 +01:00
lSW := TStreamWriter.Create(OutputStream);
try
lSW.Write(UTF8Tostring(lViewEngine.RenderJSON(FJSONModel, fPartials, nil, nil)));
finally
lSW.Free;
end;
end;
procedure TMVCMustacheViewEngine.LoadPartials;
var
lViewsExtension: string;
lViewPath: string;
lPartialFileNames: TArray<string>;
I: Integer;
begin
if gPartialsLoaded then
begin
Exit
end
else
begin
TMonitor.Enter(gLock);
try
if not gPartialsLoaded then
2020-11-05 17:14:39 +01:00
begin
lViewsExtension := Config[TMVCConfigKey.DefaultViewFileExtension];
lViewPath := Config[TMVCConfigKey.ViewPath];
lPartialFileNames := TDirectory.GetFiles(lViewPath, '*.' + lViewsExtension);
fPartials := TSynMustachePartials.Create;
for I := 0 to High(lPartialFileNames) do
begin
fPartials.Add(TPath.GetFileNameWithoutExtension(lPartialFileNames[i]), TFile.ReadAllText(lPartialFileNames[i]));
end;
gPartialsLoaded := True;
2020-11-05 17:14:39 +01:00
end;
finally
TMonitor.Exit(gLock);
2020-11-05 17:14:39 +01:00
end;
2018-01-29 17:30:53 +01:00
end;
2017-05-25 16:57:49 +02:00
end;
{$WARNINGS ON}
2017-05-25 16:57:49 +02:00
procedure TMVCMustacheViewEngine.PrepareModels;
var
lFirst: Boolean;
lList: IMVCList;
DataObj: TPair<string, TObject>;
lDSPair: TPair<string, TDataSet>;
2017-05-25 16:57:49 +02:00
lSJSON: string;
lJSON: string;
lSer: IMVCSerializer;
begin
{TODO -oDanieleT -cGeneral : Quite inefficient to generate JSON in this way. Why don't use a JSONObject directly?}
2017-05-25 16:57:49 +02:00
if (FJSONModel <> '{}') and (not FJSONModel.IsEmpty) then
Exit;
FJSONModel := '{}';
lSer := TMVCJsonDataObjectsSerializer.Create;
RegisterOptionalCustomTypesSerializers(lSer);
lSJSON := '{';
lFirst := True;
if Assigned(ViewModel) then
begin
for DataObj in ViewModel do
begin
2017-05-25 16:57:49 +02:00
lList := TDuckTypedList.Wrap(DataObj.Value);
if lList <> nil then
lJSON := lSer.SerializeCollection(DataObj.Value)
else
lJSON := lSer.SerializeObject(DataObj.Value);
if not lFirst then
lSJSON := lSJSON + ',';
2017-05-25 16:57:49 +02:00
lSJSON := lSJSON + '"' + DataObj.Key + '":' + lJSON;
lFirst := False;
end;
end;
if Assigned(ViewDataSets) then
begin
for lDSPair in ViewDataSets do
begin
lJSON := lSer.SerializeDataSet(lDSPair.Value);
if not lFirst then
lSJSON := lSJSON + ',';
lSJSON := lSJSON + '"' + lDSPair.Key + '":' + lJSON;
lFirst := False;
end;
end;
lSJSON := lSJSON + '}';
FJSONModel := lSJSON;
end;
end.