Подтвердить что ты не робот

Как я могу использовать "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
4b9b3361

Ответ 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

Ответ 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 -%>