Template Engine

From PCGen Wiki
Revision as of 11:19, 29 October 2013 by James (talk | contribs) (Show initial implementation results)
Jump to: navigation, search

Template Engine Sub Project

The aim of the project is to examine the Freemarker library with a view to using it to replace our export engine. In particular it would cover functionality logic functions like looping and if tests. This would leave our code to deal with the important things like TO HIT values etc.

We would do this in three phases:

  1. Add support for Freemarker templates for output
  2. Once sufficient sheets have been converted/rewritten, deprecate support for olod sheet types
  3. Finally after a few releases, remove support for the old sheet types.

There are a few reasons for exploring this change

  • Add a richer set of logic commands to enable more customisable output
  • Simpler more consistent logic commands and more widely understood syntax
  • Remove one of the more complex and difficult to maintain sections of the PCGen code base.

Phase 1

The aim would be to utilise the already built output system and add the ability to

  1. Detect Freemarker templates and use the Freemarker engine to process them
  2. Add custom directives that can be called from a template (output sheet) to allow the

existing output tags to be used.

This would give us a simple change-over and use our existing well tested output code base.


Example of outputting an export tag

<@pcstring tag="NAME"/>
<@pcstring tag="SPELLMEM.0.1.1.1.NAME"/>
<@pcstring tag="PLAYERNAME"/>

Example of accessing a character variable value:

${pcvar("CL=Fighter")}
${pcvar("CL=Rogue")}
${pcvar("count(\"ABILITIES\",\"CATEGORY=FEAT\",\"TYPE=Metamagic\",\"VISIBILITY=DEFAULT[or]VISIBILITY=OUTPUT_ONLY\")")}

We would need to ensure that freemarker templates can be used in html, and PDF output styles.

See User Defined Directives and Shared Variables for information on creating custom directives/tags.

See freemarker_output Git branch for the first cut code.

For example, the sheet

<!-- Produced on ${.now?date} at ${.now?time} using template ${.template_name} -->
<h1>Freemarker Sheet for <@pcstring tag="NAME"/> - ${.now?date}</h1>

<p> Character Name: <@pcstring tag="NAME"/> </p>
<p> Player Name: <@pcstring tag="PLAYERNAME"/> </p>

<p> Fighter: ${pcvar("CL=Fighter")} </p>
<p> Rogue: ${pcvar("CL=Rogue")} </p>

produced the output

<!-- Produced on 29/10/2013 at 9:44:47 PM using template csheet_test-html.ftl -->
<h1>Freemarker Sheet for Ronald 'Surefingers' Millbridge - 29/10/2013</h1>

<p> Character Name: Ronald 'Surefingers' Millbridge </p>
<p> Player Name: James </p>

<p> Fighter: 0 </p>
<p> Rogue: 11 </p>

Looping

Freemarker's inbuilt looping mechanism is object oriented. It expects to receive a list and process through it. In contrast, PCGen currently only outputs strings and numbers. The technique described at For loop can be used, but a macro should be created to make this easier and more concise. The macro will also need to deal with the empty loop scenario.