1
0
Fork 1
impress/app/models/alt_style.rb

155 lines
5 KiB
Ruby
Raw Normal View History

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
2024-01-24 03:54:43 -08:00
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"))
}
Better ordering for NC Styles in the outfit editor Previously, when opening the pose picker and looking at Styles, the Cybunny options were sorted like this: - Default - Celebratory 25th Anniversary - Festive Christmas - Nostalgic Baby - Nostalgic Blue - Nostalgic Christmas - Nostalgic Darigan - Nostalgic Faerie - Nostalgic Grey - Nostalgic Maraquan - Nostalgic Mutant - Nostalgic Plushie - Nostalgic Robot - Nostalgic Royalboy - Nostalgic Royalgirl - Nostalgic Snow - Nostalgic Tyrannian - Prismatic Cocoa: Festive Christmas - Prismatic Cocoa: Nostalgic Christmas - Prismatic Tinsel: Festive Christmas - Prismatic Tinsel: Nostalgic Christmas - Spooky Halloween Now, they're sorted like this: - Default - Celebratory 25th Anniversary - Nostalgic Baby - Nostalgic Blue - Festive Christmas - Prismatic Cocoa: Festive Christmas - Prismatic Tinsel: Festive Christmas - Nostalgic Christmas - Prismatic Cocoa: Nostalgic Christmas - Prismatic Tinsel: Nostalgic Christmas - Nostalgic Darigan - Nostalgic Faerie - Nostalgic Grey - Spooky Halloween - Nostalgic Maraquan - Nostalgic Mutant - Nostalgic Plushie - Nostalgic Robot - Nostalgic Royalboy - Nostalgic Royalgirl - Nostalgic Snow - Nostalgic Tyrannian Note especially the Christmas case, which is all together now! I think it's also more in line with people's expectations for Halloween to be alphabetically among the rest, instead of being at the bottom for being "Spooky". There's enough styles now that I'm starting to wonder if there's other UI affordances worth having here, like e.g. only showing (or at least prioritizing) styles that match the chosen color? But I don't want to mislead people about compatibility, either.
2025-01-12 11:57:43 -08:00
scope :by_series_main_name, -> {
# The main part of the series name, like "Nostalgic".
order(Arel.sql("SUBSTRING_INDEX(series_name, ': ', -1)"))
}
scope :by_series_variant_name, -> {
# The variant part of the series name, like "Prismatic Cyan".
order(Arel.sql("SUBSTRING_INDEX(series_name, ': ', 1)"))
}
scope :by_color_name, -> {
joins(:color).order(Color.arel_table[:name])
}
scope :by_name_grouped, -> {
# Sort by the color name, then the main part of the series name, then the
# variant part of the series name. This way, all the, say, Christmas colors
# and their Prismatic variants will be together, including both Festive and
# Nostalgic cases.
by_color_name.by_series_main_name.by_series_variant_name
}
scope :unlabeled, -> { where(series_name: nil) }
scope :newest, -> { order(created_at: :desc) }
def pet_name
2024-01-24 03:54:43 -08:00
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 series_main_name
series_name.split(': ').last
end
def series_variant_name
series_name.split(': ').first
end
def adjective_name
"#{series_name} #{color.human_name}"
end
def full_name
"#{series_name} #{name}"
end
EMPTY_IMAGE_URL = "data:image/gif;base64,R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw=="
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
Make thumbnail_url a manually overridable field for Alt Styles Oh jeez, okay, the latest batch of Alt Styles are using a different URL format for the thumbnail image! This isn't something we can import via modeling, so we're gonna have to keep on top of it manually. For now, I'll keep inferring the previous format in case they keep using it, but here's also a console script to fix up the latest batch. (At time of writing, not all of these are in our database, which is fine; when pasting it into the console, those lines will error and the script will continue.) ```rb def update_style(color_name, species_name, thumbnail_url) AltStyle.find_by_color_id_and_species_id( Color.find_by_name(color_name), Species.find_by_name(species_name) ).update!(thumbnail_url:) end update_style "Grey", "Blumaroo", "https://images.neopets.com/items/c0gk16fk.gif" update_style "Grey", "Bori", "https://images.neopets.com/items/55qvx6mr.gif" update_style "Grey", "Bruce", "https://images.neopets.com/items/6y6pyiuw.gif" update_style "Grey", "Buzz", "https://images.neopets.com/items/7fh4avry.gif" update_style "Grey", "Chia", "https://images.neopets.com/items/7b2jtn10.gif" update_style "Grey", "Elephante", "https://images.neopets.com/items/0ne41rao.gif" update_style "Grey", "Gnorbu", "https://images.neopets.com/items/75mwtqmh.gif" update_style "Grey", "Hissi", "https://images.neopets.com/items/dxdi2mhm.gif" update_style "Grey", "Kiko", "https://images.neopets.com/items/b9yiruxt.gif" update_style "Grey", "Lenny", "https://images.neopets.com/items/c6cboc7e.gif" update_style "Grey", "Lutari", "https://images.neopets.com/items/33fs4eqf.gif" update_style "Grey", "Nimmo", "https://images.neopets.com/items/4karmgbl.gif" update_style "Grey", "Ogrin", "https://images.neopets.com/items/dlw78fhk.gif" update_style "Grey", "Quiggle", "https://images.neopets.com/items/0aipl0iw.gif" update_style "Grey", "Ruki", "https://images.neopets.com/items/bjnjxsem.gif" update_style "Grey", "Tuskaninny", "https://images.neopets.com/items/7rh57a0o.gif" update_style "Grey", "Vandagyre", "https://images.neopets.com/items/6p8sgs69.gif" update_style "Grey", "Xweetok", "https://images.neopets.com/items/bge9vp5e.gif" ```
2024-06-15 17:35:12 -07:00
# 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?
Make thumbnail_url a manually overridable field for Alt Styles Oh jeez, okay, the latest batch of Alt Styles are using a different URL format for the thumbnail image! This isn't something we can import via modeling, so we're gonna have to keep on top of it manually. For now, I'll keep inferring the previous format in case they keep using it, but here's also a console script to fix up the latest batch. (At time of writing, not all of these are in our database, which is fine; when pasting it into the console, those lines will error and the script will continue.) ```rb def update_style(color_name, species_name, thumbnail_url) AltStyle.find_by_color_id_and_species_id( Color.find_by_name(color_name), Species.find_by_name(species_name) ).update!(thumbnail_url:) end update_style "Grey", "Blumaroo", "https://images.neopets.com/items/c0gk16fk.gif" update_style "Grey", "Bori", "https://images.neopets.com/items/55qvx6mr.gif" update_style "Grey", "Bruce", "https://images.neopets.com/items/6y6pyiuw.gif" update_style "Grey", "Buzz", "https://images.neopets.com/items/7fh4avry.gif" update_style "Grey", "Chia", "https://images.neopets.com/items/7b2jtn10.gif" update_style "Grey", "Elephante", "https://images.neopets.com/items/0ne41rao.gif" update_style "Grey", "Gnorbu", "https://images.neopets.com/items/75mwtqmh.gif" update_style "Grey", "Hissi", "https://images.neopets.com/items/dxdi2mhm.gif" update_style "Grey", "Kiko", "https://images.neopets.com/items/b9yiruxt.gif" update_style "Grey", "Lenny", "https://images.neopets.com/items/c6cboc7e.gif" update_style "Grey", "Lutari", "https://images.neopets.com/items/33fs4eqf.gif" update_style "Grey", "Nimmo", "https://images.neopets.com/items/4karmgbl.gif" update_style "Grey", "Ogrin", "https://images.neopets.com/items/dlw78fhk.gif" update_style "Grey", "Quiggle", "https://images.neopets.com/items/0aipl0iw.gif" update_style "Grey", "Ruki", "https://images.neopets.com/items/bjnjxsem.gif" update_style "Grey", "Tuskaninny", "https://images.neopets.com/items/7rh57a0o.gif" update_style "Grey", "Vandagyre", "https://images.neopets.com/items/6p8sgs69.gif" update_style "Grey", "Xweetok", "https://images.neopets.com/items/bge9vp5e.gif" ```
2024-06-15 17:35:12 -07:00
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
Better ordering for NC Styles in the outfit editor Previously, when opening the pose picker and looking at Styles, the Cybunny options were sorted like this: - Default - Celebratory 25th Anniversary - Festive Christmas - Nostalgic Baby - Nostalgic Blue - Nostalgic Christmas - Nostalgic Darigan - Nostalgic Faerie - Nostalgic Grey - Nostalgic Maraquan - Nostalgic Mutant - Nostalgic Plushie - Nostalgic Robot - Nostalgic Royalboy - Nostalgic Royalgirl - Nostalgic Snow - Nostalgic Tyrannian - Prismatic Cocoa: Festive Christmas - Prismatic Cocoa: Nostalgic Christmas - Prismatic Tinsel: Festive Christmas - Prismatic Tinsel: Nostalgic Christmas - Spooky Halloween Now, they're sorted like this: - Default - Celebratory 25th Anniversary - Nostalgic Baby - Nostalgic Blue - Festive Christmas - Prismatic Cocoa: Festive Christmas - Prismatic Tinsel: Festive Christmas - Nostalgic Christmas - Prismatic Cocoa: Nostalgic Christmas - Prismatic Tinsel: Nostalgic Christmas - Nostalgic Darigan - Nostalgic Faerie - Nostalgic Grey - Spooky Halloween - Nostalgic Maraquan - Nostalgic Mutant - Nostalgic Plushie - Nostalgic Robot - Nostalgic Royalboy - Nostalgic Royalgirl - Nostalgic Snow - Nostalgic Tyrannian Note especially the Christmas case, which is all together now! I think it's also more in line with people's expectations for Halloween to be alphabetically among the rest, instead of being at the bottom for being "Spooky". There's enough styles now that I'm starting to wonder if there's other UI affordances worth having here, like e.g. only showing (or at least prioritizing) styles that match the chosen color? But I don't want to mislead people about compatibility, either.
2025-01-12 11:57:43 -08:00
distinct.where.not(series_name: nil).
by_series_main_name.by_series_variant_name.
pluck(:series_name)
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
2024-01-24 06:12:03 -08:00
# 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