require "asciidoctor"
require "isodoc/csd/html_convert"
require "isodoc/csd/word_convert"
require "metanorma/csd"
require "asciidoctor/standoc/converter"
require "fileutils"
module Asciidoctor
module Csd
CSD_NAMESPACE = "https://www.calconnect.org/standards/csd"
# A {Converter} implementation that generates CSD output, and a document
# schema encapsulation of the document for validation
class Converter < Standoc::Converter
register_for "csd"
def initialize(backend, opts)
super
end
def metadata_author(node, xml)
xml.contributor do |c|
c.role **{ type: "author" }
c.organization do |a|
a.name "CalConnect"
end
end
end
def metadata_publisher(node, xml)
xml.contributor do |c|
c.role **{ type: "publisher" }
c.organization do |a|
a.name "CalConnect"
end
end
end
def metadata_committee(node, xml)
xml.editorialgroup do |a|
a.technical_committee node.attr("technical-committee"),
**attr_code(type: node.attr("technical-committee-type"))
i = 2
while node.attr("technical-committee_#{i}") do
a.technical_committee node.attr("technical-committee_#{i}"),
**attr_code(type: node.attr("technical-committee-type_#{i}"))
i += 1
end
end
end
def metadata_status(node, xml)
status = node.attr("status")
unless status && ::Metanorma::Csd::DOCSTATUS.keys.include?(status)
warn "#{status} is not a legal status"
end
xml.status(**{ format: "plain" }) { |s| s << status }
end
def prefix_id(node)
prefix = "CC"
typesuffix = ::Metanorma::Csd::DOCSUFFIX[node.attr("doctype")] || ""
prefix += "/#{typesuffix}" unless typesuffix.empty?
status = ::Metanorma::Csd::DOCSTATUS[node.attr("status")] || ""
prefix += "/#{status}" unless status.empty?
prefix
end
def metadata_id(node, xml)
id = node.attr("docnumber") || "???"
prefix = prefix_id(node)
id = "#{prefix} #{id}"
year = node.attr("copyright-year")
id += ":#{year}" if year
xml.docidentifier id, **{type: "csd"}
xml.docnumber node.attr("docnumber")
end
def doctype(node)
d = node.attr("doctype")
unless ::Metanorma::Csd::DOCSUFFIX.keys.include?(d)
warn "#{d} is not a legal document type: reverting to 'standard'"
d = "standard"
end
d
end
def metadata_copyright(node, xml)
from = node.attr("copyright-year") || Date.today.year
xml.copyright do |c|
c.from from
c.owner do |owner|
owner.organization do |o|
o.name "CalConnect"
end
end
end
end
def title_validate(root)
nil
end
def makexml(node)
result = ["\n"]
@draft = node.attributes.has_key?("draft")
result << noko { |ixml| front node, ixml }
result << noko { |ixml| middle node, ixml }
result << ""
result = textcleanup(result.flatten * "\n")
ret1 = cleanup(Nokogiri::XML(result))
validate(ret1)
ret1.root.add_namespace(nil, CSD_NAMESPACE)
ret1
end
def document(node)
init(node)
ret1 = makexml(node)
ret = ret1.to_xml(indent: 2)
unless node.attr("nodoc") || !node.attr("docfile")
filename = node.attr("docfile").gsub(/\.adoc$/, ".xml").
gsub(%r{^.*/}, "")
File.open(filename, "w") { |f| f.write(ret) }
html_converter(node).convert filename
word_converter(node).convert filename
pdf_converter(node).convert filename
end
@files_to_delete.each { |f| FileUtils.rm f }
ret
end
def validate(doc)
content_validate(doc)
schema_validate(formattedstr_strip(doc.dup),
File.join(File.dirname(__FILE__), "csd.rng"))
end
def literal(node)
noko do |xml|
xml.figure **id_attr(node) do |f|
figure_title(node, f)
f.pre node.lines.join("\n")
end
end
end
def sections_cleanup(x)
super
x.xpath("//*[@inline-header]").each do |h|
h.delete("inline-header")
end
end
def style(n, t)
return
end
def html_converter(node)
IsoDoc::Csd::HtmlConvert.new(html_extract_attributes(node))
end
def pdf_converter(node)
IsoDoc::Csd::PdfConvert.new(html_extract_attributes(node))
end
def word_converter(node)
IsoDoc::Csd::WordConvert.new(doc_extract_attributes(node))
end
def inline_quoted(node)
noko do |xml|
case node.type
when :emphasis then xml.em node.text
when :strong then xml.strong node.text
when :monospaced then xml.tt node.text
when :double then xml << "\"#{node.text}\""
when :single then xml << "'#{node.text}'"
when :superscript then xml.sup node.text
when :subscript then xml.sub node.text
when :asciimath then stem_parse(node.text, xml)
else
case node.role
when "strike" then xml.strike node.text
when "smallcap" then xml.smallcap node.text
when "keyword" then xml.keyword node.text
else
xml << node.text
end
end
end.join
end
end
end
end