2017-09-19 10:42:40 +10:00
|
|
|
# frozen_string_literal: true
|
2023-02-20 16:58:28 +11:00
|
|
|
|
2017-09-19 10:42:40 +10:00
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: custom_emojis
|
|
|
|
#
|
2020-04-27 07:29:08 +10:00
|
|
|
# id :bigint(8) not null, primary key
|
|
|
|
# shortcode :string default(""), not null
|
|
|
|
# domain :string
|
|
|
|
# image_file_name :string
|
|
|
|
# image_content_type :string
|
|
|
|
# image_file_size :integer
|
|
|
|
# image_updated_at :datetime
|
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
|
|
|
# disabled :boolean default(FALSE), not null
|
|
|
|
# uri :string
|
|
|
|
# image_remote_url :string
|
|
|
|
# visible_in_picker :boolean default(TRUE), not null
|
|
|
|
# category_id :bigint(8)
|
|
|
|
# image_storage_schema_version :integer
|
2017-09-19 10:42:40 +10:00
|
|
|
#
|
|
|
|
|
|
|
|
class CustomEmoji < ApplicationRecord
|
2021-09-30 07:52:36 +10:00
|
|
|
include Attachmentable
|
|
|
|
|
2022-07-10 06:07:17 +10:00
|
|
|
LIMIT = 256.kilobytes
|
2018-03-26 23:02:10 +11:00
|
|
|
|
2017-09-19 10:42:40 +10:00
|
|
|
SHORTCODE_RE_FRAGMENT = '[a-zA-Z0-9_]{2,}'
|
|
|
|
|
|
|
|
SCAN_RE = /(?<=[^[:alnum:]:]|\n|^)
|
|
|
|
:(#{SHORTCODE_RE_FRAGMENT}):
|
|
|
|
(?=[^[:alnum:]:]|$)/x
|
2022-11-10 15:49:30 +11:00
|
|
|
SHORTCODE_ONLY_RE = /\A#{SHORTCODE_RE_FRAGMENT}\z/
|
2017-09-19 10:42:40 +10:00
|
|
|
|
2022-05-28 04:06:40 +10:00
|
|
|
IMAGE_MIME_TYPES = %w(image/png image/gif image/webp).freeze
|
2019-08-09 07:03:09 +10:00
|
|
|
|
2019-06-28 23:54:10 +10:00
|
|
|
belongs_to :category, class_name: 'CustomEmojiCategory', optional: true
|
2023-04-30 22:06:53 +10:00
|
|
|
|
2023-12-02 02:52:47 +11:00
|
|
|
has_one :local_counterpart, -> { where(domain: nil) }, class_name: 'CustomEmoji', primary_key: :shortcode, foreign_key: :shortcode, inverse_of: false, dependent: nil
|
2017-11-08 00:49:32 +11:00
|
|
|
|
2024-06-07 19:27:59 +10:00
|
|
|
has_attached_file :image, styles: { static: { format: 'png', convert_options: '-coalesce +profile "!icc,*" +set date:modify +set date:create +set date:timestamp', file_geometry_parser: FastGeometryParser } }, validate_media_type: false, processors: [:lazy_thumbnail]
|
2017-09-19 10:42:40 +10:00
|
|
|
|
2024-01-08 22:20:59 +11:00
|
|
|
normalizes :domain, with: ->(domain) { domain.downcase }
|
2018-12-11 15:30:57 +11:00
|
|
|
|
2019-08-09 07:03:09 +10:00
|
|
|
validates_attachment :image, content_type: { content_type: IMAGE_MIME_TYPES }, presence: true, size: { less_than: LIMIT }
|
2022-11-10 15:49:30 +11:00
|
|
|
validates :shortcode, uniqueness: { scope: :domain }, format: { with: SHORTCODE_ONLY_RE }, length: { minimum: 2 }
|
2017-09-19 10:42:40 +10:00
|
|
|
|
2019-09-14 00:01:09 +10:00
|
|
|
scope :local, -> { where(domain: nil) }
|
|
|
|
scope :remote, -> { where.not(domain: nil) }
|
2024-09-10 23:21:40 +10:00
|
|
|
scope :enabled, -> { where(disabled: false) }
|
2017-10-06 08:42:05 +11:00
|
|
|
scope :alphabetic, -> { order(domain: :asc, shortcode: :asc) }
|
2022-08-26 04:39:40 +10:00
|
|
|
scope :by_domain_and_subdomains, ->(domain) { where(domain: domain).or(where(arel_table[:domain].matches("%.#{domain}"))) }
|
2024-09-10 23:21:40 +10:00
|
|
|
scope :listed, -> { local.enabled.where(visible_in_picker: true) }
|
2017-09-23 09:57:23 +10:00
|
|
|
|
2018-03-26 23:02:10 +11:00
|
|
|
remotable_attachment :image, LIMIT
|
2017-09-19 10:42:40 +10:00
|
|
|
|
2018-04-27 09:38:10 +10:00
|
|
|
after_commit :remove_entity_cache
|
|
|
|
|
2017-10-06 08:42:05 +11:00
|
|
|
def local?
|
|
|
|
domain.nil?
|
|
|
|
end
|
|
|
|
|
2017-10-08 02:43:42 +11:00
|
|
|
def object_type
|
|
|
|
:emoji
|
|
|
|
end
|
|
|
|
|
2019-09-10 06:44:17 +10:00
|
|
|
def copy!
|
|
|
|
copy = self.class.find_or_initialize_by(domain: nil, shortcode: shortcode)
|
|
|
|
copy.image = image
|
2019-09-18 07:20:48 +10:00
|
|
|
copy.tap(&:save!)
|
2019-09-10 06:44:17 +10:00
|
|
|
end
|
|
|
|
|
2022-08-26 04:39:40 +10:00
|
|
|
def to_log_human_identifier
|
|
|
|
shortcode
|
|
|
|
end
|
|
|
|
|
2017-09-19 10:42:40 +10:00
|
|
|
class << self
|
2020-01-24 08:00:13 +11:00
|
|
|
def from_text(text, domain = nil)
|
2017-09-19 10:42:40 +10:00
|
|
|
return [] if text.blank?
|
2017-09-27 12:14:03 +10:00
|
|
|
|
|
|
|
shortcodes = text.scan(SCAN_RE).map(&:first).uniq
|
|
|
|
|
|
|
|
return [] if shortcodes.empty?
|
|
|
|
|
2018-04-27 09:38:10 +10:00
|
|
|
EntityCache.instance.emoji(shortcodes, domain)
|
2017-09-19 10:42:40 +10:00
|
|
|
end
|
2018-04-10 23:46:27 +10:00
|
|
|
|
|
|
|
def search(shortcode)
|
2024-01-09 00:26:12 +11:00
|
|
|
where(arel_table[:shortcode].matches("%#{sanitize_sql_like(shortcode)}%"))
|
2018-04-10 23:46:27 +10:00
|
|
|
end
|
2017-09-19 10:42:40 +10:00
|
|
|
end
|
2018-04-27 09:38:10 +10:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def remove_entity_cache
|
|
|
|
Rails.cache.delete(EntityCache.instance.to_key(:emoji, shortcode, domain))
|
|
|
|
end
|
2017-09-19 10:42:40 +10:00
|
|
|
end
|