Как я могу использовать тег "форма", а не "форма" в этом файле


Я новичок в 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
5   28  

5 ответов:

Я только что завершил это упражнение, так что я ни в коем случае не эксперт; однако это код, который работал на меня и прошел все тесты:

../приложение/просмотров/сессий/новый.формат 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 / controllers/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' новый.формат 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>

И исходный sessions_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

Я работаю над тем же этапом урока, и ваш вопрос помог мне найти свой путь.

Использование label и 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

Руководства RoR описывают, как работает form_tag.

Http://guides.rubyonrails.org/form_helpers.html

Меня поймали на том, что я не назвал поля правильно. Когда вы просматриваете исходный код оригинала, он показывает вам схему именования.

<%= 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 -%>

У вас есть 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, Когда вы удалите его тесты пройдут.