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>

Læses lige nu

    Erhvervsakademi Aarhus

    Undervisere til it, digitalisering og AI

    Midtjylland

    PensionDanmark

    Incident & Problem Manager

    Københavnsområdet

    Netcompany A/S

    Software Developer

    Københavnsområdet

    Annonceindlæg fra Conscia

    Internettets smutveje er smarte – men hvad gør I for at sikre dem?

    Det er fristende at spare tid med lækre internetapps og GenAI. Men pas på. Uautoriseret brug risikerer at underminere hele forretningsgevinsten.

    Navnenyt fra it-Danmark

    Norriq Danmark A/S har pr. 1. september 2025 ansat Niels Bjørndal Nygaard som Digital Product Lead. Han skal især beskæftige sig med designe og implementere effektive IT-løsninger. Han har tidligere beskæftiget sig med at være digital consultant og project Manager hos Peytz & Co. Nyt job

    Niels Bjørndal Nygaard

    Norriq Danmark A/S

    Signifly har pr. 1. august 2025 ansat Morten Eskildsen som UX Lead. Han skal især beskæftige sig med styrke bureauets kompetencer inden for UX og digital strategi i konsulentteamet og arbejde på tværs af alle afdelinger. Han kommer fra en stilling som CDO, UX & Digital Strategy Consultant hos Zupa. Nyt job

    Morten Eskildsen

    Signifly

    Danske Spil har pr. 1. oktober 2025 ansat Jesper Krogh Heitmann som Brand Manager for Oddset. Han skal især beskæftige sig med at udvikle og drive brandets strategi og sikre en rød tråd på tværs af alle platforme og aktiviteter. Han kommer fra en stilling som Marketing & Communications Manager hos Intellishore. Nyt job

    Jesper Krogh Heitmann

    Danske Spil

    Netip A/S har pr. 15. september 2025 ansat Jimmi Overgaard som Key Account Manager ved netIP's kontor i Viborg. Han kommer fra en stilling som Sales Executive hos Globalconnect A/S. Nyt job

    Jimmi Overgaard

    Netip A/S