<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DZone Snippets: adobe code</title>
    <link>http://snippets.dzone.com/posts</link>
    <pubDate>Thu, 21 Aug 2008 04:51:21 GMT</pubDate>
    <description>DZone Snippets: adobe code</description>
    <item>
      <title>Export Adobe FDF and XFDF from ActiveRecord</title>
      <link>http://snippets.dzone.com/posts/show/3487</link>
      <description>I needed to export XFDF from an application.  This code is kind of untested, but is based on the solution I came up with.  I would appreciate responses/modifications.  It's very straightforward mixin stuff, for the most part.&lt;br /&gt;&lt;code&gt;&lt;br /&gt;# (X)FDF Export for ActiveRecord&lt;br /&gt;# Based on Justin Koivisto's FDF library for PHP&lt;br /&gt;# Author: Sean Cribbs, seancribbs_AT_gmail_DOT_com, http://seancribbs.com&lt;br /&gt;module FDF&lt;br /&gt;  def self.included(base)&lt;br /&gt;    base.extend ClassMethods&lt;br /&gt;  end&lt;br /&gt;  &lt;br /&gt;  module ClassMethods&lt;br /&gt;    # Options:&lt;br /&gt;    #  &lt;tt&gt;:filename&lt;/tt&gt; - The filename of the associated PDF document.  REQUIRED!&lt;br /&gt;    #  &lt;tt&gt;:indentation&lt;/tt&gt; - How much to indent the resulting XFDF (XML)&lt;br /&gt;    #  &lt;tt&gt;:include&lt;/tt&gt; - Which associated models to include in the generated XFDF.&lt;br /&gt;    #  &lt;tt&gt;:exclude_attributes&lt;/tt&gt; - Which attributes of the current model should not be exported.  By default all non-internal attributes are exported (i.e. everything but _id fields).&lt;br /&gt;    #  &lt;tt&gt;:include_attributes&lt;/tt&gt; - Which attributes of the current model should be exported in addition to the default. By default all non-internal attributes are exported (i.e. everything but _id fields).&lt;br /&gt;    #  &lt;tt&gt;:attributes&lt;/tt&gt; - Override which attributes to export.&lt;br /&gt;    def exports_xfdf(options = {})&lt;br /&gt;      raise ArgumentError, "A :filename option must be specified." unless options[:filename]&lt;br /&gt;      options[:indentation] ||= 2&lt;br /&gt;      options[:include] = options[:include].is_a?(Array) ? options[:include] : [options[:include]].compact&lt;br /&gt;      unless included_modules.include? XFDFMethods&lt;br /&gt;        class_inheritable_accessor :xfdf_options&lt;br /&gt;        extend ClassMethods&lt;br /&gt;        include XFDFMethods&lt;br /&gt;      end&lt;br /&gt;      self.xfdf_options = options&lt;br /&gt;    end&lt;br /&gt;    &lt;br /&gt;    # Options:&lt;br /&gt;    #  &lt;tt&gt;:filename&lt;/tt&gt; - The filename of the associated PDF document.  REQUIRED!&lt;br /&gt;    #  &lt;tt&gt;:include&lt;/tt&gt; - Which associated models to include in the generated FDF.&lt;br /&gt;    #  &lt;tt&gt;:exclude_attributes&lt;/tt&gt; - Which attributes of the current model should not be exported.  By default all non-internal attributes are exported (i.e. everything but _id fields).&lt;br /&gt;    #  &lt;tt&gt;:include_attributes&lt;/tt&gt; - Which attributes of the current model should be exported in addition to the default. By default all non-internal attributes are exported (i.e. everything but _id fields).&lt;br /&gt;    #  &lt;tt&gt;:attributes&lt;/tt&gt; - Override which attributes to export.&lt;br /&gt;    def exports_fdf(options = {})&lt;br /&gt;      raise ArgumentError, "A :filename option must be specified." unless options[:filename]&lt;br /&gt;      options[:include] = options[:include].is_a?(Array) ? options[:include] : [options[:include]].compact&lt;br /&gt;      unless included_modules.include? FDFMethods&lt;br /&gt;        class_inheritable_accessor :fdf_options&lt;br /&gt;        extend ClassMethods&lt;br /&gt;        include FDFMethods&lt;br /&gt;      end&lt;br /&gt;      self.fdf_options = options&lt;br /&gt;    end&lt;br /&gt;  end&lt;br /&gt;&lt;br /&gt;  module XFDFMethods&lt;br /&gt;    def to_xfdf(options = {})&lt;br /&gt;      options.reverse_merge! self.class.xfdf_options&lt;br /&gt;      fields = Util.collect_values(self, self.class.content_columns.map(&amp;:name), options)&lt;br /&gt;      filename = options[:filename]&lt;br /&gt;      xml = Builder::XmlMarkup.new :indentation =&gt; options[:indentation]&lt;br /&gt;      xml.instruct!&lt;br /&gt;      xml.xfdf("xmlns" =&gt; "http://ns.adobe.com/xfdf/", "xml:space" =&gt; "preserve") {&lt;br /&gt;        xml.f :href =&gt; filename&lt;br /&gt;        xml.fields {&lt;br /&gt;          fields.each do |field, value|&lt;br /&gt;            xml.field(:name =&gt; field) {&lt;br /&gt;              if value.is_a? Array&lt;br /&gt;                  value.each {|item| xml.value(item.to_s) }&lt;br /&gt;              else&lt;br /&gt;                xml.value(value.to_s)&lt;br /&gt;              end&lt;br /&gt;            }&lt;br /&gt;          end&lt;br /&gt;        }&lt;br /&gt;      }&lt;br /&gt;      xml.target!&lt;br /&gt;    end    &lt;br /&gt;  end&lt;br /&gt;  &lt;br /&gt;  module FDFMethods&lt;br /&gt;    def to_fdf(options={})&lt;br /&gt;      options.reverse_merge! self.class.fdf_options&lt;br /&gt;      fields = Util.collect_values(self, self.class.content_columns.map(&amp;:name), options)&lt;br /&gt;      filename = options[:filename]&lt;br /&gt;      data = "%FDF-1.2\n%&#226;&#227;&#65533;?&#211;\n1 0 obj\n&lt;&lt; \n/FDF &lt;&lt; /Fields [ "&lt;br /&gt;      fields.each do |field, value|&lt;br /&gt;        if value.is_a? Array&lt;br /&gt;          data &lt;&lt; "&lt;&lt;/T(#{field})/V["&lt;br /&gt;          value.each {|v| data &lt;&lt; "(#{v.strip})"}&lt;br /&gt;          data &lt;&lt; "]&gt;&gt;"&lt;br /&gt;        else&lt;br /&gt;          data &lt;&lt; "&lt;&lt;/T(#{field})/V(#{value.strip})&gt;&gt;"&lt;br /&gt;        end&lt;br /&gt;      end&lt;br /&gt;    end&lt;br /&gt;    data &lt;&lt; "] \n/F (#{filename}) /ID [ &lt;#{MD5.md5(Time.now).to_s}&gt;\n ] &gt;&gt;" &lt;&lt;&lt;br /&gt;            " \n&gt;&gt; \nendobj\ntrailer\n" &lt;&lt; "&lt;&lt;\n/Root 1 0 R \n\n&gt;&gt;\n%%EOF\n"&lt;br /&gt;  end&lt;br /&gt;  &lt;br /&gt;  module Util&lt;br /&gt;    def self.collect_values(object, defaults, options = {})&lt;br /&gt;      attrs = []&lt;br /&gt;      if options[:attributes]&lt;br /&gt;        attrs = stringify_all(options[:attributes]) rescue []&lt;br /&gt;      else        &lt;br /&gt;        [:include_attributes, :exclude_attributes].each do |opt|&lt;br /&gt;          options[opt] = stringify_all(options[opt]) rescue []&lt;br /&gt;        end&lt;br /&gt;        attrs = stringify_all(defaults) + options[:include_attributes] - options[:exclude_attributes]&lt;br /&gt;      end&lt;br /&gt;      fields = attrs.inject({}) do |hash, key|&lt;br /&gt;        value = object.send(key) rescue nil&lt;br /&gt;        hash.merge key =&gt; value&lt;br /&gt;      end&lt;br /&gt;      fields.merge collect_association_values(object, options)&lt;br /&gt;    end&lt;br /&gt;    &lt;br /&gt;    def self.collect_association_values(object, options = {})&lt;br /&gt;      return {} if options[:include].blank?&lt;br /&gt;      values = {}&lt;br /&gt;      options[:include].each do |association|&lt;br /&gt;        unless object.send(association).blank?&lt;br /&gt;          models = object.send(association)&lt;br /&gt;          unless models.is_a? Array&lt;br /&gt;            columns = models.class.content_columns.map(&amp;:name)&lt;br /&gt;            values.merge! association_dump(association, models, columns)&lt;br /&gt;          else&lt;br /&gt;            models.each_with_index do |model, index|&lt;br /&gt;              columns = model.class.content_columns.map(&amp;:name)&lt;br /&gt;              values.merge! association_dump("#{association.singularize}_#{index+1}", model, columns)&lt;br /&gt;            end&lt;br /&gt;          end&lt;br /&gt;        end&lt;br /&gt;      end&lt;br /&gt;      values&lt;br /&gt;    end&lt;br /&gt;    &lt;br /&gt;    def self.association_dump(prefix, object, attributes)&lt;br /&gt;      attributes.inject({}) do |hash, attr|&lt;br /&gt;        value = object.send(attr) rescue nil&lt;br /&gt;        hash.merge "#{prefix}_#{attr}" =&gt; value&lt;br /&gt;      end&lt;br /&gt;    end&lt;br /&gt;    &lt;br /&gt;    def self.stringify_all(ary)&lt;br /&gt;      ary.compact.map(&amp;:to_s).uniq&lt;br /&gt;    end&lt;br /&gt;  end&lt;br /&gt;end&lt;br /&gt;ActiveRecord::Base.send :include, FDF&lt;br /&gt;&lt;/code&gt;&lt;br /&gt;&lt;br /&gt;Also available from: http://pastie.caboo.se/38835</description>
      <pubDate>Sat, 10 Feb 2007 00:43:34 GMT</pubDate>
      <guid>http://snippets.dzone.com/posts/show/3487</guid>
      <author>seancribbs (Sean Cribbs)</author>
    </item>
  </channel>
</rss>
