require File.expand_path("#{File.dirname(__FILE__)}/../spec_helper")
require 'benchmark'
require 'nokogiri'
module WidgetSpec
describe Erector::Widget do
include Erector::Mixin
describe "#to_html" do
it "invokes #content and returns the string representation of the rendered widget" do
Class.new(Erector::Widget) do
def content
text "Hello"
end
end.new.to_html.should == "Hello"
end
it "supports other content methods via :content_method_name" do
Class.new(Erector::Widget) do
def alternate
text "Alternate"
end
end.new.to_html(:content_method_name => :alternate).should == "Alternate"
end
it "returns an HTML-safe string" do
Erector::Widget.new.to_html.should be_html_safe
end
it "accepts an existing string as an output buffer" do
s = "foo"
Erector.inline { text "bar" }.to_html(:output => s)
s.should == "foobar"
end
it "accepts an existing Output as an output buffer" do
output = Erector::Output.new
output << "foo"
Erector.inline { text "bar" }.to_html(:output => output)
output.to_s.should == "foobar"
end
end
describe "#to_a" do
it "returns an array" do
a = Erector.inline { div "Hello" }.to_a
a.should be_an(Array)
a.join.should == "
Hello
"
end
end
describe '#widget' do
class Orphan < Erector::Widget
def content
p @name
end
end
it "renders a widget class" do
erector do
div do
widget Orphan, :name => "Annie"
end
end.should == ""
end
it "renders a widget instance" do
erector do
div do
widget Orphan.new(:name => "Oliver")
end
end.should == ""
end
it "adds the widget to the parent's output widgets" do
inner = Class.new(Erector::Widget)
outer = Erector.inline { widget inner }
outer.to_html
outer.output.widgets.should include(inner)
end
it "supports specifying content_method_name" do
inner = Class.new(Erector::Widget) do
def foo; text "foo"; end
end
erector do
widget inner, {}, :content_method_name => :foo
end.should == "foo"
end
it "renders nested widgets in the correct order" do
class Parent < Erector::Widget
def content
text 1
widget Erector::Widget do
text 2
third
end
end
def third
text 3
end
end
Parent.new.to_html.should == '123'
end
context "when nested" do
module WhenNested
class Parent < Erector::Widget
def content
div :id => "parent_widget" do
super
end
end
end
class Child < Erector::Widget
def content
div :id => "child_widget" do
super
end
end
end
class Grandchild < Erector::Widget
needs :parent_widget, :child_widget
def content
widget(@parent_widget) do
widget(@child_widget) do
div :id => "grandchild"
end
end
end
end
end
it "renders the tag around the rest of the block" do
WhenNested::Grandchild.new(
:parent_widget => WhenNested::Parent,
:child_widget => WhenNested::Child
).to_html.should == ''
end
it "renders the tag around the rest of the block with proper indentation" do
WhenNested::Grandchild.new(
:parent_widget => WhenNested::Parent,
:child_widget => WhenNested::Child
).to_pretty.should ==
"\n"
end
it "passes a pointer to the child object back into the parent object's block" do
child_widget = Erector::Widget.new
class Parent2 < Erector::Widget
needs :child_widget
def content
div do
widget @child_widget do |child|
b child.widget_dom_id
end
end
end
end
Parent2.new(:child_widget => child_widget).to_html.should == "#{child_widget.widget_dom_id}
"
end
end
end
describe "#call_block" do
it "calls the block with a pointer to self" do
inside_arg = nil
inside_self = nil
x = Erector::Widget.new do |y|
inside_arg = y.object_id
inside_self = self.object_id
end
x.call_block
# inside the block...
inside_arg.should == x.object_id # the argument is the child
inside_self.should == self.object_id # and self is the parent
end
end
describe '#capture' do
it "should return content rather than write it to the buffer" do
erector do
captured = capture_content do
p 'Captured Content'
end
div do
text captured
end
end.should == ''
end
it "returns a RawString" do
captured = nil
erector do
captured = capture_content {}
end.should == ""
captured.should be_a_kind_of(Erector::RawString)
end
it "works with nested captures" do
erector do
captured = capture_content do
captured = capture_content do
p 'Nested Capture'
end
p 'Captured Content'
text captured
end
div do
text captured
end
end.should == 'Captured Content
Nested Capture
'
end
it "also is callable as #capture_content in case someone stole the capture name (I'm talking to you, rails 3.1)" do
captured = nil
erector do
captured = capture_content { text "A" }
end.should == ""
captured.should == "A"
end
end
describe '#widget' do
it "renders a widget" do
erector do
text "B"
widget Erector.inline { p "A" }
text "B"
end.should == "BA
B"
end
end
describe '#text' do
it "fails to render a widget" do
lambda do
erector do
text "B"
text Erector.inline { p "A" }
text "B"
end
end.should raise_error
end
it "can take several text values" do
erector do
text "a", "b", "c"
end.should == "abc"
end
it "can take several text values with a text delimiter" do
erector do
text "a", "b", "c", :join => " "
end.should == "a b c"
end
it "can take several text values including a promise" do
erector do
text "i ", b("love"), " cheese"
end.should == "i love cheese"
end
it "can take several text values including several promises and a delimiter" do
erector do
text "i", b("love"), i("green"), "cheese", :join => " "
end.should == "i love green cheese"
end
it "can take several text values including a dot-class promise" do
erector do
text "i ", b.wow("love"), " cheese"
end.should == "i love cheese"
end
end
describe "assigning instance variables" do
it "handles instance variable names with and without '@' in the beginning" do
html = Erector.inline(:foo => "bar", '@baz' => 'quux') do
div do
p @foo
p @baz
end
end.to_html
doc = Nokogiri::HTML(html)
doc.css("p").map {|p| p.inner_html}.should == ["bar", "quux"]
end
end
describe "named elements" do
it "can emit boolean values" do
erector { div true }.should == "true
"
erector { div false }.should == "false
"
end
end
end
end