How loop through child node in XSLT - json

I have a JSON object returned from a HTTP request and I've converted this into XML however the object that's being returned become an array objects. Now I am using XSL to render out the returned converted objects.
This is the kind of object I'm getting back after converting JSON to XML
<Results>
<0>
<Codes>5051946526135</Codes>
<Images>http://www.cheaney.co.uk/images/cheaney-arthur-iii-brogue-in-original-chestnut-calf-leather-p6-1109_image.jpg</Images>
</0>
<1>
<Codes>5051946523974</Codes>
<Images>http://www.bagsbydiamonds.co.uk/ekmps/shops/bagsbydiamonds/images/radley-al…-black-leather-shoulder-bag-workbag-3-compartments-rrp-199-[2]-11903-p.jpg</Images>
</1>
</Results>
I've been trying to use <xsl:for-each> however the objects are not being shown in my HTML, so I'm assuming that there's something wrong with my XSL template.
<xsl:template match="/Results">
<xsl:for-each select="*">
<div class="col-sm-3">
<img>
<xsl:attribute name="src">
<xsl:apply-templates select="Images"/>
</xsl:attribute>
</img>
</div>
<div class="col-sm-9">
<p> <xsl:apply-templates select="Codes"/> </p>
</div>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
I thought using * or . would selected all child nodes but clearly it didn't work for me. It would have been easier if the child nodes are the same like <collection></collection> rather than 0 and 1.
How can I loop through these kind of collections?

Related

Second XSL template disables the first one

The following template fixes the img/src attribute and is there since years:
<xsl:template match="xh:img/#src">
<xsl:attribute name="src">
<xsl:value-of select="
if( string-length(substring-before(substring-after(
subsequence(parent::node()/following-sibling::comment(),1,1),'src="'),'.eps')) > 0 )
then
concat('images/',tokenize(concat(substring-before(substring-after(
subsequence(parent::node()/following-sibling::comment(),1,1),'src="'),'.eps'),'.png'),'/')[last()])
else
data(self::node())"/>
</xsl:attribute>
</xsl:template>
Now I added the following template to move width and height to style:
<xsl:template match="xh:img">
<img style="width:{#width}; height:{#height};">
<xsl:copy-of select="#*[not(name()='width' or name()='height')]"/>
</img>
</xsl:template>
The second one works but it "disables" the first one.
If I comment the second one, the first works.
Is there a way to merge them?
I really don't have experience with XSL so any help would be appreciated.
This is because you are using xsl:copy-of to copy attributes in the template matching xh:img. This will not apply any matching templates, but just copy them exactly.
Simply change to using xsl:apply-templates...
<xsl:apply-templates select="#*[not(name()='width' or name()='height')]"/>
However, you might need to add an additional template to match attributes other than src should you wish them to still be created too.
<xsl:template match="#*">
<xsl:copy />
</xsl:template>

Import HTML Drupal module with custom XSL template

I have a requirement to import a html website to Drupal and I have decided to using Import HTML module to do it.
I have to be able to grab just the text from html page (inside tag) without the html tags.
For this, I'm trying to create a custom xsl template based on the default template: html2simplehtml.xsl.
Currently my import is working fine with html2simplehtml.xsl template.
here is example of the result node body from the import:
<div class="container-narrow">
<div class="masthead">
<ul class="nav nav-pills pull-right">
<li class="active">
Home
</li>
<li>
Applications
</li>
<li>
Middleware
</li>
now, the requirement is to only get:
Home
Applications
Middleware
I have found this to remove html tags:
<!-- This will remove the tag -->
<xsl:template name="remove-html">
<xsl:param name="text"/>
<xsl:choose>
<xsl:when test="contains($text, '<')">
<xsl:value-of select="substring-before($text, '<')"/>
<xsl:call-template name="remove-html">
<xsl:with-param name="text" select="substring-after($text, '>')"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$text"/>
</xsl:otherwise>
</xsl:choose>
but I am not sure where to put and how to call it using this:
<!-- Calling the template that removes tag -->
<xsl:call-template name="remove-html">
<xsl:with-param name="text" select="{HtmlBody}"/>
</xsl:call-template>
How can I do this?
I'm not quite familiar with the way that Drupal calls your XSLT but let's assume it's a simple XSLT 1.0 processor using some HTML page as input and generating the output that you showed above. Let's further assume that the original HTML is well formed with all required closing tags, so that it's in fact XHTML which can be processed by the XSLT processor. (This is not true for the HTML you included in your question, by the way.)
So what you want to do is basically prevent all tags in the XML/XHTML input from showing up in the output. I think the easiest way to achieve this to use the <xsl:value-of select> tag. Assuming that you copy all the child tags of the <body></body> section of your XHTML like this:
<xsl:template match="body">
<xsl:copy-of select="*">
</xsl:template>
instead you could do this:
<xsl:template match="body">
<xsl:value-of select=".">
</xsl:template>
<xsl:value-of> forces the evaluation of the XML sub tree into a string which is done (simply put) by concatenating all contained text elements. This does not, however, take care of white space yet. If you want to the eliminate disturbing white space you could brace the call like this:
<xsl:template match="body">
<xsl:value-of select="normalize-space(.)">
</xsl:template>
Now for the template you originally wanted to use: This does in fact remove tags from the input, too. But if I interpret the code right the input is NOT an XML node set but it must already be a STRING. So this works for other context in which you have a literal XML representation in a string. If you tried to use it here you would have to explicitly convert your XML representation into a string beforehand by using e.g. <xsl:value-of>. In this case the template would already be stripped off the tags (as described above) and would effectively not do anything at all but return the same string that it was passed as parameter. So IMHO, you will not need this template at all.

In HTML code use another data tag in href=Variable

<!-- Section: /report/detail -->
<xsl:template match="/report/detail">
<div href="./lf_web" style="left: 0.0ex; position: absolute"><xsl:value-of select="./lf_po"/></div>
<br/>
</xsl:template>
In this code I am trying to use the content of the taq element "./lf_web".
Can any one help me please.
It is in and .xsl file and linked to .xml data source
You should tag your question with XSL instead of div and href which are not important to your question.
One way is to first retrieve the value in a variable. I like that way because you can use functions:
<xsl:variable name="website_uri" as="xs:string">
<xsl:value-of select="info/uri"/>
</xsl:variable>
<xsl:variable name="protocol" as="xs:string">
<xsl:value-of select="substring-before($website_uri, '://')"/>
</xsl:variable>
And I show how you use a variable: with a $ sign before the name of the value. However, to do so inside a different attribute, you need to put it between curly brackets;
...
Now I think you can do the query you're trying to achieve inline too, but I don't see a good example for it.

Define CSS style using XML / XSL parser

I have a XSL/XML parser to produce html code.
The xml is like this.
<root>
<a>
<url> http://image.jpg </url>
<x> 100 </x>
<y> 200 </y>
...
</a>
</root>
and the XST should be something like this. In style I want do define a background url using the value of xml node. How can I do it?
<xsl:template match="root">
<xsl:for-each select="a">
<div class="crop_image" style="background:url("<xsl:value-of select="url"/>") -<xsl:value-of select="x"/>px -<xsl:value-of select="y"/>px">"</div>
</xsl:for-each>
</xsl:template>
Thanks
What you are looking for are "Attribute Value Templates", which allow you to write values from your XML directly into attributes
<xsl:template match="root">
<xsl:for-each select="a">
<div class="crop_image" style="background:url({url}) -{x}px -{y}px">"</div>
</xsl:for-each>
</xsl:template>
The curly braces indicate an expression to be evaluated, rather than output literally, so {url} for example will be replaced with http://image.jpg in your output.

How to add an ID attribut to xml elements using xslt transformation

I am very new to xml and xslt transformation. I need some advice.
Basically I have managed to display a range of images from my XML page using XLTR Transformation.
The original xml has no ID's associated with each image, I cant alter the xml. Therefore I was wondering if there was a way in xslt to add an ID attribute to each image before it is outputted to the browser? As I want to be able to control each element/image individually using CSS.
I hope that makes sense?
The xslt is as follows.
<xsl:for-each select=" cars/car_type">
<!-- <h5> <xsl:value-of select="car-type"/> </h5> -->
<div class ="images">
<img>
<xsl:attribute name="src">
<xsl:value-of select="image"/>
</xsl:attribute>
</img>
</div>
</xsl:for-each>
Thanks again.
You can use generate-id (http://www.w3.org/TR/xslt#function-generate-id) function to generate an id for an input node e.g.
<xsl:for-each select="cars/car_type">
<div class="images">
<img src="{image}" id="{generate-id(image)}"/>
</div>
</xsl:for-each>
When you create the CSS elsewhere in your stylesheet you simply need to make sure you process the image elements again and use generate-id again.