<?xml version="1.0" encoding="UTF-8" ?> <!-- Application Description Language framework adl2activerecord.xslt (c) 2008 Cygnet Solutions Ltd Transform ADL into ActiveRecord conforming entity classes $Author: sb $ $Revision: 1.3 $ $Date: 2008-10-02 10:52:40 $ --> <!-- WARNING WARNING WARNING: Do NOT reformat this file! Whitespace (or lack of it) is significant! --> <xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:exsl="http://exslt.org/common" xmlns:adl="http://libs.cygnets.co.uk/adl/1.1/" xmlns:msxsl="urn:schemas-microsoft-com:xslt"> <!-- NOTE NOTE NOTE: deliberately does NOT include 'csharp-type-include' as ActiveRecord appears to need slightly different base types from NHibernate --> <xsl:include href="base-type-include.xslt"/> <xsl:output encoding="UTF-8" method="text"/> <!-- The locale for which these entities are generated TODO: Entities should NOT be locale specific. Instead, the entity should generate messages based on the client's locale. However, there may still need to be a concept of a 'default locale', for when we don't have messages which suit the client's locale --> <xsl:param name="locale" select="en-GB"/> <!-- The C# namespace within which I shall generate controllers --> <xsl:param name="controllerns" select="Unset"/> <!-- The C# namespace within which I shall generate entities --> <xsl:param name="entityns" select="Unset"/> <!-- the name and version of the product being built --> <xsl:param name="product-version" select="'Application Description Language Framework'"/> <!-- strings used in normalising names for constants. NOTE NOTE NOTE: this deliberately converts space and punctuation to underscore --> <xsl:variable name="notpermittedinconstantname">abcdefghijklmnopqrstuvwxyz .,;:?/()</xsl:variable> <xsl:variable name="permittedinconstantname">ABCDEFGHIJKLMNOPQRSTUVWXYZ_________</xsl:variable> <xsl:template match="adl:application"> /* ---- [ cut here: next file '<xsl:value-of select="@name"/>.auto.cs'] ---------------- */ //------------------------------------------------------------- // // <xsl:value-of select="$product-version"/> // <xsl:value-of select="@name"/>.auto.cs // // (c)2007 Cygnet Solutions Ltd // // Main application entry point for <xsl:value-of select="@name"/> // Automatically generated from application description using // adl2activerecord.xsl revision <xsl:value-of select="substring( '$Revision: 1.3 $', 10)"/> // // <xsl:value-of select="/adl:application/@revision"/> // // This file is automatically generated; DO NOT EDIT IT. // //------------------------------------------------------------- using System; using System.Windows.Forms; using Castle.ActiveRecord; using Castle.ActiveRecord.Framework.Config; using Cygnet.Entities; namespace <xsl:value-of select="$entityns"/> { /// <summary> /// <xsl:value-of select="normalize-space( adl:documentation)"/> /// </summary> /// <remarks> /// Automatically generated from description of application <xsl:value-of select="@name"/> /// using adl2activerecord.xsl revision <xsl:value-of select="substring( '$Revision: 1.3 $', 10)"/>. /// Note that manually maintained parts of this class may be defined in /// a separate file called <xsl:value-of select="@name"/>.manual.cs, q.v. /// /// DO NOT EDIT THIS FILE! /// </remarks> public class <xsl:value-of select="@name"/> { public static void Main() { XmlConfigurationSource source = new XmlConfigurationSource("appconfig.xml"); ActiveRecordStarter.Initialize( source <xsl:for-each select="adl:entity"> , typeof( <xsl:value-of select="@name"/>) </xsl:for-each>); } } } /* ---- [ cut here: next file 'junk'] ------------------------- */ <xsl:apply-templates select="adl:entity"/> </xsl:template> <!-- Don't bother generating anything for foreign entities --> <xsl:template match="adl:entity[@foreign='true']"/> <xsl:template match="adl:entity"> /* ---- [ cut here: next file '<xsl:value-of select="@name"/>.auto.cs'] ---------------- */ //------------------------------------------------------------- // // <xsl:value-of select="$product-version"/> // <xsl:value-of select="@name"/>.auto.cs // // (c)2007 Cygnet Solutions Ltd // // Automatically generated from application description using // adl2activerecord.xsl revision <xsl:value-of select="substring( '$Revision: 1.3 $', 10)"/> // // <xsl:value-of select="/adl:application/@revision"/> // // This file is automatically generated; DO NOT EDIT IT. // //------------------------------------------------------------- using System; using System.Text; using System.Collections.Generic; using Castle.ActiveRecord; using Cygnet.Entities; using Cygnet.Exceptions; using Cygnet.Entities.Exceptions; namespace <xsl:value-of select="$entityns"/> { /// <summary> /// <xsl:value-of select="normalize-space( adl:documentation)"/> /// </summary> /// <remarks> /// Automatically generated from description of entity <xsl:value-of select="@name"/> /// using adl2activerecord.xsl revision <xsl:value-of select="substring( '$Revision: 1.3 $', 10)"/>. /// Note that manually maintained parts of this class may be defined in /// a separate file called <xsl:value-of select="@name"/>.manual.cs, q.v. /// /// DO NOT EDIT THIS FILE! /// </remarks> [ActiveRecord] public partial class <xsl:value-of select="@name"/> : AdlActiveRecord<<xsl:value-of select="@name"/>> { /// <summary> /// Auto-generated no-args constructor; does nothing (but probably should /// ensure ID slot is initialised correctly) /// </summary> public <xsl:value-of select="@name"/>() : base(){ <xsl:call-template name="initialise-lists"/> } <xsl:choose> <xsl:when test="@natural-key"> /* natural primary key exists - not generating abstract key */ </xsl:when> <xsl:when test="adl:key"> /* primary key exists - not generating abstract key */ /// <summary> /// Auto-generated constructor; initialises each of the slots within /// the primary key and also all one-to-many and many-to-many slots /// </summary> public <xsl:value-of select="@name"/>( <xsl:for-each select="adl:key/adl:property"> <xsl:variable name="csharp-type"> <xsl:call-template name="csharp-type"> <xsl:with-param name="property" select="."/> </xsl:call-template> </xsl:variable> <xsl:value-of select="concat( $csharp-type, ' ', @name)"/> <xsl:if test="not( position() = last())">, </xsl:if> </xsl:for-each>){ <xsl:call-template name="initialise-lists"/> <xsl:call-template name="initialise-messages"/> <xsl:for-each select="adl:key/adl:property"> this.<xsl:value-of select="@name"/> = <xsl:value-of select="@name"/>; </xsl:for-each> } </xsl:when> <xsl:otherwise> <xsl:message terminate="yes"> ADL: ERROR: Entity '<xsl:value-of select="@name"/>' has no key. Was the canonicalise stage missed in the build process? </xsl:message> </xsl:otherwise> </xsl:choose> /// <summary> /// Auto-generated overridden property for the Key slot, maps onto /// <xsl:for-each select="adl:key/adl:property"> /// <xsl:value-of select="@name"/><xsl:if test="@type='entity'">.KeyString</xsl:if></xsl:for-each> /// </summary> public override string KeyString { get { StringBuilder result = new StringBuilder(); <xsl:for-each select="adl:key/adl:property"> result.Append(<xsl:value-of select="@name"/><xsl:if test="@type='entity'">.KeyString</xsl:if>); <xsl:if test="position()!=last()"> result.Append('|'); </xsl:if> </xsl:for-each> return result.ToString(); } } /// <summary> /// A user readable distinct identifying string /// </summary> public override string UserIdentifier { get { StringBuilder result = new StringBuilder(); <xsl:choose> <xsl:when test="descendant::adl:property[@distinct='user' or @distinct='all']"> <xsl:for-each select="descendant::adl:property[@distinct='user' or @distinct='all']"> <xsl:choose> <xsl:when test="@type='message'"> if ( <xsl:value-of select="@name"/> != null){ result.Append( <xsl:value-of select="concat( @name, '.LocalText')"/>); } </xsl:when> <xsl:when test="@type='entity'"> <!-- TODO: this is dangerous and could potentially give rise to infinite loops; find a way of stopping it running away! --> if ( <xsl:value-of select="@name"/> != null){ result.Append( <xsl:value-of select="concat( @name, '.UserIdentifier')"/>); } </xsl:when> <xsl:when test="@type='date'"> <!-- if what we've got is just a date, we only want to see the date part of it --> if ( <xsl:value-of select="@name"/> != null){ result.Append(<xsl:value-of select="@name"/>.ToString( "d")); } </xsl:when> <xsl:when test="@type='time'"> <!-- if what we've got is just a time, we only want to see the time part of it --> if ( <xsl:value-of select="@name"/> != null){ result.Append(<xsl:value-of select="@name"/>.ToString( "t")); } </xsl:when> <xsl:when test="@required = 'true' and (@type = 'integer' or @type = 'real' or @type = 'boolean' or @type = 'money')"> <!-- can't be null because we will have generated a non-nullable type --> result.Append(<xsl:value-of select="@name"/>.ToString()); </xsl:when> <xsl:otherwise> if ( <xsl:value-of select="@name"/> != null){ result.Append(<xsl:value-of select="@name"/>); } </xsl:otherwise> </xsl:choose> <xsl:choose> <xsl:when test="position() = last()"/> <xsl:otherwise> result.Append( ","); </xsl:otherwise> </xsl:choose> </xsl:for-each> </xsl:when> <xsl:otherwise> result.AppendFormat( "<xsl:value-of select="@name"/>#{0}", KeyString); </xsl:otherwise> </xsl:choose> return result.ToString(); } } /// <summary> /// If I should not be deleted, return a message explaining why I should not be deleted; else null. /// </summary> /// <returns>a message explaining why I should not be deleted; else null</returns> public override string NoDeleteReason { get { string result = null; <xsl:if test="adl:property[@type='list']|adl:property[@type='link']"> StringBuilder bob = new StringBuilder(); <!-- TODO: we ought to start worrying about internationalisation NOW, not later! --> <xsl:for-each select="adl:property[@type='list']|adl:property[@type='link']"> <xsl:choose> <xsl:when test="@cascade='manual'"/> <xsl:when test="@cascade='all'"/> <xsl:when test="@cascade='all-delete-orphan'"/> <xsl:when test="@cascade='delete'"/> <xsl:when test="@concrete='false'"/> <xsl:otherwise> if ( <xsl:value-of select="concat( ' _', @name)"/> != null && <xsl:value-of select="concat( ' _', @name)"/>.Count > 0) { bob.AppendFormat("Cannot delete this <xsl:value-of select="../@name"/> as it has {0} dependent <xsl:value-of select="@name"/>; ", <xsl:value-of select="concat( ' _', @name)"/>.Count); } </xsl:otherwise> </xsl:choose> </xsl:for-each> if (bob.Length > 0) { result = bob.ToString(); } </xsl:if> return result; } } <!-- 'descendant' to catch properties inside keys as well as properties which are direct children --> <xsl:apply-templates select="descendant::adl:property"/> } } /* ---- [ cut here: next file 'junk'] ------------------------- */ </xsl:template> <xsl:template match="adl:property[@concrete='false']"> <!-- generate nothing for non-concrete properties --> /* NOTE: property '<xsl:value-of select="@name"/>' is marked as being abstract; it must * be supported by manually maintained code */ </xsl:template> <xsl:template match="adl:property"> // auto generating iv/property pair for slot with name <xsl:value-of select="@name"/> <xsl:apply-templates select="help"/> <xsl:variable name="base-type"> <xsl:call-template name="base-type"> <xsl:with-param name="property" select="."/> </xsl:call-template> </xsl:variable> <xsl:variable name="csharp-type"> <xsl:call-template name="csharp-type"> <xsl:with-param name="property" select="."/> </xsl:call-template> </xsl:variable> <xsl:variable name="nullable-decoration"> <xsl:choose> <xsl:when test="@required='true'"/> <!-- when required is 'true' null is not permitted anyway; otherwise... --> <xsl:when test="@type='message'"/> <xsl:when test="$base-type='entity'"/> <xsl:when test="$base-type='string'"/> <xsl:when test="$base-type='text'"/> <!-- entities and strings are always nullable, don't need decoration --> <xsl:when test="$base-type='list'"/> <xsl:when test="$base-type='link'"/> <!-- things which are collections are not nullable --> <xsl:otherwise>?</xsl:otherwise> </xsl:choose> </xsl:variable> <xsl:variable name="initialiser"> <xsl:choose> <xsl:when test="@default"> <xsl:choose> <xsl:when test="$csharp-type = 'String'"> = "<xsl:value-of select="@default"/>" </xsl:when> <xsl:otherwise> = <xsl:value-of select="@default"/> </xsl:otherwise> </xsl:choose> </xsl:when> <xsl:when test="normalize-space( $nullable-decoration) = '?'"> = null</xsl:when> <xsl:when test="$base-type = 'Boolean'"> = false</xsl:when> <xsl:when test="$base-type = 'int'"> = 0</xsl:when> <xsl:when test="$csharp-type = 'Decimal'"> = 0.0M</xsl:when> <xsl:when test="$base-type = 'real'"> = 0.0</xsl:when> <xsl:when test="$base-type='String'"> = null</xsl:when> </xsl:choose> </xsl:variable> <xsl:variable name="validationpattern"> <xsl:choose> <xsl:when test="@type='defined'"> <xsl:variable name="definition"> <xsl:value-of select="@typedef"/> </xsl:variable> <xsl:value-of select="//adl:typedef[@name=$definition]/@pattern"/> </xsl:when> </xsl:choose> </xsl:variable> <xsl:if test="string-length( $validationpattern) > 0"> private Regex <xsl:value-of select="@name"/>Validator = new Regex( "<xsl:value-of select="$validationpattern"/>"); </xsl:if> <xsl:if test="$base-type = 'integer'"> <xsl:for-each select="adl:option"> <xsl:variable name="constname"> <xsl:call-template name="constname"> <xsl:with-param name="option" select="."/> <xsl:with-param name="locale" select="$locale"/> </xsl:call-template> </xsl:variable> public const int <xsl:value-of select="$constname"/> = <xsl:choose> <xsl:when test="@value"> <xsl:value-of select="@value"/> </xsl:when> <xsl:when test="adl:prompt[@locale='default']/@prompt"> <xsl:value-of select="adl:prompt[@locale='default']/@prompt"/> </xsl:when> <xsl:when test="adl:prompt[@locale=$locale]/@prompt"> <xsl:value-of select="adl:prompt[@locale=$locale]/@prompt"/> </xsl:when> <xsl:otherwise> <xsl:message terminate="yes"> ADL: ERROR: Cannot infer a value for option </xsl:message> </xsl:otherwise> </xsl:choose>; </xsl:for-each> </xsl:if> private <xsl:value-of select="normalize-space( $csharp-type)"/><xsl:value-of select="normalize-space( $nullable-decoration)"/> <xsl:value-of select="concat( ' _', @name)"/> <xsl:value-of select="normalize-space( $initialiser)"/>; /// <summary> <xsl:choose> <xsl:when test="adl:documentation">/// <xsl:value-of select="normalize-space(adl:documentation)"/></xsl:when> <xsl:otherwise>/// Auto generated property for field <xsl:value-of select="@name"/></xsl:otherwise> </xsl:choose> /// <xsl:value-of select="normalize-space( help[@locale=$locale])"/> /// </summary> <xsl:choose> <xsl:when test="parent::adl:key">[PrimaryKey]</xsl:when> <xsl:when test="$base-type='text'">[Property(ColumnType="StringClob")]</xsl:when> <xsl:when test="$base-type='list'"> <xsl:choose> <xsl:when test="@cascade='all'"> [HasMany(Inverse=true, Cascade=ManyRelationCascadeEnum.All)] </xsl:when> <xsl:when test="@cascade='all-delete-orphan'"> [HasMany(Inverse=true, Cascade=ManyRelationCascadeEnum.AllDeleteOrphan)] </xsl:when> <xsl:when test="@cascade='delete'">[HasMany(Inverse=true, Cascade=ManyRelationCascadeEnum.Delete)]</xsl:when> <xsl:when test="@cascade='save-update'">[HasMany(Inverse=true, Cascade=ManyRelationCascadeEnum.SaveUpdate)]</xsl:when> <xsl:otherwise>[HasMany]</xsl:otherwise> </xsl:choose> </xsl:when> <xsl:when test="$base-type='entity'">[BelongsTo(<xsl:value-of select="@entity"/>)]</xsl:when> <xsl:otherwise>[Property]</xsl:otherwise> </xsl:choose> <xsl:if test="@distinct = 'user' or @distinct = 'all'">[IsUserDistinct()]</xsl:if> public virtual <xsl:value-of select="normalize-space( $csharp-type)"/><xsl:value-of select="normalize-space( $nullable-decoration)"/><xsl:text> </xsl:text> <xsl:value-of select="@name"/> { get { <xsl:if test="$base-type='list'"> if ( <xsl:value-of select="concat( ' _', @name)"/> == null) { <xsl:value-of select="concat( '_', @name)"/> = new List<<xsl:value-of select="@entity"/>>(); } </xsl:if> return <xsl:value-of select="concat( ' _', @name)"/>; } set { <xsl:if test="@required='true'"> <!-- If we could generate a non-nullable type we have done so; otherwise, must catch null --> <xsl:choose> <xsl:when test="@type='integer'"/> <xsl:when test="@type='real'"/> <xsl:when test="@type='money'"> <!-- not quite certain of that - check! --> </xsl:when> <xsl:when test="@type='boolean'"/> <xsl:otherwise> if ( value == null) { throw new DataRequiredException( <xsl:choose> <xsl:when test="ifmissing[@locale=$locale]"> <xsl:apply-templates select="ifmissing"/> </xsl:when> <xsl:otherwise> "The value for <xsl:value-of select="@name"/> may not be set to null" </xsl:otherwise> </xsl:choose> ); } </xsl:otherwise> </xsl:choose> </xsl:if> <xsl:if test="@type='defined'"> <xsl:variable name="definition"> <xsl:value-of select="@typedef"/> </xsl:variable> <xsl:variable name="maximum"> <xsl:value-of select="//adl:typedef[@name=$definition]/@maximum"/> </xsl:variable> <xsl:variable name="minimum"> <xsl:value-of select="//adl:typedef[@name=$definition]/@minimum"/> </xsl:variable> <xsl:if test="string-length( $maximum) > 0"> if ( value > <xsl:value-of select="$maximum"/>) { throw new DataRangeException( "The maximum permitted value for <xsl:value-of select="@name"/> is <xsl:value-of select="$maximum"/>"); } </xsl:if> <xsl:if test="string-length( $minimum) > 0"> if ( value < <xsl:value-of select="$minimum"/>) { throw new DataRangeException( "The minimum permitted value for <xsl:value-of select="@name"/> is <xsl:value-of select="$minimum"/>"); } </xsl:if> <xsl:if test="string-length( $validationpattern) > 0"> if ( value != null && ! <xsl:value-of select="@name"/>Validator.IsMatch( value)) { throw new DataFormatException( string.Format( "The value supplied ({0}) does not match the format required by <xsl:value-of select="@name"/>", value)); } </xsl:if> </xsl:if> <xsl:if test="adl:option"> <xsl:choose> <xsl:when test="$base-type = 'integer'"> switch ( value ) { <xsl:for-each select="adl:option"> case <xsl:call-template name="constname"> <xsl:with-param name="option" select="."/> <xsl:with-param name="locale" select="$locale"/> </xsl:call-template>: </xsl:for-each> break; default: throw new DataRangeException( string.Format( "The value supplied ({0}) is not an acceptable value for <xsl:value-of select="@name"/>", value)); } </xsl:when> </xsl:choose> </xsl:if> <xsl:if test="@size and $csharp-type='String'"> if ( value != null && value.Length > <xsl:value-of select="@size"/>) { value = value.Substring( 0, <xsl:value-of select="@size"/>); } </xsl:if> <xsl:value-of select="concat( ' _', @name)"/> = value; } } <xsl:if test="parent::adl:key and @type='entity'"> /// <summary> /// auto generated primitive value for key property of type entity (experimental) /// </summary> <xsl:variable name="csharp-base-type"> <xsl:call-template name="csharp-base-type"> <xsl:with-param name="property" select="."/> </xsl:call-template> </xsl:variable> private <xsl:value-of select="concat( $csharp-base-type, ' _', @name, '_Value')"/> <xsl:choose> <xsl:when test="$csharp-base-type = 'Boolean'"> = false</xsl:when> <xsl:when test="$csharp-base-type = 'int'"> = 0</xsl:when> <xsl:when test="$csharp-base-type = 'Decimal'"> = 0.0M</xsl:when> <xsl:when test="$csharp-base-type = 'real'"> = 0.0</xsl:when> <xsl:when test="$csharp-base-type='String'"> = null</xsl:when> <xsl:otherwise> [unknown? <xsl:value-of select="$csharp-base-type"/>] </xsl:otherwise> </xsl:choose>; /// <summary> /// auto generated primitive value getter/setter for key property of type entity (experimental) /// </summary> public virtual <xsl:value-of select="concat( $csharp-base-type, ' ', @name, '_Value')"/> { get { return <xsl:value-of select="concat( '_', @name, '_Value')"/>; } set { <xsl:value-of select="concat( '_', @name, '_Value')"/> = value; } } </xsl:if> </xsl:template> <xsl:template match="adl:help"> <xsl:if test="@locale=$locale"> <!-- might conceivably be more than one line --> <xsl:text> /* </xsl:text><xsl:apply-templates/> */ </xsl:if> </xsl:template> <xsl:template match="ifmissing"> <xsl:if test="@locale=$locale"> "<xsl:value-of select="normalize-space(.)"/>" </xsl:if> </xsl:template> <xsl:template match="adl:documentation"> /// <xsl:apply-templates/> </xsl:template> <xsl:template name="initialise-messages"> <!-- each IV of type message needs to be initialised --> <xsl:for-each select="adl:property[@type='message']"> <xsl:choose> <xsl:when test="@concrete='false'"/> <xsl:otherwise> <xsl:value-of select="concat( ' _', @name)"/> = new Message(); </xsl:otherwise> </xsl:choose> </xsl:for-each> </xsl:template> <xsl:template name="initialise-lists"> <!-- initialise all concrete lists and links --> <xsl:for-each select="adl:property[@type='list' or @type='link']"> <xsl:choose> <xsl:when test="@concrete='false'"/> <xsl:otherwise> <xsl:value-of select="concat( ' _', @name)"/> = new List<<xsl:value-of select="@entity"/>>(); </xsl:otherwise> </xsl:choose> </xsl:for-each> </xsl:template> <!-- produce a name for a constant value from an option entity --> <xsl:template name="constname"> <xsl:param name="option"/> <xsl:param name="locale"/> <xsl:variable name="unnormalisedname"> <xsl:choose> <xsl:when test="$option/adl:prompt[@locale='default']"> <xsl:value-of select="concat( $option/ancestor::adl:property/@name, '_', $option/adl:prompt[@locale='default']/@prompt, '_value')"/> </xsl:when> <xsl:when test="$option/adl:prompt[@locale=$locale]"> <xsl:value-of select="concat( $option/ancestor::adl:property/@name, '_', $option/adl:prompt[@locale=$locale]/@prompt, '_value')"/> </xsl:when> <xsl:when test="$option/@value"> <xsl:value-of select="concat( $option/ancestor::adl:property/@name, '_', $option/@value, '_value')"/> </xsl:when> <xsl:otherwise> <!-- this one's dodgy but shouldn't be hit often (perhaps never) --> <xsl:value-of select="concat( $option/ancestor::adl:property/@name, '_', position(), '_value')"/></xsl:otherwise> </xsl:choose> </xsl:variable> <xsl:value-of select="translate($unnormalisedname, $notpermittedinconstantname, $permittedinconstantname)"/> </xsl:template> <xsl:template name="csharp-base-type"> <xsl:param name="property"/> <xsl:param name="entityns"/> <xsl:choose> <xsl:when test="$property/@type = 'entity'"> <xsl:variable name="entityname" select="$property/@entity"/> <xsl:choose> <xsl:when test="//adl:entity[@name=$entityname]/adl:key/adl:property"> <!-- recurse... --> <xsl:call-template name="csharp-base-type"> <xsl:with-param name="property" select="//adl:entity[@name=$entityname]/adl:key/adl:property[position()=1]"/> <xsl:with-param name="entityns" select="$entityns"/> </xsl:call-template> </xsl:when> <xsl:otherwise> <xsl:message terminate="yes"> ADL: ERROR: could not find C# base type of property <xsl:value-of select="$property/@name"/> </xsl:message> </xsl:otherwise> </xsl:choose> </xsl:when> <xsl:otherwise> <xsl:call-template name="csharp-type"> <xsl:with-param name="property" select="$property"/> <xsl:with-param name="entityns" select="$entityns"/> </xsl:call-template> </xsl:otherwise> </xsl:choose> </xsl:template> <!-- return the C# type of the property which is passed as a parameter --> <xsl:template name="csharp-type"> <xsl:param name="property"/> <xsl:param name="entityns"/> <xsl:variable name="base-type"> <xsl:call-template name="base-type"> <xsl:with-param name="property" select="$property"/> </xsl:call-template> </xsl:variable> <xsl:choose> <xsl:when test="$property/@type = 'message'">Message</xsl:when> <xsl:when test="$base-type = 'link'"> IList<<xsl:value-of select="@entity"/>> </xsl:when> <xsl:when test="$base-type = 'list'"> IList<<xsl:value-of select="@entity"/>> </xsl:when> <xsl:when test="$base-type = 'date'">DateTime</xsl:when> <xsl:when test="$base-type = 'time'">DateTime</xsl:when> <xsl:when test="$base-type = 'string'">String</xsl:when> <xsl:when test="$base-type = 'text'">String</xsl:when> <xsl:when test="$base-type = 'boolean'">Boolean</xsl:when> <xsl:when test="$base-type = 'timestamp'">DateTime</xsl:when> <xsl:when test="$base-type = 'integer'">int</xsl:when> <xsl:when test="$base-type = 'real'">double</xsl:when> <xsl:when test="$base-type = 'money'">Decimal</xsl:when> <xsl:when test="$base-type = 'entity'"> <xsl:choose> <xsl:when test="$entityns"> <xsl:value-of select="concat( $entityns, '.', $property/@entity)"/> </xsl:when> <xsl:otherwise> <xsl:value-of select="$property/@entity"/> </xsl:otherwise> </xsl:choose> </xsl:when> <xsl:otherwise>[unknown?]</xsl:otherwise> </xsl:choose> </xsl:template> </xsl:stylesheet>