# Copyright 2007 Suraj N. Kurapati # See the file named LICENSE for details. desc: web page for the Internet code: | # load the String#to_html method require 'gerbil/html' class String # Transforms this UTF-8 string into HTML entities. def to_html_entities unpack('U*').map! {|c| "&##{c};"}.join end # Transforms this string into a valid URI fragment. # See http://www.nmt.edu/tcc/help/pubs/xhtml/id-type.html def to_uri_fragment # remove HTML tags from the input buf = gsub(/<.*?>/, '') # The first or only character must be a letter. buf.insert(0, 'a') unless buf[0,1] =~ /[[:alpha:]]/ # The remaining characters must be letters, digits, hyphens (-), # underscores (_), colons (:), or periods (.) or Unicode characters buf.unpack('U*').map! do |code| if code > 0xFF or code.chr =~ /[[:alnum:]\-_:\.]/ code else 32 # ASCII character code for a single space end end.pack('U*').strip.gsub(/[[:space:]-]+/, '-') end end class Node # Returns the user-defined title for this node's content. def title @title ||= args[0] end # Returns the user-defined indentifer for this node. def id @id ||= args[1] end @@frags = [] # Returns a unique URI fragment for this node. def url unless defined? @url frag = (id || title || object_id).to_s.to_uri_fragment frag << frag.object_id.to_s while @@frags.include? frag # make it unique @@frags << frag @url = frag end @url end # Returns the URI fragment for the location in the table # of contents / list of figures that points this node. def list_url @list_url ||= trace.object_id.to_s.to_uri_fragment end end # Returns a hyperlink to the given URL of # the given name and mouse-hover title. def link aUrl, aName = aUrl, aTitle = nil %{#{aName}} end # Encodes the given input in base64 format. def encode_base_64 aInput [aInput].pack('m') end # Returns a HTML tag that embeds the given image data. # # aData:: raw image image data in base64 encoding # aFormat:: format of the image data (e.g. PNG, JPG, GIF, etc.) # aAttrs:: additional attributes for the tag # def embed_image_data aData, aFormat, aAttrs atts = aAttrs.inject('') {|s,(k,v)| s << %( #{k}="#{v}") } %{} end # Returns a HTML tag that embeds the given image file. # # aPath:: path to the image file # aFormat:: format of the image data (e.g. PNG, JPG, GIF, etc.) # aAttrs:: additional attributes for the tag # def embed_image_file aPath, aFormat = aPath[/\w+$/], aAttrs = {} data = encode_base_64 File.read(aPath) embed_image_data data, aFormat, aAttrs end # load admonition icons Icon = Struct.new(:path, :name, :format, :data) class Icon # Returns a HTML image tag containing embedded image data. The given # attributes (name => value) are applied to the HTML tag declaration. def to_html aAttributes = {} embed_image_data data, format, aAttributes end end ICONS = {} Dir[File.join(Gerbil[:format_home], 'html.icons', '*.*')].each do |path| ext = File.extname(path) name = File.basename(path, ext) data = encode_base_64 File.read(path) ICONS[name] = Icon.new(path, name, ext.sub('.', ''), data) end nodes: ## # Structure header: &header toc: false lof: false index: false number: false silent: true output: <%= @node.content.to_html %> header_outside_above: *header header_inside_above: *header header_inside_below: *header header_outside_below: *header footer: *header footer_outside_above: *header footer_inside_above: *header footer_inside_below: *header footer_outside_below: *header abstract: toc: false lof: false index: false number: false silent: true output: |

<%= @node.type.capitalize %>

<%= @node.content.to_s.to_html %>
## # Organization part: &latex toc: true lof: false index: true number: true output: |

<%= @node.type.capitalize %> <%= @node.index %>
<%= @node.title.to_s.to_html %>

<%= @node.content.to_s.to_html %>
chapter: *latex section: toc: true lof: false index: true number: true output: | <% level = [6, @node.depth + 1].min %>
class="title"> <%= @node.index %>  <%= @node.title.to_s.to_html %> >
<%= @node.content.to_s.to_html %>
paragraph: toc: false lof: false index: false number: false output: |

<%= @node.title.to_s.to_html %>

<%= @node.content.to_s.to_html %>
## # Admonitions tip: &admonition toc: false lof: true index: false number: true output: |

<%= @node.type.capitalize %> <%= @node.number %>.  <%= @node.title.to_s.to_html %>

<% $use_icons = true unless defined? $use_icons if $use_icons && icon = ICONS[@node.type] ($icons_used ||= []) << icon %><%= icon.to_html :class => :icon %><% end %>
<%= @node.content.to_s.to_html %>
note: *admonition caution: *admonition warning: *admonition important: *admonition ## # Auxilary materials (formal blocks) # see http://www.sagehill.net/docbookxsl/FormalTitles.html figure: &formal toc: false lof: true index: false number: true output: |

<%= @node.type.capitalize %> <%= @node.number %>.  <%= @node.title.to_s.to_html %>

<%= @node.content.to_s.to_html %>
table: *formal example: *formal equation: *formal procedure: *formal ## # cross-references xref: toc: false lof: false index: false number: false output: | <%= scope = @nodes.select {|n| n.content} target = scope.find {|n| n.id == @node.args[0]} || # id has 1st priority scope.find {|n| n.title == @node.args[0]} if target prefix = '%s %s' % [ target.type.capitalize, target.index || target.number ] unless target.type == 'reference' title = "#{target.title.to_s.to_html}" end '%s' % [ @node.type, target.url, if @node.args.length > 1 @node.args[1].to_s.to_html else [prefix, title].compact.join(': ') end ] else raise ArgumentError, "invalid cross-reference to #{@node.title.inspect}", @node.trace end %> ## # Bibliography reference: toc: false lof: false index: false number: true silent: true cite: toc: false lof: false index: false number: false output: | <%= target = @types['reference'].find {|n| n.title == @node.args[0]} if target '[%s]' % [ '%s' % [ @node.type, target.url, target.number ], *@node.args[1..-1] ].join(', ') else raise ArgumentError, "invalid citation for #{@node.title.inspect}", @node.trace end %> output: | <% # set default parameter values $title = '$title' unless defined? $title $authors = {'$authors' => nil} unless defined? $authors $date = Time.now.strftime("%d %B %Y") unless defined? $date %> <% $feeds.each_pair do |url, fmt| %> <% end if $feeds %> <%= $title.to_s.to_html %> <% styles = @spec['styles'] %w[screen print].each do |media| %> <% end %> <%= node = @types['header_outside_above'].first and node.output %> <%= node = @types['header_outside_below'].first and node.output %> <% if abstract = @types['abstract'].first %> <%= abstract.output %> <% end %> <% # table of contents builder = lambda do |n| if @spec['nodes'][n.type]['toc'] entry = '%s%s' % [ (n.index.to_s + '  ' if n.index), n.list_url, n.url, n.title.to_s.to_html ] nested = n.children.inject('') {|s,c| s << builder[c] } %{
  • #{entry}#{ "" unless nested.empty? }
  • } else '' end end toc = @roots.inject('') {|s,n| s << builder[n] } unless toc.empty? %>

    Contents

    <% end %> <% # list of figures enums = {} # type => nodes @spec['nodes'].each_pair do |name, info| if info['lof'] nodes = @types[name] enums[name] = nodes unless nodes.empty? end end lof = enums.sort.map do |(type, nodes)| nested = nodes.map do |n| %{
  • #{n.title.to_s.to_html}
  • } end %{

    #{type.capitalize}s

      #{nested}
    } end unless lof.empty? %>
    <%= lof %>
    <% end %>
    <%= @content %>
    <% unless (refs = @types['reference']).empty? %>

    References

      <% refs.each do |n| %>
    1. <%= n.content.to_s.to_html %>
    2. <% end %>
    <% end %>


    <%= node = @types['footer_outside_above'].first and node.output %> <%= node = @types['footer_outside_below'].first and node.output %> styles: common: | body { color : #000000; background-color : #FFFFFF; line-height : 1.5em; font-family : Calibri, Verdana, sans-serif; } /* emphasis */ em, blockquote { font-family : Cambria, Georgia, serif; } /* lists */ #content li:first-child { margin-top : 1em; } #content li { margin-bottom : 1em; } /* headings */ h1, h2, h3, h4, h5, h6, .title { font-weight : lighter; font-family : Constantia, "Book Antiqua", "URW Bookman L", serif; } #lof h1, #lof h2, #lof h3, #lof h4, #lof h5, #lof h6 { margin-top : 1.25em; } #content h1, #content h2, #content h3, #content h4, #content h5, #content h6 { margin-top : 2.5em; line-height : 1.25em; } #content h1 { font-size : 2.0em; } #content h2 { font-size : 1.8em; } #content h3 { font-size : 1.6em; } #content h4 { font-size : 1.4em; } #content h5 { font-size : 1.2em; } #content h6 { font-size : 1.0em; } /* tables */ table { border : none; margin : auto; /* center horizontally */ margin-top : 1em; } th, td { padding : 1em; border : 1px solid #C0C0C0; vertical-align : top; background-color : inherit; } th { background-color : #F5F5F5; } /* document structure */ #header { margin-bottom : 5em; text-align : center; } #abstract { margin-bottom : 5em; } #toc li { list-style-type : none; } #toc li ul { padding-bottom : 1em; border-left : thick solid #F5F5F5; _border-left : none; /* for IE6 */ } #toc li ul:hover { border-color : #DCDCDC; } #toc > ul { padding-left : 1em; } #references { margin-top : 5em; } #footer { border-top : thick dotted #DCDCDC; padding-top : 1em; margin-top : 5em; text-align : center; } /* document nodes */ .part > .title, .chapter > .title { padding-bottom : 0.5em; } .part > .title > big, .chapter > .title > big { display : block; margin-top : 0.25em; } .part .title big, .chapter .title big { _display : block; /* for IE6 */ _margin-top : 0.25em; /* for IE6 */ _margin-bottom : 0.75em; /* for IE6 */ } .paragraph > .title, .tip > .title, .note > .title, .caution > .title, .warning > .title, .important > .title, .figure > .title, .table > .title, .example > .title, .equation > .title, .procedure > .title { font-size : large; margin-top : 2em; } .paragraph .title, .tip .title, .note .title, .caution .title, .warning .title, .important .title, .figure .title, .table .title, .example .title, .equation .title, .procedure .title { _font-size : large; /* for IE6 */ _font-weight : bold; /* large is not bold in IE6 */ _margin-top : 2em; /* for IE6 */ } .tip , .note , .caution , .warning , .important, .figure , .table , .example , .equation , .procedure { margin : 3em; } .tip > .icon, .note > .icon, .caution > .icon, .warning > .icon, .important > .icon { float : left; margin : 0 1em 1em 0; /* top right bottom left */ } .tip .icon, .note .icon, .caution .icon, .warning .icon, .important .icon { _display : none; /* IE6 cannot display embedded images */ } .figure > .title { text-align : center; } .figure .title { _text-align : center; /* for IE6 */ } .figure > .content img { display : block; margin : auto; } .figure .content img { _display : block; /* for IE6 */ _margin : auto; /* for IE6 */ } screen: | body { margin : auto; padding : 0.5em; max-width : 36em; } /* emphasis */ blockquote { margin : 1em; border : 5px dotted #C0C0C0; padding : 1em; color : #444; } hr { color : #FF0000; /* for IE6 */ background-color : #FF0000; /* for Firefox */ } /* source code */ tt, code, pre { font-family : Consolas, "Lucida Console", monospace; } tt { font-weight : bold; color : #A52A2A; background-color : #FFFAF0; } /* output of the syntax coloring library */ .code { background-color : #FFFFF0; } pre { cursor : text; line-height : normal; border : 1px dashed #C0C0C0; background-color : #F5FFDF; padding : 1em; overflow : auto; } /* pre:hover { border : none; position : fixed; z-index : 1; margin : 0; top : 0; left : 0; right : 0; bottom : 0; overflow : auto; cursor : text; } */ /* hyperlinks */ a > img { border : none; } a:link { color : #0000FF; text-decoration : none; } a:visited { color : #9400D3; text-decoration : none; } a:hover { color : #FF0000; text-decoration : underline; } a:target { color : #FF0000; font-weight : bold; } a.toc:link, a.toc:visited { text-decoration : none; z-index : 1; } print: | /* source code */ tt { color : inherit; background-color : inherit; font-weight : normal; } pre, .code { border : none; overflow : visible; background-color : inherit; } /* headings */ h1, h2, h3, h4, h5, h6 { font-weight : normal; } /* hyperlinks */ a:link, a:visited, a:active { color : #0000FF; font-weight : bold; text-decoration : underline; } a:after { content : " " attr(href); font-family : Consolas, "Lucida Console", monospace; font-weight : normal; font-size : 90%; } a[href^="#"]:after { content : ""; } a[href^="#"] { color : #A52A2A; font-weight : lighter; text-decoration : none; font-style : italic; } a.toc:link, a.toc:visited { color : inherit; .color : #000000; /* for IE6 and IE7 */ font-weight : inherit; text-decoration : none; font-style : normal; } /* document structure */ #lof { display : none; } /* document nodes */ .part > .title > big, .chapter > .title > big { padding-bottom : 0.5em; } .part .title big, .chapter .title big { _padding-bottom : 0.5em; /* for IE6 */ }