require 'spec_helper' module Alchemy describe Admin::ElementsController do let(:alchemy_page) { create(:page) } let(:element) { create(:element, :page_id => alchemy_page.id) } let(:element_in_clipboard) { create(:element, :page_id => alchemy_page.id) } let(:clipboard) { session[:alchemy_clipboard] = {} } before { sign_in(author_user) } describe '#index' do let(:alchemy_page) { build_stubbed(:page) } before do Page.stub(find: alchemy_page) end context 'with cells' do let(:cell) { build_stubbed(:cell, page: alchemy_page) } before { alchemy_page.stub(cells: [cell]) } it "groups elements by cell" do alchemy_page.should_receive(:elements_grouped_by_cells) get :index, {page_id: alchemy_page.id} assigns(:cells).should eq([cell]) end end context 'without cells' do before { alchemy_page.stub(cells: []) } it "assigns page elements" do alchemy_page.should_receive(:elements).and_return(double(not_trashed: [])) get :index, {page_id: alchemy_page.id} end end end describe '#list' do let(:alchemy_page) { build_stubbed(:page) } before do Page.stub(find: alchemy_page) end context 'without page_id, but with page_urlname' do it "loads page from urlname" do Language.stub(:current).and_return(double(code: 'en', pages: double(find_by: double(id: 1001)))) xhr :get, :list, {page_urlname: 'contact'} end describe 'view' do render_views it "should return a select tag with elements" do xhr :get, :list, {page_urlname: alchemy_page.urlname} response.body.should match(/select(.*)elements_from_page_selector(.*)option/) end end end context 'with page_id' do it "loads page from urlname" do xhr :get, :list, {page_id: alchemy_page.id} assigns(:page_id).should eq(alchemy_page.id.to_s) end end end describe '#order' do let(:element_1) { FactoryGirl.create(:element) } let(:element_2) { FactoryGirl.create(:element) } let(:element_3) { FactoryGirl.create(:element) } let(:element_ids) { [element_1.id, element_3.id, element_2.id] } it "sets new position for given element ids" do xhr :post, :order, element_ids: element_ids expect(Element.all.pluck(:id)).to eq(element_ids) end context "untrashing" do let(:trashed_element) { FactoryGirl.create(:element, public: false, position: nil, page_id: 58, cell_id: 32) } before do # Because of a before_create filter it can not be created with a nil position and needs to be trashed here trashed_element.trash! end it "sets a list of trashed element ids" do xhr :post, :order, element_ids: [trashed_element.id] expect(assigns(:trashed_elements).to_a).to eq [trashed_element.id] end it "should set a new position to the element" do xhr :post, :order, element_ids: [trashed_element.id] trashed_element.reload trashed_element.position.should_not == nil end it "should assign the (new) page_id to the element" do xhr :post, :order, element_ids: [trashed_element.id], page_id: 1, cell_id: nil trashed_element.reload trashed_element.page_id.should == 1 end it "should assign the (new) cell_id to the element" do xhr :post, :order, element_ids: [trashed_element.id], page_id: 1, cell_id: 5 trashed_element.reload trashed_element.cell_id.should == 5 end end end describe '#new' do let(:alchemy_page) { build_stubbed(:page) } before { Page.stub(:find_by_id).and_return(alchemy_page) } it "assign variable for all available element definitions" do alchemy_page.should_receive(:available_element_definitions) get :new, {page_id: alchemy_page.id} end context "with elements in clipboard" do let(:clipboard_items) { [{'id' => element.id.to_s, 'action' => 'copy'}] } before { clipboard['elements'] = clipboard_items } it "should load all elements from clipboard" do Element.should_receive(:all_from_clipboard_for_page).and_return(clipboard_items) get :new, {page_id: alchemy_page.id} assigns(:clipboard_items).should == clipboard_items end end end describe '#create' do describe 'insertion position' do before { element } it "should insert the element at bottom of list" do xhr :post, :create, {element: {name: 'news', page_id: alchemy_page.id}} alchemy_page.elements.count.should == 2 alchemy_page.elements.last.name.should == 'news' end context "on a page with a setting for insert_elements_at of top" do before do PageLayout.stub(:get).and_return({ 'name' => 'news', 'elements' => ['news'], 'insert_elements_at' => 'top' }) end it "should insert the element at top of list" do xhr :post, :create, {element: {name: 'news', page_id: alchemy_page.id}} alchemy_page.elements.count.should == 2 alchemy_page.elements.first.name.should == 'news' end end end context "if page has cells" do context "" do before do @page = create(:public_page, :do_not_autogenerate => false) @cell = create(:cell, :name => 'header', :page => @page) PageLayout.stub(:get).and_return({ 'name' => 'standard', 'elements' => ['article'], 'cells' => ['header'] }) Cell.stub(:definition_for).and_return({'name' => 'header', 'elements' => ['article']}) end context "and cell name in element name" do it "should put the element in the correct cell" do xhr :post, :create, {:element => {:name => "article#header", :page_id => @page.id}} @cell.elements.first.should be_an_instance_of(Element) end end context "and no cell name in element name" do it "should put the element in the main cell" do xhr :post, :create, {:element => {:name => "article", :page_id => @page.id}} @page.elements.not_in_cell.first.should be_an_instance_of(Element) end end end context "with paste_from_clipboard in parameters" do context "" do before do @page = create(:public_page, :do_not_autogenerate => false) @cell = create(:cell, :name => 'header', :page => @page) PageLayout.stub(:get).and_return({ 'name' => 'standard', 'elements' => ['article'], 'cells' => ['header'] }) Cell.stub(:definition_for).and_return({'name' => 'header', 'elements' => ['article']}) clipboard['elements'] = [{'id' => element_in_clipboard.id.to_s}] end context "and cell name in element name" do it "should create the element in the correct cell" do xhr :post, :create, {:element => {:page_id => @page.id}, :paste_from_clipboard => "#{element_in_clipboard.id}##{@cell.name}"} @cell.elements.first.should be_an_instance_of(Element) end end context "and no cell name in element name" do it "should create the element in the nil cell" do xhr :post, :create, {:element => {:page_id => @page.id}, :paste_from_clipboard => "#{element_in_clipboard.id}"} @page.elements.first.cell.should == nil end end context "" do before { @cell.elements.create(:page_id => @page.id, :name => "article", :create_contents_after_create => false) } it "should set the correct position for the element" do xhr :post, :create, {:element => {:page_id => @page.id}, :paste_from_clipboard => "#{element_in_clipboard.id}##{@cell.name}"} @cell.elements.last.position.should == @cell.elements.count end end end context "on a page with a setting for insert_elements_at of top" do let(:alchemy_page) { create(:public_page, :name => 'News') } let(:element_in_clipboard) { create(:element, :page => alchemy_page, :name => 'news') } let(:cell) { alchemy_page.cells.first } let(:element) { create(:element, :name => 'news', :page => alchemy_page, :cell => cell) } before do PageLayout.stub(:get).and_return({ 'name' => 'news', 'elements' => ['news'], 'insert_elements_at' => 'top', 'cells' => ['news'] }) Cell.stub(:definition_for).and_return({'name' => 'news', 'elements' => ['news']}) clipboard['elements'] = [{'id' => element_in_clipboard.id.to_s}] cell.elements << element end it "should insert the element at top of list" do xhr :post, :create, {:element => {:name => 'news', :page_id => alchemy_page.id}, :paste_from_clipboard => "#{element_in_clipboard.id}##{cell.name}"} cell.elements.count.should == 2 cell.elements.first.name.should == 'news' cell.elements.first.should_not == element end end end end context "with paste_from_clipboard in parameters" do render_views before do clipboard['elements'] = [{'id' => element_in_clipboard.id.to_s, 'action' => 'cut'}] end it "should create an element from clipboard" do xhr :post, :create, {paste_from_clipboard: element_in_clipboard.id, element: {page_id: alchemy_page.id}} response.status.should == 200 response.body.should match(/Successfully added new element/) end context "and with cut as action parameter" do it "should also remove the element id from clipboard" do xhr :post, :create, {paste_from_clipboard: element_in_clipboard.id, element: {page_id: alchemy_page.id}} session[:alchemy_clipboard]['elements'].detect { |item| item['id'] == element_in_clipboard.id.to_s }.should be_nil end end end context 'if element could not be saved' do subject { post :create, {element: {page_id: alchemy_page.id}} } before { Element.any_instance.stub(save: false) } it "renders the new template" do expect(subject).to render_template(:new) end end end describe '#find_or_create_cell' do before do Cell.stub(:definition_for).and_return({'name' => 'header', 'elements' => ['header']}) controller.instance_variable_set(:@page, alchemy_page) end context "with element name and cell name in the params" do before do controller.stub(params: {element: {name: 'header#header'}}) end context "with cell not existing" do it "should create the cell" do expect { controller.send(:find_or_create_cell) }.to change(alchemy_page.cells, :count).from(0).to(1) end end context "with the cell already present" do before do create(:cell, page: alchemy_page, name: 'header') end it "should load the cell" do expect { controller.send(:find_or_create_cell) }.to_not change(alchemy_page.cells, :count) end end end context "with only the element name in the params" do before do controller.stub(params: {element: {name: 'header'}}) end it "should return nil" do controller.send(:find_or_create_cell).should be_nil end end context 'with cell definition not found' do before do controller.stub(params: {element: {name: 'header#header'}}) Cell.stub(definition_for: nil) end it "raises error" do expect { controller.send(:find_or_create_cell) }.to raise_error(CellDefinitionError) end end end describe '#update' do let(:page) { build_stubbed(:page) } let(:element) { build_stubbed(:element, page: page) } let(:contents_parameters) { ActionController::Parameters.new(1 => {ingredient: 'Title'}) } let(:element_parameters) { ActionController::Parameters.new(tag_list: 'Tag 1', public: false) } before do Element.stub(:find).and_return element controller.should_receive(:contents_params).and_return(contents_parameters) end it "updates all contents in element" do element.should_receive(:update_contents).with(contents_parameters) xhr :put, :update, {id: element.id} end it "updates the element" do controller.should_receive(:element_params).and_return(element_parameters) element.should_receive(:update_contents).and_return(true) element.should_receive(:update_attributes!).with(element_parameters).and_return(true) xhr :put, :update, {id: element.id} end context "failed validations" do it "displays validation failed notice" do element.should_receive(:update_contents).and_return(false) xhr :put, :update, {id: element.id} assigns(:element_validated).should be_false end end end describe 'params security' do context "contents params" do let(:parameters) { ActionController::Parameters.new(contents: {1 => {ingredient: 'Title'}}) } specify ":contents is required" do controller.params.should_receive(:fetch).and_return(parameters) controller.send :contents_params end specify "everything is permitted" do controller.should_receive(:params).and_return(parameters) parameters.should_receive(:fetch).and_return(parameters) parameters.should_receive(:permit!) controller.send :contents_params end end context "element params" do let(:parameters) { ActionController::Parameters.new(element: {public: true}) } specify ":element is required" do controller.params.should_receive(:require).with(:element).and_return(parameters) controller.send :element_params end specify ":public and :tag_list is permitted" do controller.should_receive(:params).and_return(parameters) parameters.should_receive(:require).with(:element).and_return(parameters) parameters.should_receive(:permit).with(:public, :tag_list) controller.send :element_params end end end describe '#trash' do subject { xhr :delete, :trash, {id: element.id} } let(:element) { build_stubbed(:element) } before { Element.stub(find: element) } it "trashes the element instead of deleting it" do element.should_receive(:trash!).and_return(true) subject end end describe '#fold' do subject { xhr :post, :fold, {id: element.id} } let(:element) { build_stubbed(:element) } before do element.stub(save: true) Element.stub(find: element) end context 'if element is folded' do before { element.stub(folded: true) } it "sets folded to false." do element.should_receive(:folded=).with(false).and_return(true) subject end end context 'if element is not folded' do before { element.stub(folded: false) } it "sets folded to true." do element.should_receive(:folded=).with(true).and_return(true) subject end end end end end