Webudviklerens drøm: Cross-browser uden tårer

Med XML, XSLT og CSS er det ingen sag at levere dynamisk indhold i så forskellige formater som HTML4, HTML3.2, XHTML1.0 og alt muligt andet. I artiklen vises hvordan det gøres med få liniers kode.

Tre slags uddata

Vi har tidligere gennemgået teknologierne XML, XSLT, CSS, HTML4, HTML3.2, XHTML1.0 i adskillige artikler. I denne artikel viser vi helt konkret, hvorledes XML-baseret inddata sammen med XSLT-ark hurtigt og nemt kan generere uddata så forskelligt som HTML3.2 til gamle browsere, XHTML til moderne browsere og en version til håndholdte - alt sammen ud fra samme inddata.

Vi har i en tidligere artikel gennemgået, hvorledes XML transformeres med XSLT.

Som udgangspunkt har vi et XML-dokument med en eller anden form for indhold. Her i eksemplet er det en af vores egne artikler, der danner inddata, men det kunne lige så godt være produktinformation eller alt mulig andet dynamisk generet indhold.


<?xml version="1.0" encoding="ISO-8859-1"?>
<side>
  <titel>Comanche - administration af Apache</titel>
  <overskrift>Grafisk administration af Apache</overskrift>
  <manchet>
    Fuldblods Unix-hajer redigerer deres
    konfigurationsfiler i Vi og Emacs.
  </manchet>
  <tekst>
    <afsnit>Comanche kan downloades fra
      www.covalent.net og findes i
      forskellige versioner.</afsnit>
    <afsnit>Comanche har sit udspring i
      Apache GUI-projektet, der
      startede i 1997.</afsnit>
  </tekst>
</side>

Først skriver vi et XSLT-ark, til ældre HTML3.2 browsere. Her formaterer vi al tekst med gammeldags <FONT>-tags.


<xsl:template match="overskrift">
  <p><font face="arial,helvetica" size="3"><b>
      <xsl:value-of select="." />
  </b></font></p>
</xsl:template>

Hele arket kan ses sidst i denne artikel.

Derefter er det simpelt at skrive et XSLT-ark til XHTML baseret på et CSS-ark. Vi tager simpelthen vores ark til HTML3.2-browsere og erstatter alle formateringtags med CSS-klasser.

Toppen af arket ser således ud:


<xsl:template match="side">
<html>
<head>
  <title><xsl:value-of select="titel" /></title>
  <link rel="stylesheet"
  type="text/css"
  href="xhtml_styles.css" />
</head>
<body>

<xsl:apply-templates />

</body>
</html>
</xsl:template>

Og de enkelte dele af XML-arket formateres således:


<xsl:template match="overskrift">
  <h1 class="overskrift">
    <xsl:value-of select="." />
  </h1>
</xsl:template>

Hele arket kan ses sidst i artiklen.

Til sidst laver vi en version til PDA-tjenesten AvantGo, hvor brugere af lommecomputere baseret på PalmOS og Windows CE kan downloade offline-versioner af webindhold. Der er stort set blot tale om en neddroslet version af vores HTML3.2-ark. Mere information kan findes på AvantGo's side for udviklere. Meta-tagget <meta name="HandheldFriendly" content="True" /> er det vigtigeste i denne sammenhæng - det fortæller, at dokumentet er optimeret til håndholdt udstyr:


<xsl:template match="side">
<html>
<head>
  <meta name="HandheldFriendly" content="True" />
  <title><xsl:value-of select="titel" /></title>
</head>
<body>

<xsl:apply-templates />

</body>
</html>
</xsl:template>

Og de enkelte dele formateres som gammeldags HTML, men uden <FONT>-tags:


<xsl:template match="overskrift">
  <h1 class="overskrift">
    <xsl:value-of select="." />
  </h1>
</xsl:template>

Nu da vi har de tre relevante XSLT-dokumenter, er det bare at sende de rigtige versioner afsted til de rigtige brugeragenter.

Browserdetektering

Vores uddata til håndholdte lægger vi på en særskilt adresse, for der skal brugerne alligvel navigere frem til sitet på egen hånd. Men alle de almindelige browsere kommer ind på den samme adresse, så her skal vi skelne imellem dem der kan håndtere XHTML og CSS, og dem der skal have gammeldags HTML3.2.

Det gøres ved at kigge på browserens navn og versionsnummer. Netscape og Internet Explorer er CSS-kapable i versioner 4 og opefter, og Opera for version 5 og op. Så disse versioner får vores CSS-formaterede XHTML1 version. I PHP gøres det for eksempel som vist herunder, forudsat at PHP er kompileret med XSLT-udvidelsen Sablotron. I eksemplet har vi kilde-xml-data liggende i en variabel ved navn $input.


// Vi ser, hvad for en browser, brugeren benytter

$ua = get_browser();

$browser_xhtml_capable =
  (( $ua->browser == "Netscape" )
  && ( $ua->version >= 4 ))
||
  (($ua->browser == "Internet Explorer" )
  && ( $ua->version >= 4 ))
||
  (( $ua->browser == "Opera" )
  && ( $ua->version >= 5 ));

if($browser_xhtml_capable) {
  // Disse browsere kan håndtere CSS,
  // så de får vores XHTML1.0-formaterede
  // dokument:
  $filename =
  "/usr/local/apache/htdocs/xml/xhtml.xsl";
} else {

  // Øvrige browsere får HTML3.2-versionen:
  $filename =
  "/usr/local/apache/htdocs/xml/html32.xsl";
}

// Så åbner vi det relevante XSLT-dokument...

$fd = fopen ($filename, "r");
$xsl_sheet = fread ($fd, filesize ($filename));
fclose ($fd);

// ...og processerer uddata:

$result = @xslt_process($xsl_sheet, $input,  $output);

if($result) {
  echo $output;
} else {
  echo "Der opstod desværre følgende fejl: "
  .xslt_error(xslt_errno());
}

I ASP skal man benytte objektet MSXML2.DOMDocument, og det foregår - i grove rids - som vist herunder i dialekten JScript, hvor input og filename er identisk med vores PHP-variabler $input og $filename:

¨
// Indlæs XML-inddata
var oSource = Server.CreateObject("MSXML2.DOMDocument.3.0");
oSource.async = false;
oSource.loadXML(input);

// Hent det rigtige XSLT-ark
var oStylesheet = Server.CreateObject("MSXML2.DOMDocument.3.0");
oStylesheet.async = false;
oStylesheet.load(filename);

// Processor uddata, og send det til klienten:
output = oSource.transformNode(oStylesheet);

Response.write(output);

Som det forhåbentligt kan ses, er det nemt at generere uddata til forskellige klienttyper. Ligeledes er der store fordele at hente når sitet skal redesignes, og når der skal fabrikeres uddata til nye klienttyper.

Koden bag XSL-arkene

XML-inddata blev vist først i artiklen, men man skal huske, at data skal gemmes som Unicode - ellers kan XSLT-processoren løbe sur i æ, ø og å.

Arkene kan testes på statisk vis i Internet Explorer ved at gemme XSL-arkene og XML-kilden som flade filer, og derefter tilføje denne streng som anden linie i XML-arket:

<?xml-stylesheet type="text/xsl" href=" HTML32.xsl" ?>

- hvor href-attributten skal pege på det XSL-ark, man ønsker at teste.

XSL-arket til HTML3.2-browserne ser sådan ud:


<?xml version="1.0" encoding="ISO-8859-1"?>

<xsl:template match="/">
  <xsl:apply-templates />
</xsl:template>

<xsl:template match="side">
<html>
<head>
   <title><xsl:value-of select="titel" /></title>
</head>
<body>

<xsl:apply-templates />

</body>
</html>
</xsl:template>

<xsl:template match="overskrift">
  <p><font face="arial,helvetica" size="3"><b><xsl:value-of select="." /></b></font></p>
</xsl:template>

<xsl:template match="manchet">
  <p><font face="arial,helvetica" size="2"><b><xsl:value-of select="." /></b></font></p>
</xsl:template>

<xsl:template match="tekst">
  <xsl:apply-templates />
</xsl:template>

<xsl:template match="afsnit">
  <p><font face="arial,helvetica" size="2"><xsl:value-of select="." /></font></p>
</xsl:template>

<xsl:template match="titel">
</xsl:template>

</xsl:stylesheet>

XSL-arket til XHTML-browserne ser således ud:


<?xml version="1.0" encoding="ISO-8859-1"?>

<xsl:template match="/">
    <xsl:apply-templates />
</xsl:template>

<xsl:template match="side">
  <html>
  <head>
    <title><xsl:value-of select="titel" /></title>
    <link rel="stylesheet" type="text/css" href="xhtml_styles.css" />
  </head>
  <body>

  <xsl:apply-templates />
  
  </body>
  </html>
</xsl:template>

<xsl:template match="overskrift">
    <h1 class="overskrift"><xsl:value-of select="." /></h1>
</xsl:template>

<xsl:template match="manchet">
    <p class="manchet"><xsl:value-of select="." /></p>
</xsl:template>

<xsl:template match="tekst">
    <xsl:apply-templates />
</xsl:template>

<xsl:template match="afsnit">
    <p class="broedtekst"><xsl:value-of select="." /></p>
</xsl:template>

<xsl:template match="titel">
</xsl:template>

</xsl:stylesheet>

Til sidst er der XSL-arket til AvantGo-tjenesten:


<?xml version="1.0" encoding="ISO-8859-1"?>

<xsl:template match="/">
  <xsl:apply-templates />
</xsl:template>

<xsl:template match="side">
  <html>
  <head>
      <meta name="HandheldFriendly" content="True" />
    <title><xsl:value-of select="titel" /></title>
  </head>
  <body>

  <xsl:apply-templates />
  
  </body>
  </html>
</xsl:template>

<xsl:template match="overskrift">
  <h1><xsl:value-of select="." /></h1>
</xsl:template>

<xsl:template match="manchet">
  <p><b><xsl:value-of select="." /></b></p>
</xsl:template>

<xsl:template match="tekst">
  <xsl:apply-templates />
</xsl:template>

<xsl:template match="afsnit">
  <p><xsl:value-of select="." /></p>
</xsl:template>

<xsl:template match="titel">
</xsl:template>

</xsl:stylesheet>




Brancheguiden
Brancheguide logo
Opdateres dagligt:
Den største og
mest komplette
oversigt
over danske
it-virksomheder
Hvad kan de? Hvor store er de? Hvor bor de?
Ed A/S
Salg af hard- og software.

Nøgletal og mere info om virksomheden
Skal din virksomhed med i Guiden? Klik her

Kommende events
Computerworld Cloud & AI Festival 2025

Med den eksplosive udvikling indenfor cloud & AI er behovet for at følge med og vidensdeling større end nogensinde før. Glæd dig til to dage, hvor du kan netværke med over 2.400 it-professionelle, møde mere end 50 it-leverandører og høre indlæg fra +90 talere. Vi sætter fokus på emner som AI; infrastruktur, compliance, sikkerhed og løsninger for både private og offentlige organisationer.

17. september 2025 | Læs mere


IT og OT i harmoni: Sikring uden at gå på kompromis med effektiviteten

IT og OT smelter sammen – men med risiko for dyre fejl. Få metoder til sikker integration med ERP, kundesystemer og produktion. Tilmeld dig og få styr på forskellene og faldgruberne.

24. september 2025 | Læs mere


NIS2: Vi gør status efter tre måneder og lærer af erfaringerne

Vær med, når vi deler oplevelser med implementering af NIS2 og drøfter, hvordan du undgår at gentage erfaringerne fra GDPR – og særligt undgår kostbar overimplementering.

30. september 2025 | Læs mere