- Super Powerful and Flexible Syntax
- HTML Templating (and not a text templating)
- Templates are Compiled into the Swift Code
Here is an example of the Reparse
syntax as used in the bundled Example
project (with few lines removed for brevity):
<r-require name="context" type="[String]" label="superheroes" />
<r-extend name="base" />
<r-extend name="body" />
<r-eval line='req.logger.info("Index Debug Message")' />
<main>
<h1>
<r-set name="class" value=" red" append r-if="context.isEmpty" />
Hello
<r-include name="components.world" r-if="!context.isEmpty">
Ultra Heroes!
</r-include>
<r-block r-else> World?</r-block>
</h1>
<ol>
<li r-for-every="context" r-with-item r-with-index>
<p>
<r-include name="components.hello-me">\(item)</r-include>
</p>
<p>Index: \(index)</p>
</li>
<li r-else>No more heroes...</li>
</ol>
<p><r-value of="req.url.string" /> or \(req.url.string)</p>
</main>
<title r-add-to-slot="head">Hero List</title>
Here is the default command to build the example pages manually: swift run reparse ./Resources/Pages ./Sources/Example --parameters req:Request --imports Vapor
And then run the Example project with: swift run ReparseExample
If you would like to use it in your own project, firstly, add it to your package dependencies like this:
.package(url: "https://github.com/RussBaz/experimental-reparse-html.git", from: "0.0.19"),
Then add the ReparseRuntime
as a dependency to your target like this:
.product(name: "ReparseRuntime", package: "experimental-reparse-html"),
To use it as a standalone tool, you would need to git clone
this repo and then compile the tool like this: swift build -c release
. Then copy the built tool form the build folder to anywhere you want and run it. Otherwise, you can just call swift run reparse
from inside this project folder, providing valid arguments and options.
Here is the help page for the tool:
USAGE: reparse <location> <destination> [--file-name <file-name>] [--file-extension <file-extension>] [--enum-name <enum-name>] [--imports <imports> ...] [--parameters <parameters> ...] [--protocols <protocols> ...] [--dry-run]
ARGUMENTS:
<location> The target data folder location.
<destination> The destination folder for the output file.
OPTIONS:
--file-name <file-name> Output file name (default: pages.swift)
--file-extension <file-extension>
The file extension to be searched for (default: html)
--enum-name <enum-name> The name of the generated enum (default: Pages)
--imports <imports> List of global imports
--parameters <parameters>
List of shared parameters (parameters to be added to
every 'include' function) in a form of
'[?][label:]name:type[=default]' where the optional
parts are in square brackets. The question mark at the
beginning indicates that the parameter will be
overriden by a local requirement if it is present.
--protocols <protocols> A list of protocols to apply to enums with render
functions in them in a form of
'name[:associatedName:associatedType]' where the
optional parts are in square brackets. Optional part
can be repeated any number of times.
--dry-run Write the output to the console instead of file
-h, --help Show help information.
You can also run it as a command plugin from the terminal. You only have to follow the installation instructions for it to be automatically available.
To list available plugins:
swift package plugin --list
To use an automatic Vapor settings, use:
swift package plugin reparse --preset vapor
NOTE: Every Vapor based preset adds 'req: Request' as a first parameter to the generated functions.
of if you are using VaporHX
:
swift package plugin reparse --preset vaporhx
NOTE: If you are to use it with VaporHX, then you can only 'require' context
variables because otherwise the genrated code will no longer conform to the HXTemplateable
protocol, required for automatic use of generated templates.
In addition to to 'req: Request' coming from a Vapor preset, it also adds 'isPage: Bool' and 'context: Context' parameters. 'isPage' is true when the template is expected to return the whole page and it is false when a fragment of the page is expected (in-place update). Then 'Context' type is EmptyContext
struct by default, provided by the VaporHX. However, if you specify the 'context' using a 'r-require' tag, then it will be replaced with your own type without breaking compatibility with the VaporHX.
If you do not use any preset, then all the relative routes would be pointing at the project root folder.
All the options you can pass are the same except the location and destination arguments are replaced by the following options:
--source
to specify a different root folder other than than the project root
--target
to pick a target from the package targets as a destination for the generated file
--destination
as a list of path components to provide a different output folder relative to the source root folder for the target (or project root if none selected)
All special attributes and tags will be removed by the compilation and must not appear in the output of the render function. If they do, then it is a bug.
You can use the standard swift syntax for string interpolation \(expression)
to insert any swift expression into a text part of the html template - inside the attribute values and the text between tags.
There are a few types of control attributes and can be separated in 3 groups:
Most html tags can be equipped with one of the conditional control attributes:
- r-if="condition"
- r-else-if="condition"
- r-else
If the condition (which must be a valid Swift expression) is satisfied than the html tag and its contents are rendered.
If the condition is satisfied, then a special variable called previousUnnamedIfTaken
will be set to true. Otherwise, it will be set to false.
Optionally, you can save the result of the condition to a different variable using an additional attribute:
- r-tag="tag-name"
- r-for-every="sequence" to repeat the tag for evey item in a sequence
- r-with-item="optional name" to include the value of the item in the loop, optionally specifying an alternative name for such a variable. The default name is 'item'
- r-with-index="optional name" to include the index of the item in the loop, optionally specifying an alternative name for such a variable. The default name is 'index'
- r-add-to-slot="slot-name"
- r-replace-slot="slot-name"
<r-extend name="template-name" />
(must be before any tag other than r-require) to wrap the current template into a default slot of the specified template<r-require label="optional-label" name="var-name" type="var-type" default="optional-default" />
(must be before any tag other than r-extend) to define a variable that must be passed into the template from the caller<r-require label="optional-label" name="var-name" type="var-type" default="optional-default" mutable />
or if you need to remap it to a mutable variable of the same name- or you can add
local-only
attribute to prevent it from being required in outer scopes. You have to provide your own argument for such a parameter in such a case. WARNING! This will disable all protocol conformance for the enums withlocal-only
attributes. <r-include name="template-name" />
to include another template or<r-include name="template-name"> default slot </r-include>
to include a template with a default slot provided<r-block> some data </r-block>
to group some part of template, e.g. wrap some text with it and now you can apply control attributes to it.<r-set name="attr-name" value="attr-value" />
to replace an attribute in a preceding tag (skipping other set/unset tags) or<r-set name="attr-name" value="attr-value" append />
to append to it instead<r-unset name="attr-name" />
to remove an atttribute from a preceding tag (skipping other set/unset tags)<r-var name="var-name" line="expression" />
to assign the result of an expression to a variable<r-value of="name" default="optional-val" />
to paste the value of the specified variable or the provided default value if the value wasnil
<r-eval line="expression" />
to paste the expression as is into the generated code<r-slot name="optional-name" />
to mark an area to be filled by the incoming outer slot. If no name is provided, it will be known as 'default' or<r-slot name="optional-name"> default slot </r-slot>
to declare a slot and provide the default contents if no matching slot found in the incoming outer slots
It operates in two stages. There is the compilation stage and the runtime stage.
- All the templates are discovered, and their names and full paths are recorded. The names are derived from the relative path to the root folder and the name of file.
- For each template, a tokeniser converts a stream of characters into a stream of tokens.
- The tokens are now parsed into an abstract syntax tree.
- The syntax tree is flattened and transpiled into a series of commands for a file builder. References between files and other meta data are recorded.
- Function signatures are resolved for every template
- All individual command sequences for every template are combined into a single one based and executed.
- The final string is saved into the specified location.
- The caller passes the paremters to the render function.
- The render function passes them to the include function of the specified template.
- The include function returns the line storage object with a render method.
- Then the render method is called and it will call its resolve method with empty outer slots.
- The returned sequence of text constants is then merged into a string and returned to the caller.
- Calling the template will call an auto generated function that builds a simple internal model for each template by calling a very limited number of its commands. (done when the include method is called)
- The slots are passed into the template and the default slot is resolved. The slots passed into the template are referred as outer slots. (done when the resolve method is called)
- All slot declaration sites (including the inside of include commands) are recuresively replaced with either contents of outer slots (if found) and mark them for deletion, or replaces with the defaults. It does nothing if neither are found. Nested declarations are allowed but if the parent 'comsumes' the slot, then it will no longer appear to the nested declarations. Sibling declarations do not have this restriction.
- Inner slots are computed - the slots to be passed to included templates. All the commands affecting the slots are resolved.
- Default inner slots are computed for every include command and passed into the specified templates. Nested includes are allowed.
- Resolved inner templates are merged into the current template
- The contents, now consisting of text constant commands only, are returned to the caller.