Add featured hashtags to profiles (#9755)
* Add hashtag filter to profiles GET /@:username/tagged/:hashtag GET /api/v1/accounts/:id/statuses?tagged=:hashtag * Display featured hashtags on public profile * Use separate model for featured tags * Update featured hashtag counters on-write * Limit featured tags to 10
This commit is contained in:
		| @ -55,5 +55,6 @@ module AccountAssociations | ||||
|  | ||||
|     # Hashtags | ||||
|     has_and_belongs_to_many :tags | ||||
|     has_many :featured_tags, -> { includes(:tag) }, dependent: :destroy, inverse_of: :account | ||||
|   end | ||||
| end | ||||
|  | ||||
							
								
								
									
										46
									
								
								app/models/featured_tag.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								app/models/featured_tag.rb
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | ||||
| # frozen_string_literal: true | ||||
| # == Schema Information | ||||
| # | ||||
| # Table name: featured_tags | ||||
| # | ||||
| #  id             :bigint(8)        not null, primary key | ||||
| #  account_id     :bigint(8) | ||||
| #  tag_id         :bigint(8) | ||||
| #  statuses_count :bigint(8)        default(0), not null | ||||
| #  last_status_at :datetime | ||||
| #  created_at     :datetime         not null | ||||
| #  updated_at     :datetime         not null | ||||
| # | ||||
|  | ||||
| class FeaturedTag < ApplicationRecord | ||||
|   belongs_to :account, inverse_of: :featured_tags, required: true | ||||
|   belongs_to :tag, inverse_of: :featured_tags, required: true | ||||
|  | ||||
|   delegate :name, to: :tag, allow_nil: true | ||||
|  | ||||
|   validates :name, presence: true | ||||
|   validate :validate_featured_tags_limit, on: :create | ||||
|  | ||||
|   def name=(str) | ||||
|     self.tag = Tag.find_or_initialize_by(name: str.delete('#').mb_chars.downcase.to_s) | ||||
|   end | ||||
|  | ||||
|   def increment(timestamp) | ||||
|     update(statuses_count: statuses_count + 1, last_status_at: timestamp) | ||||
|   end | ||||
|  | ||||
|   def decrement(deleted_status_id) | ||||
|     update(statuses_count: [0, statuses_count - 1].max, last_status_at: account.statuses.where(visibility: %i(public unlisted)).tagged_with(tag).where.not(id: deleted_status_id).select(:created_at).first&.created_at) | ||||
|   end | ||||
|  | ||||
|   def reset_data | ||||
|     self.statuses_count = account.statuses.where(visibility: %i(public unlisted)).tagged_with(tag).count | ||||
|     self.last_status_at = account.statuses.where(visibility: %i(public unlisted)).tagged_with(tag).select(:created_at).first&.created_at | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def validate_featured_tags_limit | ||||
|     errors.add(:base, I18n.t('featured_tags.errors.limit')) if account.featured_tags.count >= 10 | ||||
|   end | ||||
| end | ||||
| @ -14,6 +14,7 @@ class Tag < ApplicationRecord | ||||
|   has_and_belongs_to_many :accounts | ||||
|   has_and_belongs_to_many :sample_accounts, -> { searchable.discoverable.popular.limit(3) }, class_name: 'Account' | ||||
|  | ||||
|   has_many :featured_tags, dependent: :destroy, inverse_of: :tag | ||||
|   has_one :account_tag_stat, dependent: :destroy | ||||
|  | ||||
|   HASHTAG_NAME_RE = '[[:word:]_]*[[:alpha:]_·][[:word:]_]*' | ||||
| @ -23,6 +24,7 @@ class Tag < ApplicationRecord | ||||
|  | ||||
|   scope :discoverable, -> { joins(:account_tag_stat).where(AccountTagStat.arel_table[:accounts_count].gt(0)).where(account_tag_stats: { hidden: false }).order(Arel.sql('account_tag_stats.accounts_count desc')) } | ||||
|   scope :hidden, -> { where(account_tag_stats: { hidden: true }) } | ||||
|   scope :most_used, ->(account) { joins(:statuses).where(statuses: { account: account }).group(:id).order(Arel.sql('count(*) desc')) } | ||||
|  | ||||
|   delegate :accounts_count, | ||||
|            :accounts_count=, | ||||
|  | ||||
		Reference in New Issue
	
	Block a user