TwP-webby 0.9.0
Sign up to get free protection for your applications and to get access to all the features.
- data/History.txt +176 -0
- data/Manifest.txt +173 -0
- data/README.txt +92 -0
- data/Rakefile +50 -0
- data/bin/webby +8 -0
- data/bin/webby-gen +8 -0
- data/examples/blog/Sitefile +7 -0
- data/examples/blog/tasks/blog.rake +72 -0
- data/examples/blog/templates/atom_feed.erb +40 -0
- data/examples/blog/templates/blog/month.erb +22 -0
- data/examples/blog/templates/blog/post.erb +16 -0
- data/examples/blog/templates/blog/year.erb +22 -0
- data/examples/presentation/Sitefile +10 -0
- data/examples/presentation/content/css/uv/twilight.css +137 -0
- data/examples/presentation/content/presentation/_sample_code.txt +10 -0
- data/examples/presentation/content/presentation/index.txt +63 -0
- data/examples/presentation/content/presentation/s5/blank.gif +0 -0
- data/examples/presentation/content/presentation/s5/bodybg.gif +0 -0
- data/examples/presentation/content/presentation/s5/framing.css +23 -0
- data/examples/presentation/content/presentation/s5/iepngfix.htc +42 -0
- data/examples/presentation/content/presentation/s5/opera.css +7 -0
- data/examples/presentation/content/presentation/s5/outline.css +15 -0
- data/examples/presentation/content/presentation/s5/pretty.css +86 -0
- data/examples/presentation/content/presentation/s5/print.css +1 -0
- data/examples/presentation/content/presentation/s5/s5-core.css +9 -0
- data/examples/presentation/content/presentation/s5/slides.css +3 -0
- data/examples/presentation/content/presentation/s5/slides.js +553 -0
- data/examples/presentation/layouts/presentation.txt +43 -0
- data/examples/presentation/templates/_code_partial.erb +13 -0
- data/examples/presentation/templates/presentation.erb +40 -0
- data/examples/tumblog/Sitefile +9 -0
- data/examples/tumblog/content/css/tumblog.css +308 -0
- data/examples/tumblog/content/images/tumblog/permalink.gif +0 -0
- data/examples/tumblog/content/images/tumblog/rss.gif +0 -0
- data/examples/tumblog/content/tumblog/200806/the-noble-chicken/index.txt +12 -0
- data/examples/tumblog/content/tumblog/200807/historical-perspectives-on-the-classic-chicken-joke/index.txt +12 -0
- data/examples/tumblog/content/tumblog/200807/mad-city-chickens/index.txt +10 -0
- data/examples/tumblog/content/tumblog/200807/the-wisdom-of-the-dutch/index.txt +11 -0
- data/examples/tumblog/content/tumblog/200807/up-a-tree/index.txt +13 -0
- data/examples/tumblog/content/tumblog/index.txt +37 -0
- data/examples/tumblog/content/tumblog/rss.txt +37 -0
- data/examples/tumblog/layouts/tumblog/default.txt +44 -0
- data/examples/tumblog/layouts/tumblog/post.txt +15 -0
- data/examples/tumblog/lib/tumblog_helper.rb +32 -0
- data/examples/tumblog/tasks/tumblog.rake +30 -0
- data/examples/tumblog/templates/atom_feed.erb +40 -0
- data/examples/tumblog/templates/tumblog/conversation.erb +12 -0
- data/examples/tumblog/templates/tumblog/link.erb +10 -0
- data/examples/tumblog/templates/tumblog/photo.erb +13 -0
- data/examples/tumblog/templates/tumblog/post.erb +12 -0
- data/examples/tumblog/templates/tumblog/quote.erb +11 -0
- data/examples/webby/Sitefile +19 -0
- data/examples/webby/content/communicate/index.txt +28 -0
- data/examples/webby/content/css/background.gif +0 -0
- data/examples/webby/content/css/blueprint/print.css +76 -0
- data/examples/webby/content/css/blueprint/screen.css +696 -0
- data/examples/webby/content/css/coderay.css +96 -0
- data/examples/webby/content/css/site.css +196 -0
- data/examples/webby/content/css/uv/twilight.css +137 -0
- data/examples/webby/content/index.txt +37 -0
- data/examples/webby/content/learn/index.txt +28 -0
- data/examples/webby/content/reference/index.txt +204 -0
- data/examples/webby/content/release-notes/rel-0-9-0/index.txt +73 -0
- data/examples/webby/content/robots.txt +6 -0
- data/examples/webby/content/script/jquery.corner.js +152 -0
- data/examples/webby/content/script/jquery.js +31 -0
- data/examples/webby/content/sitemap.txt +31 -0
- data/examples/webby/content/tips_and_tricks/index.txt +96 -0
- data/examples/webby/content/tutorial/index.txt +131 -0
- data/examples/webby/content/user-manual/index.txt +419 -0
- data/examples/webby/layouts/default.txt +49 -0
- data/examples/webby/templates/page.erb +10 -0
- data/examples/website/Sitefile +7 -0
- data/examples/website/content/css/blueprint/License.txt +21 -0
- data/examples/website/content/css/blueprint/Readme.txt +100 -0
- data/examples/website/content/css/blueprint/compressed/print.css +76 -0
- data/examples/website/content/css/blueprint/compressed/screen.css +696 -0
- data/examples/website/content/css/blueprint/lib/forms.css +45 -0
- data/examples/website/content/css/blueprint/lib/grid.css +193 -0
- data/examples/website/content/css/blueprint/lib/grid.png +0 -0
- data/examples/website/content/css/blueprint/lib/ie.css +30 -0
- data/examples/website/content/css/blueprint/lib/reset.css +39 -0
- data/examples/website/content/css/blueprint/lib/typography.css +116 -0
- data/examples/website/content/css/blueprint/plugins/buttons/Readme +31 -0
- data/examples/website/content/css/blueprint/plugins/buttons/buttons.css +97 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/cross.png +0 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/key.png +0 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/tick.png +0 -0
- data/examples/website/content/css/blueprint/plugins/css-classes/Readme +14 -0
- data/examples/website/content/css/blueprint/plugins/css-classes/css-classes.css +24 -0
- data/examples/website/content/css/blueprint/plugins/fancy-type/Readme +22 -0
- data/examples/website/content/css/blueprint/plugins/fancy-type/fancy-type-compressed.css +5 -0
- data/examples/website/content/css/blueprint/plugins/fancy-type/fancy-type.css +74 -0
- data/examples/website/content/css/blueprint/print.css +68 -0
- data/examples/website/content/css/blueprint/screen.css +22 -0
- data/examples/website/content/css/coderay.css +111 -0
- data/examples/website/content/css/site.css +67 -0
- data/examples/website/content/index.txt +19 -0
- data/examples/website/layouts/default.txt +58 -0
- data/examples/website/lib/breadcrumbs.rb +28 -0
- data/examples/website/templates/_partial.erb +10 -0
- data/examples/website/templates/page.erb +18 -0
- data/examples/website/templates/presentation.erb +40 -0
- data/lib/webby/apps/generator.rb +283 -0
- data/lib/webby/apps/main.rb +221 -0
- data/lib/webby/apps.rb +12 -0
- data/lib/webby/auto_builder.rb +83 -0
- data/lib/webby/builder.rb +183 -0
- data/lib/webby/core_ext/enumerable.rb +11 -0
- data/lib/webby/core_ext/hash.rb +28 -0
- data/lib/webby/core_ext/kernel.rb +21 -0
- data/lib/webby/core_ext/string.rb +163 -0
- data/lib/webby/core_ext/time.rb +9 -0
- data/lib/webby/filters/basepath.rb +97 -0
- data/lib/webby/filters/erb.rb +9 -0
- data/lib/webby/filters/haml.rb +18 -0
- data/lib/webby/filters/markdown.rb +16 -0
- data/lib/webby/filters/outline.rb +309 -0
- data/lib/webby/filters/sass.rb +17 -0
- data/lib/webby/filters/slides.rb +56 -0
- data/lib/webby/filters/textile.rb +16 -0
- data/lib/webby/filters/tidy.rb +76 -0
- data/lib/webby/filters.rb +91 -0
- data/lib/webby/helpers/capture_helper.rb +141 -0
- data/lib/webby/helpers/coderay_helper.rb +69 -0
- data/lib/webby/helpers/graphviz_helper.rb +136 -0
- data/lib/webby/helpers/tag_helper.rb +65 -0
- data/lib/webby/helpers/tex_img_helper.rb +133 -0
- data/lib/webby/helpers/ultraviolet_helper.rb +63 -0
- data/lib/webby/helpers/url_helper.rb +235 -0
- data/lib/webby/helpers.rb +30 -0
- data/lib/webby/link_validator.rb +152 -0
- data/lib/webby/renderer.rb +379 -0
- data/lib/webby/resources/db.rb +251 -0
- data/lib/webby/resources/file.rb +221 -0
- data/lib/webby/resources/layout.rb +63 -0
- data/lib/webby/resources/page.rb +118 -0
- data/lib/webby/resources/partial.rb +79 -0
- data/lib/webby/resources/resource.rb +160 -0
- data/lib/webby/resources/static.rb +52 -0
- data/lib/webby/resources.rb +96 -0
- data/lib/webby/stelan/mktemp.rb +135 -0
- data/lib/webby/stelan/paginator.rb +150 -0
- data/lib/webby/stelan/spawner.rb +339 -0
- data/lib/webby/tasks/build.rake +27 -0
- data/lib/webby/tasks/create.rake +22 -0
- data/lib/webby/tasks/deploy.rake +22 -0
- data/lib/webby/tasks/growl.rake +15 -0
- data/lib/webby/tasks/heel.rake +28 -0
- data/lib/webby/tasks/validate.rake +19 -0
- data/lib/webby.rb +227 -0
- data/spec/core_ext/hash_spec.rb +47 -0
- data/spec/core_ext/string_spec.rb +110 -0
- data/spec/core_ext/time_spec.rb +19 -0
- data/spec/spec.opts +1 -0
- data/spec/spec_helper.rb +14 -0
- data/spec/webby/apps/generator_spec.rb +111 -0
- data/spec/webby/apps/main_spec.rb +75 -0
- data/spec/webby/helpers/capture_helper_spec.rb +56 -0
- data/spec/webby/resources/file_spec.rb +104 -0
- data/spec/webby/resources_spec.rb +17 -0
- data/tasks/ann.rake +81 -0
- data/tasks/bones.rake +21 -0
- data/tasks/gem.rake +126 -0
- data/tasks/git.rake +41 -0
- data/tasks/manifest.rake +49 -0
- data/tasks/notes.rake +28 -0
- data/tasks/post_load.rake +39 -0
- data/tasks/rdoc.rake +51 -0
- data/tasks/rubyforge.rake +57 -0
- data/tasks/setup.rb +268 -0
- data/tasks/spec.rake +55 -0
- data/tasks/website.rake +38 -0
- metadata +289 -0
@@ -0,0 +1,183 @@
|
|
1
|
+
require 'find'
|
2
|
+
require 'fileutils'
|
3
|
+
require 'erb'
|
4
|
+
|
5
|
+
module Webby
|
6
|
+
|
7
|
+
# The Builder class performs the work of scanning the content folder,
|
8
|
+
# creating Resource objects, and converting / copying the contents to the
|
9
|
+
# output folder as needed.
|
10
|
+
#
|
11
|
+
class Builder
|
12
|
+
|
13
|
+
class << self
|
14
|
+
|
15
|
+
# call-seq:
|
16
|
+
# Builder.run( :rebuild => false )
|
17
|
+
#
|
18
|
+
# Create a new instance of the Builder class and invoke the run method.
|
19
|
+
# If the <code>:rebuild</code> option is given as +true+, then all pages
|
20
|
+
# will be recreated / copied.
|
21
|
+
#
|
22
|
+
def run( opts = {} )
|
23
|
+
self.new.run opts
|
24
|
+
end
|
25
|
+
|
26
|
+
# call-seq:
|
27
|
+
# Builder.create( page, :from => template, :locals => {} )
|
28
|
+
#
|
29
|
+
# This mehod is used to create a new _page_ in the content folder based
|
30
|
+
# on the specified template. _page_ is the relative path to the new page
|
31
|
+
# from the <code>content/</code> folder. The _template_ is the name of
|
32
|
+
# the template to use from the <code>templates/</code> folder.
|
33
|
+
#
|
34
|
+
def create( page, opts = {} )
|
35
|
+
tmpl = opts[:from]
|
36
|
+
raise Error, "template not given" unless tmpl
|
37
|
+
|
38
|
+
name = ::Webby::Resources::File.basename(page)
|
39
|
+
ext = ::Webby::Resources::File.extname(page)
|
40
|
+
dir = ::File.dirname(page)
|
41
|
+
dir = '' if dir == '.'
|
42
|
+
|
43
|
+
if tmpl.pathmap('%n') =~ %r/^_/
|
44
|
+
page = ::File.join(::Webby.site.content_dir, dir, '_'+name)
|
45
|
+
page << '.' << (ext.empty? ? 'txt' : ext)
|
46
|
+
elsif ::Webby.site.create_mode == 'directory' and name != 'index'
|
47
|
+
page = ::File.join(::Webby.site.content_dir, dir, name, 'index')
|
48
|
+
page << '.' << (ext.empty? ? 'txt' : ext)
|
49
|
+
else
|
50
|
+
page = ::File.join(::Webby.site.content_dir, page)
|
51
|
+
page << '.txt' if ext.empty?
|
52
|
+
end
|
53
|
+
raise Error, "#{page} already exists" if test ?e, page
|
54
|
+
|
55
|
+
Logging::Logger[self].info "creating #{page}"
|
56
|
+
FileUtils.mkdir_p ::File.dirname(page)
|
57
|
+
|
58
|
+
context = scope
|
59
|
+
opts[:locals].each do |k,v|
|
60
|
+
Thread.current[:value] = v
|
61
|
+
definition = "#{k} = Thread.current[:value]"
|
62
|
+
eval(definition, context)
|
63
|
+
end if opts.has_key?(:locals)
|
64
|
+
|
65
|
+
str = ERB.new(::File.read(tmpl), nil, '-').result(context)
|
66
|
+
::File.open(page, 'w') {|fd| fd.write str}
|
67
|
+
|
68
|
+
page
|
69
|
+
end
|
70
|
+
|
71
|
+
# call-seq:
|
72
|
+
# Builder.new_page_info => [page, title, directory]
|
73
|
+
#
|
74
|
+
def new_page_info
|
75
|
+
args = Webby.site.args
|
76
|
+
|
77
|
+
if args.raw.empty?
|
78
|
+
task_name = Rake.application.top_level_tasks.first
|
79
|
+
raise "Usage: webby #{task_name} path"
|
80
|
+
end
|
81
|
+
|
82
|
+
[args.page, args.title, args.dir]
|
83
|
+
end
|
84
|
+
|
85
|
+
|
86
|
+
private
|
87
|
+
|
88
|
+
# Returns the binding in the scope of the Builder class object.
|
89
|
+
#
|
90
|
+
def scope() binding end
|
91
|
+
|
92
|
+
end # class << self
|
93
|
+
|
94
|
+
# call-seq:
|
95
|
+
# Builder.new
|
96
|
+
#
|
97
|
+
# Creates a new Builder object for creating pages from the content and
|
98
|
+
# layout directories.
|
99
|
+
#
|
100
|
+
def initialize
|
101
|
+
@log = Logging::Logger[self]
|
102
|
+
end
|
103
|
+
|
104
|
+
# call-seq:
|
105
|
+
# run( :rebuild => false, :load_files => true )
|
106
|
+
#
|
107
|
+
# Runs the Webby builder by loading in the layout files from the
|
108
|
+
# <code>layouts/</code> folder and the content from the
|
109
|
+
# <code>contents/</code> folder. Content is analyzed, and those that need
|
110
|
+
# to be copied or compiled (filtered using ERB, Texttile, Markdown, etc.)
|
111
|
+
# are handled. The results are placed in the <code>output/</code> folder.
|
112
|
+
#
|
113
|
+
# If the <code>:rebuild</code> flag is set to +true+, then all content is
|
114
|
+
# copied and/or compiled to the output folder.
|
115
|
+
#
|
116
|
+
# A content file can mark itself as dirty by setting the +dirty+ flag to
|
117
|
+
# +true+ in the meta-data of the file. This will cause the contenet to
|
118
|
+
# always be compiled when the builder is run. Conversely, setting the
|
119
|
+
# dirty flag to +false+ will cause the content to never be compiled or
|
120
|
+
# copied to the output folder.
|
121
|
+
#
|
122
|
+
# A content file needs to be built if the age of the file is less then the
|
123
|
+
# age of the output product -- i.e. the content file has been modified
|
124
|
+
# more recently than the output file.
|
125
|
+
#
|
126
|
+
def run( opts = {} )
|
127
|
+
opts[:load_files] = true unless opts.has_key?(:load_files)
|
128
|
+
|
129
|
+
unless test(?d, output_dir)
|
130
|
+
@log.info "creating #{output_dir}"
|
131
|
+
FileUtils.mkdir output_dir
|
132
|
+
end
|
133
|
+
|
134
|
+
load_files if opts[:load_files]
|
135
|
+
|
136
|
+
Resources.pages.each do |page|
|
137
|
+
next unless page.dirty? or opts[:rebuild]
|
138
|
+
|
139
|
+
@log.info "creating #{page.destination}"
|
140
|
+
|
141
|
+
# make sure the directory exists
|
142
|
+
FileUtils.mkdir_p ::File.dirname(page.destination)
|
143
|
+
|
144
|
+
# copy the resource to the output directory if it is static
|
145
|
+
if page.instance_of? Resources::Static
|
146
|
+
FileUtils.cp page.path, page.destination
|
147
|
+
FileUtils.chmod 0644, page.destination
|
148
|
+
|
149
|
+
# otherwise, layout the resource and write the results to
|
150
|
+
# the output directory
|
151
|
+
else Renderer.write(page) end
|
152
|
+
end
|
153
|
+
|
154
|
+
# touch the cairn so we know when the website was last generated
|
155
|
+
FileUtils.touch ::Webby.cairn
|
156
|
+
|
157
|
+
nil
|
158
|
+
end
|
159
|
+
|
160
|
+
|
161
|
+
private
|
162
|
+
|
163
|
+
# Scan the <code>layouts/</code> folder and the <code>content/</code>
|
164
|
+
# folder and create a new Resource object for each file found there.
|
165
|
+
#
|
166
|
+
def load_files
|
167
|
+
::Find.find(layout_dir, content_dir) do |path|
|
168
|
+
next unless test ?f, path
|
169
|
+
next if path =~ ::Webby.exclude
|
170
|
+
Resources.new path
|
171
|
+
end
|
172
|
+
end
|
173
|
+
|
174
|
+
%w(output_dir layout_dir content_dir).each do |key|
|
175
|
+
self.class_eval <<-CODE
|
176
|
+
def #{key}( ) ::Webby.site.#{key} end
|
177
|
+
CODE
|
178
|
+
end
|
179
|
+
|
180
|
+
end # class Builder
|
181
|
+
end # module Webby
|
182
|
+
|
183
|
+
# EOF
|
@@ -0,0 +1,28 @@
|
|
1
|
+
|
2
|
+
class Hash
|
3
|
+
|
4
|
+
def sanitize!
|
5
|
+
h = self.injecting({}) do |h, (k, v)|
|
6
|
+
h[k] = case v
|
7
|
+
when 'none', 'nil'; nil
|
8
|
+
when 'true', 'yes'; true
|
9
|
+
when 'false', 'no'; false
|
10
|
+
else v end
|
11
|
+
end
|
12
|
+
self.replace h
|
13
|
+
end
|
14
|
+
|
15
|
+
def stringify_keys
|
16
|
+
h = {}
|
17
|
+
self.each {|k,v| h[k.to_s] = v}
|
18
|
+
return h
|
19
|
+
end
|
20
|
+
|
21
|
+
def symbolize_keys
|
22
|
+
h = {}
|
23
|
+
self.each {|k,v| h[k.to_sym] = v}
|
24
|
+
return h
|
25
|
+
end
|
26
|
+
end # class Hash
|
27
|
+
|
28
|
+
# EOF
|
@@ -0,0 +1,21 @@
|
|
1
|
+
|
2
|
+
module Kernel
|
3
|
+
|
4
|
+
# :stopdoc:
|
5
|
+
WINDOWS = %r/djgpp|(cyg|ms|bcc)win|mingw/ =~ RUBY_PLATFORM
|
6
|
+
DEV_NULL = WINDOWS ? 'NUL:' : '/dev/null'
|
7
|
+
# :startdoc:
|
8
|
+
|
9
|
+
def cmd_available?( *args )
|
10
|
+
io = [STDOUT.dup, STDERR.dup]
|
11
|
+
STDOUT.reopen DEV_NULL
|
12
|
+
STDERR.reopen DEV_NULL
|
13
|
+
system(*(args.flatten))
|
14
|
+
ensure
|
15
|
+
STDOUT.reopen io.first
|
16
|
+
STDERR.reopen io.last
|
17
|
+
$stdout, $stderr = STDOUT, STDERR
|
18
|
+
end
|
19
|
+
end # module Kernel
|
20
|
+
|
21
|
+
# EOF
|
@@ -0,0 +1,163 @@
|
|
1
|
+
|
2
|
+
class String
|
3
|
+
|
4
|
+
def self.small_words
|
5
|
+
@small_words ||= %w(a an and as at but by en for if in of on or the to v[.]? via vs[.]?)
|
6
|
+
end
|
7
|
+
|
8
|
+
def /( path )
|
9
|
+
::File.join(self, path)
|
10
|
+
end
|
11
|
+
|
12
|
+
def titlecase
|
13
|
+
swrgxp = self.class.small_words.join('|')
|
14
|
+
|
15
|
+
parts = self.split( %r/( [:.;?!][ ] | (?:[ ]|^)["“] )/x )
|
16
|
+
parts.each do |part|
|
17
|
+
part.gsub!(%r/\b[[:alpha:]][[:lower:].'’]*\b/) do |s|
|
18
|
+
s =~ %r/\w+\.\w+/ ? s : s.capitalize
|
19
|
+
end
|
20
|
+
|
21
|
+
# Lowercase the small words
|
22
|
+
part.gsub!(%r/\b(#{swrgxp})\b/i) {|w| w.downcase}
|
23
|
+
|
24
|
+
# If the first word is a small word, then capitalize it
|
25
|
+
part.gsub!(%r/\A([[:punct:]]*)(#{swrgxp})\b/) {$1 + $2.capitalize}
|
26
|
+
|
27
|
+
# If the last word is a small word, then capitalize it
|
28
|
+
part.gsub!(%r/\b(#{swrgxp})([^\w\s]*)\z/) {$1.capitalize + $2}
|
29
|
+
end
|
30
|
+
|
31
|
+
str = parts.join
|
32
|
+
|
33
|
+
# Special cases:
|
34
|
+
str.gsub!(/ V(s?)\. /, ' v\1. ') # "v." and "vs."
|
35
|
+
str.gsub!(/(['’])S\b/, '\1s') # 'S (otherwise you get "the SEC'S decision")
|
36
|
+
str.gsub!(/\b(AT&T|Q&A)\b/i) { |w| w.upcase } # "AT&T" and "Q&A", which get tripped up.
|
37
|
+
|
38
|
+
str
|
39
|
+
end
|
40
|
+
|
41
|
+
# Borrowed from the excellent StringEx library: git://github.com/rsl/stringex.git
|
42
|
+
|
43
|
+
# Create a URI-friendly representation of the string.
|
44
|
+
def to_url
|
45
|
+
remove_formatting.downcase.replace_whitespace("-").collapse("-")
|
46
|
+
end
|
47
|
+
|
48
|
+
# Performs multiple text manipulations. Essentially a shortcut for typing them all. View source
|
49
|
+
# below to see which methods are run.
|
50
|
+
def remove_formatting
|
51
|
+
strip_html_tags.convert_accented_entities.convert_misc_entities.convert_misc_characters.collapse
|
52
|
+
end
|
53
|
+
|
54
|
+
# Removes HTML tags from text. This code is simplified from Tobias Luettke's regular expression
|
55
|
+
# in Typo[http://typosphere.org].
|
56
|
+
def strip_html_tags(leave_whitespace = false)
|
57
|
+
name = /[\w:_-]+/
|
58
|
+
value = /([A-Za-z0-9]+|('[^']*?'|"[^"]*?"))/
|
59
|
+
attr = /(#{name}(\s*=\s*#{value})?)/
|
60
|
+
rx = /<[!\/?\[]?(#{name}|--)(\s+(#{attr}(\s+#{attr})*))?\s*([!\/?\]]+|--)?>/
|
61
|
+
(leave_whitespace) ? gsub(rx, "").strip : gsub(rx, "").gsub(/\s+/, " ").strip
|
62
|
+
end
|
63
|
+
|
64
|
+
# Converts HTML entities into the respective non-accented letters. Examples:
|
65
|
+
#
|
66
|
+
# "á".convert_accented_entities # => "a"
|
67
|
+
# "ç".convert_accented_entities # => "c"
|
68
|
+
# "è".convert_accented_entities # => "e"
|
69
|
+
# "î".convert_accented_entities # => "i"
|
70
|
+
# "ø".convert_accented_entities # => "o"
|
71
|
+
# "ü".convert_accented_entities # => "u"
|
72
|
+
#
|
73
|
+
# Note: This does not do any conversion of Unicode/Ascii accented-characters. For that
|
74
|
+
# functionality please use <tt>to_ascii</tt>.
|
75
|
+
def convert_accented_entities
|
76
|
+
gsub(/&([A-Za-z])(grave|acute|circ|tilde|uml|ring|cedil|slash);/, '\1')
|
77
|
+
end
|
78
|
+
|
79
|
+
# Converts HTML entities (taken from common Textile/RedCloth formattings) into plain text formats.
|
80
|
+
#
|
81
|
+
# Note: This isn't an attempt at complete conversion of HTML entities, just those most likely
|
82
|
+
# to be generated by Textile.
|
83
|
+
def convert_misc_entities
|
84
|
+
dummy = dup
|
85
|
+
{
|
86
|
+
"#822[01]" => "\"",
|
87
|
+
"#821[67]" => "'",
|
88
|
+
"#8230" => "...",
|
89
|
+
"#8211" => "-",
|
90
|
+
"#8212" => "--",
|
91
|
+
"#215" => "x",
|
92
|
+
"gt" => ">",
|
93
|
+
"lt" => "<",
|
94
|
+
"(#8482|trade)" => "(tm)",
|
95
|
+
"(#174|reg)" => "(r)",
|
96
|
+
"(#169|copy)" => "(c)",
|
97
|
+
"(#38|amp)" => "and",
|
98
|
+
"nbsp" => " ",
|
99
|
+
"(#162|cent)" => " cent",
|
100
|
+
"(#163|pound)" => " pound",
|
101
|
+
"(#188|frac14)" => "one fourth",
|
102
|
+
"(#189|frac12)" => "half",
|
103
|
+
"(#190|frac34)" => "three fourths",
|
104
|
+
"(#176|deg)" => " degrees"
|
105
|
+
}.each do |textiled, normal|
|
106
|
+
dummy.gsub!(/&#{textiled};/, normal)
|
107
|
+
end
|
108
|
+
dummy.gsub(/&[^;]+;/, "")
|
109
|
+
end
|
110
|
+
|
111
|
+
# Converts various common plaintext characters to a more URI-friendly representation.
|
112
|
+
# Examples:
|
113
|
+
#
|
114
|
+
# "foo & bar".convert_misc_characters # => "foo and bar"
|
115
|
+
# "Chanel #9".convert_misc_characters # => "Chanel number nine"
|
116
|
+
# "user@host".convert_misc_characters # => "user at host"
|
117
|
+
# "google.com".convert_misc_characters # => "google dot com"
|
118
|
+
# "$10".convert_misc_characters # => "10 dollars"
|
119
|
+
# "*69".convert_misc_characters # => "star 69"
|
120
|
+
# "100%".convert_misc_characters # => "100 percent"
|
121
|
+
# "windows/mac/linux".convert_misc_characters # => "windows slash mac slash linux"
|
122
|
+
#
|
123
|
+
# Note: Because this method will convert any & symbols to the string "and",
|
124
|
+
# you should run any methods which convert HTML entities (convert_html_entities and convert_misc_entities)
|
125
|
+
# before running this method.
|
126
|
+
def convert_misc_characters
|
127
|
+
dummy = dup.gsub(/\.{3,}/, " dot dot dot ") # Catch ellipses before single dot rule!
|
128
|
+
{
|
129
|
+
/\s*&\s*/ => "and",
|
130
|
+
/\s*#/ => "number",
|
131
|
+
/\s*@\s*/ => "at",
|
132
|
+
/(\S|^)\.(\S)/ => '\1 dot \2',
|
133
|
+
/(\s|^)\$(\d*)(\s|$)/ => '\2 dollars',
|
134
|
+
/\s*\*\s*/ => "star",
|
135
|
+
/\s*%\s*/ => "percent",
|
136
|
+
/\s*(\\|\/)\s*/ => "slash",
|
137
|
+
}.each do |found, replaced|
|
138
|
+
replaced = " #{replaced} " unless replaced =~ /\\1/
|
139
|
+
dummy.gsub!(found, replaced)
|
140
|
+
end
|
141
|
+
dummy = dummy.gsub(/(^|\w)'(\w|$)/, '\1\2').gsub(/[\.,:;()\[\]\/\?!\^'"_]/, " ")
|
142
|
+
end
|
143
|
+
|
144
|
+
# Replace runs of whitespace in string. Defaults to a single space but any replacement
|
145
|
+
# string may be specified as an argument. Examples:
|
146
|
+
#
|
147
|
+
# "Foo bar".replace_whitespace # => "Foo bar"
|
148
|
+
# "Foo bar".replace_whitespace("-") # => "Foo-bar"
|
149
|
+
def replace_whitespace(replace = " ")
|
150
|
+
gsub(/\s+/, replace)
|
151
|
+
end
|
152
|
+
|
153
|
+
# Removes specified character from the beginning and/or end of the string and then performs
|
154
|
+
# <tt>String#squeeze(character)</tt>, condensing runs of the character within the string.
|
155
|
+
#
|
156
|
+
# Note: This method has been superceded by ActiveSupport's squish method.
|
157
|
+
def collapse(character = " ")
|
158
|
+
sub(/^#{character}*/, "").sub(/#{character}*$/, "").squeeze(character)
|
159
|
+
end
|
160
|
+
|
161
|
+
end # class String
|
162
|
+
|
163
|
+
# EOF
|
@@ -0,0 +1,97 @@
|
|
1
|
+
require 'hpricot'
|
2
|
+
|
3
|
+
module Webby
|
4
|
+
module Filters
|
5
|
+
|
6
|
+
# The BasePath filter is used to rewrite URI paths in HTML documents. This
|
7
|
+
# is useful when the server location of the website is not located at the
|
8
|
+
# root of the webserver (e.g. http://my.site.com/foo/bar).
|
9
|
+
#
|
10
|
+
# The BasePath filter will adjust the URI paths in a given HTML document by
|
11
|
+
# prepending a base path to the URI. This only works for URIs that start
|
12
|
+
# with a leading slash "/". Any other character will exclude the URI from
|
13
|
+
# being modified.
|
14
|
+
#
|
15
|
+
# Assume the user specifies a new URI base in the <tt>Webby.site.base</tt>
|
16
|
+
# property:
|
17
|
+
#
|
18
|
+
# Webby.site.base = '/foo/bar'
|
19
|
+
#
|
20
|
+
# Here is a snippet from some HTML document.
|
21
|
+
#
|
22
|
+
# <a href="/some/other/page.html">Page</a>
|
23
|
+
# <img src="fractal.jpg" alt="a fractal" />
|
24
|
+
#
|
25
|
+
# When run through the BasePath filter, the resulting snippet would look
|
26
|
+
# like this.
|
27
|
+
#
|
28
|
+
# <a href="/foo/bar/some/other/page.html">Page</a>
|
29
|
+
# <img src="fractal.jpg" alt="a fractal" />
|
30
|
+
#
|
31
|
+
# The +href+ attribute of the anchor tag is modified because it started
|
32
|
+
# with a leading slash. The +src+ attribute of the image tag is not
|
33
|
+
# modified because it lacks the leading slash.
|
34
|
+
#
|
35
|
+
class BasePath
|
36
|
+
|
37
|
+
# call-seq:
|
38
|
+
# BasePath.new( html, mode )
|
39
|
+
#
|
40
|
+
# Creates a new BasePath filter that will operate on the given _html_
|
41
|
+
# string. The _mode_ is either 'xml' or 'html' and determines how Hpricot
|
42
|
+
# will handle the parsing of the input string.
|
43
|
+
#
|
44
|
+
def initialize( str, mode )
|
45
|
+
@str = str
|
46
|
+
@mode = mode.downcase.to_sym
|
47
|
+
end
|
48
|
+
|
49
|
+
# call-seq:
|
50
|
+
# filter => html
|
51
|
+
#
|
52
|
+
# Process the original html document passed to the filter when it was
|
53
|
+
# created. The document will be scanned and the basepath for certain
|
54
|
+
# elements will be modified.
|
55
|
+
#
|
56
|
+
# For example, if a document contains the following line:
|
57
|
+
#
|
58
|
+
# <a href="/link/to/another/page.html">Page</a>
|
59
|
+
#
|
60
|
+
# and the user has requested for the base path to be some other directory
|
61
|
+
# on the webserver -- <tt>/some/other/directory</tt>. The result of the
|
62
|
+
# BasePath filter would be:
|
63
|
+
#
|
64
|
+
# <a href="/some/other/directory/link/to/another/page.html">Page</a>
|
65
|
+
#
|
66
|
+
def filter
|
67
|
+
doc = @mode == :xml ? Hpricot.XML(@str) : Hpricot(@str)
|
68
|
+
base_path = ::Webby.site.base
|
69
|
+
attr_rgxp = %r/\[@(\w+)\]$/o
|
70
|
+
sub_rgxp = %r/\A(?=\/)/o
|
71
|
+
|
72
|
+
::Webby.site.xpaths.each do |xpath|
|
73
|
+
@attr_name = nil
|
74
|
+
|
75
|
+
doc.search(xpath).each do |element|
|
76
|
+
@attr_name ||= attr_rgxp.match(xpath)[1]
|
77
|
+
a = element.get_attribute(@attr_name)
|
78
|
+
element.set_attribute(@attr_name, a) if a.sub!(sub_rgxp, base_path)
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
doc.to_html
|
83
|
+
end
|
84
|
+
|
85
|
+
end # class BasePath
|
86
|
+
|
87
|
+
# Rewrite base URIs in the input HTML text.
|
88
|
+
#
|
89
|
+
register :basepath do |input, cursor|
|
90
|
+
if ::Webby.site.base then BasePath.new(input, cursor.page.extension).filter
|
91
|
+
else input end
|
92
|
+
end
|
93
|
+
|
94
|
+
end # module Filters
|
95
|
+
end # module Webby
|
96
|
+
|
97
|
+
# EOF
|
@@ -0,0 +1,18 @@
|
|
1
|
+
|
2
|
+
# Render text via the Haml library
|
3
|
+
if try_require('haml', 'haml')
|
4
|
+
|
5
|
+
Webby::Filters.register :haml do |input, cursor|
|
6
|
+
opts = ::Webby.site.haml_options.merge(cursor.page.haml_options || {})
|
7
|
+
b = cursor.renderer.get_binding
|
8
|
+
Haml::Engine.new(input, opts).to_html(b)
|
9
|
+
end
|
10
|
+
|
11
|
+
# Otherwise raise an error if the user tries to use haml
|
12
|
+
else
|
13
|
+
Webby::Filters.register :haml do |input, cursor|
|
14
|
+
raise Webby::Error, "'haml' must be installed to use the haml filter"
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
# EOF
|
@@ -0,0 +1,16 @@
|
|
1
|
+
|
2
|
+
# Render text via markdown using the RDiscount library.
|
3
|
+
if try_require('rdiscount', 'rdiscount')
|
4
|
+
|
5
|
+
Webby::Filters.register :markdown do |input|
|
6
|
+
RDiscount.new(input).to_html
|
7
|
+
end
|
8
|
+
|
9
|
+
# Otherwise raise an error if the user tries to use markdown
|
10
|
+
else
|
11
|
+
Webby::Filters.register :markdown do |input|
|
12
|
+
raise Webby::Error, "'rdiscount' must be installed to use the markdown filter"
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
# EOF
|