用户可以登录(devise gem),创建文本帖子,&标记该帖子(通过acts-as-taggable-on gem)。如果在帖子中多次使用标记,则该标记的CSS会发生变化。例如,在下面的示例中,#foobar上的下划线样式消失了。但是,就目前而言,这是在用户帐户中发生的。这意味着用户输入标记其帖子的标签并非其帐户所独有。换句话说,如果 User1 标记 #foobar ,之后,在另一个帐户中, User2 也会标记 #foobar - 两个用户帐户的下划线都消失了。我的目标是让用户在该用户独有的发布表单中输入标签。
示例:
User1 帐户
User2 帐户
我尝试了很多解决方案 - 其中没有一个正常工作!请帮忙!
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
答案 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语句中,我们像之前一样生成链接,只是再次使用用户特定的计数。