Помогите преобразовать RDF в HTML через XSL - PullRequest
0 голосов
/ 03 января 2011

У меня есть файл xml / rdf, который я получаю от сервиса. Итак, мне нужно преобразовать его в блок HTML через XSL.

Вот часть файла xml / rdf:

<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:c="http://s.opencalais.com/1/pred/">
    <rdf:Description rdf:about="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d/Instance/2">
        <rdf:type rdf:resource="http://s.opencalais.com/1/type/sys/InstanceInfo"/>
        <c:docId rdf:resource="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d"/>
        <c:subject rdf:resource="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5"/><!--Company: Zila Inc.; -->
        <c:detection>[                PHOENIX, March 28 /PRNewswire/ -- ]Zila, Inc.[ (Nasdaq: ZILA) Chairman and President Joseph]</c:detection>
        <c:prefix>                PHOENIX, March 28 /PRNewswire/ -- </c:prefix>
        <c:exact>Zila, Inc.</c:exact>
        <c:suffix> (Nasdaq: ZILA) Chairman and President Joseph</c:suffix>
        <c:offset>131</c:offset>
        <c:length>10</c:length>
    </rdf:Description>
    <rdf:Description rdf:about="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d/Instance/3">
        <rdf:type rdf:resource="http://s.opencalais.com/1/type/sys/InstanceInfo"/>
        <c:docId rdf:resource="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d"/>
        <c:subject rdf:resource="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5"/><!--Company: Zila Inc.; -->
        <c:detection>[March 28 /PRNewswire/ -- Zila, Inc. (Nasdaq: ]ZILA[) Chairman and President Joseph Hines announced]</c:detection>
        <c:prefix>March 28 /PRNewswire/ -- Zila, Inc. (Nasdaq: </c:prefix>
        <c:exact>ZILA</c:exact>
        <c:suffix>) Chairman and President Joseph Hines announced</c:suffix>
        <c:offset>151</c:offset>
        <c:length>4</c:length>
    </rdf:Description>
    <rdf:Description rdf:about="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5">
        <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Company"/>
        <c:name>Zila Inc.</c:name>
        <c:nationality>N/A</c:nationality>
    </rdf:Description>
    <rdf:Description rdf:about="http://d.opencalais.com/pershash-1/31337438-fcbd-3137-9705-ef2f1a752b88">
        <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Person"/>
        <c:name>Joseph Hines</c:name>
        <c:persontype>economic</c:persontype>
        <c:nationality>N/A</c:nationality>
        <c:commonname>Joseph Hines</c:commonname>
    </rdf:Description>
    ...
</rdf:RDF>

Итак, я должен построить HTML из него. HTML состоит из списков объектов, которые имеют заголовок и элементы.

Заголовки - это последнее слово в атрибуте <rdf:type> узла. Пример:

<rdf:Description rdf:about="http://d.opencalais.com/pershash-1/31337438-fcbd-3137-9705-ef2f1a752b88">
            <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Person"/>
            <c:name>Joseph Hines</c:name>
            <c:persontype>economic</c:persontype>
            <c:nationality>N/A</c:nationality>
            <c:commonname>Joseph Hines</c:commonname>
        </rdf:Description>

Этот узел должен быть преобразован в

<div class="Block">
    <div class="Header Person">Person</div>
    <div class="Item">Joseph Hines</div> 
</div>

Итак, у меня есть много таких блоков, в которых есть заголовок Персона (или другой) и элемент в узле <c:name>.

Как я могу преобразовать этот xml / rdf в блоки html и поместить все элементы с одинаковым заголовком (например, Person) в один и тот же блок?

Это должно выглядеть как

<div class="Block">
    <div class="Header Person">Person</div>
    <div class="Item">Joseph Hines</div> 
    <div class="Item">another item</div>
    ...
</div>
<div class="Block">
    <div class="Header Company">Company</div>
    <div class="Item">Zila Inc.</div> 
    <div class="Item">another company item</div>
    ...
</div>
...

Извините, если проблема не ясна. Я отвечу на все вопросы, чтобы помочь понять мою проблему.

Заранее спасибо.

Ответы [ 2 ]

2 голосов
/ 03 января 2011

Этот XSLT:

<xsl:stylesheet version="1.0"
            xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
            xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
            xmlns:c="http://s.opencalais.com/1/pred/"
            exclude-result-prefixes="rdf c">
<xsl:output method="html" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="entityByType" match="rdf:Description" use="substring-after(rdf:type/@rdf:resource, 'type/em/e/')"/>

<xsl:template match="/*">
    <xsl:apply-templates select="key('entityByType', 'Person')[1]"/>
    <xsl:apply-templates select="key('entityByType', 'Company')[1]"/>
</xsl:template>

<xsl:template match="rdf:Description">
    <div class="block">
        <xsl:variable name="entityType" select="substring-after(rdf:type/@rdf:resource, 'type/em/e/')"/>
        <div class="header {$entityType}">
            <xsl:value-of select="$entityType"/>
        </div>
        <xsl:apply-templates select="key('entityByType', $entityType)" mode="inner-content"/>
    </div>
</xsl:template>

<xsl:template match="rdf:Description" mode="inner-content">
    <div class="item">
        <xsl:value-of select="c:name"/>
    </div>
</xsl:template>

</xsl:stylesheet>

Применяется для более описательного ввода XML:

<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:c="http://s.opencalais.com/1/pred/">
<rdf:Description rdf:about="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d/Instance/2">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/sys/InstanceInfo"/>
    <c:docId rdf:resource="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d"/>
    <c:subject rdf:resource="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5"/><!--Company: Zila Inc.; -->
    <c:detection>[                PHOENIX, March 28 /PRNewswire/ -- ]Zila, Inc.[ (Nasdaq: ZILA) Chairman and President Joseph]</c:detection>
    <c:prefix>                PHOENIX, March 28 /PRNewswire/ -- </c:prefix>
    <c:exact>Zila, Inc.</c:exact>
    <c:suffix> (Nasdaq: ZILA) Chairman and President Joseph</c:suffix>
    <c:offset>131</c:offset>
    <c:length>10</c:length>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d/Instance/3">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/sys/InstanceInfo"/>
    <c:docId rdf:resource="http://d.opencalais.com/dochash-1/f1a1cace-8df1-3247-b8e2-331a8fa8363d"/>
    <c:subject rdf:resource="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5"/><!--Company: Zila Inc.; -->
    <c:detection>[March 28 /PRNewswire/ -- Zila, Inc. (Nasdaq: ]ZILA[) Chairman and President Joseph Hines announced]</c:detection>
    <c:prefix>March 28 /PRNewswire/ -- Zila, Inc. (Nasdaq: </c:prefix>
    <c:exact>ZILA</c:exact>
    <c:suffix>) Chairman and President Joseph Hines announced</c:suffix>
    <c:offset>151</c:offset>
    <c:length>4</c:length>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Company"/>
    <c:name>Zila Inc.1</c:name>
    <c:nationality>N/A</c:nationality>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/pershash-1/31337438-fcbd-3137-9705-ef2f1a752b88">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Person"/>
    <c:name>Joseph Hines1</c:name>
    <c:persontype>economic</c:persontype>
    <c:nationality>N/A</c:nationality>
    <c:commonname>Joseph Hines</c:commonname>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Company"/>
    <c:name>Zila Inc.2</c:name>
    <c:nationality>N/A</c:nationality>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/pershash-1/31337438-fcbd-3137-9705-ef2f1a752b88">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Person"/>
    <c:name>Joseph Hines2</c:name>
    <c:persontype>economic</c:persontype>
    <c:nationality>N/A</c:nationality>
    <c:commonname>Joseph Hines</c:commonname>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/comphash-1/f69211ed-b366-31a8-9e04-b85c50b8e0a5">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Company"/>
    <c:name>Zila Inc.3</c:name>
    <c:nationality>N/A</c:nationality>
</rdf:Description>
<rdf:Description rdf:about="http://d.opencalais.com/pershash-1/31337438-fcbd-3137-9705-ef2f1a752b88">
    <rdf:type rdf:resource="http://s.opencalais.com/1/type/em/e/Person"/>
    <c:name>Joseph Hines3</c:name>
    <c:persontype>economic</c:persontype>
    <c:nationality>N/A</c:nationality>
    <c:commonname>Joseph Hines</c:commonname>
</rdf:Description>
</rdf:RDF>

Создает этот правильный результат:

<div class="block">
    <div class="header Person">Person</div>
    <div class="item">Joseph Hines1</div>
    <div class="item">Joseph Hines2</div>
    <div class="item">Joseph Hines3</div>
</div>
<div class="block">
    <div class="header Company">Company</div>
    <div class="item">Zila Inc.1</div>
    <div class="item">Zila Inc.2</div>
    <div class="item">Zila Inc.3</div>
</div>

РЕДАКТИРОВАТЬ:

<xsl:stylesheet version="1.0"
            xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
            xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
            xmlns:c="http://s.opencalais.com/1/pred/"
            exclude-result-prefixes="rdf c">
<xsl:output method="html" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="entityByType" match="rdf:Description" use="substring-after(rdf:type/@rdf:resource, 'type/em/e/')"/>

<xsl:template match="/*">
    <xsl:apply-templates select="
    rdf:Description[
    generate-id() = generate-id(key('entityByType', substring-after(rdf:type/@rdf:resource, 'type/em/e/')))
    ]"/>
</xsl:template>

<xsl:template match="rdf:Description">
    <div class="block">
        <xsl:variable name="entityType">
            <xsl:call-template name="substring-after-last">
                <xsl:with-param name="input" select="rdf:type/@rdf:resource"/>
                <xsl:with-param name="substr" select="'/'"/>
            </xsl:call-template>
        </xsl:variable>
        <div class="header {$entityType}">
            <xsl:value-of select="$entityType"/>
        </div>
        <xsl:apply-templates select="key('entityByType', $entityType)" mode="inner-content"/>
    </div>
</xsl:template>

<xsl:template match="rdf:Description" mode="inner-content">
    <div class="item">
        <xsl:value-of select="c:name"/>
    </div>
</xsl:template>

<xsl:template name="substring-after-last">
    <xsl:param name="input"/>
    <xsl:param name="substr"/>
    <xsl:variable name="temp" select="substring-after($input,$substr)"/>
    <xsl:choose>
        <xsl:when test="$substr and contains($temp,$substr)">
            <xsl:call-template name="substring-after-last">
            <xsl:with-param name="input"  select="$temp" />
            <xsl:with-param name="substr" select="$substr" />
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:value-of select="$temp"/>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>

</xsl:stylesheet>

И результат:

<div class="block">
    <div class="header InstanceInfo">InstanceInfo</div>
</div>
<div class="block">
    <div class="header Company">Company</div>
    <div class="item">Zila Inc.1</div>
    <div class="item">Zila Inc.2</div>
    <div class="item">Zila Inc.3</div>
</div>
<div class="block">
    <div class="header Person">Person</div>
    <div class="item">Joseph Hines1</div>
    <div class="item">Joseph Hines2</div>
    <div class="item">Joseph Hines3</div>
</div>

Вы можете исключить любые ненужные типы самостоятельно или предоставить критерии для такого исключения.

1 голос
/ 03 января 2011

Вот таблица стилей XSLT 2.0, которую вы можете запустить с процессорами XSLT 2.0, такими как Saxon 9 или AltovaXML Tools:

<xsl:stylesheet
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
  xmlns:c="http://s.opencalais.com/1/pred/"
  exclude-result-prefixes="rdf c"
  version="2.0">

  <xsl:output method="html" indent="yes"/>

  <xsl:template match="rdf:RDF">
    <xsl:for-each-group select="rdf:Description[rdf:type/@rdf:resource]"
      group-by="tokenize(rdf:type/@rdf:resource, '/')[last()]">

      <div class="Block">
        <div class="Header {current-grouping-key()}">
          <xsl:value-of select="current-grouping-key()"/>
        </div>
        <xsl:apply-templates select="current-group()/c:name"/>
      </div>
    </xsl:for-each-group>
  </xsl:template>

  <xsl:template match="c:name">
    <div class="Item">
      <xsl:value-of select="."/>
    </div>
  </xsl:template>

</xsl:stylesheet>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...