2

我正在尝试实现一个简单的like/unlike 函数。我在这里看到的所有示例似乎都适用于 ajax 或 jquery。我仍然是初学者,我也不完全理解,我只想要一个简单的解决方案。

我的想法是,我有书,我有用户。用户可以喜欢书籍。所以我通过 Like 模型创建了一个多对多关联。Like 模型有一个对应的数据库,其中包含 book_id 和 user_id 列。因此:

class Book < ActiveRecord::Base
  has_many :likes
  has_many :users, through: :likes

class User < ActiveRecord::Base 
  has_many :likes
  has_many :books, through: :likes

class Like < ActiveRecord::Base
  belongs_to :book
  belongs_to :user
end

不幸的是,这是我的理解。我不知道如何利用这种关系来创建喜欢并将它们与适当的书籍和用户相关联。

我的想法是如果他不喜欢这本书,或者如果他不喜欢,就给用户一个“喜欢”按钮。所以基本上,我想要这样简单的东西:

<%  if user likes? %>

<div class="unlike_button"><%= link_to "Unlike", '#' %></div>

<% else %>

<div class="like_button"><%= link_to "Like", '#' %></div>

<% end %>

我使用了 # 存根,因为我不知道它应该使用什么路由。但无论解决方案是什么,我都希望它重定向到同一页面,并带有“喜欢”或“不喜欢”的快速通知。我已经有like_button 和like_button 类的div 背景图像,这就是我将它们实现为上面的图像链接的原因。

任何形式的指导或帮助,将不胜感激。谢谢

更新

我在下面遵循 Bennick 的指导,但我仍然卡在 Rails 控制台中。我认为如果我在控制台中遇到错误,那么继续前进是没有意义的。

正如建议的那样,我在控制台中尝试了这个:

因为我已经有用户user = User.find(1)book = Book.find(1)

但是在下一行 like = Like.create(:user => user, :book => book)返回了一个质量分配错误。Can't mass-assign protected attributes: book, user 我想也许这会对attr_accessible :book_id, :user_id类似的模型有所帮助,但我仍然会遇到错误。我错过了什么吗?

解决了

我终于让它工作了!使用like = Like.create(:user => user.id, :book => book.id).

4

1 回答 1

6

好的,这里有很多项目。我将带你上堆栈模型->关联->控制器->视图->路由器)。通常,当您设计 Web 应用程序时,您会从数据库层开始,然后逐步向上。所以我们会在这里做。

模型

这是您决定需要哪些数据库对象并创建数据库表来表示它们的地方。如果您还没有,请阅读 Rails 迁移指南:http: //guides.rubyonrails.org/migrations.html

在您的情况下,此设置将是合适的:

class Book < ActiveRecord::Base
  attr_accessible :title

  has_many :likes
  has_many :users, through: :likes
end

class User < ActiveRecord::Base
  attr_accessible :name

  has_many :likes
  has_many :books, through: :likes
end

class Like < ActiveRecord::Base
  attr_accessible :book, :user

  belongs_to :book
  belongs_to :user
end

请注意,我们需要包含attr_accessible,因此我们不会收到任何批量分配错误。请注意,在 Rails 4 中,此安全功能已移至控制器中。有关这方面的更多信息,请参阅这些或搜索互联网:http: //blog.teamtreehouse.com/rails-4-strong-parameters http://weblog.rubyonrails.org/2012/3/21/strong-parameters/

协会

您应该阅读有关关联的 Rails 指南:http: //guides.rubyonrails.org/association_basics.html

这将使您对数据库对象(Active Record 对象)如何相互交互有一个很好的了解。在您的问题中,您已经设置了这些。一旦建立关联,Rails 提供了许多用于访问它们的方法。这是一个示例 Rails 控制台会话:rails c

# Create a user
user = User.create(:name => "Ryan") # I'm assuming User just requires a name for simplicity
  => #<User id: 1, name: "Ryan">
# Create two a books
book = Book.create(:title => "Game of Thrones")
  => #<Book id: 1, title: "Game of Thrones">
book2 = Book.create(:title => "The Well-Grounded Rubyist")
  => #<Book id: 2, title: "The Well-Grounded Rubyist">
# Create a two likes from the books and the user record
like = Like.create(:user => user, :book => book)
  => #<Like id: 1, user_id: 1, book_id: 1>
like2 = Like.create(:user => user, :book => book2)
  => #<Like id: 2, user_id: 1, book_id: 2>
# Notice how the keys glue the associations

# Query a user's likes
user.likes.count
  => 2
user.likes
  => #<ActiveRecord::Associations::CollectionProxy [#<Like id: 1, user_id: 1, book_id: 1>, #<Like id: 2, user_id: 1, book_id: 2>]
# Query a user's books
user.books
  => #<ActiveRecord::Associations::CollectionProxy [#<Book id: 1, title: "Game of Thrones">, #<Book id: 1, title: "The Well-Grounded Rubyist">]

如有疑问,请使用 rails 控制台。你会从中学到很多东西。

控制器

为了让最终用户与您的数据库对象进行交互,需要一个控制器来促进交换。再次阅读相关的 Rails 指南: http: //guides.rubyonrails.org/action_controller_overview.html 如果您现在还没有猜到,我强烈建议您阅读其中的大部分内容。

在您的事业中,我们正在创建类似的对象,因此让我们制作一个喜欢的控制器:

rails g controller likes index

这将创建具有索引操作和视图文件的控制器。

# app/controllers/likes_controller.rb
class LikesController < ApplicationController

  # This action will show our likes for a user.
  # Lets assume you have an authentication system (ex Devise) that logs a user in and provides a `current_user` object
  # GET /likes
  def index
    # Assign the logged in user to @user
    @user = current_user
    # Grab all of the books and put them into an array in @books
    @books = Book.all
  end

  # This is our key action. We will use this action to create a Like
  # POST /likes
  def create
    # Grab our book from the DB. Note that this syntax is for Rails 3.2 and below. Rails 4 uses something called Strong Parameters, but that is for another time. 
    book = Book.find(params[:book_id])
    # Create a like
    Like.create(:book => book, :user => current_user)
    # redirect back to the Like index page and assign a flash
    redirect_to likes_path, :notice => "You just liked the book #{book.title}" 
  end

  # here is where we will destroy a Like
  # DELETE /likes/:id
  def destroy
    # Get the like form the DB
    like = Like.find(params[:id])
    # destroy it
    like.destroy
    redirect_to likes_path, :notice => "You destroyed a like"
  end
end

路由器

路由器将外部 http 请求连接到您的控制器操作。在您的情况下,您只需要这样:

# config/routers.rb
 MyApp::Application.routes.draw do

  resources :likes

 end

这是一个 Rails 快捷方式,它设置了 7 条标准路线以及相关的助手:

    likes GET    /likes(.:format)          likes#index
          POST   /likes(.:format)          likes#create
 new_like GET    /likes/new(.:format)      likes#new
edit_like GET    /likes/:id/edit(.:format) likes#edit
     like GET    /likes/:id(.:format)      likes#show
          PUT    /likes/:id(.:format)      likes#update
          DELETE /likes/:id(.:format)      likes#destroy

帮自己一个忙,阅读本指南: http: //guides.rubyonrails.org/routing.html 它将解释这些路线是什么以及它们是如何工作的。Rails 与大多数现代 Web 开发世界一样遵循 REST。

看法

在您看来,您将需要一个供用户交互的表单。此表单会将数据发送到应用程序,特别是您的 LikesController 操作。

# app/views/likes/index.html.erb

# show your flash messages
<% flash.each do |name, msg| %>
  <div class="alert <%= "alert-#{name}" %>">
    <%= msg %>
  </div>
<% end %>

<h1>Books you may or may not like</h1>
# For each book
<% @books.each do |book| %>
  <% unless @user.books.include?(book) %> # Prob want to move this into a User instance method
    # Create a like form if the user does not have a like for this book
    <%= form_tag likes_path do %>
      <%= hidden_field_tag 'book_id', book.id %>
      # Clicking this sends a request: POST /likes with params of: book_id=123
      <%= submit_tag "Like this book", :class => "like_button" %>
    <% end %>
  <% else %>
    # Find the like. I'll admit there is probably a better way to do this but it's getting past my bed time. 
    <% like = book.likes.where(:user_id => @user.id).first %>
    # Destroy the like associated with this book and user
    <div class="unlike_button">
      # Clicking this sends a request to: DELETE /likes/123
      <%= link_to "destroy like", likes_path(like.id), :method => :delete %>
    </div>
  <% end %>
<% end %>

结论

我希望这能给你一些指导。

将来尝试使您的问题更加具体,因为这个问题涵盖了很大的领域。我刚刚开始积极回馈,所以我可能做得过火了。刚开始时,我收到了大量免费指导和帮助。是我回报人情的时候了。

慢慢来,当您遇到错误时,只需将其发布到 Google。您可能最终会遇到 Stack Overflow 问题。

干杯!

于 2013-05-09T04:23:24.273 回答