From f40b1fb1681e90cdc5ece5c79c95d0f0c81b364f Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 22 Sep 2019 15:18:46 +0200 Subject: [PATCH] NPCs in the NPC list are now filtered according to the selected area. Objects can now be added via drag and drop from the object list. --- src/core/data_formats/parsing/quest/areas.ts | 4 +- .../data_formats/parsing/quest/npc_types.ts | 245 +++++++++++++++++- src/quest_editor/gui/EntityListView.css | 1 + src/quest_editor/gui/EntityListView.ts | 11 +- src/quest_editor/gui/NpcListView.ts | 29 ++- src/quest_editor/gui/ObjectListView.ts | 12 + src/quest_editor/gui/QuestEditorView.ts | 8 + .../rendering/QuestEntityControls.ts | 60 +++-- 8 files changed, 331 insertions(+), 39 deletions(-) create mode 100644 src/quest_editor/gui/ObjectListView.ts diff --git a/src/core/data_formats/parsing/quest/areas.ts b/src/core/data_formats/parsing/quest/areas.ts index fbcd90d3..22506fb6 100644 --- a/src/core/data_formats/parsing/quest/areas.ts +++ b/src/core/data_formats/parsing/quest/areas.ts @@ -27,7 +27,7 @@ export function get_area_variant( const variant = area.area_variants[variant_id]; if (!variant) throw new Error(`No area variant with id ${variant_id}.`); - return variant; + return Object.freeze(variant); } const AREAS: { [episode: number]: Area[] } = []; @@ -39,7 +39,7 @@ function create_area(id: number, name: string, order: number, variants: number): area.area_variants.push({ id, area }); } - return area; + return Object.freeze(area); } // The IDs match the PSO IDs for areas. diff --git a/src/core/data_formats/parsing/quest/npc_types.ts b/src/core/data_formats/parsing/quest/npc_types.ts index b5b17b30..eb5b4d66 100644 --- a/src/core/data_formats/parsing/quest/npc_types.ts +++ b/src/core/data_formats/parsing/quest/npc_types.ts @@ -198,7 +198,14 @@ export type NpcTypeData = { readonly ultimate_name: string; readonly episode?: Episode; readonly enemy: boolean; + /** + * The type of this NPC's rare variant if it has one. + */ readonly rare_type?: NpcType; + /** + * IDs of the areas this NPC can be found in. + */ + readonly area_ids: number[]; /** * Type ID used by the game. */ @@ -249,6 +256,7 @@ function define_npc_type_data( episode: Episode | undefined, enemy: boolean, rare_type: NpcType | undefined, + area_ids: number[], pso_type_id: number | undefined, pso_roaming: number | undefined, pso_regular: boolean | undefined, @@ -259,17 +267,18 @@ function define_npc_type_data( ENEMY_NPC_TYPES.push(npc_type); } - NPC_TYPE_DATA[npc_type] = { + NPC_TYPE_DATA[npc_type] = Object.freeze({ name, simple_name, ultimate_name, episode, enemy, rare_type, + area_ids, pso_type_id, pso_roaming, pso_regular, - }; + }); if (episode) { const map = EP_AND_NAME_TO_NPC_TYPE[episode]; @@ -293,6 +302,7 @@ define_npc_type_data( undefined, false, undefined, + [], undefined, undefined, undefined, @@ -310,6 +320,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x004, 0, true, @@ -322,6 +333,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x005, 0, true, @@ -334,6 +346,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x007, 0, true, @@ -346,6 +359,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x00a, 0, true, @@ -358,6 +372,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x00b, 0, true, @@ -370,6 +385,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x00c, 0, true, @@ -382,6 +398,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x00d, 0, true, @@ -394,6 +411,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x019, 0, true, @@ -406,6 +424,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01a, 0, true, @@ -418,6 +437,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01b, 0, true, @@ -430,6 +450,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01c, 0, true, @@ -442,6 +463,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01d, 0, true, @@ -454,6 +476,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01e, 0, true, @@ -466,6 +489,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x01f, 0, true, @@ -478,6 +502,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x020, 0, true, @@ -490,6 +515,7 @@ define_npc_type_data( undefined, false, undefined, + [0], 0x0f1, 0, true, @@ -502,6 +528,7 @@ define_npc_type_data( 2, false, undefined, + [0], 0x0fe, 0, true, @@ -521,6 +548,7 @@ define_npc_type_data( 1, true, NpcType.Hildeblue, + [1, 2], 0x040, 0, true, @@ -533,6 +561,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x040, 1, true, @@ -545,6 +574,7 @@ define_npc_type_data( 1, true, NpcType.AlRappy, + [1, 2], 0x041, 0, true, @@ -557,6 +587,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x041, 1, true, @@ -569,6 +600,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x042, 0, true, @@ -581,6 +613,7 @@ define_npc_type_data( 1, true, undefined, + [], undefined, undefined, undefined, @@ -593,6 +626,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x043, 0, true, @@ -605,11 +639,24 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x043, 0, false, ); -define_npc_type_data(NpcType.Booma, "Booma", "Booma", "Bartle", 1, true, undefined, 0x044, 0, true); +define_npc_type_data( + NpcType.Booma, + "Booma", + "Booma", + "Bartle", + 1, + true, + undefined, + [1, 2], + 0x044, + 0, + true, +); define_npc_type_data( NpcType.Gobooma, "Gobooma", @@ -618,6 +665,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x044, 1, true, @@ -630,6 +678,7 @@ define_npc_type_data( 1, true, undefined, + [1, 2], 0x044, 2, true, @@ -642,6 +691,7 @@ define_npc_type_data( 1, true, undefined, + [11], 0x0c0, 0, true, @@ -657,6 +707,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x060, 0, true, @@ -669,6 +720,7 @@ define_npc_type_data( 1, true, NpcType.NarLily, + [3, 4, 5], 0x061, 0, true, @@ -681,6 +733,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x061, 1, true, @@ -693,6 +746,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x062, 0, true, @@ -705,6 +759,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x063, 0, true, @@ -717,6 +772,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x063, 1, true, @@ -729,6 +785,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x063, 2, true, @@ -741,6 +798,7 @@ define_npc_type_data( 1, true, NpcType.PouillySlime, + [3, 4, 5], 0x064, 0, true, @@ -753,6 +811,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x064, 0, false, @@ -765,6 +824,7 @@ define_npc_type_data( 1, true, undefined, + [3, 4, 5], 0x065, 0, true, @@ -777,6 +837,7 @@ define_npc_type_data( 1, true, undefined, + [], undefined, undefined, undefined, @@ -789,6 +850,7 @@ define_npc_type_data( 1, true, undefined, + [], undefined, undefined, undefined, @@ -801,6 +863,7 @@ define_npc_type_data( 1, true, undefined, + [12], 0x0c1, 0, true, @@ -816,6 +879,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x080, 0, true, @@ -828,6 +892,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x080, 1, true, @@ -840,6 +905,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x081, 0, true, @@ -852,6 +918,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x082, 0, true, @@ -864,6 +931,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x082, 0, false, @@ -876,6 +944,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x083, 0, true, @@ -888,6 +957,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x084, 0, true, @@ -900,6 +970,7 @@ define_npc_type_data( 1, true, undefined, + [6, 7], 0x085, 0, true, @@ -912,6 +983,7 @@ define_npc_type_data( 1, true, undefined, + [13], 0x0c5, 0, true, @@ -927,6 +999,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a0, 0, true, @@ -939,6 +1012,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a1, 0, true, @@ -951,6 +1025,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a2, 0, true, @@ -963,6 +1038,7 @@ define_npc_type_data( 1, true, undefined, + [], undefined, undefined, undefined, @@ -975,6 +1051,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a4, 0, true, @@ -987,6 +1064,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a5, 0, true, @@ -999,6 +1077,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a6, 0, true, @@ -1011,6 +1090,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a6, 1, true, @@ -1023,6 +1103,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a6, 2, true, @@ -1035,6 +1116,7 @@ define_npc_type_data( 1, true, undefined, + [8, 9, 10], 0x0a7, 0, true, @@ -1047,11 +1129,24 @@ define_npc_type_data( 1, true, undefined, + [], undefined, undefined, undefined, ); -define_npc_type_data(NpcType.Claw, "Claw", "Claw", "Claw", 1, true, undefined, 0x0a8, 0, true); +define_npc_type_data( + NpcType.Claw, + "Claw", + "Claw", + "Claw", + 1, + true, + undefined, + [8, 9, 10], + 0x0a8, + 0, + true, +); define_npc_type_data( NpcType.DarkFalz, "Dark Falz", @@ -1060,6 +1155,7 @@ define_npc_type_data( 1, true, undefined, + [14], 0x0c8, 0, true, @@ -1075,6 +1171,7 @@ define_npc_type_data( 2, true, NpcType.Hildeblue2, + [1, 2], 0x040, 0, true, @@ -1087,6 +1184,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x040, 1, true, @@ -1099,6 +1197,7 @@ define_npc_type_data( 2, true, NpcType.LoveRappy, + [1, 2], 0x041, 0, true, @@ -1111,6 +1210,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x041, 1, true, @@ -1123,6 +1223,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1135,6 +1236,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1147,6 +1249,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1159,6 +1262,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x042, 0, true, @@ -1171,6 +1275,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1183,6 +1288,7 @@ define_npc_type_data( 2, true, NpcType.NarLily2, + [1, 2], 0x061, 0, true, @@ -1195,6 +1301,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x061, 1, true, @@ -1207,6 +1314,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x060, 0, true, @@ -1219,6 +1327,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x0a6, 0, true, @@ -1231,6 +1340,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x0a6, 1, true, @@ -1243,6 +1353,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x0a6, 2, true, @@ -1255,6 +1366,7 @@ define_npc_type_data( 2, true, undefined, + [1, 2], 0x0a5, 0, true, @@ -1267,6 +1379,7 @@ define_npc_type_data( 2, true, undefined, + [14], 0x0cb, 0, true, @@ -1282,6 +1395,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x043, 0, true, @@ -1294,6 +1408,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x043, 0, false, @@ -1306,6 +1421,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x065, 0, true, @@ -1318,6 +1434,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1330,6 +1447,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1342,6 +1460,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x080, 0, true, @@ -1354,6 +1473,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x080, 1, true, @@ -1366,6 +1486,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x081, 0, true, @@ -1378,6 +1499,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x085, 0, true, @@ -1390,6 +1512,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x0a0, 0, true, @@ -1402,6 +1525,7 @@ define_npc_type_data( 2, true, undefined, + [3, 4], 0x0a1, 0, true, @@ -1414,6 +1538,7 @@ define_npc_type_data( 2, true, undefined, + [15], 0x0cc, 0, true, @@ -1429,6 +1554,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d4, 0, true, @@ -1441,6 +1567,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d4, 1, true, @@ -1453,6 +1580,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d5, 0, true, @@ -1465,6 +1593,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d5, 1, true, @@ -1477,6 +1606,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16, 17], 0x0d6, 0, true, @@ -1489,6 +1619,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16, 17], 0x0d6, 1, true, @@ -1501,6 +1632,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16, 17], 0x0d6, 2, true, @@ -1513,6 +1645,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d7, 0, true, @@ -1525,6 +1658,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16], 0x0d7, 1, true, @@ -1537,11 +1671,24 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16, 17], 0x0d8, 0, true, ); -define_npc_type_data(NpcType.Gee, "Gee", "Gee", "Gee", 2, true, undefined, 0x0d9, 0, true); +define_npc_type_data( + NpcType.Gee, + "Gee", + "Gee", + "Gee", + 2, + true, + undefined, + [5, 6, 7, 8, 9, 16], + 0x0d9, + 0, + true, +); define_npc_type_data( NpcType.GiGue, "Gi Gue", @@ -1550,6 +1697,7 @@ define_npc_type_data( 2, true, undefined, + [5, 6, 7, 8, 9, 16, 17], 0x0da, 0, true, @@ -1562,6 +1710,7 @@ define_npc_type_data( 2, true, undefined, + [17], 0x0e1, 0, true, @@ -1574,6 +1723,7 @@ define_npc_type_data( 2, true, undefined, + [17], 0x061, 0, true, @@ -1586,6 +1736,7 @@ define_npc_type_data( 2, true, undefined, + [17], 0x0e0, 0, true, @@ -1598,6 +1749,7 @@ define_npc_type_data( 2, true, undefined, + [12], 0x0c0, 0, true, @@ -1613,6 +1765,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0db, 0, true, @@ -1625,6 +1778,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11, 17], 0x0dc, 0, true, @@ -1637,6 +1791,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0dd, 0, true, @@ -1649,6 +1804,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0dd, 1, true, @@ -1661,6 +1817,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0de, 0, true, @@ -1673,6 +1830,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11, 17], 0x0df, 0, true, @@ -1685,6 +1843,7 @@ define_npc_type_data( 2, true, undefined, + [], undefined, undefined, undefined, @@ -1697,6 +1856,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0e0, 0, true, @@ -1709,6 +1869,7 @@ define_npc_type_data( 2, true, undefined, + [10, 11], 0x0e0, 1, true, @@ -1721,6 +1882,7 @@ define_npc_type_data( 2, true, undefined, + [13], 0x0ca, 0, true, @@ -1736,6 +1898,7 @@ define_npc_type_data( 4, true, NpcType.DelRappy, + [1, 2, 3, 4, 5, 6, 7, 8], 0x041, 0, true, @@ -1748,6 +1911,7 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5, 6, 7, 8], 0x041, 1, true, @@ -1760,6 +1924,7 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5, 6, 7, 8], 0x110, 0, true, @@ -1772,11 +1937,24 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5, 6, 7, 8], 0x111, 0, true, ); -define_npc_type_data(NpcType.Yowie, "Yowie", "Yowie", "Yowie", 4, true, undefined, 0x111, 0, false); +define_npc_type_data( + NpcType.Yowie, + "Yowie", + "Yowie", + "Yowie", + 4, + true, + undefined, + [1, 2, 3, 4, 5, 6, 7, 8], + 0x111, + 0, + false, +); define_npc_type_data( NpcType.MerissaA, "Merissa A", @@ -1785,6 +1963,7 @@ define_npc_type_data( 4, true, NpcType.MerissaAA, + [6, 7, 8], 0x112, 0, true, @@ -1797,6 +1976,7 @@ define_npc_type_data( 4, true, undefined, + [6, 7, 8], 0x112, 1, true, @@ -1809,11 +1989,24 @@ define_npc_type_data( 4, true, undefined, + [6, 7, 8], 0x113, 0, true, ); -define_npc_type_data(NpcType.Zu, "Zu", "Zu", "Zu", 4, true, NpcType.Pazuzu, 0x114, 0, true); +define_npc_type_data( + NpcType.Zu, + "Zu", + "Zu", + "Zu", + 4, + true, + NpcType.Pazuzu, + [1, 2, 3, 4, 5, 6, 7, 8], + 0x114, + 0, + true, +); define_npc_type_data( NpcType.Pazuzu, "Pazuzu", @@ -1822,11 +2015,24 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5, 6, 7, 8], 0x114, 1, true, ); -define_npc_type_data(NpcType.Boota, "Boota", "Boota", "Boota", 4, true, undefined, 0x115, 0, true); +define_npc_type_data( + NpcType.Boota, + "Boota", + "Boota", + "Boota", + 4, + true, + undefined, + [1, 2, 3, 4, 5], + 0x115, + 0, + true, +); define_npc_type_data( NpcType.ZeBoota, "Ze Boota", @@ -1835,6 +2041,7 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5], 0x115, 1, true, @@ -1847,6 +2054,7 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5], 0x115, 2, true, @@ -1859,6 +2067,7 @@ define_npc_type_data( 4, true, NpcType.DorphonEclair, + [1, 2, 3, 4, 5], 0x116, 0, true, @@ -1871,11 +2080,24 @@ define_npc_type_data( 4, true, undefined, + [1, 2, 3, 4, 5], 0x116, 1, true, ); -define_npc_type_data(NpcType.Goran, "Goran", "Goran", "Goran", 4, true, undefined, 0x117, 0, true); +define_npc_type_data( + NpcType.Goran, + "Goran", + "Goran", + "Goran", + 4, + true, + undefined, + [6, 7, 8], + 0x117, + 0, + true, +); define_npc_type_data( NpcType.PyroGoran, "Pyro Goran", @@ -1884,6 +2106,7 @@ define_npc_type_data( 4, true, undefined, + [6, 7, 8], 0x117, 1, true, @@ -1896,6 +2119,7 @@ define_npc_type_data( 4, true, undefined, + [6, 7, 8], 0x117, 2, true, @@ -1908,6 +2132,7 @@ define_npc_type_data( 4, true, NpcType.Kondrieu, + [9], 0x119, 0, true, @@ -1920,6 +2145,7 @@ define_npc_type_data( 4, true, NpcType.Kondrieu, + [9], 0x119, 1, true, @@ -1932,6 +2158,7 @@ define_npc_type_data( 4, true, undefined, + [9], 0x119, 0, false, diff --git a/src/quest_editor/gui/EntityListView.css b/src/quest_editor/gui/EntityListView.css index ec75f860..963f2ca9 100644 --- a/src/quest_editor/gui/EntityListView.css +++ b/src/quest_editor/gui/EntityListView.css @@ -8,6 +8,7 @@ grid-column-gap: 6px; grid-row-gap: 6px; justify-content: center; + margin: 6px; } .quest_editor_EntityListView_entity { diff --git a/src/quest_editor/gui/EntityListView.ts b/src/quest_editor/gui/EntityListView.ts index 796e9e87..f7562e81 100644 --- a/src/quest_editor/gui/EntityListView.ts +++ b/src/quest_editor/gui/EntityListView.ts @@ -2,14 +2,17 @@ import { ResizableWidget } from "../../core/gui/ResizableWidget"; import { bind_children_to, el } from "../../core/gui/dom"; import "./EntityListView.css"; import { entity_data, EntityType } from "../../core/data_formats/parsing/quest/entities"; -import { ListProperty } from "../../core/observable/property/list/ListProperty"; import { entity_dnd_source } from "./entity_dnd"; import { render_entity_to_image } from "../rendering/render_entity_to_image"; +import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty"; +import { list_property } from "../../core/observable"; export abstract class EntityListView extends ResizableWidget { readonly element: HTMLElement; - protected constructor(private readonly class_name: string, entities: ListProperty) { + protected readonly entities: WritableListProperty = list_property(); + + protected constructor(class_name: string) { super(); const list_element = el.div({ class: "quest_editor_EntityListView_entity_list" }); @@ -17,7 +20,7 @@ export abstract class EntityListView extends ResizableWidg this.element = el.div({ class: `${class_name} quest_editor_EntityListView` }, list_element); this.disposables( - bind_children_to(list_element, entities, this.create_entity_element), + bind_children_to(list_element, this.entities, this.create_entity_element), entity_dnd_source(list_element, target => { let element: HTMLElement | null = target; @@ -28,7 +31,7 @@ export abstract class EntityListView extends ResizableWidg if (index != undefined) { return [ element.querySelector("img")!.cloneNode(true) as HTMLElement, - entities.get(parseInt(index, 10)), + this.entities.get(parseInt(index, 10)), ]; } diff --git a/src/quest_editor/gui/NpcListView.ts b/src/quest_editor/gui/NpcListView.ts index 6c371d80..35a825cb 100644 --- a/src/quest_editor/gui/NpcListView.ts +++ b/src/quest_editor/gui/NpcListView.ts @@ -1,11 +1,34 @@ import { EntityListView } from "./EntityListView"; -import { NPC_TYPES, NpcType } from "../../core/data_formats/parsing/quest/npc_types"; -import { list_property } from "../../core/observable"; +import { npc_data, NPC_TYPES, NpcType } from "../../core/data_formats/parsing/quest/npc_types"; +import { quest_editor_store } from "../stores/QuestEditorStore"; +import { Episode } from "../../core/data_formats/parsing/quest/Episode"; export class NpcListView extends EntityListView { constructor() { - super("quest_editor_NpcListView", list_property(undefined, ...NPC_TYPES)); + super("quest_editor_NpcListView"); + this.disposables( + quest_editor_store.current_quest.observe(this.filter_npcs), + quest_editor_store.current_area.observe(this.filter_npcs), + ); + + this.filter_npcs(); this.finalize_construction(NpcListView.prototype); } + + private filter_npcs = (): void => { + const quest = quest_editor_store.current_quest.val; + const area = quest_editor_store.current_area.val; + + const episode = quest ? quest.episode : Episode.I; + const area_id = area ? area.id : 0; + + this.entities.val = NPC_TYPES.filter(npc => { + const data = npc_data(npc); + return ( + (data.episode == undefined || data.episode === episode) && + data.area_ids.includes(area_id) + ); + }); + }; } diff --git a/src/quest_editor/gui/ObjectListView.ts b/src/quest_editor/gui/ObjectListView.ts new file mode 100644 index 00000000..eca2c8ac --- /dev/null +++ b/src/quest_editor/gui/ObjectListView.ts @@ -0,0 +1,12 @@ +import { EntityListView } from "./EntityListView"; +import { OBJECT_TYPES, ObjectType } from "../../core/data_formats/parsing/quest/object_types"; + +export class ObjectListView extends EntityListView { + constructor() { + super("quest_editor_ObjectListView"); + + this.entities.val = OBJECT_TYPES; + + this.finalize_construction(ObjectListView.prototype); + } +} diff --git a/src/quest_editor/gui/QuestEditorView.ts b/src/quest_editor/gui/QuestEditorView.ts index 9477c5d2..637fdd2b 100644 --- a/src/quest_editor/gui/QuestEditorView.ts +++ b/src/quest_editor/gui/QuestEditorView.ts @@ -13,6 +13,7 @@ import { EntityInfoView } from "./EntityInfoView"; import { gui_store, GuiTool } from "../../core/stores/GuiStore"; import { quest_editor_store } from "../stores/QuestEditorStore"; import { NpcListView } from "./NpcListView"; +import { ObjectListView } from "./ObjectListView"; import Logger = require("js-logger"); const logger = Logger.get("quest_editor/gui/QuestEditorView"); @@ -25,6 +26,7 @@ const VIEW_TO_NAME = new Map ResizableWidget, string>([ [AsmEditorView, "asm_editor"], [EntityInfoView, "entity_info"], [NpcListView, "npc_list_view"], + [ObjectListView, "object_list_view"], ]); const DEFAULT_LAYOUT_CONFIG = { @@ -99,6 +101,12 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [ componentName: VIEW_TO_NAME.get(NpcListView), isClosable: false, }, + { + title: "Objects", + type: "component", + componentName: VIEW_TO_NAME.get(ObjectListView), + isClosable: false, + }, ], }, ], diff --git a/src/quest_editor/rendering/QuestEntityControls.ts b/src/quest_editor/rendering/QuestEntityControls.ts index c2a0d235..3c6d82c3 100644 --- a/src/quest_editor/rendering/QuestEntityControls.ts +++ b/src/quest_editor/rendering/QuestEntityControls.ts @@ -17,6 +17,7 @@ import { remove_entity_dnd_listener, } from "../gui/entity_dnd"; import { vec3_to_threejs } from "../../core/rendering/conversion"; +import { QuestObjectModel } from "../model/QuestObjectModel"; const UP_VECTOR = Object.freeze(new Vector3(0, 1, 0)); const DOWN_VECTOR = Object.freeze(new Vector3(0, -1, 0)); @@ -198,18 +199,14 @@ export class QuestEntityControls implements Disposable { if (!area || !quest) return; + let entity: QuestEntityModel; + if (is_npc_type(e.entity_type)) { const data = npc_data(e.entity_type); if (data.pso_type_id == undefined || data.pso_roaming == undefined) return; - e.drag_element.style.display = "none"; - - if (e.event.dataTransfer) { - e.event.dataTransfer.dropEffect = "copy"; - } - - const npc = new QuestNpcModel( + entity = new QuestNpcModel( e.entity_type, data.pso_type_id, 0, @@ -223,21 +220,42 @@ export class QuestEntityControls implements Disposable { // TODO: do the following values make sense? [[0, 0, 0, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0], [0, 0, 0, 0]], ); - const grab_offset = new Vector3(0, 0, 0); - const drag_adjust = new Vector3(0, 0, 0); - this.translate_entity_horizontally(npc, grab_offset, drag_adjust); - quest.add_npc(npc); - - quest_editor_store.set_selected_entity(npc); - - this.pick = { - mode: PickMode.Creating, - initial_section: npc.section.val, - initial_position: npc.world_position.val, - grab_offset, - drag_adjust, - }; + } else { + entity = new QuestObjectModel( + e.entity_type, + 0, + 0, + area.id, + 0, + new Vec3(0, 0, 0), + new Vec3(0, 0, 0), + // TODO: which default properties? + new Map(), + // TODO: do the following values make sense? + [[0, 0, 0, 0, 0, 0], [0, 0]], + ); } + + e.drag_element.style.display = "none"; + + if (e.event.dataTransfer) { + e.event.dataTransfer.dropEffect = "copy"; + } + + const grab_offset = new Vector3(0, 0, 0); + const drag_adjust = new Vector3(0, 0, 0); + this.translate_entity_horizontally(entity, grab_offset, drag_adjust); + quest.add_entity(entity); + + quest_editor_store.set_selected_entity(entity); + + this.pick = { + mode: PickMode.Creating, + initial_section: entity.section.val, + initial_position: entity.world_position.val, + grab_offset, + drag_adjust, + }; }; private dragover = (e: EntityDragEvent) => {