From e885e3fa62de0f2fa6ba0c9c243e505a7ba8fcdd Mon Sep 17 00:00:00 2001 From: gabe Date: Sun, 28 Jul 2024 20:04:30 -0400 Subject: [PATCH 1/4] feat: add new `sprites` field to `type` via adding `TypeSprites` to db containing type name icons from all games (excluding tera types, for now) --- data/v2/build.py | 39 +++++++++++++++++++++ data/v2/sprites | 2 +- pokemon_v2/migrations/0016_typesprites.py | 41 +++++++++++++++++++++++ pokemon_v2/models.py | 4 +++ pokemon_v2/serializers.py | 12 +++++++ 5 files changed, 97 insertions(+), 1 deletion(-) create mode 100644 pokemon_v2/migrations/0016_typesprites.py diff --git a/data/v2/build.py b/data/v2/build.py index 9e6459eb7..cfabe10a4 100644 --- a/data/v2/build.py +++ b/data/v2/build.py @@ -629,6 +629,45 @@ def csv_record_to_objects(info): build_generic((TypeEfficacyPast,), "type_efficacy_past.csv", csv_record_to_objects) + def csv_record_to_objects(info): + game_map = { + "generation-iii": [ + "colosseum", + "emerald", + "firered-leafgreen", + "ruby-saphire", + "xd", + ], + "generation-iv": ["diamond-pearl", "heartgold-soulsilver", "platinum"], + "generation-v": ["black-2-white-2", "black-white"], + "generation-vi": ["omega-ruby-alpha-sapphire", "x-y"], + "generation-vii": [ + "lets-go-pikachu-lets-go-eevee", + "sun-moon", + "ultra-sun-ultra-moon", + ], + "generation-viii": [ + "brilliant-diamond-and-shining-pearl", + "legends-arceus", + "sword-shield", + ], + "generation-ix": ["scarlet-violet"], + } + sprites = {} + for generation in game_map.keys(): + for game in game_map[generation]: + if generation not in sprites: + sprites[generation] = {} + sprites[generation][game] = { + "name_icon": file_path_or_none( + f"types/{generation}/{game}/{info[0]}.png" + ) + } + + yield TypeSprites(type_id=int(info[0]), sprites=sprites) + + build_generic((TypeSprites,), "types.csv", csv_record_to_objects) + ############# # CONTEST # diff --git a/data/v2/sprites b/data/v2/sprites index ca5a7886c..d607f0df0 160000 --- a/data/v2/sprites +++ b/data/v2/sprites @@ -1 +1 @@ -Subproject commit ca5a7886c10753144e6fae3b69d45a4d42a449b4 +Subproject commit d607f0df04218c564167f654802262ef23a3cac1 diff --git a/pokemon_v2/migrations/0016_typesprites.py b/pokemon_v2/migrations/0016_typesprites.py new file mode 100644 index 000000000..a9e752d63 --- /dev/null +++ b/pokemon_v2/migrations/0016_typesprites.py @@ -0,0 +1,41 @@ +# Generated by Django 3.2.23 on 2024-07-29 02:09 + +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + dependencies = [ + ("pokemon_v2", "0015_pokemoncries"), + ] + + operations = [ + migrations.CreateModel( + name="TypeSprites", + fields=[ + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("sprites", models.JSONField()), + ( + "type", + models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.CASCADE, + related_name="typesprites", + to="pokemon_v2.type", + ), + ), + ], + options={ + "abstract": False, + }, + ), + ] diff --git a/pokemon_v2/models.py b/pokemon_v2/models.py index d7d0e48a1..dfe6bc0a5 100644 --- a/pokemon_v2/models.py +++ b/pokemon_v2/models.py @@ -773,6 +773,10 @@ class TypeEfficacyPast(HasTypeEfficacy, HasGeneration): pass +class TypeSprites(HasType): + sprites = models.JSONField() + + ################# # STAT MODELS # ################# diff --git a/pokemon_v2/serializers.py b/pokemon_v2/serializers.py index 3b5df51f6..c59bb3e54 100644 --- a/pokemon_v2/serializers.py +++ b/pokemon_v2/serializers.py @@ -2340,6 +2340,12 @@ class Meta: fields = ("name", "language") +class TypeSpriteSerializer(serializers.ModelSerializer): + class Meta: + model = TypeSprites + fields = ("sprites",) + + class TypeDetailSerializer(serializers.ModelSerializer): """ Serializer for the Type resource @@ -2357,6 +2363,7 @@ class TypeDetailSerializer(serializers.ModelSerializer): ) pokemon = serializers.SerializerMethodField("get_type_pokemon") moves = MoveSummarySerializer(many=True, read_only=True, source="move") + sprites = serializers.SerializerMethodField("get_type_sprites") class Meta: model = Type @@ -2371,8 +2378,13 @@ class Meta: "names", "pokemon", "moves", + "sprites", ) + def get_type_sprites(self, obj): + sprites_object = TypeSprites.objects.get(type_id=obj) + return sprites_object.sprites + # adds an entry for the given type with the given damage # factor in the given direction to the set of relations From 554747a707fa9ceb058a89811e99fe93385b74af Mon Sep 17 00:00:00 2001 From: gabe Date: Mon, 29 Jul 2024 22:26:15 -0400 Subject: [PATCH 2/4] feat: add type sprites test also rename `type` to `type_obj` to avoid reassigning the `type` function --- pokemon_v2/tests.py | 99 ++++++++++++++++++++++++++++++++++++--------- 1 file changed, 79 insertions(+), 20 deletions(-) diff --git a/pokemon_v2/tests.py b/pokemon_v2/tests.py index efa42567d..43c99ce27 100644 --- a/pokemon_v2/tests.py +++ b/pokemon_v2/tests.py @@ -692,6 +692,54 @@ def setup_type_game_index_data(cls, type, game_index=0): return type_game_index + def setup_type_sprites_data(cls, type): + game_map = { + "generation-iii": [ + "colosseum", + "emerald", + "firered-leafgreen", + "ruby-saphire", + "xd", + ], + "generation-iv": ["diamond-pearl", "heartgold-soulsilver", "platinum"], + "generation-v": ["black-2-white-2", "black-white"], + "generation-vi": ["omega-ruby-alpha-sapphire", "x-y"], + "generation-vii": [ + "lets-go-pikachu-lets-go-eevee", + "sun-moon", + "ultra-sun-ultra-moon", + ], + "generation-viii": [ + "brilliant-diamond-and-shining-pearl", + "legends-arceus", + "sword-shield", + ], + "generation-ix": ["scarlet-violet"], + } + sprites = {} + for generation in game_map.keys(): + for game in game_map[generation]: + if generation not in sprites: + sprites[generation] = {} + + if type.id == 18 and generation.endswith(("-iii", "-iv", "-v")): + sprites[generation][game] = None + elif type.id == 19 and generation.endswith( + ("-iii", "-iv", "-v", "-vi", "-vii", "-viii") + ): + sprites[generation][game] = None + else: + sprites[generation][game] = { + "name_icon": f"https://raw.githubusercontent.com/PokeAPI/sprites/master/sprites/types/{generation}/{game}/{type.id}.png" + } + + type_sprites = TypeSprites.objects.create( + type=type, sprites=json.dumps(sprites) + ) + type_sprites.save() + + return type_sprites, game_map + # Move Data @classmethod def setup_move_ailment_data(cls, name="mv almnt"): @@ -3313,12 +3361,12 @@ def test_super_contest_effect_api(self): # Type Tests def test_type_api(self): - type = self.setup_type_data(name="base tp") - type_name = self.setup_type_name_data(type, name="base tp nm") - type_game_index = self.setup_type_game_index_data(type, game_index=10) - move = self.setup_move_data(name="mv for base tp", type=type) + type_obj = self.setup_type_data(name="base tp") + type_name = self.setup_type_name_data(type_obj, name="base tp nm") + type_game_index = self.setup_type_game_index_data(type_obj, game_index=10) + move = self.setup_move_data(name="mv for base tp", type=type_obj) pokemon = self.setup_pokemon_data(name="pkmn for base tp") - pokemon_type = self.setup_pokemon_type_data(pokemon=pokemon, type=type) + pokemon_type = self.setup_pokemon_type_data(pokemon=pokemon, type=type_obj) generation = self.setup_generation_data(name="past gen") @@ -3345,39 +3393,41 @@ def test_type_api(self): newer_type = self.setup_type_data(name="newer tp", generation=newer_generation) + type_sprites, game_map = self.setup_type_sprites_data(type=type_obj) + # type relations no_damage_to_relation = TypeEfficacy( - damage_type=type, target_type=no_damage_to, damage_factor=0 + damage_type=type_obj, target_type=no_damage_to, damage_factor=0 ) no_damage_to_relation.save() half_damage_to_type_relation = TypeEfficacy( - damage_type=type, target_type=half_damage_to, damage_factor=50 + damage_type=type_obj, target_type=half_damage_to, damage_factor=50 ) half_damage_to_type_relation.save() double_damage_to_type_relation = TypeEfficacy( - damage_type=type, target_type=double_damage_to, damage_factor=200 + damage_type=type_obj, target_type=double_damage_to, damage_factor=200 ) double_damage_to_type_relation.save() no_damage_from_relation = TypeEfficacy( - damage_type=no_damage_from, target_type=type, damage_factor=0 + damage_type=no_damage_from, target_type=type_obj, damage_factor=0 ) no_damage_from_relation.save() half_damage_from_type_relation = TypeEfficacy( - damage_type=half_damage_from, target_type=type, damage_factor=50 + damage_type=half_damage_from, target_type=type_obj, damage_factor=50 ) half_damage_from_type_relation.save() double_damage_from_type_relation = TypeEfficacy( - damage_type=double_damage_from, target_type=type, damage_factor=200 + damage_type=double_damage_from, target_type=type_obj, damage_factor=200 ) double_damage_from_type_relation.save() double_damage_from_newer_type_relation = TypeEfficacy( - damage_type=newer_type, target_type=type, damage_factor=200 + damage_type=newer_type, target_type=type_obj, damage_factor=200 ) double_damage_from_newer_type_relation.save() @@ -3385,20 +3435,20 @@ def test_type_api(self): # type used to deal half damage rather than no damage past_no_damage_to_relation = TypeEfficacyPast( - damage_type=type, + damage_type=type_obj, target_type=no_damage_to, damage_factor=50, generation=generation, ) past_no_damage_to_relation.save() - response = self.client.get("{}/type/{}/".format(API_V2, type.pk)) + response = self.client.get("{}/type/{}/".format(API_V2, type_obj.pk)) self.assertEqual(response.status_code, status.HTTP_200_OK) # base params - self.assertEqual(response.data["id"], type.pk) - self.assertEqual(response.data["name"], type.name) + self.assertEqual(response.data["id"], type_obj.pk) + self.assertEqual(response.data["name"], type_obj.name) # name params self.assertEqual(response.data["names"][0]["name"], type_name.name) self.assertEqual( @@ -3409,19 +3459,19 @@ def test_type_api(self): "{}{}/language/{}/".format(TEST_HOST, API_V2, type_name.language.pk), ) # generation params - self.assertEqual(response.data["generation"]["name"], type.generation.name) + self.assertEqual(response.data["generation"]["name"], type_obj.generation.name) self.assertEqual( response.data["generation"]["url"], - "{}{}/generation/{}/".format(TEST_HOST, API_V2, type.generation.pk), + "{}{}/generation/{}/".format(TEST_HOST, API_V2, type_obj.generation.pk), ) # damage class params self.assertEqual( - response.data["move_damage_class"]["name"], type.move_damage_class.name + response.data["move_damage_class"]["name"], type_obj.move_damage_class.name ) self.assertEqual( response.data["move_damage_class"]["url"], "{}{}/move-damage-class/{}/".format( - TEST_HOST, API_V2, type.move_damage_class.pk + TEST_HOST, API_V2, type_obj.move_damage_class.pk ), ) # move params @@ -3570,6 +3620,15 @@ def test_type_api(self): ), ) + sprites_data = json.loads(type_sprites.sprites) + + for generation in game_map.keys(): + for game in game_map[generation]: + self.assertEqual( + json.loads(response.data["sprites"])[generation][game]["name_icon"], + sprites_data[generation][game]["name_icon"], + ) + # Pokedex Tests def test_pokedex_api(self): pokedex = self.setup_pokedex_data(name="base pkdx") From 9841870055ef8e05ae656d6536d2efd91ba74575 Mon Sep 17 00:00:00 2001 From: gabe Date: Mon, 29 Jul 2024 22:36:43 -0400 Subject: [PATCH 3/4] fix: use items sprites in items test instead of pokemon sprites --- pokemon_v2/tests.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/pokemon_v2/tests.py b/pokemon_v2/tests.py index 43c99ce27..1400e8057 100644 --- a/pokemon_v2/tests.py +++ b/pokemon_v2/tests.py @@ -312,7 +312,7 @@ def setup_item_category_name_data(cls, item_category, name="itm ctgry nm"): @classmethod def setup_item_sprites_data(cls, item, default=True): - sprite_path = "https://raw.githubusercontent.com/PokeAPI/sprites/master/sprites/pokemon/%s.png" + sprite_path = "https://raw.githubusercontent.com/PokeAPI/sprites/master/sprites/items/%s.png" sprites = { "default": sprite_path % item.id if default else None, @@ -2801,9 +2801,14 @@ def test_item_api(self): "{}{}/evolution-chain/{}/".format(TEST_HOST, API_V2, evolution_chain.pk), ) - sprites_data = json.loads(response.data["sprites"]) + sprites_data = json.loads(item_sprites.sprites) + response_sprites_data = json.loads(response.data["sprites"]) # sprites + self.assertEqual( + sprites_data["default"], + response_sprites_data["default"], + ) self.assertEqual( sprites_data["default"], "{}".format(sprites_data["default"]), From 5598df2eaf83e9ab5a6d87f1b552ff9f510cfcf2 Mon Sep 17 00:00:00 2001 From: gabe Date: Mon, 29 Jul 2024 23:51:07 -0400 Subject: [PATCH 4/4] chore: update sprites submodule to latest --- data/v2/sprites | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/data/v2/sprites b/data/v2/sprites index d607f0df0..6e5b8ac35 160000 --- a/data/v2/sprites +++ b/data/v2/sprites @@ -1 +1 @@ -Subproject commit d607f0df04218c564167f654802262ef23a3cac1 +Subproject commit 6e5b8ac354ddc347104840cbd14ad6e0b2fdb551