2023-08-02 16:05:02 -07:00
|
|
|
class Pet < ApplicationRecord
|
2010-10-07 07:46:23 -07:00
|
|
|
belongs_to :pet_type
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2024-01-24 03:25:23 -08:00
|
|
|
attr_reader :items, :pet_state, :alt_style
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2024-04-06 02:31:24 -07:00
|
|
|
def load!(timeout: nil)
|
2024-10-18 17:40:31 -07:00
|
|
|
viewer_data = Neopets::CustomPets.fetch_viewer_data(name, timeout:)
|
2024-04-06 02:31:24 -07:00
|
|
|
use_viewer_data(viewer_data)
|
2010-10-07 07:46:23 -07:00
|
|
|
end
|
2013-12-08 20:59:36 -08:00
|
|
|
|
2024-04-06 02:31:24 -07:00
|
|
|
def use_viewer_data(viewer_data)
|
2013-12-08 20:59:36 -08:00
|
|
|
pet_data = viewer_data[:custom_pet]
|
|
|
|
|
2024-01-24 03:25:23 -08:00
|
|
|
raise UnexpectedDataFormat unless pet_data[:species_id]
|
|
|
|
raise UnexpectedDataFormat unless pet_data[:color_id]
|
|
|
|
raise UnexpectedDataFormat unless pet_data[:body_id]
|
2024-01-24 00:54:30 -08:00
|
|
|
|
2024-11-02 21:05:28 -07:00
|
|
|
@pet_state = Pet.pet_state_from_pet_data(pet_data)
|
|
|
|
self.pet_type = @pet_state.pet_type
|
2024-04-06 02:25:22 -07:00
|
|
|
|
2024-04-16 10:03:36 -07:00
|
|
|
begin
|
2024-11-02 21:05:28 -07:00
|
|
|
pet_type.consider_pet_image(name)
|
2024-04-16 10:03:36 -07:00
|
|
|
rescue => error
|
2024-11-02 21:05:28 -07:00
|
|
|
Rails.logger.warn "Failed to load pet image: #{error.full_message}"
|
2024-01-24 06:12:35 -08:00
|
|
|
end
|
|
|
|
|
2024-11-02 21:05:28 -07:00
|
|
|
if pet_data[:alt_style]
|
2024-01-24 03:25:23 -08:00
|
|
|
raise UnexpectedDataFormat unless pet_data[:alt_color]
|
|
|
|
raise UnexpectedDataFormat if pet_data[:biology_by_zone].empty?
|
|
|
|
|
2024-01-24 04:01:34 -08:00
|
|
|
@alt_style = AltStyle.find_or_initialize_by(id: pet_data[:alt_style].to_i)
|
|
|
|
@alt_style.assign_attributes(
|
2024-01-24 03:25:23 -08:00
|
|
|
color_id: pet_data[:alt_color].to_i,
|
|
|
|
species_id: pet_data[:species_id].to_i,
|
|
|
|
body_id: pet_data[:body_id].to_i,
|
|
|
|
biology: pet_data[:biology_by_zone],
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2013-12-08 20:59:36 -08:00
|
|
|
@items = Item.collection_from_pet_type_and_registries(self.pet_type,
|
2024-04-06 02:31:24 -07:00
|
|
|
viewer_data[:object_info_registry], viewer_data[:object_asset_registry])
|
2013-12-08 20:59:36 -08:00
|
|
|
end
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2010-10-10 19:18:42 -07:00
|
|
|
def wardrobe_query
|
|
|
|
{
|
Oops, fix bug with saving outfits of pets loaded from Neopets.com
Okay right, the wardrobe-2020 app treats `state` as a bit of an
override thing, and `pose` is the main canonical field for how a pet
looks. We were missing a few pieces here:
1. After loading a pet, we weren't including the `pose` field in the
initial query string for the wardrobe URL, but we _were_ including
the `state` field, so the outfit would get set up with a conflicting
pet state ID vs pose.
2. When saving an outfit, we weren't taking the `state` field into
account at all. This could cause the saved outfit to not quite match
how it actually looked in-app, because the default pet state for
that species/color/pose trio could be different; and regardless, the
outfit state would come back with `appearanceId` set to `null`,
which wouldn't match the local outfit state, which would trigger an
infinite loop.
Here, we complete the round-trip of the `state` field, from pet loading
to outfit saving to the outfit data that comes back after saving!
2024-02-08 09:51:31 -08:00
|
|
|
name: self.name,
|
2024-02-16 23:22:41 -08:00
|
|
|
color: self.pet_type.color_id,
|
|
|
|
species: self.pet_type.species_id,
|
Oops, fix bug with saving outfits of pets loaded from Neopets.com
Okay right, the wardrobe-2020 app treats `state` as a bit of an
override thing, and `pose` is the main canonical field for how a pet
looks. We were missing a few pieces here:
1. After loading a pet, we weren't including the `pose` field in the
initial query string for the wardrobe URL, but we _were_ including
the `state` field, so the outfit would get set up with a conflicting
pet state ID vs pose.
2. When saving an outfit, we weren't taking the `state` field into
account at all. This could cause the saved outfit to not quite match
how it actually looked in-app, because the default pet state for
that species/color/pose trio could be different; and regardless, the
outfit state would come back with `appearanceId` set to `null`,
which wouldn't match the local outfit state, which would trigger an
infinite loop.
Here, we complete the round-trip of the `state` field, from pet loading
to outfit saving to the outfit data that comes back after saving!
2024-02-08 09:51:31 -08:00
|
|
|
pose: self.pet_state.pose,
|
|
|
|
state: self.pet_state.id,
|
|
|
|
objects: self.items.map(&:id),
|
2024-10-18 19:16:41 -07:00
|
|
|
style: self.alt_style ? self.alt_style.id : nil,
|
2010-10-10 19:18:42 -07:00
|
|
|
}.to_query
|
2010-10-10 11:33:54 -07:00
|
|
|
end
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2010-11-06 16:07:15 -07:00
|
|
|
def contributables
|
2024-01-24 03:54:43 -08:00
|
|
|
contributables = [pet_type, @pet_state, @alt_style].filter(&:present?)
|
2010-11-06 16:07:15 -07:00
|
|
|
items.each do |item|
|
|
|
|
contributables << item
|
|
|
|
contributables += item.pending_swf_assets
|
|
|
|
end
|
|
|
|
contributables
|
|
|
|
end
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2010-11-06 15:08:42 -07:00
|
|
|
before_validation do
|
2010-11-06 16:07:15 -07:00
|
|
|
pet_type.save!
|
2024-11-02 21:05:28 -07:00
|
|
|
@pet_state.save! if @pet_state
|
|
|
|
|
2011-05-21 19:32:01 -07:00
|
|
|
if @items
|
|
|
|
@items.each do |item|
|
2013-01-28 00:10:25 -08:00
|
|
|
item.save! if item.changed?
|
2012-01-13 13:56:31 -08:00
|
|
|
item.handle_assets!
|
2011-05-21 19:32:01 -07:00
|
|
|
end
|
2010-11-06 15:08:42 -07:00
|
|
|
end
|
2024-01-24 03:25:23 -08:00
|
|
|
|
|
|
|
if @alt_style
|
|
|
|
@alt_style.save!
|
|
|
|
end
|
2010-11-06 15:08:42 -07:00
|
|
|
end
|
2011-05-21 19:32:01 -07:00
|
|
|
|
2024-04-06 02:31:24 -07:00
|
|
|
def self.load(name, **options)
|
2023-10-12 18:05:01 -07:00
|
|
|
pet = Pet.find_or_initialize_by(name: name)
|
2024-04-06 02:31:24 -07:00
|
|
|
pet.load!(**options)
|
2010-10-07 07:46:23 -07:00
|
|
|
pet
|
|
|
|
end
|
2024-10-18 18:14:01 -07:00
|
|
|
|
2024-11-02 21:05:28 -07:00
|
|
|
def self.pet_state_from_pet_data(pet_data)
|
|
|
|
# First, set up the pet type.
|
|
|
|
pet_type = PetType.find_or_initialize_by(
|
|
|
|
species_id: pet_data[:species_id],
|
|
|
|
color_id: pet_data[:color_id],
|
|
|
|
)
|
|
|
|
|
|
|
|
# Apply the pet's body ID to the pet type, unless it's wearing an alt
|
|
|
|
# style, in which case ignore it, because it's the *alt style*'s body ID.
|
|
|
|
# (This can theoretically cause a problem saving a new pet type when
|
|
|
|
# there's an alt style too!)
|
|
|
|
pet_type.body_id = pet_data[:body_id] unless pet_data[:alt_style]
|
|
|
|
if pet_type.body_id.nil?
|
|
|
|
raise UnexpectedDataFormat,
|
|
|
|
"can't process alt style on first occurrence of pet type"
|
|
|
|
end
|
|
|
|
|
|
|
|
# Then, set up the biology assets.
|
|
|
|
pet_state_biology = pet_data[:alt_style].present? ?
|
|
|
|
pet_data[:original_biology] :
|
|
|
|
pet_data[:biology_by_zone]
|
|
|
|
body_id = pet_data[:body_id].to_i
|
|
|
|
swf_assets = pet_state_biology.values.map do |biology_data|
|
|
|
|
SwfAsset.from_biology_data(body_id, biology_data)
|
|
|
|
end
|
|
|
|
swf_asset_ids = swf_assets.map(&:remote_id).sort.join(",")
|
|
|
|
|
|
|
|
# Finally, set up the pet state, and use these biology assets.
|
|
|
|
pet_type.pet_states.find_or_initialize_by(swf_asset_ids:).tap do |pet_state|
|
|
|
|
pet_state.swf_assets = swf_assets
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-10-18 18:14:01 -07:00
|
|
|
class UnexpectedDataFormat < RuntimeError;end
|
2010-10-07 07:46:23 -07:00
|
|
|
end
|
2011-05-21 19:32:01 -07:00
|
|
|
|