# encoding: utf-8 =begin * Name: SiSU * Description: a framework for document structuring, publishing and search * Author: Ralph Amissah * Copyright: (C) 1997 - 2011, Ralph Amissah, All Rights Reserved. * License: GPL 3 or later: SiSU, a framework for document structuring, publishing and search Copyright (C) Ralph Amissah This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . If you have Internet connection, the latest version of the GPL should be available at these locations: * SiSU uses: * Standard SiSU markup syntax, * Standard SiSU meta-markup syntax, and the * Standard SiSU object citation numbering and system * Hompages: * Download: * Ralph Amissah ** Description: document abstraction =end module SiSU_document_structure_extract class Instantiate < SiSU_Param::Parameters::Instructions @@flag={} #Beware!! def initialize @@flag['table_to']=false @@counter=@@column=@@columns=0 @@line_mode='' end end class Build @@flag={} #Beware!! def initialize(md,data) @md,@data=md,data Instantiate.new @pb=SiSU_document_structure::Object_layout.new.break(Hx[:br_page]) @pbn=SiSU_document_structure::Object_layout.new.break(Hx[:br_page_new]) end def ln_get(lv) ln=case lv when /A/; 1 when /B/; 2 when /C/; 3 when /1/; 4 when /2/; 5 when /3/; 6 when /4/; 7 when /5/; 8 when /6/; 9 end end def image_test(str) boolean=(str=~/\{\s*\S+?\.png.+?\}https?:\/\/\S+/ ? true : false) end def bullet_test(str) bool=((str=~/\*/) ? true : false) end def hang_and_indent_test(str) hang_indent=if str=~/^_([1-9])[^_]/ [$1,$1] elsif str=~/^__([1-9])/ [0,$1] elsif str=~/^_([0-9])_([0-9])/ [$1,$2] else [0,0] end hang,indent=hang_indent[0],hang_indent[1] [hang,indent] end def hang_and_indent_def_test(str1,str2) hang_indent=if str1=~/^_([1-9])[^_]/ [$1,$1] elsif str1=~/^__([1-9])/ [0,$1] elsif str1=~/^_([0-9])_([0-9])/ [$1,$2] else [0,0] end obj=if str2 =~/^(.+?)\s+\\\\(?:\s+|\n)/ str2.gsub(/^(.+?)(\s+\\\\(?:\s+|\n))/,"#{Mx[:fa_bold_o]}\\1#{Mx[:fa_bold_c]}\\2") else str2.gsub(/^(.+?)\n/,"#{Mx[:fa_bold_o]}\\1#{Mx[:fa_bold_c]}\n") end hang,indent=hang_indent[0],hang_indent[1] [hang,indent,obj] end def endnote_test?(str) bool=((str=~/~\{.+?\}~|~\[.+?\]~/) ? true : false) end def extract_tags(str,nametag=nil) tags=[] if str.nil? else if str =~/(?:^|[ ])\*~([a-z0-9._-]+)(?=[ #{Mx[:br_nl]}]|$)/ str.gsub!(/(^|[ ])\*~([a-z0-9._-]+)(?=[ #{Mx[:br_nl]}]|$)/i, "\\1#{Mx[:tag_o]}\\2#{Mx[:tag_c]}") str.gsub!(/ [ ]+/i,' ') tags=str.scan(/#{Mx[:tag_o]}(\S+?)#{Mx[:tag_c]}/).flatten str.gsub!(/[ ]?#{Mx[:tag_o]}\S+?#{Mx[:tag_c]}[ ]?/,' ') #may be issues with spaces would leave one, but "code" blocks? end tags=nametag ? (tags << nametag) : tags end [str,tags] end def identify_parts data=@data tuned_file=[] @tuned_block,@tuned_code=[],[] @@counter,@verse_count=0,0 @metadata={} @data.each do |t_o| t_o.gsub!(/(?:\n\s*\n)+/m,"\n") unless @@flag['code'] if t_o !~/^(?:code|poem|alt|group|block)\{|^\}(?:code|poem|alt|group|block)|^(?:table\{|\{table)[ ~]/ \ and not @@flag['code'] \ and not @@flag['poem'] \ and not @@flag['group'] \ and not @@flag['block'] \ and not @@flag['alt'] \ and not @@flag['table'] unless t_o =~/^(?:@\S+?:|%+)\s/ # extract book index for paragraph if any idx=if t_o=~/^=\{(.+)\}\s*$\Z/m; m=$1 t_o.gsub!(/\n=\{.+\}\s*$\Z/m,'') m else nil end end t_o=case t_o when /^#{Mx[:meta_o]}\S+?#{Mx[:meta_c]}/ #metadata, header if t_o=~/^#{Mx[:meta_o]}(\S+?)#{Mx[:meta_c]}\s*(.+)/m tag,obj=$1,$2 @metadata[tag]=obj end t_o=nil when /^%+\s/ #comment t_o=if t_o=~/^%+\s+(.+)/ h={obj: $1} SiSU_document_structure::Object_comment.new.comment(h) else nil end when /^:?([A-C1-6])\~/ #heading / lv lv=$1 ln=ln_get(lv) t_o=if t_o=~/^:?[A-C1-6]\~\s+(.+)/m obj=$1 note=endnote_test?(obj) obj,tags=extract_tags(obj) h={ lv: lv, ln: ln, obj: obj, idx: idx, tags: tags } SiSU_document_structure::Object_heading.new.heading(h) elsif t_o=~/^:?[A-C1-6]\~(\S+?)-\s+(.+)/m name,obj=$1,$2 note=endnote_test?(obj) obj,tags=extract_tags(obj) h={ lv: lv, name: name, obj: obj, idx: idx, autonum_: false, tags: tags} SiSU_document_structure::Object_heading.new.heading(h) elsif t_o=~/^:?[A-C1-6]\~(\S+)\s+(.+)/m name,obj=$1,$2 note=endnote_test?(obj) obj,tags=extract_tags(obj,name) h={ lv: lv, name: name, obj: obj, idx: idx, tags: tags } SiSU_document_structure::Object_heading.new.heading(h) else nil end when /^_(?:[1-9]!?|[1-9]?\*)\s+/ #indented and/or bullet paragraph t_o=if t_o=~/^(_(?:[1-9]?\*|[1-9]!?)\s+)(.+)/m tst,obj=$1,$2 if t_o=~/^_[1-9]!\s+.+/m hang,indent,obj=hang_and_indent_def_test(tst,obj) else hang,indent=hang_and_indent_test(tst) end bullet=bullet_test(tst) image=image_test(obj) note=endnote_test?(obj) obj,tags=extract_tags(obj) unless obj=~/\A\s*\Z/m h={ bullet_: bullet, hang: hang, indent: indent, obj: obj, idx: idx, note_: note, image_: image, tags: tags } SiSU_document_structure::Object_para.new.paragraph(h) end else nil end when /^_[0-9]?_[0-9]!?\s+/ #hanging indent paragraph t_o=if t_o=~/^(_[0-9]?_[0-9]!?\s+)(.+)/m tst,obj=$1,$2 if t_o=~/^_[0-9]?_[0-9]!\s+.+/m hang,indent,obj=hang_and_indent_def_test(tst,obj) else hang,indent=hang_and_indent_test(tst) end image=image_test(obj) note=endnote_test?(obj) obj,tags=extract_tags(obj) unless obj=~/\A\s*\Z/m h={ hang: hang, indent: indent, obj: obj, idx: idx, note_: note, image_: image, tags: tags } SiSU_document_structure::Object_para.new.paragraph(h) end else nil end when /^[<\[](?:br)?:(?:pa?r|o(?:bj|---)?)[>\]]\s*$/ #[br:par] #[br:obj] SiSU_document_structure::Object_layout.new.break(Hx[:br_obj]) when /^(?:[<\[](?:br)?:pg[>\]]|?)\s*$/ #[br:pg] SiSU_document_structure::Object_layout.new.break(Hx[:br_page]) when /^[<\[](?:br)?:pg?n[>\]]\s*$/ #[br:pgn] SiSU_document_structure::Object_layout.new.break(Hx[:br_page_new]) else #paragraph image=image_test(t_o) note=endnote_test?(t_o) obj,tags=extract_tags(t_o) unless obj=~/\A\s*\Z/m h={ bullet_: false, indent: 0, hang: 0, obj: obj, idx: idx, note_: note, image_: image, tags: tags } SiSU_document_structure::Object_para.new.paragraph(h) end end elsif not @@flag['code'] if t_o =~/^code\{/ @@flag['code']=true @@counter=1 @codeblock_numbered=(t_o =~/^code\{#/) ? true : false h={ obj: 'code block start' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) elsif t_o =~/^poem\{/ @@flag['poem']=true h={ obj: 'poem start' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) tuned_file << t_o elsif t_o =~/^group\{/ @@flag['group']=true h={ obj: 'group text start' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) tuned_file << t_o elsif t_o =~/^block\{/ @@flag['block']=true h={ obj: 'block text start' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) tuned_file << t_o elsif t_o =~/^alt\{/ @@flag['alt']=true h={ obj: 'alt text start' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) tuned_file << t_o elsif t_o =~/^(?:table\{|\{table)[ ~]/ h={ obj: 'table start' } #introduce a counter ins=SiSU_document_structure::Object_comment.new.comment(h) #ins=SiSU_document_structure::Object_layout.new.insert(h) tuned_file << ins if t_o=~/^table\{(?:~h)?\s+/ @@flag['table']=true @rows='' case t_o when /table\{~h\s+c(\d+);\s+(.+)/ cols=$1 col=$2.scan(/\d+/) heading=true when /table\{\s+c(\d+);\s+(.+)/ cols=$1 col=$2.scan(/\d+/) heading=false end @h={ head_: heading, cols: cols, widths: col, idx: idx } elsif t_o=~/^\{table(?:~h)?(?:\s+\d+;?)?\}\n.+\Z/m m1,m2,hd=nil,nil,nil tbl=/^\{table(?:~h)?(?:\s+\d+;?)?\}\n(.+)\Z/m.match(t_o)[1] #two table representations should be consolidated as one hd=((t_o =~/^\{table~h/) ? true : false) tbl,tags=extract_tags(tbl) rws=tbl.split(/\n/) rows='' cols=nil rws.each do |r| cols=(cols ? cols : (r.scan('|').length) +1) r.gsub!(/\s*\|\s*/m,"#{Mx[:tc_p]}") #r.gsub!(/\|/m,"#{Mx[:tc_p]}") rows += r + Mx[:tc_c] end col=[] if t_o =~/^\{table(?:~h)?\s+(\d+);?\}/ #width of col 1 given as %, usually when wider than rest that are even c1=$1.to_i width=(100 - c1)/(cols - 1) col=[ c1 ] (cols - 1).times { col << width } else #all columns of equal width width=100.00/cols cols.times { col << width } end h={ head_: hd, cols: cols, widths: col, obj: rows, idx: idx, tags: tags } t_o=SiSU_document_structure::Object_table.new.table(h) unless h.nil? tuned_file << t_o h={ obj: 'table end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) t_o elsif t_o=~/^\{table(?:~h)?\s+/ m1,m2,hd=nil,nil,nil h=case t_o when /\{table~h\s+(.+?)\}\n(.+)\Z/m #two table representations should be consolidated as one m1,tbl,hd=$1,$2,true when /\{table\s+(.+?)\}\n(.+)\Z/m #two table representations should be consolidated as one m1,tbl,hd=$1,$2,false else nil end tbl,tags=extract_tags(tbl) col=m1.scan(/\d+/) rws=tbl.split(/\n/) rows='' rws.each do |r| r.gsub!(/\s*\|\s*/m,"#{Mx[:tc_p]}") #r.gsub!(/\|/m,"#{Mx[:tc_p]}") rows += r + Mx[:tc_c] end h={ head_: hd, cols: col.length, widths: col, obj: rows, idx: idx, tags: tags } t_o=SiSU_document_structure::Object_table.new.table(h) unless h.nil? tuned_file << t_o h={ obj: 'table end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) t_o end end t_o end if @@flag['table'] if @@flag['table'] \ and t_o =~/^\}table/ #two table representations should be consolidated as one @@flag['table']=false headings,columns,widths,idx=@h[:head_],@h[:cols],@h[:widths],@h[:idx] @h={ head_: headings, cols: columns, widths: widths, idx: idx, obj: @rows } t_o=SiSU_document_structure::Object_table.new.table(@h) tuned_file << t_o @h,@rows=nil,'' t_o h={ obj: 'table end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) t_o else if t_o !~/^table\{/ \ and not t_o.nil? t_o.gsub!(/^\n+/m,'') #check added for ruby 1.9.2 not needed in 1.8 series (tested in v2) t_o.gsub!(/\n+/m,"#{Mx[:tc_p]}") @rows += t_o + Mx[:tc_c] end t_o=nil end end if @@flag['code'] if t_o =~/^\}code/ @@flag['code']=false obj,tags=extract_tags(@tuned_code.join("\n")) h={ obj: obj, tags: tags, number_: @codeblock_numbered } t_o=SiSU_document_structure::Object_block_txt.new.code(h) @tuned_code=[] tuned_file << t_o h={ obj: 'code block end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) end if @@flag['code'] \ and t_o.class==String \ and not t_o.nil? #you may need to introduce t_o.class==String test more widely sub_array=t_o.dup + "#{Mx[:br_nl]}" @line_mode=sub_array.scan(/.+/) @line_mode=[] sub_array.scan(/.+/) {|w| @line_mode << w if w =~/[\S]+/} t_o=SiSU_document_structure_extract::Build.new(@md,@line_mode).build_lines('code').join @tuned_code << t_o t_o=nil end elsif @@flag['poem'] \ or @@flag['group'] \ or @@flag['block'] \ or @@flag['alt'] if @@flag['poem'] \ and t_o =~/^\}poem/ @@flag['poem']=false h={ obj: 'poem end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) elsif ( @@flag['group'] \ and t_o =~/^\}group/ ) @@flag['group']=false obj,tags=extract_tags(@tuned_block.join("\n")) h={ obj: obj, tags: tags } @tuned_block=[] t_o=SiSU_document_structure::Object_block_txt.new.group(h) tuned_file << t_o h={ obj: 'group text end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) elsif ( @@flag['block'] \ and t_o =~/^\}block/ ) @@flag['block']=false obj,tags=extract_tags(@tuned_block.join("\n")) h={ obj: obj, tags: tags } @tuned_block=[] t_o=SiSU_document_structure::Object_block_txt.new.block(h) tuned_file << t_o h={ obj: 'block text end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) elsif ( @@flag['alt'] \ and t_o =~/^\}alt/ ) @@flag['alt']=false obj,tags=extract_tags(@tuned_block.join("\n")) h={ obj: obj, tags: tags } t_o=SiSU_document_structure::Object_block_txt.new.alt(h) @tuned_block=[] tuned_file << t_o h={ obj: 'alt text end' } #introduce a counter t_o=SiSU_document_structure::Object_comment.new.comment(h) #t_o=SiSU_document_structure::Object_layout.new.insert(h) end if @@flag['poem'] \ or @@flag['group'] \ or @@flag['alt'] \ and t_o =~/\S/ \ and t_o !~/^(?:\}(?:verse|code|alt|group|block)|(?:verse|code|alt|group|block)\{)/ # fix logic sub_array=t_o.dup @line_mode=sub_array.scan(/.+/) type=if @@flag['poem']; 'poem' t_o=SiSU_document_structure_extract::Build.new(@md,@line_mode).build_lines(type).join poem=t_o.split(/\n\n/) poem.each do |v| v.gsub!(/\n/m,"#{Mx[:br_nl]}\n") obj,tags=extract_tags(v) h={ obj: obj, tags: tags } t_o=SiSU_document_structure::Object_block_txt.new.verse(h) tuned_file << t_o end else 'group' end @verse_count+=1 if @@flag['poem'] end end if not @@flag['code'] if @@flag['poem'] \ or @@flag['group'] \ or @@flag['alt'] if t_o.class==String t_o.gsub!(/\n/m,"#{Mx[:br_nl]}") t_o.gsub!(/[ ][ ]/m,"#{Mx[:nbsp]*2}") t_o.gsub!(/#{Mx[:nbsp]}\s/,"#{Mx[:nbsp]*2}") t_o=t_o + Mx[:br_nl] if t_o =~/\S+/ elsif t_o.is=='group' \ or t_o.is=='block' \ or t_o.is=='alt' \ or t_o.is=='verse' t_o.obj.gsub!(/\n/m,"#{Mx[:br_nl]}") t_o.obj.gsub!(/[ ][ ]/m,"#{Mx[:nbsp]*2}") t_o.obj.gsub!(/#{Mx[:nbsp]}\s/,"#{Mx[:nbsp]*2}") end @tuned_block << t_o if t_o =~/\S+/ else tuned_file << t_o end else tuned_file << t_o end end if @md.flag_endnotes tuned_file << @pb h={ ln: 2, obj: 'Endnotes', autonum_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ ln: 4, obj: 'Endnotes', name: 'endnotes', autonum_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ obj: 'Endnotes' } end if @md.book_idx tuned_file << @pb h={ ln: 2, obj: 'Index', autonum_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ ln: 4, obj: 'Index', name: 'book_index', autonum_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ obj: 'Index' } end tuned_file << @pb h={ ln: 2, obj: 'Metadata', autonum_: false, ocn_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ ln: 4, obj: 'SiSU Metadata, document information', name: 'metadata', autonum_: false, ocn_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) tuned_file << @pb h={ ln: 2, obj: 'Manifest', autonum_: false, ocn_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) h={ ln: 4, obj: 'SiSU Manifest, alternative outputs etc.', name: 'sisu_manifest', autonum_: false, ocn_: false } tuned_file << SiSU_document_structure::Object_heading.new.heading_insert(h) tuned_file h={ obj: 'eof' } meta=SiSU_document_structure::Object_metadata.new.metadata(@metadata) [tuned_file,meta] end def table_rows_and_columns_array(table_str) table=[] table_str.split(/#{Mx[:tc_c]}/).each do |table_row| table_row_with_columns=table_row.split(/#{Mx[:tc_p]}/) table << table_row_with_columns end table end def meta_heading(h) h={ lv: h[:lv], ln: h[:ln], name: h[:name], obj: h[:obj], ocn: '0' } SiSU_document_structure::Object_heading.new.heading(h) end def meta_para(str) h={ obj: str, ocn_: false } SiSU_document_structure::Object_para.new.paragraph(h) end def metadata meta=[] dir=SiSU_Env::Info_env.new(@md.fns) base_html="#{dir.url.root}/#{@md.fnb}" l=SiSU_Env::Standardise_language.new(@md.opt.lng).language language=l[:n] tr=SiSU_Translate::Source.new(@md,language) meta << @pb h={ ln: 2, obj: 'Metadata', ocn_: false } meta << SiSU_document_structure::Object_heading.new.heading(h) h={ ln: 4, name: 'metadata', obj: 'Metadata', autonum_: false, ocn_: false } meta << SiSU_document_structure::Object_heading.new.heading(h) #add ocnm s="Document Manifest @\n #{base_html}/#{@md.fn[:manifest]}" meta << meta_para(s) s="#{Mx[:fa_bold_o]}Dublin Core#{Mx[:fa_bold_c]} (DC)" #add ocnm meta << meta_para(s) s="#{Mx[:fa_italics_o]}DC tags included with this document are provided here.#{Mx[:fa_italics_c]}" #add ocnm meta << meta_para(s) if defined? @md.title.full \ and @md.title.full=~/\S+/ s="#{tr.full_title}: #{Mx[:fa_underscore_o]}#{@md.title.full}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.author \ and @md.creator.author=~/\S+/ s="\n#{tr.author}: #{Mx[:fa_underscore_o]}#{@md.creator.author}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.translator \ and @md.creator.translator=~/\S+/ s="#{tr.translator}: #{Mx[:fa_underscore_o]}#{@md.creator.translator}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.illustrator \ and @md.creator.illustrator=~/\S+/ s="#{tr.illustrator}: #{Mx[:fa_underscore_o]}#{@md.creator.illustrator}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.prepared_by \ and @md.creator.prepared_by=~/\S+/ s="\n#{tr.prepared_by}: #{Mx[:fa_underscore_o]}#{@md.creator.prepared_by}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.digitized_by \ and @md.creator.digitized_by=~/\S+/ s="#{tr.digitized_by}: #{Mx[:fa_underscore_o]}#{@md.creator.digitized_by}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.rights.all \ and @md.rights.all=~/\S+/ s="\n#{tr.rights}: #{Mx[:fa_underscore_o]}#{@md.rights.all}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.notes.description \ and @md.notes.description=~/\S+/ s="#{tr.description}: #{Mx[:fa_underscore_o]}#{@md.notes.description}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.subject \ and @md.classify.subject=~/\S+/ s="#{tr.subject}: #{Mx[:fa_underscore_o]}#{@md.classify.subject}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.publisher \ and @md.publisher=~/\S+/ s="\n#{tr.publisher}: #{Mx[:fa_underscore_o]}#{@md.publisher}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.creator.contributor \ and @md.creator.contributor=~/\S+/ s="\n#{tr.contributor}: #{Mx[:fa_underscore_o]}#{@md.creator.contributor}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.notes.abstract \ and @md.notes.abstract=~/\S+/ s="\n#{tr.abstract}: #{Mx[:fa_underscore_o]}#{@md.notes.abstract}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.created \ and @md.date.created=~/\S+/ s="\n#{tr.date_created}: #{Mx[:fa_underscore_o]}#{@md.date.created}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.issued \ and @md.date.issued=~/\S+/ s="\n#{tr.date_issued}: #{Mx[:fa_underscore_o]}#{@md.date.issued}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.available \ and @md.date.available=~/\S+/ s="\n#{tr.date_available}: #{Mx[:fa_underscore_o]}#{@md.date.available}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.modified \ and @md.date.modified=~/\S+/ s="\n#{tr.date_modified}: #{Mx[:fa_underscore_o]}#{@md.date.modified}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.valid \ and @md.date.valid=~/\S+/ s="\n#{tr.date_valid}: #{Mx[:fa_underscore_o]}#{@md.date.valid}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.date.published \ and @md.date.published=~/\S+/ s="\n#{tr.date}: #{Mx[:fa_underscore_o]}#{@md.date.published}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.loc \ and @md.classify.loc=~/\S+/ s="\n#{tr.cls_loc}: #{Mx[:fa_underscore_o]}#{@md.classify.loc}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.dewey \ and @md.classify.dewey=~/\S+/ s="\n#{@cls_dewey}: #{Mx[:fa_underscore_o]}#{@md.classify.dewey}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.pg \ and @md.classify.pg=~/\S+/ s="\n#{tr.cls_gutenberg}: #{Mx[:fa_underscore_o]}#{@md.classify.pg}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.isbn \ and @md.classify.isbn=~/\S+/ s="\n#{tr.cls_isbn}: #{Mx[:fa_underscore_o]}#{@md.classify.isbn}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.notes.comment \ and @md.notes.comment=~/\S+/ s="\n#{tr.comments}: #{Mx[:fa_underscore_o]}#{@md.notes.comment}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.notes.prefix_a \ and @md.notes.prefix_a=~/\S+/ s="\n#{tr.prefix_a}: #{Mx[:fa_underscore_o]}#{@md.notes.prefix_a}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.notes.prefix_b \ and @md.notes.prefix_b=~/\S+/ s="\n#{tr.prefix_b}: #{Mx[:fa_underscore_o]}#{@md.notes.prefix_b}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.identifier \ and @md.classify.identifier=~/\S+/ s="\n#{tr.identifier}: #{Mx[:fa_underscore_o]}#{@md.classify.identifier}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.original.source \ and @md.original.source=~/\S+/ s="\n#{tr.source}: #{Mx[:fa_underscore_o]}#{@md.original.source}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.title.language \ and @md.title.language=~/\S+/ s="\n#{tr.language}: #{Mx[:fa_underscore_o]}#{@md.title.language}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.original.language \ and @md.original.language=~/\S+/ s="\n#{tr.language_original}: #{Mx[:fa_underscore_o]}#{@md.original.language}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.format \ and @md.classify.format=~/\S+/ s="\n#{tr.format}: #{Mx[:fa_underscore_o]}#{@md.classify.format}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.relation \ and @md.classify.relation=~/\S+/ s="\n#{tr.relation}: #{Mx[:fa_underscore_o]}#{@md.classify.relation}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.coverage \ and @md.classify.coverage=~/\S+/ s="\n#{tr.coverage}: #{Mx[:fa_underscore_o]}#{@md.classify.coverage}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.classify.keywords \ and @md.classify.keywords=~/\S+/ s="\n#{tr.keywords}: #{Mx[:fa_underscore_o]}#{@md.classify.keywords}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end s="#{Mx[:fa_bold_o]}Version Information#{Mx[:fa_bold_c]}" meta << meta_para(s) if defined? @md.fns \ and @md.fns=~/\S+/ s="#{tr.sourcefile}: #{Mx[:fa_underscore_o]}#{@md.fns}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.file_encoding \ and @md.file_encoding=~/\S+/ s="Filetype: #{Mx[:fa_underscore_o]}#{@md.file_encoding}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.dgst \ and @md.dgst.class==Array s="Source Digest: #{@md.dgst[0]} #{Mx[:fa_underscore_o]}#{@md.dgst[1]}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end if defined? @md.dgst_skin \ and @md.dgst_skin.class==Array s="Skin Digest: #{@md.dgst_skin[0]} #{Mx[:fa_underscore_o]}#{@md.dgst_skin[1]}#{Mx[:fa_underscore_c]}" meta << meta_para(s) end s="#{Mx[:fa_bold_o]}Generated#{Mx[:fa_bold_c]}" meta << meta_para(s) s="#{tr.last_generated}: #{Mx[:fa_underscore_o]}#{Time.now}#{Mx[:fa_underscore_c]}" meta << meta_para(s) s="#{tr.sisu_version}: #{Mx[:fa_underscore_o]}#{@md.sisu_version[:project]}#{Mx[:fa_underscore_c]} #{Mx[:fa_underscore_o]}#{@md.sisu_version[:version]}#{Mx[:fa_underscore_c]} of #{@md.sisu_version[:date_stamp]} (#{@md.sisu_version[:date]})" meta << meta_para(s) meta end def build_lines(type='') data=@data data.each do |line| if line =~/\S/ \ and line !~/^code\{|^\}code/ \ and line.class != Hash line.gsub!(/\s\s/,"#{Mx[:nbsp]*2}") line.gsub!(/#{Mx[:nbsp]}\s/,"#{Mx[:nbsp]*2}") line.gsub!(/^/,"#{Mx[:gr_o]}codeline#{Mx[:gr_c]}") if type=='code' # REMOVE try sort for texpdf special case if line =~/(?:https?|file|ftp):\/\/\S+$/ line.gsub!(/\s*$/," #{Mx[:br_nl]}") else line.gsub!(/\s*$/,"#{Mx[:br_nl]}") #unless type=='code' end if @@flag['code']; @@counter+=1 else end elsif line =~/^\s*$/ line.gsub!(/\s*$/,"#{Mx[:br_nl]}") end end data end end class Structure # this must happen early def initialize(md,dob) @md,@dob=md,dob end def structure structure_markup_normalize structure_markup @dob end def structure_markup #build structure where structure provided only in meta header @dob=if @dob.is =~/para/ \ and ((@dob.hang !~/[1-9]/ and @dob.indent !~/[1-9]/) \ or (@dob.hang != @dob.indent)) \ and not @dob.bullet_ @dob=case @dob.obj when /^#{@md.lv1}/ h={ lv: 'A', ln: 1 } SiSU_document_structure::Object_heading.new.heading(h,@dob) when /^#{@md.lv2}/ h={ lv: 'B', ln: 2 } SiSU_document_structure::Object_heading.new.heading(h,@dob) when /^#{@md.lv3}/ h={ lv: 'C', ln: 3 } SiSU_document_structure::Object_heading.new.heading(h,@dob) when /^#{@md.lv4}/ h={ lv: '1', ln: 4 } SiSU_document_structure::Object_heading.new.heading(h,@dob) when /^#{@md.lv5}/ h={ lv: '2', ln: 5 } SiSU_document_structure::Object_heading.new.heading(h,@dob) when /^#{@md.lv6}/ h={ lv: '3', ln: 6 } SiSU_document_structure::Object_heading.new.heading(h,@dob) else @dob end else @dob end @dob end def structure_markup_normalize #needs a bit of thinking dob=if @md.markup_version.determined < 0.38 #%convert internal representation, consider making 0.38 structure default ([A-C1-6] instead of [1-9]), requires downstream changes @dob.gsub!(/^[456]~/,'!_') @dob.gsub!(/^3~(\S+)/,"#{Mx[:lv_o]}6:\\1#{Mx[:lv_c]}") @dob.gsub!(/^3~\s+/,"#{Mx[:lv_o]}6:#{Mx[:lv_c]}") @dob.gsub!(/^2~(\S+)/,"#{Mx[:lv_o]}5:\\1#{Mx[:lv_c]}") @dob.gsub!(/^2~\s+/,"#{Mx[:lv_o]}5:#{Mx[:lv_c]}") @dob.gsub!(/^1~(\S+)/,"#{Mx[:lv_o]}4:\\1#{Mx[:lv_c]}") @dob.gsub!(/^1~\s+/,"#{Mx[:lv_o]}4:#{Mx[:lv_c]}") @dob.gsub!(/^:?C~(\S+)/,"#{Mx[:lv_o]}3:\\1#{Mx[:lv_c]}") @dob.gsub!(/^:?C~\s+/,"#{Mx[:lv_o]}3:#{Mx[:lv_c]}") @dob.gsub!(/^:?B~(\S+)/,"#{Mx[:lv_o]}2:\\1#{Mx[:lv_c]}") @dob.gsub!(/^:?B~\s+/,"#{Mx[:lv_o]}2:#{Mx[:lv_c]}") @dob.gsub!(/^:?A~(\S+)/,"#{Mx[:lv_o]}1:\\1#{Mx[:lv_c]}") @dob.gsub!(/^:?A~\s+/,"#{Mx[:lv_o]}1:#{Mx[:lv_c]}") @dob=if @dob =~/^@(?:level|markup):\s/ @dob.gsub!(/3/,'6') @dob.gsub!(/2/,'5') @dob.gsub!(/1/,'4') @dob.gsub!(/:?C/,'3') @dob.gsub!(/:?B/,'2') @dob.gsub!(/:?A/,'1') @dob else @dob end else @dob end end def structure_marks t_o=if @md.markup_version.determined < 0.38 @t_o.gsub!(/^1~(\S+)/,"#{Mx[:lv_o]}1:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^1~\s+/,"#{Mx[:lv_o]}1:#{Mx[:lv_c]}") @t_o.gsub!(/^2~(\S+)/,"#{Mx[:lv_o]}2:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^2~\s+/,"#{Mx[:lv_o]}2:#{Mx[:lv_c]}") @t_o.gsub!(/^3~(\S+)/,"#{Mx[:lv_o]}3:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^3~\s+/,"#{Mx[:lv_o]}3:#{Mx[:lv_c]}") @t_o.gsub!(/^4~(\S+)/,"#{Mx[:lv_o]}4:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^4~\s+/,"#{Mx[:lv_o]}4:#{Mx[:lv_c]}") @t_o.gsub!(/^5~(\S+)/,"#{Mx[:lv_o]}5:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^5~\s+/,"#{Mx[:lv_o]}5:#{Mx[:lv_c]}") @t_o.gsub!(/^6~(\S+)/,"#{Mx[:lv_o]}6:\\1#{Mx[:lv_c]}") @t_o.gsub!(/^6~\s+/,"#{Mx[:lv_o]}6:#{Mx[:lv_c]}") @t_o.gsub!(/^[789]~/,'!_') @t_o else @t_o end end end class OCN def initialize(md,data) @md,@data=md,data end def ocn #and auto segment numbering increment data=@data @o_array=[] node=ocn=ocn_dv=ocn_sp=ocnh=ocnh1=ocnh2=ocnh3=ocnh4=ocnh5=ocnh6=ocno=ocnp=ocnt=ocnc=ocng=ocni=ocnm=ocnu=ocnk=nm=0 # h heading, o other, t table, g group, i image node_count_flag=false regex_exclude_ocn_and_node = /#{Rx[:meta]}|^@\S+?:\s|^4~endnotes|^#{Mx[:lv_o]}4:endnotes#{Mx[:lv_c]}|^\^~ |<:e[:_]\d+?>|^<:\#|<:- |<[:!]!4|
if dob.is=='heading' ln=case dob.lv when 'A'; 1 when 'B'; 2 when 'C'; 3 when '1'; 4 when '2'; 5 when '3'; 6 when '4'; 7 when '5'; 8 when '6'; 9 end end if not dob.obj =~/<:#>|~#|-#/ \ or not dob.toc_ # fix this no longer in dob.obj ocn+=1 if dob.is=='heading' \ and (ln.to_s =~/^[1-9]/ \ or ln.to_s =~@md.lv1 \ or ln.to_s =~@md.lv2 \ or ln.to_s =~@md.lv3 \ or ln.to_s =~@md.lv4 \ or ln.to_s =~@md.lv5 \ or ln.to_s =~@md.lv6) ocnh+=1 if ln==1 \ or ln=~@md.lv1; ocnh1+=1 #heading node1="1:#{ocnh1};#{ocn}" node,ocn_sp,parent=node1,"h#{ocnh}",0 #FIX elsif ln==2 \ or ln=~@md.lv2; ocnh2+=1 node2="2:#{ocnh2};#{ocn}" node,ocn_sp,parent=node2,"h#{ocnh}",node1 elsif ln==3 \ or ln=~@md.lv3; ocnh3+=1 node3="3:#{ocnh3};#{ocn}" node,ocn_sp,parent=node3,"h#{ocnh}",node2 elsif ln==4 \ or ln=~@md.lv4; ocnh4+=1 node4="4:#{ocnh4};#{ocn}" node,ocn_sp,parent=node4,"h#{ocnh}",node3 elsif ln==5 \ or ln=~@md.lv5; ocnh5+=1 node5="5:#{ocnh5};#{ocn}" node,ocn_sp,parent=node5,"h#{ocnh}",node4 elsif ln==6 \ or ln=~@md.lv6; ocnh6+=1 node6="6:#{ocnh6};#{ocn}" node,ocn_sp,parent=node6,"h#{ocnh}",node5 end else ocno+=1 if dob.is=='table' ocnt+=1 ocn_sp,parent="t#{ocnt}",node elsif dob.is=='code' ocnc+=1 ocn_sp,parent="c#{ocnc}",node elsif dob.is=~/^(?:group|block|alt|verse)/ ocng+=1 #group, poem ocn_sp,parent="g#{ocng}",node elsif dob.is=~/image|#{Mx[:lnk_o]}\S+?\.(?:png|jpg|gif)\s+/m ocni+=1 ocn_sp,parent="i#{ocni}",node else ocnp+=1 #paragraph ocn_sp,parent="p#{ocnp}",node end end if dob.is=='heading' dob.ln,dob.node,dob.ocn,dob.odv,dob.osp,dob.parent=ln,node,ocn,ocn_dv,ocn_sp,parent else unless dob.of=~/meta|comment|layout/ dob.ocn,dob.odv,dob.osp,dob.parent=ocn,ocn_dv,ocn_sp,parent end end else ocnu+=1 dob.obj.gsub!(/#{Mx[:fa_o]}~##{Mx[:fa_c]}/,'') if dob.obj ocn_dv,ocn_sp="u#{ocnu}","u#{ocnu}" dob.ocn,dob.odv,dob.osp=ocn,ocn_dv,ocn_sp end h elsif dob.obj=~/#{Mx[:pa_non_object_no_heading]}/ dob.obj.gsub!(/#{Mx[:pa_non_object_no_heading]}/,'') if dob.is=='para' h={ obj: dob.obj, ocn_: false, ocn: nil } dob=SiSU_document_structure::Object_para.new.paragraph(h,dob) elsif dob.is=='heading' h={ obj: dob.obj, ocn_: false, ocn: nil, toc_: true } dob=SiSU_document_structure::Object_heading.new.heading(h,dob) end elsif dob.obj=~/#{Mx[:pa_non_object_dummy_heading]}/ dob.obj.gsub!(/#{Mx[:pa_non_object_dummy_heading]}/,'') if dob.is=='para' h={ obj: dob.obj, ocn_: false, ocn: nil } dob=SiSU_document_structure::Object_para.new.paragraph(h,dob) elsif dob.is=='heading' h={ obj: dob.obj, ocn_: false, ocn: nil, toc_: false } dob=SiSU_document_structure::Object_heading.new.heading(h,dob) end else dob end dob.obj.gsub!(/\n\n/,"\n") if dob.is =~/(?:code|verse|alt|group|block)/ #newlines taken out @o_array << dob end @o_array end end class XML def initialize(md,data) @data,@md=data,md end def dom @s=['0', 'A', 'B', 'C', '1', '2', '3' ] @sp=' ' tuned_file=structure_build tuned_file end def structure_build data=@data tuned_file=[] hs=[0,false,false,false] t={ lv: @s[0], status: 'open' } tuned_file << tags(t) if @md.opt.cmd =~/V/ puts "\nXML sisu structure outline --->\n" puts "<#{@s[0]}>" end data.each_with_index do |o,i| if o.is =~/^heading/ case o.ln when 1 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs=[1,true,false,false] when 2 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs=[2,true,true,false] when 3 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs=[3,true,true,true] when 4 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs[0]=4 when 5 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs[0]=5 when 6 tuned_file << tag_close(o.ln,hs) tuned_file << tag_open(o,@s) if @md.opt.cmd =~/V/ puts_tag_close(o.ln,hs) puts_tag_open(o,@s) end hs[0]=6 end end tuned_file << o end puts_tag_close(0,hs) if @md.opt.cmd =~/V/ tuned_file << tag_close(0,hs) tuned_file.flatten! tuned_file end def tags(o) tag=if o[:status]=='open' %{<#{o[:lv]} id="#{o[:node]}">} else "" end ln=case o[:lv] when 'A'; 1 when 'B'; 2 when 'C'; 3 when '1'; 4 when '2'; 5 when '3'; 6 when '4'; 7 when '5'; 8 when '6'; 9 end h={ tag: tag, node: o[:node], lv: o[:lv], ln: ln, status: o[:status] } SiSU_document_structure::Object_structure.new.xml_dom(h) #downstream code utilise else ignore like comments end def tag_open(o,tag) t={ lv: tag[o.ln], node: o.node, status: 'open' } t_o=tags(t) t_o end def tag_close(lev,hs) ary=[] case hs[0] when 1 if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end when 2 if (lev <= 2) and hs[2] t={ lv: @s[2], status: 'close' } ary << tags(t) end if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end when 3 if (lev <= 3) and hs[3] t={ lv: @s[3], status: 'close' } ary << tags(t) end if (lev <= 2) and hs[2] t={ lv: @s[2], status: 'close' } ary << tags(t) end if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end when 4 if (lev <= 4) t={ lv: @s[4], status: 'close' } ary << tags(t) end if (lev <= 3) and hs[3] t={ lv: @s[3], status: 'close' } ary << tags(t) end if (lev <= 2) and hs[2] t={ lv: @s[2], status: 'close' } ary << tags(t) end if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end when 5 if (lev <= 5) t={ lv: @s[5], status: 'close' } ary << tags(t) end if (lev <= 4) t={ lv: @s[4], status: 'close' } ary << tags(t) end if (lev <= 3) and hs[3] t={ lv: @s[3], status: 'close' } ary << tags(t) end if (lev <= 2) and hs[2] t={ lv: @s[2], status: 'close' } ary << tags(t) end if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end when 6 if (lev <= 6) t={ lv: @s[6], status: 'close' } ary << tags(t) end if (lev <= 5) t={ lv: @s[5], status: 'close' } ary << tags(t) end if (lev <= 4) t={ lv: @s[4], status: 'close' } ary << tags(t) end if (lev <= 3) and hs[3] t={ lv: @s[3], status: 'close' } ary << tags(t) end if (lev <= 2) and hs[2] t={ lv: @s[2], status: 'close' } ary << tags(t) end if (lev <= 1) and hs[1] t={ lv: @s[1], status: 'close' } ary << tags(t) end if (lev==0) t={ lv: @s[0], status: 'close' } ary << tags(t) end end ary end def puts_tag_open(o,tag) puts %{#{@sp*o.ln}<#{tag[o.ln]} id="#{o.node}">} end def puts_tag_close(lev,hs) case hs[0] when 1 puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) when 2 puts "#{@sp*2}" if (lev <= 2) and hs[2] puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) when 3 puts "#{@sp*3}" if (lev <= 3) and hs[3] puts "#{@sp*2}" if (lev <= 2) and hs[2] puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) when 4 puts "#{@sp*4}" if (lev <= 4) puts "#{@sp*3}" if (lev <= 3) and hs[3] puts "#{@sp*2}" if (lev <= 2) and hs[2] puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) when 5 puts "#{@sp*5}" if (lev <= 5) puts "#{@sp*4}" if (lev <= 4) puts "#{@sp*3}" if (lev <= 3) and hs[3] puts "#{@sp*2}" if (lev <= 2) and hs[2] puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) when 6 puts "#{@sp*6}" if (lev <= 6) puts "#{@sp*5}" if (lev <= 5) puts "#{@sp*4}" if (lev <= 4) puts "#{@sp*3}" if (lev <= 3) and hs[3] puts "#{@sp*2}" if (lev <= 2) and hs[2] puts "#{@sp*1}" if (lev <= 1) and hs[1] puts "" if (lev==0) end end end end __END__