Emi Matchu
7f62417294
Before this change, the sort order when searching for "Prismatic Pine: Nostalgic" showed: - [Added Dec 18, 2024] Prismatic Pine: Nostalgic Christmas Flotsam - [Added Dec 19, 2024] Prismatic Pine: Nostalgic Christmas Gelert - [Added Dec 18, 2024] Prismatic Pine: Nostalgic Christmas Bruce - [Added Dec 17, 2024] Prismatic Pine: Nostalgic Christmas Scorchio - <more> This is because the Gelert was created at 11:37 NST on Dec 19, whereas the Flotsam was created at 18:11 NST on Dec 18—but in UTC, which is how timestamps are stored in the database, these are both Dec 19, so the Flotsam was sorting first alphabetically. In this change, we do a hacky transform from UTC to NST-ish. I didn't want to set up the deploy process to pull named time zones into MySQL, and then have this as a potential gotcha for the dev environment later—so instead, I pretend `-08:00` is a good-enough specification of NST. This will sometimes create slightly incorrect sort ordering when it *is* Daylight Savings, and a record was created around midnight. I'm okay with that!
126 lines
4.1 KiB
Ruby
126 lines
4.1 KiB
Ruby
require "addressable/template"
|
|
|
|
class AltStyle < ApplicationRecord
|
|
belongs_to :species
|
|
belongs_to :color
|
|
|
|
has_many :parent_swf_asset_relationships, as: :parent, dependent: :destroy
|
|
has_many :swf_assets, through: :parent_swf_asset_relationships
|
|
has_many :contributions, as: :contributed, inverse_of: :contributed
|
|
|
|
validates :body_id, presence: true
|
|
validates :series_name, presence: true, allow_nil: true
|
|
validates :thumbnail_url, presence: true
|
|
|
|
before_validation :infer_thumbnail_url, unless: :thumbnail_url?
|
|
|
|
scope :matching_name, ->(series_name, color_name, species_name) {
|
|
color = Color.find_by_name!(color_name)
|
|
species = Species.find_by_name!(species_name)
|
|
where(series_name:, color_id: color.id, species_id: species.id)
|
|
}
|
|
scope :by_creation_date, -> {
|
|
# HACK: Setting up named time zones in MySQL takes effort, so we assume
|
|
# it's not Daylight Savings. This will produce slightly incorrect
|
|
# sorting when it *is* Daylight Savings, and records happen to be
|
|
# created around midnight.
|
|
order(Arel.sql("DATE(CONVERT_TZ(created_at, '+00:00', '-08:00')) DESC"))
|
|
}
|
|
scope :unlabeled, -> { where(series_name: nil) }
|
|
scope :newest, -> { order(created_at: :desc) }
|
|
|
|
def pet_name
|
|
I18n.translate('pet_types.human_name', color_human_name: color.human_name,
|
|
species_human_name: species.human_name)
|
|
end
|
|
|
|
alias_method :name, :pet_name
|
|
|
|
# If the series_name hasn't yet been set manually by support staff, show the
|
|
# string "<New?>" instead. But it won't be searchable by that string—that is,
|
|
# `fits:<New?>-faerie-draik` intentionally will not work, and the canonical
|
|
# filter name will be `fits:alt-style-IDNUMBER`, instead.
|
|
def series_name
|
|
real_series_name || AltStyle.placeholder_name
|
|
end
|
|
|
|
def real_series_name=(new_series_name)
|
|
self[:series_name] = new_series_name
|
|
end
|
|
|
|
def real_series_name
|
|
self[:series_name]
|
|
end
|
|
|
|
# You can use this to check whether `series_name` is returning the actual
|
|
# value or its placeholder value.
|
|
def real_series_name?
|
|
real_series_name.present?
|
|
end
|
|
|
|
def adjective_name
|
|
"#{series_name} #{color.human_name}"
|
|
end
|
|
|
|
def full_name
|
|
"#{series_name} #{name}"
|
|
end
|
|
|
|
EMPTY_IMAGE_URL = ""
|
|
def preview_image_url
|
|
# Use the image URL for the first asset. Or, fall back to an empty image.
|
|
swf_assets.first&.image_url || EMPTY_IMAGE_URL
|
|
end
|
|
|
|
# Given a list of items, return how they look on this alt style.
|
|
def appearances_for(items, ...)
|
|
Item.appearances_for(items, self, ...)
|
|
end
|
|
|
|
# At time of writing, most batches of Alt Styles thumbnails used a simple
|
|
# pattern for the item thumbnail URL, but that's not always the case anymore.
|
|
# For now, let's keep using this format as the default value when creating a
|
|
# new Alt Style, but the database field can be manually overridden as needed!
|
|
THUMBNAIL_URL_TEMPLATE = Addressable::Template.new(
|
|
"https://images.neopets.com/items/{series}_{color}_{species}.gif"
|
|
)
|
|
DEFAULT_THUMBNAIL_URL = "https://images.neopets.com/items/mall_bg_circle.gif"
|
|
def infer_thumbnail_url
|
|
if real_series_name?
|
|
self.thumbnail_url = THUMBNAIL_URL_TEMPLATE.expand(
|
|
series: series_name.gsub(/\s+/, '').downcase,
|
|
color: color.name.gsub(/\s+/, '').downcase,
|
|
species: species.name.gsub(/\s+/, '').downcase,
|
|
).to_s
|
|
else
|
|
self.thumbnail_url = DEFAULT_THUMBNAIL_URL
|
|
end
|
|
end
|
|
|
|
def real_thumbnail_url?
|
|
thumbnail_url != DEFAULT_THUMBNAIL_URL
|
|
end
|
|
|
|
def self.placeholder_name
|
|
"<New?>"
|
|
end
|
|
|
|
def self.all_series_names
|
|
distinct.where.not(series_name: nil).pluck(:series_name).sort
|
|
end
|
|
|
|
def self.all_supported_colors
|
|
Color.find(distinct.pluck(:color_id))
|
|
end
|
|
|
|
def self.all_supported_species
|
|
Species.find(distinct.pluck(:species_id))
|
|
end
|
|
|
|
# For convenience in the console!
|
|
def self.find_by_name(color_name, species_name)
|
|
color = Color.find_by_name(color_name)
|
|
species = Species.find_by_name(species_name)
|
|
where(color_id: color, species_id: species).first
|
|
end
|
|
end
|