require 'spec_helper'
if opal?
describe 'the param macro', type: :component do
it 'defines collect_other_params_as method on params proxy' do
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
collect_other_params_as :foo
def render
div { params.foo[:bar] }
end
end
expect(Foo).to render_static_html('
biz
').with_params(bar: 'biz')
end
it "can create and access a required param" do
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
param :foo
def render
div { params.foo }
end
end
expect(Foo).to render_static_html('bar
').with_params(foo: :bar)
end
it "can create and access an optional params" do
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
param foo1: :no_bar1
param foo2: :no_bar2
param :foo3, default: :no_bar3
param :foo4, default: :no_bar4
def render
div { "#{params.foo1}-#{params.foo2}-#{params.foo3}-#{params.foo4}" }
end
end
expect(Foo).to render_static_html('bar1-no_bar2-bar3-no_bar4
').with_params(foo1: :bar1, foo3: :bar3)
end
it 'can specify validation rules with the type option' do
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
param :foo, type: String
end
expect(Foo.prop_types).to have_key(:_componentValidator)
end
it "can type check params" do
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
param :foo1, type: String
param :foo2, type: String
def render
div { "#{params.foo1}-#{params.foo2}" }
end
end
%x{
var log = [];
var org_warn_console = window.console.warn;
var org_error_console = window.console.error;
window.console.warn = window.console.error = function(str){log.push(str)}
}
expect(Foo).to render_static_html('12-string
').with_params(foo1: 12, foo2: "string")
`window.console.warn = org_warn_console; window.console.error = org_error_console;`
expect(`log[0]`).to match(/Warning: Failed prop( type|Type): In component `Foo`\nProvided prop `foo1` could not be converted to String/)
end
it 'logs error in warning if validation failed' do
stub_const 'Lorem', Class.new
stub_const 'Foo2', Class.new(React::Component::Base)
Foo2.class_eval do
param :foo
param :lorem, type: Lorem
param :bar, default: nil, type: String
def render; div; end
end
%x{
var log = [];
var org_warn_console = window.console.warn;
var org_error_console = window.console.error;
window.console.warn = window.console.error = function(str){log.push(str)}
}
renderToDocument(Foo2, bar: 10, lorem: Lorem.new)
`window.console.warn = org_warn_console; window.console.error = org_error_console;`
expect(`log[0]`).to match(/Warning: Failed prop( type|Type): In component `Foo2`\nRequired prop `foo` was not specified\nProvided prop `bar` could not be converted to String/)
end
it 'should not log anything if validation passes' do
stub_const 'Lorem', Class.new
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval do
param :foo
param :lorem, type: Lorem
param :bar, default: nil, type: String
def render; div; end
end
%x{
var log = [];
var org_warn_console = window.console.warn;
var org_error_console = window.console.error;
window.console.warn = window.console.error = function(str){log.push(str)}
}
renderToDocument(Foo, foo: 10, bar: '10', lorem: Lorem.new)
`window.console.warn = org_warn_console; window.console.error = org_error_console;`
expect(`log`).to eq([])
end
describe 'advanced type handling' do
before(:each) do
%x{
window.dummy_log = [];
window.org_warn_console = window.console.warn;
window.org_error_console = window.console.warn
window.console.warn = window.console.error = function(str){window.dummy_log.push(str)}
}
stub_const 'Foo', Class.new(React::Component::Base)
Foo.class_eval { def render; ""; end }
end
after(:each) do
`window.console.warn = window.org_warn_console; window.console.error = window.org_error_console`
end
it "can use the [] notation for arrays" do
Foo.class_eval do
param :foo, type: []
param :bar, type: []
end
renderToDocument(Foo, foo: 10, bar: [10])
expect(`window.dummy_log[0]`).to match(/Warning: Failed prop( type|Type): In component `Foo`\nProvided prop `foo` could not be converted to Array/)
end
it "can use the [xxx] notation for arrays of a specific type" do
Foo.class_eval do
param :foo, type: [String]
param :bar, type: [String]
end
renderToDocument(Foo, foo: [10], bar: ["10"])
expect(`window.dummy_log[0]`).to match(/Warning: Failed prop( type|Type): In component `Foo`\nProvided prop `foo`\[0\] could not be converted to String/)
end
it "can convert a json hash to a type" do
stub_const "BazWoggle", Class.new
BazWoggle.class_eval do
def initialize(kind)
@kind = kind
end
attr_accessor :kind
def self._react_param_conversion(json, validate_only)
new(json[:bazwoggle]) if json[:bazwoggle]
end
end
Foo.class_eval do
param :foo, type: BazWoggle
param :bar, type: BazWoggle
param :baz, type: [BazWoggle]
def render
"#{params.bar.kind}, #{params.baz[0].kind}"
end
end
params = { foo: "", bar: { bazwoggle: 1 }, baz: [{ bazwoggle: 2 }] }
expect(Foo).to render_static_html('1, 2').with_params(params)
expect(`window.dummy_log[0]`).to match(/Warning: Failed prop( type|Type): In component `Foo`\nProvided prop `foo` could not be converted to BazWoggle/)
end
describe "converts params only once" do
it "not on every access" do
stub_const "BazWoggle", Class.new
BazWoggle.class_eval do
def initialize(kind)
@kind = kind
end
attr_accessor :kind
def self._react_param_conversion(json, validate_only)
new(json[:bazwoggle]) if json[:bazwoggle]
end
end
Foo.class_eval do
param :foo, type: BazWoggle
def render
params.foo.kind = params.foo.kind+1
"#{params.foo.kind}"
end
end
expect(Foo).to render_static_html('2').with_params(foo: {bazwoggle: 1})
end
it "even if contains an embedded native object" do
pending 'Fix after merging'
stub_const "Bar", Class.new(React::Component::Base)
stub_const "BazWoggle", Class.new
BazWoggle.class_eval do
def initialize(kind)
@kind = kind
end
attr_accessor :kind
def self._react_param_conversion(json, validate_only)
new(JSON.from_object(json[0])[:bazwoggle]) if JSON.from_object(json[0])[:bazwoggle]
end
end
Bar.class_eval do
param :foo, type: BazWoggle
def render
params.foo.kind.to_s
end
end
Foo.class_eval do
export_state :change_me
before_mount do
Foo.change_me! "initial"
end
def render
Bar(foo: Native([`{bazwoggle: #{Foo.change_me}}`]))
end
end
div = `document.createElement("div")`
React.render(React.create_element(Foo, {}), div)
Foo.change_me! "updated"
expect(`div.children[0].innerHTML`).to eq("updated")
end
end
it "will alias a Proc type param" do
Foo.class_eval do
param :foo, type: Proc
def render
params.foo
end
end
expect(Foo).to render_static_html('works!').with_params(foo: lambda { 'works!' })
end
it "will create a 'bang' (i.e. update) method if the type is React::Observable" do
Foo.class_eval do
param :foo, type: React::Observable
before_mount do
params.foo! "ha!"
end
def render
params.foo
end
end
current_state = ""
observer = React::Observable.new(current_state) { |new_state| current_state = new_state }
expect(Foo).to render_static_html('ha!').with_params(foo: observer)
expect(current_state).to eq("ha!")
end
end
end
end