Rails - 设计/作为可标记的宝石 - 使标签所有权对用户来说是唯一的吗?

时间:2016-03-03 14:45:31

标签: ruby-on-rails ruby ruby-on-rails-4 devise acts-as-taggable-on

用户可以登录(devise gem),创建文本帖子,&标记该帖子(通过acts-as-taggable-on gem)。如果在帖子中多次使用标记,则该标记的CSS会发生变化。例如,在下面的示例中,#foobar上的下划线样式消失了。但是,就目前而言,这是在用户帐户中发生的。这意味着用户输入标记其帖子的标签并非其帐户所独有。换句话说,如果 User1 标记 #foobar ,之后,在另一个帐户中, User2 也会标记 #foobar - 两个用户帐户的下划线都消失了。我的目标是让用户在该用户独有的发布表单中输入标签。

示例:

User1 帐户

http://imgur.com/VnnO7IQ

User2 帐户

http://imgur.com/29gzutj

我尝试了很多解决方案 - 其中没有一个正常工作!请帮忙!

posts_controller.rb

class PostsController < ApplicationController

 def index    
    if params[:tag]
     @posts = current_user.posts.tagged_with(params[:tag]).order(created_at: :desc)
    else 
     @posts = current_user.posts.includes(:things).all.order(created_at: :desc)
    end 
 end 

 def new
    @post = current_user.posts.build
 end

 def create
    @post = current_user.posts.build(post_params)
    if @post.save
        redirect_to '/posts'
    else
        render 'new'
    end

end

new.html.erb

<button id='addtext'>text</button>

<%= form_for @post, html: { multipart: true } do |f| %>

<%= f.text_field :tag_list %>
 <%= f.fields_for :things do |ff| %>
<% end %> 

<%= f.submit %>
<% end %>

index.html.erb

<div id="posts">
  <%= render @posts %>
</div>

_post.html.erb

<div class="post">
   <%= raw post.tags.order("taggings_count DESC").map {|t| link_to t.name, tag_path(t.name), class: css_class_for_tag(t.taggings_count)}.join(' ') %> 

   <% post.things.each do |thing| %>
     <%= thing.try(:text) %>
   <% end %>
</div>

post_helper.rb

module PostsHelper
 def css_class_for_tag count
  case count
   when 1 
     'new_tag'
   else
    'used_tag' 
   end
 end
end

发布模型

class Post < ActiveRecord::Base
  acts_as_taggable
  ActsAsTaggableOn.delimiter = ' '
  ActsAsTaggableOn.force_lowercase = true
  belongs_to :user 
  has_many :things, dependent: :destroy
  accepts_nested_attributes_for :things, allow_destroy: true 
end

事物模型

class Thing < ActiveRecord::Base
  belongs_to :post
  has_attached_file :image, styles: { small: "300x300>", medium: "600x600>" }
  validates_attachment_content_type :image, content_type: /\Aimage\/.*\Z/
end

用户模型

class User < ActiveRecord::Base
  acts_as_tagger
  has_many :posts 
  devise :database_authenticatable, :registerable,
     :recoverable, :rememberable, :trackable, :validatable
end

架构

ActiveRecord::Schema.define(version: 20160303110608) do

create_table "posts", force: :cascade do |t|
 t.datetime "created_at", null: false
 t.datetime "updated_at", null: false
 t.integer  "user_id"
end

create_table "taggings", force: :cascade do |t|
 t.integer  "tag_id"
 t.integer  "taggable_id"
 t.string   "taggable_type"
 t.integer  "tagger_id"
 t.string   "tagger_type"
 t.string   "context",       limit: 128
 t.datetime "created_at"
end

add_index "taggings", ["tag_id", "taggable_id", "taggable_type", "context", "tagger_id", "tagger_type"], name: "taggings_idx", unique: true
add_index "taggings", ["taggable_id", "taggable_type", "context"], name: "index_taggings_on_taggable_id_and_taggable_type_and_context"

create_table "tags", force: :cascade do |t|
 t.string  "name"
 t.integer "taggings_count", default: 0
end

add_index "tags", ["name"], name: "index_tags_on_name", unique: true

create_table "things", force: :cascade do |t|
 t.text     "text"
 t.integer  "post_id"
 t.datetime "created_at",         null: false
 t.datetime "updated_at",         null: false
 t.integer  "order"
end

create_table "users", force: :cascade do |t|
 t.string   "email",                  default: "", null: false
 t.string   "encrypted_password",     default: "", null: false
 t.string   "reset_password_token"
 t.datetime "reset_password_sent_at"
 t.datetime "remember_created_at"
 t.integer  "sign_in_count",          default: 0,  null: false
 t.datetime "current_sign_in_at"
 t.datetime "last_sign_in_at"
 t.string   "current_sign_in_ip"
 t.string   "last_sign_in_ip"
 t.datetime "created_at",                          null: false
 t.datetime "updated_at",                          null: false
end

add_index "users", ["email"], name: "index_users_on_email", unique: true
add_index "users", ["reset_password_token"], name: "index_users_on_reset_password_token", unique: true

end

2 个答案:

答案 0 :(得分:1)

问题是,只要taggings_count使用任何标记,tags表上的tagger属性就会递增。所以你需要提出一个新的查询。

您可以通过选择标记表来获取您要查找的数据,进行一些表连接,这样您就可以选择由特定用户制作的标记,并从中获取您的计数寻找。

我写了一个查询来演示你需要的结构:

 class Post < ActiveRecord::Base
   acts_as_taggable
   ActsAsTaggableOn.delimiter = ' '
   ActsAsTaggableOn.force_lowercase = true
   belongs_to :user
   has_many :things, dependent: :destroy
   accepts_nested_attributes_for :things, allow_destroy: true

   def tags_with_user_specific_counts
     query = <<-SQL
       SELECT taggings.tag_id,
              tags.name,
              count(taggings.id)
         FROM taggings
         INNER JOIN tags
           ON taggings.tag_id = tags.id
         WHERE taggings.tagger_id = '#{user_id}'
           AND taggings.tagger_type = 'User'
           AND taggings.taggable_id = '#{id}'
           AND taggings.taggable_type = 'Post'
         GROUP BY taggings.tag_id, tags.name
     SQL
     ActiveRecord::Base.connection.execute(query).to_a
   end
 end

唯一需要注意的是我没有使用ActiveRecord方式编写这个,所以这会产生一个包含所需数据的哈希数组:

[{"tag_id"=>"1", "name"=>"jim", "count"=>"1"},
 {"tag_id"=>"2", "name"=>"bob", "count"=>"2"}]

但如果你真的希望这回复ActsAsTaggableOn::Tag,我认为这是可能的。

编辑 - 此帖子已更新为使用taggings.tagger_id运行查询。此帖子的先前版本建议运行此查询:

   def tags_with_user_specific_counts
     query = <<-SQL
       SELECT taggings.tag_id,
              tags.name,
              users.id as user_id,
              count(taggings.id)
         FROM taggings
         INNER JOIN tags
           ON taggings.tag_id = tags.id
         INNER JOIN posts
           ON taggings.taggable_id = posts.id
           AND taggings.taggable_type = 'Post'
         INNER JOIN users
           ON posts.user_id = users.id
         WHERE users.id = '#{user_id}'
         GROUP BY taggings.tag_id, tags.name, users.id
     SQL
     ActiveRecord::Base.connection.execute(query).to_a
   end

答案 1 :(得分:1)

一种方法:

tag_counts = ActsAsTaggableOn::Tag.joins(:taggings).
  where(taggings: { taggable_type: "Post", taggable_id: current_user.post_ids }).
  group("tags.id").count

给你这样的东西:

{ 7=>2, 1=>2, 5=>4 }

第一个数字是标签ID,第二个数字是current_user的标签出现次数。

现在你可以这样做:

<div class="post">
  <%= raw post.tags.map{ |t| [t.name, tag_counts[t.id]] }.sort_by{ |t| -t[1] }.
                    map{ |t| link_to t[0], tag_path(t[0]), class: css_class_for_tag(t[1])}.join(' ') %>
</div> 

这可以写得更有表现力,但它比...更长;;-) 基本上我们在post的标签上循环,就像我们之前做的那样,但是然后在第一个map语句中将每个标签与特定于current_user的计数连接起来。然后我们按计数按相反的顺序排序。在最终的map语句中,我们像之前一样生成链接,只是再次使用用户特定的计数。