HtmlReport
No edit summary
(Adding page to Category:TOC because it contains a TOC.)
 
(23 intermediate revisions by 3 users not shown)
Line 1: Line 1:
The HTML generating functions share most of it's code with the [[OpenDocument]] functionality.
=== Background ===
The HtmlReport functionality takes an XHTML string and processes it, adding data from a ViewModel and outputting the resulting XHTML.


XHtmlReportAsString
XHTML is a strict version of HTML that is XML-compatible. More importantly, it always has one, and only one, root node. Read more here: https://en.wikipedia.org/wiki/XHTML


==== HTML documents ====
All the functionality of the HtmlReport functionality also applies to XML if you need that.
You can also use html templates. If the template used ends with .htm or .html the logic expects a well formed html document and will expand tags just as in OpenDocument formats - but the we will check for row-builders %%+ROWBUILDERTAG& in <nowiki><tr></nowiki> context (html tables)
 
====XML documents====
The HTML-generating functions share most of their code with the [[OpenDocument]] functionality. The OpenDocument format uses XML for storing the document content.
XML is just the more generic case of html. They are treated by the same logic. You can from 20190919 use any xml-tag as row builder. Example<pre>
 
<SomeXml attrib='%SomeVMColumnAsAttributeresult%'>    %SomeOtherVMColumnAsXMLTextresult%  <Items>    <Item>%%+TheVMNestingColumn%      <ContentInItem someattrib='%VMColInNesting%'>          %StuffFromViewModelNesting%      </ContentInItem>    </Item>  </Items> </SomeXml> </pre>
=== Usage ===
====Viewmodel additions====
* ViewModel that defines data to insert
The templates is can be accessed in several different ways;
* Template to insert data into
*From an String attribute in a modeled class
Create a ViewModel that extracts the data you need from your model. Each attribute in this ViewModel can replace one or more "tags" in the source template. Note that the ViewModel also defines where the template can be found.
*From an URL
 
Example of a ViewModel that extracts two attributes to be used, "String" and "Text":
[[File:ReportingTemplate.png|none|thumb|417x417px]]
 
==== Ways to Retrieve the Template  ====
The template is accessed in several different ways:
*From a String attribute in a modeled class
*From a BLOB attribute in a modeled class (base64 encoded)
*From a URL
*From the local filesystem
*From the local filesystem
*From the database in BLOB attribute in a modeled class (base64 encoded)
The viewmodel providing the report data also has attributes with specific names providing information to the report functionality.
{| class="wikitable"
{| class="wikitable"
|'''Attribute name'''
|'''Attribute name'''
|'''Attribute should evaluate to'''
|'''Attribute should evaluate to'''
|'''Value type and content'''
|'''Value type and content'''
|-
|TemplateHtml
|A string that contains Html
|Not encoded in any special way
|-
|-
|TemplateUrl
|TemplateUrl
Line 26: Line 37:
|TemplateBlob
|TemplateBlob
|A class attribute containing the template
|A class attribute containing the template
|Blob with an uploaded document template, value is expected to be Base64Encoded
|Blob with an uploaded document template, value is expected to be Base64 encoded
|-
|ReportFileName
|A string with a filename
|Not used by XHtmlReportAsString but needed for opendocumentreportasblob
|}
|}
When creating OpenDocument documents, consider using the tag '''%meta%''' in your document to get the exact tag list for your document.


Copy the tag name including the percent signs from the generated meta tag information (preferable without style). The %meta% must be first string in element in order to be recognized - it is also case sensitive.
==== Methods ====
Depending on your input format, you can use either of these two functions:
 
===== XHtmlReportAsString =====
Easiest to use - takes the string input and returns another string. No encoding of the input HTML is required.
ResultHtmlAsString := self.XHtmlReportAsString(ReportRoot.ViewModels.ReportingViewmodel)
 
===== opendocumentreportasblob =====
Takes an array of bytes as input and output. The content should be UTF-8 encoded and in Base64 format.
 
To encode a Blob as base64, use [[OCLOperators BlobToBase64|BlobToBase64]]. (This will return a string base64 encoded, that will, in turn, be used as Blob.)
 
To encode a String as base64, use [[OCLOperators StringToBase64|StringToBase64]]. (This will return a string base64 encoded, that will, in turn, be used as Blob. )
ResultHtmlAsBlob := self.opendocumentreportasblob(ReportRoot.ViewModels.ReportingViewmodel)
 
===== opendocumentreportshow =====
Does the same as opendocumentreportasblob above, but opens the resulting HTML in the browser.
self.opendocumentreportshow(ReportRoot.ViewModels.ReportingViewmodel)
 
===== Inputs =====
* '''self''' here will become the root object of the reporting ViewModel.
* '''ReportRoot''' is the class that has the reporting ViewModel.
* '''Viewmodels''' is a method that retrieves all the ViewModels of the class. See [[OCLOperators ViewModels|ViewModels.]]
* '''ReportingViewmodel''' is the name of the reporting ViewModel used in this example.
 
==== Tags ====
Tags are the attribute name surrounded by percentage signs '''%'''. See %Date% and %Customername% below.
 
Multi-links form lists in the ViewModel. The name of the multi-link, in the example below, '''Invoices''' will repeat the HTML tag it's contained in, along with its content. The repeat tag is '''%%+''' at the start and ends with '''%.''' Invoices will become %%+Invoices%. In the example below, this will repeat the <nowiki><tr></nowiki> tag with its containing Date and Customer name forming a list in HTML.
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional //EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  <nowiki><html></nowiki>
      <body>
        <nowiki><table></nowiki>
            <tbody>
                <nowiki><tr></nowiki>
                    <nowiki><td></nowiki>Date<nowiki></td></nowiki>
                    <nowiki><td></nowiki>Customer<nowiki></td></nowiki>
                <nowiki></tr></nowiki>
                <nowiki><tr></nowiki>%%+Invoices%
                    <nowiki><td></nowiki>%Date%<nowiki></td></nowiki>
                    <nowiki><td></nowiki>%Customername%<nowiki></td></nowiki>
                <nowiki></tr></nowiki>
            </tbody>
        <nowiki></table></nowiki>
      </body>
  <nowiki></html></nowiki>
 
== Tips & Tricks  ==
 
====== Having a Collection of Strings  ======
If you have a collection of strings (or create a collection of strings from a comma list, for example) that you want to be rendered as a list of rows (paragraphs for example), let the ViewModel column return the expression return a collection of strings, then create a new ViewModel class (blue) '''without''' setting its type (derived from ViewModel), then add a column inside that ViewModel that has "self" as its expression.
 
In your HTML, you can then use:
<nowiki><p>%%+</nowiki>''Viewmodel.AttributeOfCollection''% %''NameOfValueAttribute''%<nowiki></p></nowiki>
Now the transformation will iterate the collection ''Viewmodel.AttributeOfCollection'' and fill in the value in ''NameOfValueAttribute''
 
==== XHtml ====
Because an Html document might be unloadable into a DOM tree like XHtml or XML, you should try providing a single document root node.
 
Providing plain HTML might be easier, and several HTML editors usually don't output Xhtml. Because of this, XHtmlReportAsString will try to compensate and add a <nowiki><html> tag around the content if it's missing.</nowiki> For example, this input, which has no single root node, has only two <nowiki><p> tags</nowiki>
<nowiki><p>Hello %String%</p></nowiki>
<nowiki><p>You need to do this....</p></nowiki>
It will create this output:
<nowiki><html></nowiki>
  <nowiki><p>Hello John</p></nowiki>
  <nowiki><p>You need to do this....</p></nowiki>
<nowiki></html></nowiki>
The <nowiki><html> tag was added because there were no <root>, <body>, or <html> tags in the input document.</nowiki>
 
==== Information About Available Tags (Meta Data) ====
When creating template documents, consider using the value '''%meta%''' in your document to get the exact tag list for your document.
 
In Html, provide a tag to hold the meta value, like this:
<meta>%meta%</meta>
Will create a ViewModel, for example:
<nowiki><html></nowiki>
    <meta>%TemplateHtml%, %TemplateBlob%, %ReportFileName%, %String%, %Text%, </meta>
<nowiki></html></nowiki>
Copy the tag name including the percent signs from the generated meta tag information. The %meta% must be the first string in the element in order to be recognized.


If you type it or edit a part of the tag name, your editor can insert invisible control information that's interfere with the merging.
==== XML Documents  ====
XML is the more generic case of HTML. They are treated by the same logic.<pre>
<SomeXml attrib='%SomeVMColumnAsAttributeresult%'>
    %SomeOtherVMColumnAsXMLTextresult%
  <Items>
    <Item>%%+TheVMNestingColumn%
      <ContentInItem someattrib='%VMColInNesting%'>
          %StuffFromViewModelNesting%
      </ContentInItem>
    </Item>
  </Items>
</SomeXml> </pre>
[[Category:Reports]]
{{Edited|July|12|2024}}


2020-04-06: Please also note that we have seen some issues with corrupt downloads when the ods template is created in OpenOffice 4.1.X. If the template is created in Excel is the document downloaded properly.
[[Category:TOC]]

Latest revision as of 13:46, 26 March 2024

Background

The HtmlReport functionality takes an XHTML string and processes it, adding data from a ViewModel and outputting the resulting XHTML.

XHTML is a strict version of HTML that is XML-compatible. More importantly, it always has one, and only one, root node. Read more here: https://en.wikipedia.org/wiki/XHTML

All the functionality of the HtmlReport functionality also applies to XML if you need that.

The HTML-generating functions share most of their code with the OpenDocument functionality. The OpenDocument format uses XML for storing the document content.

Usage

  • ViewModel that defines data to insert
  • Template to insert data into

Create a ViewModel that extracts the data you need from your model. Each attribute in this ViewModel can replace one or more "tags" in the source template. Note that the ViewModel also defines where the template can be found.

Example of a ViewModel that extracts two attributes to be used, "String" and "Text":

ReportingTemplate.png

Ways to Retrieve the Template

The template is accessed in several different ways:

  • From a String attribute in a modeled class
  • From a BLOB attribute in a modeled class (base64 encoded)
  • From a URL
  • From the local filesystem
Attribute name Attribute should evaluate to Value type and content
TemplateHtml A string that contains Html Not encoded in any special way
TemplateUrl The URL or local filename String with for example http://www.mdriven.net/templates/report.odt or for prototyping c:\\temp\\mytemplate.odt
TemplateBlob A class attribute containing the template Blob with an uploaded document template, value is expected to be Base64 encoded
ReportFileName A string with a filename Not used by XHtmlReportAsString but needed for opendocumentreportasblob

Methods

Depending on your input format, you can use either of these two functions:

XHtmlReportAsString

Easiest to use - takes the string input and returns another string. No encoding of the input HTML is required.

ResultHtmlAsString := self.XHtmlReportAsString(ReportRoot.ViewModels.ReportingViewmodel)
opendocumentreportasblob

Takes an array of bytes as input and output. The content should be UTF-8 encoded and in Base64 format.

To encode a Blob as base64, use BlobToBase64. (This will return a string base64 encoded, that will, in turn, be used as Blob.)

To encode a String as base64, use StringToBase64. (This will return a string base64 encoded, that will, in turn, be used as Blob. )

ResultHtmlAsBlob := self.opendocumentreportasblob(ReportRoot.ViewModels.ReportingViewmodel)
opendocumentreportshow

Does the same as opendocumentreportasblob above, but opens the resulting HTML in the browser.

self.opendocumentreportshow(ReportRoot.ViewModels.ReportingViewmodel)
Inputs
  • self here will become the root object of the reporting ViewModel.
  • ReportRoot is the class that has the reporting ViewModel.
  • Viewmodels is a method that retrieves all the ViewModels of the class. See ViewModels.
  • ReportingViewmodel is the name of the reporting ViewModel used in this example.

Tags

Tags are the attribute name surrounded by percentage signs %. See %Date% and %Customername% below.

Multi-links form lists in the ViewModel. The name of the multi-link, in the example below, Invoices will repeat the HTML tag it's contained in, along with its content. The repeat tag is %%+ at the start and ends with %. Invoices will become %%+Invoices%. In the example below, this will repeat the <tr> tag with its containing Date and Customer name forming a list in HTML.

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional //EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
 <html>
     <body>
        <table>
            <tbody>
                <tr>
                    <td>Date</td>
                    <td>Customer</td>
                </tr>
                <tr>%%+Invoices%
                    <td>%Date%</td>
                    <td>%Customername%</td>
                </tr>
            </tbody>
        </table>
     </body>
 </html> 

Tips & Tricks

Having a Collection of Strings

If you have a collection of strings (or create a collection of strings from a comma list, for example) that you want to be rendered as a list of rows (paragraphs for example), let the ViewModel column return the expression return a collection of strings, then create a new ViewModel class (blue) without setting its type (derived from ViewModel), then add a column inside that ViewModel that has "self" as its expression.

In your HTML, you can then use:

<p>%%+Viewmodel.AttributeOfCollection% %NameOfValueAttribute%</p>

Now the transformation will iterate the collection Viewmodel.AttributeOfCollection and fill in the value in NameOfValueAttribute

XHtml

Because an Html document might be unloadable into a DOM tree like XHtml or XML, you should try providing a single document root node.

Providing plain HTML might be easier, and several HTML editors usually don't output Xhtml. Because of this, XHtmlReportAsString will try to compensate and add a <html> tag around the content if it's missing. For example, this input, which has no single root node, has only two <p> tags

<p>Hello %String%</p>
<p>You need to do this....</p>

It will create this output:

<html>
  <p>Hello John</p>
  <p>You need to do this....</p>
</html>

The <html> tag was added because there were no <root>, <body>, or <html> tags in the input document.

Information About Available Tags (Meta Data)

When creating template documents, consider using the value %meta% in your document to get the exact tag list for your document.

In Html, provide a tag to hold the meta value, like this:

<meta>%meta%</meta>

Will create a ViewModel, for example:

<html>
    <meta>%TemplateHtml%, %TemplateBlob%, %ReportFileName%, %String%, %Text%, </meta>
</html>

Copy the tag name including the percent signs from the generated meta tag information. The %meta% must be the first string in the element in order to be recognized.

XML Documents

XML is the more generic case of HTML. They are treated by the same logic.

 <SomeXml attrib='%SomeVMColumnAsAttributeresult%'>
     %SomeOtherVMColumnAsXMLTextresult%
   <Items>
     <Item>%%+TheVMNestingColumn%
       <ContentInItem someattrib='%VMColInNesting%'>
          %StuffFromViewModelNesting%
       </ContentInItem>
     </Item>
   </Items>
 </SomeXml> 
This page was edited more than 9 months ago on 03/26/2024. What links here