svghmi/gen_index_xhtml.xslt
author Edouard Tisserant
Sat, 05 Oct 2019 09:45:58 +0200
branchsvghmi
changeset 2790 8fab1886ebec
parent 2789 ba0dd2ec6dc4
child 2791 d022523cb621
permissions -rw-r--r--
SVGHI: compute hmitree variables ordered index in xslt
<?xml version="1.0"?>
<xsl:stylesheet xmlns:svg="http://www.w3.org/2000/svg" xmlns:ns="beremiz" xmlns:cc="http://creativecommons.org/ns#" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:str="http://exslt.org/strings" xmlns:xhtml="http://www.w3.org/1999/xhtml" xmlns:regexp="http://exslt.org/regular-expressions" xmlns:exsl="http://exslt.org/common" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" exclude-result-prefixes="ns" extension-element-prefixes="ns" version="1.0">
  <xsl:output method="xml" cdata-section-elements="script"/>
  <xsl:variable name="geometry" select="ns:GetSVGGeometry()"/>
  <xsl:variable name="hmitree" select="ns:GetHMITree()"/>
  <xsl:variable name="_categories">
    <noindex>
      <xsl:text>HMI_ROOT</xsl:text>
    </noindex>
    <noindex>
      <xsl:text>HMI_LABEL</xsl:text>
    </noindex>
    <noindex>
      <xsl:text>HMI_CLASS</xsl:text>
    </noindex>
    <noindex>
      <xsl:text>HMI_PLC_STATUS</xsl:text>
    </noindex>
    <noindex>
      <xsl:text>HMI_CURRENT_PAGE</xsl:text>
    </noindex>
  </xsl:variable>
  <xsl:variable name="categories" select="exsl:node-set($_categories)"/>
  <xsl:variable name="indexed_hmitree">
    <xsl:apply-templates mode="index" select="$hmitree"/>
  </xsl:variable>
  <xsl:template mode="index" match="node()">
    <xsl:param name="index" select="0"/>
    <xsl:variable name="content">
      <xsl:choose>
        <xsl:when test="not(local-name() = $categories/noindex)">
          <xsl:copy>
            <xsl:attribute name="index">
              <xsl:value-of select="$index"/>
            </xsl:attribute>
            <xsl:for-each select="@*">
              <xsl:copy/>
            </xsl:for-each>
          </xsl:copy>
        </xsl:when>
        <xsl:otherwise>
          <xsl:apply-templates mode="index" select="*[1]">
            <xsl:with-param name="index" select="$index"/>
          </xsl:apply-templates>
        </xsl:otherwise>
      </xsl:choose>
    </xsl:variable>
    <xsl:copy-of select="$content"/>
    <xsl:apply-templates mode="index" select="following-sibling::*[1]">
      <xsl:with-param name="index" select="$index + count(exsl:node-set($content)/*)"/>
    </xsl:apply-templates>
  </xsl:template>
  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>
  <xsl:variable name="mark">
    <xsl:text>=HMI=
</xsl:text>
  </xsl:variable>
  <xsl:template match="/">
    <html xmlns="http://www.w3.org/1999/xhtml">
      <head/>
      <body style="margin:0;">
        <xsl:copy>
          <xsl:comment>
            <xsl:apply-templates mode="testgeo" select="$geometry"/>
          </xsl:comment>
          <xsl:comment>
            <xsl:apply-templates mode="testtree" select="$hmitree"/>
          </xsl:comment>
          <xsl:comment>
            <xsl:apply-templates mode="testtree" select="exsl:node-set($indexed_hmitree)"/>
          </xsl:comment>
          <xsl:apply-templates select="@* | node()"/>
        </xsl:copy>
        <script>
          <xsl:text>var subscriptions = {
</xsl:text>
          <xsl:text>    return res;
</xsl:text>
          <xsl:text>}
</xsl:text>
          <xsl:text>// svghmi.js
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>(function(){
</xsl:text>
          <xsl:text>    // Open WebSocket to relative "/ws" address
</xsl:text>
          <xsl:text>    var ws = new WebSocket(window.location.href.replace(/^http(s?:\/\/[^\/]*)\/.*$/, 'ws$1/ws'));
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    // Register message reception handler 
</xsl:text>
          <xsl:text>    ws.onmessage = function (evt) {
</xsl:text>
          <xsl:text>        // TODO : dispatch and cache hmi tree updates
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>        var received_msg = evt.data;
</xsl:text>
          <xsl:text>        // TODO : check for hmitree hash header
</xsl:text>
          <xsl:text>        //        if not matching, reload page
</xsl:text>
          <xsl:text>        alert("Message is received..."+received_msg); 
</xsl:text>
          <xsl:text>    };
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    // Once connection established
</xsl:text>
          <xsl:text>    ws.onopen = function (evt) {
</xsl:text>
          <xsl:text>        // TODO : enable the HMI (was previously offline, or just starts)
</xsl:text>
          <xsl:text>        //        show main page
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>        // TODO : prefix with hmitree hash header
</xsl:text>
          <xsl:text>        ws.send("test");
</xsl:text>
          <xsl:text>    };
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    var pending_updates = {};
</xsl:text>
          <xsl:text>    
</xsl:text>
          <xsl:text>    // subscription state, as it should be in hmi server
</xsl:text>
          <xsl:text>    // expected {index:period}
</xsl:text>
          <xsl:text>    var subscriptions = {};
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    // subscription state as needed by widget now
</xsl:text>
          <xsl:text>    // expected {index:[widgets]};
</xsl:text>
          <xsl:text>    var subscribers = {};
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    // return the diff in between curently subscribed and subscription
</xsl:text>
          <xsl:text>    function update_subscriptions() {
</xsl:text>
          <xsl:text>        let result = [];
</xsl:text>
          <xsl:text>        Object.keys(subscribers).forEach(index =&gt; {
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>            let previous_period = subscriptions[index];
</xsl:text>
          <xsl:text>            let new_period = Math.min(...widgets.map(widget =&gt; widget.period));
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>            if(previous_period != new_period) 
</xsl:text>
          <xsl:text>                result.push({index: index, period: new_period});
</xsl:text>
          <xsl:text>        })
</xsl:text>
          <xsl:text>    }
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    function update_value(index, value) {
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>    };
</xsl:text>
          <xsl:text>
</xsl:text>
          <xsl:text>})();
</xsl:text>
        </script>
      </body>
    </html>
  </xsl:template>
  <xsl:template mode="code_from_descs" match="*">
    <xsl:text>{
</xsl:text>
    <xsl:text>    var path, role, name, priv;
</xsl:text>
    <xsl:text>    var id = "</xsl:text>
    <xsl:value-of select="@id"/>
    <xsl:text>";
</xsl:text>
    <xsl:if test="@inkscape:label">
      <xsl:text>name = "</xsl:text>
      <xsl:value-of select="@inkscape:label"/>
      <xsl:text>";
</xsl:text>
    </xsl:if>
    <xsl:text>/* -------------- */
</xsl:text>
    <xsl:value-of select="substring-after(svg:desc, $mark)"/>
    <xsl:text>
</xsl:text>
    <xsl:text>    /* -------------- */
</xsl:text>
    <xsl:text>    res.push({
</xsl:text>
    <xsl:text>        path:path,
</xsl:text>
    <xsl:text>        role:role,
</xsl:text>
    <xsl:text>        name:name,
</xsl:text>
    <xsl:text>        priv:priv
</xsl:text>
    <xsl:text>    })
</xsl:text>
    <xsl:text>}
</xsl:text>
  </xsl:template>
  <xsl:template mode="testgeo" match="bbox">
    <xsl:text>ID: </xsl:text>
    <xsl:value-of select="@Id"/>
    <xsl:text> x: </xsl:text>
    <xsl:value-of select="@x"/>
    <xsl:text> y: </xsl:text>
    <xsl:value-of select="@y"/>
    <xsl:text> w: </xsl:text>
    <xsl:value-of select="@w"/>
    <xsl:text> h: </xsl:text>
    <xsl:value-of select="@h"/>
    <xsl:text>
</xsl:text>
  </xsl:template>
  <xsl:template mode="testtree" match="*">
    <xsl:param name="indent" select="''"/>
    <xsl:value-of select="$indent"/>
    <xsl:text> </xsl:text>
    <xsl:value-of select="local-name()"/>
    <xsl:text> </xsl:text>
    <xsl:for-each select="@*">
      <xsl:value-of select="local-name()"/>
      <xsl:text>=</xsl:text>
      <xsl:value-of select="."/>
      <xsl:text> </xsl:text>
    </xsl:for-each>
    <xsl:text>
</xsl:text>
    <xsl:apply-templates mode="testtree" select="*">
      <xsl:with-param name="indent">
        <xsl:value-of select="concat($indent,'&gt;')"/>
      </xsl:with-param>
    </xsl:apply-templates>
  </xsl:template>
</xsl:stylesheet>