Как я могу использовать "Form_tag" в отличие от "Form_for" в этом файле
Я новичок в Ruby on Rails и очень помог Майклу Хартлу отличная книга: Ruby on Rails Tutorial. Я перешел к главе 8, и теперь я занимаюсь упражнениями в этой главе. У меня есть (я предполагаю типичную "новичку" ) проблему с упражнением 1. В этом упражнении задается вопрос "1. Создать форму знака, чтобы использовать form_tag вместо form_for". Я попытался найти помощь в этом в Stackoverflow, Google, Railscast и многих других "веб-поисках" в течение двух дней, и я, похоже, не нашел помощи, которая мне нужна для ответа на эту проблему. Файл, который я пытаюсь изменить с помощью form_tag, приведен ниже:
<% provide(:title, "Sign in") %>
<h1>Sign in</h1>
<div class="row">
<div class="span6 offset3">
<%= form_for(:session, url: sessions_path) do |f| %>
<%= f.label :email %>
<%= f.text_field :email %>
<%= f.label :password %>
<%= f.password_field :password %>
<%= f.submit "Sign in", class: "btn btn-large btn-primary" %>
<% end %>
<p>New user? <%= link_to "Sign up now!", signup_path %></p>
</div>
</div>
Я использую Rails 3.2.3 в этом приложении. Может кто-нибудь указать мне в правильном направлении? Может ли кто-нибудь помочь мне с этой проблемой? Я был бы очень благодарен.
Это реализация, использующая form_tag:
<% provide(:title, "Sign in") %>
<h1>Sign in</h1>
<div class="row">
<div class="span6 offset3">
<%= form_tag( url: sessions_path ) do %>
<%= label_tag :email %>
<%= text_field_tag :email %>
<%= label_tag :password %>
<%= password_field_tag :password %>
<%= submit_tag "Sign in", class: "btn btn-large btn-primary" %>
<% end %>
<p>New user? <%= link_to "Sign up now!", signup_path %></p>
</div>
</div>
Я использую Rspec 2.9.0 и ниже - неудачные тесты:
describe "signin page" do
before { visit signin_path }
it { should have_selector('h1', text: 'Sign in') }
it { should have_selector('title', text: 'Sign in') }
end
и
describe "with invalid information" do
before { click_button "Sign in" }
it { should have_selector('title', text: 'Sign in') }
it { should have_selector('div.alert.alert-error', text: 'Invalid') }
describe "after visiting another page" do
before { click_link "Home" }
it { should_not have_selector('div.alert.alert-error') }
end
end
и
describe "with valid information" do
let(:user) { FactoryGirl.create(:user) }
before do
fill_in "Email", with: user.email
fill_in "Password", with: user.password
click_button "Sign in"
end
it { should have_selector('title', text: user.name) }
it { should have_link('Profile', href: user_path(user)) }
it { should have_link('Sign out', href: signout_path) }
it { should_not have_link('Sign in', href: signin_path) }
describe "followed by signout" do
before { click_link "Sign out" }
it { should have_link('Sign in') }
end
end
Здесь мой файл маршрутов:
SampleApp::Application.routes.draw do
resources :users
resources :sessions, only: [:new, :create, :destroy]
get "users/new"
root to: 'static_pages#home'
match '/signup', to: 'users#new'
match '/signin', to: 'sessions#new'
match '/signout', to: 'sessions#destroy', via: :delete
match '/help', to: 'static_pages#help'
match '/about', to: 'static_pages#about'
match '/contact', to: 'static_pages#contact'
end
Ответы
Ответ 1
Я только что закончил это упражнение, поэтому я ни в коем случае не специалист; однако это код, который работал у меня и прошел все тесты:
../приложение/просмотров/сессии/new.html.erb
<% provide(:title, "Sign in") %>
<h1>Sign in</h1>
<div class="row">
<div class="span 6 offset 3">
<%= form_tag sessions_path do %>
<%= label_tag :email %>
<%= text_field_tag :email, params[:email] %>
<%= label_tag :password %>
<%= password_field_tag :password %>
<%= submit_tag "Sign in", class: "btn btn-large btn-primary" %>
<% end %>
<p>New User?<%= link_to "Sign Up Now", signup_path %> </p>
</div>
</div>
Мне также нужно было изменить.. /app/controller/sessions _contoller
class SessionsController < ApplicationController
def new
end
def create
user = User.find_by_email(params[:email])
if user && user.authenticate(params[:password])
session[:user] = user.id
sign_in user
redirect_to user
else
flash.now[:error] = 'Invalid email/password combination'
render 'new'
end
end
def destroy
sign_out
redirect_to root_path
end
end
Пока это работает, я не совсем уверен, почему он это делает; если кто-то может объяснить, почему требуются изменения в контроллере, это было бы высоко оценено. Я знаю, что это может быть задано отдельным вопросом, но оно тесно связано с OP, и я уверен, что мы оба найдем его чрезвычайно полезным в понимании не только того, как заставить это работать, но и почему он работает таким образом. Ниже приведены исходные файлы вида и контроллера:
Оригинал 'form_for' new.html.erb:
<% provide(:title, "Sign in") %>
<h1>Sign in</h1>
<div class="row">
<div class="span6 offset3">
<%= form_for(:session, url: sessions_path) do |f| %>
<%= f.label :email %>
<%= f.text_field :email %>
<%= f.label :password %>
<%= f.password_field :password %>
<%= f.submit "Sign in", class: "btn btn-large btn-primary" %>
<% end %>
<p>New user? <%= link_to "Sign up now!", signup_path %></p>
</div>
</div>
и исходный session_controller:
class SessionsController < ApplicationController
def new
end
def create
user = User.find_by_email(params[:session][:email])
if user && user.authenticate(params[:session][:password])
sign_in user
redirect_to user
else
flash.now[:error] = 'Invalid email/password combination'
render 'new'
end
end
def destroy
sign_out
redirect_to root_path
end
end
Ответ 2
Я работаю над тем же шагом учебника, и ваш вопрос помог мне найти мой путь.
Использование метки и text_field вместо label_tag и text_field_tag работает нормально, и вам не нужно менять код контроллера (это создает тот же HTML-код код как с исходным методом form_for).
<%= form_tag(sessions_path) do %>
<%= label :session, :email %>
<%= text_field :session, :email %>
<%= label :session, :password %>
<%= password_field :session, :password %>
<%= submit_tag("Sign in", class: "btn btn-large btn-primary") %>
<% end %>
Подробности можно найти в http://guides.rubyonrails.org/form_helpers.html#dealing-with-model-objects
Ответ 3
В руководствах RoR описано, как работает form_tag.
http://guides.rubyonrails.org/form_helpers.html
Ответ 4
У вас есть Session_Helper, который вы не используете в своей реализации.
ИЗМЕНИТЬ ваши вспомогательные методы.
def sign_in(user)
cookies.permanent[:remember_token] = user.remember_token
session[:user_id] = user.id
end
def sign_out
cookies.delete(:remember_token)
session[:user_id] = nil
end
Затем вы можете просто использовать соответствующие методы в своем контроллере сеанса. Это более аккуратная реализация, которая следует за модульным подходом к дизайну. Также есть проблемы с rspec.
describe "with valid information" do
let(:user) { FactoryGirl.create(:user) }
before do
fill_in "Email", with: user.email.upcase
fill_in "Password", with: user.password
click_button "Sign in"
end
it { should have_selector('title', text: user.name) }
it { should have_link('Profile', href: user_path(user)) }
it { should have_link('Sign out', href: signout_path) }
it { should_not have_link('Sign in', href: signin_path) }
Несмотря на то, что при посещении страницы все правильно, эти тесты отмечены значком. Я предполагаю, что вам нужно изменить before do
, потому что если он делает то, что вы хотите, чтобы выполнить тесты, будет проходить.
В любом случае, мне любопытно, что проблема с rspec, если у кого-то есть идея!
Я думаю, что проблема связана с user.email.upcase
. Вам не нужно .upcase
, когда вы удалите его, тесты пройдут.
Ответ 5
Я поймал, не назвав поля правильно. Когда вы просматриваете исходный код оригинала, он показывает схему именования.
<%= form_tag(sessions_path) do %>
<%= label_tag 'session_email', 'Email' %>
<%= text_field_tag 'session[email]' %>
<%= label_tag 'session_password', 'Password' %>
<%= password_field_tag 'session[password]' %>
<%= submit_tag "Sign in", class: "btn btn-large btn-primary" %>
<% end -%>