From c9478cb542d61a71d6ef9fc751bdd54bdcdadee8 Mon Sep 17 00:00:00 2001 From: LmeSzinc <37934724+LmeSzinc@users.noreply.github.com> Date: Fri, 19 Dec 2025 02:50:13 +0800 Subject: [PATCH] Add: chapter A --- .../cn/template/TEMPLATE_SIREN_Vboss_Star.gif | Bin 0 -> 7023 bytes .../en/template/TEMPLATE_SIREN_Vboss_Star.gif | Bin 0 -> 7023 bytes .../jp/template/TEMPLATE_SIREN_Vboss_Star.gif | Bin 0 -> 7023 bytes .../tw/template/TEMPLATE_SIREN_Vboss_Star.gif | Bin 0 -> 7023 bytes campaign/event_20251218_cn/a1.py | 96 ++++++++++++++++++ campaign/event_20251218_cn/a2.py | 75 ++++++++++++++ campaign/event_20251218_cn/a3.py | 81 +++++++++++++++ dev_tools/map_extractor.py | 6 ++ module/template/assets.py | 1 + 9 files changed, 259 insertions(+) create mode 100644 assets/cn/template/TEMPLATE_SIREN_Vboss_Star.gif create mode 100644 assets/en/template/TEMPLATE_SIREN_Vboss_Star.gif create mode 100644 assets/jp/template/TEMPLATE_SIREN_Vboss_Star.gif create mode 100644 assets/tw/template/TEMPLATE_SIREN_Vboss_Star.gif create mode 100644 campaign/event_20251218_cn/a1.py create mode 100644 campaign/event_20251218_cn/a2.py create mode 100644 campaign/event_20251218_cn/a3.py diff --git a/assets/cn/template/TEMPLATE_SIREN_Vboss_Star.gif b/assets/cn/template/TEMPLATE_SIREN_Vboss_Star.gif new file mode 100644 index 0000000000000000000000000000000000000000..aa503f1369bdb4b7b95ce9ae9a9cc9749e26483b GIT binary patch literal 7023 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 literal 0 HcmV?d00001 diff --git a/assets/en/template/TEMPLATE_SIREN_Vboss_Star.gif b/assets/en/template/TEMPLATE_SIREN_Vboss_Star.gif new file mode 100644 index 0000000000000000000000000000000000000000..aa503f1369bdb4b7b95ce9ae9a9cc9749e26483b GIT binary patch literal 7023 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 literal 0 HcmV?d00001 diff --git a/assets/jp/template/TEMPLATE_SIREN_Vboss_Star.gif b/assets/jp/template/TEMPLATE_SIREN_Vboss_Star.gif new file mode 100644 index 0000000000000000000000000000000000000000..aa503f1369bdb4b7b95ce9ae9a9cc9749e26483b GIT binary patch literal 7023 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 literal 0 HcmV?d00001 diff --git a/assets/tw/template/TEMPLATE_SIREN_Vboss_Star.gif b/assets/tw/template/TEMPLATE_SIREN_Vboss_Star.gif new file mode 100644 index 0000000000000000000000000000000000000000..aa503f1369bdb4b7b95ce9ae9a9cc9749e26483b GIT binary patch literal 7023 zcmeI0X;4##8ir3ufP}ClVM|z(5FqR!paw)uLV&O-n+k}UK%mfKMQasXYePsNLZrxM z@LC5DwAI#vr4}t}SXDp;q<6s57Af>zI=xe<{-Euh@qV2%b0(wM>rXoaKbibEIg{sm z-uI1(jTVH-Z4nEk96|p0gwp|Xm4-7bm`KkpMKif+FDy%TU}jURaI3{QK8jp z%gV|$8cj(_$vf}7bNKM#Lx&FS+qbW*9!^?UVH7eyu3VxLXne`lbxL{ zlgW1N+Le`+wPVMQ%*@P;jEwEuw{P3FZR^&pX=!Pxsj11y$w^5`TefW3ym@m%LPC6e zd|X^yY;3GlDvgPWiHeGnNF(;GX zyLRoGHETjcLqkGB_ z*(?@|$z-~@xw*Q!y12MFJ3Bi$IWZUvI-Ty|;9zTOOQX?jY;3Hpt%*dUnVFfXsi}#H z39|b4H+ciBr{v91X{nK-q$qwM5088L6k+{FAhO``{r~(5e`10VClEh>C9gy&BG@@` zie?N-m8k{Wgfdg2F|~3GUuIIp1IH|B)?8T7jJmg5 zy|mxcOy|hGp)cU8$rbA|g#<3{UzR61!o1G(rKtbV1H}oSER-~hDnc^!E&Gu+A$}*+nEi|3%9mKYsl1;ls(v zN$eC14GqB^=+6GSU@#cq6u=#TBXH%)mCru=tfi%;sj2DAnKP$PpKfSqsIRZD ztE)q=!O^2fj~qGj_SVp<=gyrN^6Ba6#=NJbq+qpM+rJKXy7z6I^?v62T47i=09m+U`LcuUM zH#Y~;2?W9_>3*G2YVpehOx6%Cv0lg)q0Eb z4AOnGv2|SLp^u2AKSXJ)wwDQfvhla9{5PKxl+fnOh`~v%9i>FzKVlFD)%qsZ=0+K>omi1N-;y2iNE4=cBF%qFy zd4E}uUir@v#}FJZshWk?STMp1XPTPv1RRmVp%?Xt@Ek=IkCpSIE+$m!<-(OAczfoW zjUO%3qII>tsh2}pz3P4yB8=dCtcoTMlx(CqFT>s^lUw-k$7oDA*Xut=(1Ns>B zdcD4;rUo)TiaW|W1pI>s4+8FQz4aDI{>?Yv1l&RLknSPZ1M%Q_EZ3vZ8>=4kd}3lE z7VRO_1NUgIe4~Lt zL$&k(=$6m*lk(qx|9yFRd0}DU+i$;}nwo+Vf9K8}Xz(cS5Z=+`4zNRJ2fBgl^XJb) zaR;S?%)#W1jg3Gzk9K-sA5;&pLxVS#Jdlot`e(X36nE6^P}woS(aLVz*U>m{3^-QVpV2#(=rO=S z?DqEdfH|~w5IZD#6n3<`L!t-Cul!3|{qHOM28j0&*L(=+LK9Q6lZ#L(6ZM%=90>?h z9%*XnAkZ}xvG7_;GA%USAS~6IdrcfzN-?|Zw06&QcLudth$n2(PLIm7?=XT=8Z1e_ zmPXNoX=aq=nX=fEbfI*v_(5)kz5lkV$x*ha+Lq2AN%wi&hd)Nj>y#z$Xw{r>O6it` zEqRdip{~q2`ajz4eBjK_%4bBBk*!=5T}>3;*&#=6%#kOP9rHu70EN1pi=Wq2@TEh0v{+>W_)pR@t0qI`TqOwVXgzrp{B!V2XY@98w0CCky4~4QtY_tqs$UK$=t?_3*t5>n7Qea zsI&NE)_7luol-P-+$E{SZects-hmfaLDy4CdMg~mB|lGQYvOC^K1rPjKH9vDj8~1! zh}+6Z&RdVFskUVa^$wfI>qZw#2dN(U`k$3Ok0gwM{OdEtA`eo1V5F{#$^Ws!!fInz zI^zN17}7SjFVb!1z{~T*VRBc%^g^MK%2i4t}bR zVpdC=#Yx{aXOw4J=3;~^AP${;W@ZN9291C7%{RtN+`W7Euw8z|6ptl$beD(Ne&NCe zDD7v@o(0K+*vrey)oS(o@4t^~9F6M0H%J>L8!hKB$$_-7a{eN9!}$Ie&D>bv*zEon z4u`et~udy2_y%U+w2Fh0?3~AxP_CNw=Gd_D&M@sFFmoi91ug)RG(T7$oni8(S2rG}b|7PnMPa7IwaIswYiyi0u?> z-oFKG1zAt`j^h-lOe-5gHl?{F+e%1OhNxKAGaK1&#!qceZ{+8yDUSZ_3KClqQym;j z&q}u)7_N0)m)R;DHQ!-4St literal 0 HcmV?d00001 diff --git a/campaign/event_20251218_cn/a1.py b/campaign/event_20251218_cn/a1.py new file mode 100644 index 000000000..7e692435f --- /dev/null +++ b/campaign/event_20251218_cn/a1.py @@ -0,0 +1,96 @@ +from module.campaign.campaign_base import CampaignBase +from module.map.map_base import CampaignMap +from module.map.map_grids import SelectedGrids, RoadGrids +from module.logger import logger + +MAP = CampaignMap('A1') +MAP.shape = 'J8' +MAP.camera_data = ['D3', 'D6', 'G3', 'G6'] +MAP.camera_data_spawn_point = ['G2'] +MAP.map_data = """ + ++ -- -- -- -- -- ++ -- SP -- + ++ ME ++ ++ ++ Me -- -- -- SP + -- -- -- -- Me -- -- MS -- -- + ME -- ME -- -- -- __ -- ++ ++ + ++ -- ++ ++ -- MS -- -- ++ ++ + -- -- -- ++ ME -- ++ Me -- -- + -- MB -- -- -- -- -- -- -- -- + -- -- -- ++ ME -- ME ++ ++ ++ +""" +MAP.weight_data = """ + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 +""" +MAP.spawn_data = [ + {'battle': 0, 'enemy': 2, 'siren': 1}, + {'battle': 1, 'enemy': 1}, + {'battle': 2, 'enemy': 1}, + {'battle': 3, 'enemy': 1, 'boss': 1}, + {'battle': 4, 'enemy': 1}, +] +A1, B1, C1, D1, E1, F1, G1, H1, I1, J1, \ +A2, B2, C2, D2, E2, F2, G2, H2, I2, J2, \ +A3, B3, C3, D3, E3, F3, G3, H3, I3, J3, \ +A4, B4, C4, D4, E4, F4, G4, H4, I4, J4, \ +A5, B5, C5, D5, E5, F5, G5, H5, I5, J5, \ +A6, B6, C6, D6, E6, F6, G6, H6, I6, J6, \ +A7, B7, C7, D7, E7, F7, G7, H7, I7, J7, \ +A8, B8, C8, D8, E8, F8, G8, H8, I8, J8, \ + = MAP.flatten() + + +class Config: + # ===== Start of generated config ===== + MAP_SIREN_TEMPLATE = ['Vboss_Star'] + MOVABLE_ENEMY_TURN = (2,) + MAP_HAS_SIREN = True + MAP_HAS_MOVABLE_ENEMY = True + MAP_HAS_MAP_STORY = False + MAP_HAS_FLEET_STEP = True + MAP_HAS_AMBUSH = False + MAP_HAS_MYSTERY = False + # ===== End of generated config ===== + + MAP_CHAPTER_SWITCH_20241219 = True + STAGE_ENTRANCE = ['half', '20240725'] + MAP_HAS_MODE_SWITCH = True + STAGE_INCREASE_AB = True + INTERNAL_LINES_FIND_PEAKS_PARAMETERS = { + 'height': (80, 255 - 17), + 'width': (0.9, 10), + 'prominence': 10, + 'distance': 35, + } + EDGE_LINES_FIND_PEAKS_PARAMETERS = { + 'height': (255 - 17, 255), + 'prominence': 10, + 'distance': 50, + 'wlen': 1000 + } + HOMO_EDGE_COLOR_RANGE = (0, 17) + MAP_WALK_USE_CURRENT_FLEET = True + MAP_SWIPE_MULTIPLY = (1.232, 1.255) + MAP_SWIPE_MULTIPLY_MINITOUCH = (1.191, 1.213) + MAP_SWIPE_MULTIPLY_MAATOUCH = (1.156, 1.177) + + +class Campaign(CampaignBase): + MAP = MAP + ENEMY_FILTER = '1L > 1M > 1E > 1C > 2L > 2M > 2E > 2C > 3L > 3M > 3E > 3C' + + def battle_0(self): + if self.clear_siren(): + return True + if self.clear_filter_enemy(self.ENEMY_FILTER, preserve=0): + return True + + return self.battle_default() + + def battle_3(self): + return self.clear_boss() diff --git a/campaign/event_20251218_cn/a2.py b/campaign/event_20251218_cn/a2.py new file mode 100644 index 000000000..40e981981 --- /dev/null +++ b/campaign/event_20251218_cn/a2.py @@ -0,0 +1,75 @@ +from module.campaign.campaign_base import CampaignBase +from module.map.map_base import CampaignMap +from module.map.map_grids import SelectedGrids, RoadGrids +from module.logger import logger +from .a1 import Config as ConfigBase + +MAP = CampaignMap('A2') +MAP.shape = 'J8' +MAP.camera_data = ['E3', 'E6', 'F3', 'F6'] +MAP.camera_data_spawn_point = ['E6'] +MAP.map_data = """ + -- -- -- -- -- ++ ME ++ -- -- + -- -- MB -- -- ME -- ME -- -- + ++ ++ -- ++ ++ -- -- -- ME -- + ++ -- __ -- ++ Me -- ++ ++ ++ + -- MS -- MS -- -- -- Me -- -- + -- -- MS -- -- Me -- -- ME -- + ++ -- -- -- -- ++ -- ME ++ ++ + ++ SP -- SP -- ME -- -- ++ ++ +""" +MAP.weight_data = """ + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 +""" +MAP.spawn_data = [ + {'battle': 0, 'enemy': 2, 'siren': 1}, + {'battle': 1, 'enemy': 1}, + {'battle': 2, 'enemy': 1}, + {'battle': 3, 'enemy': 1}, + {'battle': 4, 'enemy': 1, 'boss': 1}, +] +A1, B1, C1, D1, E1, F1, G1, H1, I1, J1, \ +A2, B2, C2, D2, E2, F2, G2, H2, I2, J2, \ +A3, B3, C3, D3, E3, F3, G3, H3, I3, J3, \ +A4, B4, C4, D4, E4, F4, G4, H4, I4, J4, \ +A5, B5, C5, D5, E5, F5, G5, H5, I5, J5, \ +A6, B6, C6, D6, E6, F6, G6, H6, I6, J6, \ +A7, B7, C7, D7, E7, F7, G7, H7, I7, J7, \ +A8, B8, C8, D8, E8, F8, G8, H8, I8, J8, \ + = MAP.flatten() + + +class Config(ConfigBase): + # ===== Start of generated config ===== + MAP_SIREN_TEMPLATE = ['Vboss_Star'] + MOVABLE_ENEMY_TURN = (2,) + MAP_HAS_SIREN = True + MAP_HAS_MOVABLE_ENEMY = True + MAP_HAS_MAP_STORY = False + MAP_HAS_FLEET_STEP = True + MAP_HAS_AMBUSH = False + MAP_HAS_MYSTERY = False + # ===== End of generated config ===== + + +class Campaign(CampaignBase): + MAP = MAP + ENEMY_FILTER = '1L > 1M > 1E > 1C > 2L > 2M > 2E > 2C > 3L > 3M > 3E > 3C' + + def battle_0(self): + if self.clear_siren(): + return True + if self.clear_filter_enemy(self.ENEMY_FILTER, preserve=0): + return True + + return self.battle_default() + + def battle_4(self): + return self.clear_boss() diff --git a/campaign/event_20251218_cn/a3.py b/campaign/event_20251218_cn/a3.py new file mode 100644 index 000000000..368b00817 --- /dev/null +++ b/campaign/event_20251218_cn/a3.py @@ -0,0 +1,81 @@ +from module.campaign.campaign_base import CampaignBase +from module.map.map_base import CampaignMap +from module.map.map_grids import SelectedGrids, RoadGrids +from module.logger import logger +from .a1 import Config as ConfigBase + +MAP = CampaignMap('A3') +MAP.shape = 'K10' +MAP.camera_data = ['F3', 'F6'] +MAP.camera_data_spawn_point = ['F6'] +MAP.map_data = """ + -- ++ ++ ++ -- -- -- ++ ++ ++ -- + -- ++ -- Me -- MB -- Me -- ++ -- + -- -- ME -- Me -- Me -- ME -- -- + -- -- ++ MS -- __ -- MS ++ -- -- + -- -- ++ -- -- MS -- -- ++ -- -- + ++ ++ ME -- -- -- -- -- ME ++ ++ + ++ ++ -- -- SP -- SP -- -- ++ ++ + -- -- ME -- ++ ++ ++ -- ME -- -- + -- ++ -- -- ++ ++ ++ -- -- ++ -- + -- -- -- -- ++ ++ ++ -- -- -- -- +""" +MAP.weight_data = """ + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 + 50 50 50 50 50 50 50 50 50 50 50 +""" +MAP.spawn_data = [ + {'battle': 0, 'enemy': 2, 'siren': 1}, + {'battle': 1, 'enemy': 1}, + {'battle': 2, 'enemy': 1}, + {'battle': 3, 'enemy': 1}, + {'battle': 4, 'enemy': 1, 'boss': 1}, +] +A1, B1, C1, D1, E1, F1, G1, H1, I1, J1, K1, \ +A2, B2, C2, D2, E2, F2, G2, H2, I2, J2, K2, \ +A3, B3, C3, D3, E3, F3, G3, H3, I3, J3, K3, \ +A4, B4, C4, D4, E4, F4, G4, H4, I4, J4, K4, \ +A5, B5, C5, D5, E5, F5, G5, H5, I5, J5, K5, \ +A6, B6, C6, D6, E6, F6, G6, H6, I6, J6, K6, \ +A7, B7, C7, D7, E7, F7, G7, H7, I7, J7, K7, \ +A8, B8, C8, D8, E8, F8, G8, H8, I8, J8, K8, \ +A9, B9, C9, D9, E9, F9, G9, H9, I9, J9, K9, \ +A10, B10, C10, D10, E10, F10, G10, H10, I10, J10, K10, \ + = MAP.flatten() + + +class Config(ConfigBase): + # ===== Start of generated config ===== + MAP_SIREN_TEMPLATE = ['Vboss_Star'] + MOVABLE_ENEMY_TURN = (2,) + MAP_HAS_SIREN = True + MAP_HAS_MOVABLE_ENEMY = True + MAP_HAS_MAP_STORY = False + MAP_HAS_FLEET_STEP = True + MAP_HAS_AMBUSH = False + MAP_HAS_MYSTERY = False + # ===== End of generated config ===== + + +class Campaign(CampaignBase): + MAP = MAP + ENEMY_FILTER = '1L > 1M > 1E > 1C > 2L > 2M > 2E > 2C > 3L > 3M > 3E > 3C' + + def battle_0(self): + if self.clear_siren(): + return True + if self.clear_filter_enemy(self.ENEMY_FILTER, preserve=0): + return True + + return self.battle_default() + + def battle_4(self): + return self.clear_boss() diff --git a/dev_tools/map_extractor.py b/dev_tools/map_extractor.py index f9e1eb44a..b61bbf6cd 100644 --- a/dev_tools/map_extractor.py +++ b/dev_tools/map_extractor.py @@ -311,6 +311,12 @@ DIC_SIREN_NAME_CHI_TO_ENG = { 'jiulaimu_shixianggui': 'Jiulaimu_Statue', 'jiulaimu_emo': 'Jiulaimu_Demon', 'youlin_ylsb': 'Jiulaimu_Ghost', + + # A Note Through the Firmament + 'unknownV_boss_star': 'Vboss_Star', + 'unknownV_boss_hermit': 'Vboss_Hermit', + 'unknownV_boss_lovers': 'Vboss_Lovers', + 'unknownV_boss_chariot': 'Vboss_Chariot', } diff --git a/module/template/assets.py b/module/template/assets.py index 59b6b26bd..05b2a6d12 100644 --- a/module/template/assets.py +++ b/module/template/assets.py @@ -239,6 +239,7 @@ TEMPLATE_SIREN_U81 = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_U TEMPLATE_SIREN_UDFFortress2 = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_UDFFortress2.gif', 'en': './assets/en/template/TEMPLATE_SIREN_UDFFortress2.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_UDFFortress2.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_UDFFortress2.gif'}) TEMPLATE_SIREN_UlrichVonHutten = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_UlrichVonHutten.gif', 'en': './assets/en/template/TEMPLATE_SIREN_UlrichVonHutten.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_UlrichVonHutten.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_UlrichVonHutten.gif'}) TEMPLATE_SIREN_Vauquelin = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_Vauquelin.gif', 'en': './assets/en/template/TEMPLATE_SIREN_Vauquelin.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_Vauquelin.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_Vauquelin.gif'}) +TEMPLATE_SIREN_Vboss_Star = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_Vboss_Star.gif', 'en': './assets/en/template/TEMPLATE_SIREN_Vboss_Star.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_Vboss_Star.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_Vboss_Star.gif'}) TEMPLATE_SIREN_Victorious = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_Victorious.gif', 'en': './assets/cn/template/TEMPLATE_SIREN_Victorious.gif', 'jp': './assets/cn/template/TEMPLATE_SIREN_Victorious.gif', 'tw': './assets/cn/template/TEMPLATE_SIREN_Victorious.gif'}) TEMPLATE_SIREN_Volga = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_Volga.gif', 'en': './assets/en/template/TEMPLATE_SIREN_Volga.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_Volga.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_Volga.gif'}) TEMPLATE_SIREN_Wakaba = Template(file={'cn': './assets/cn/template/TEMPLATE_SIREN_Wakaba.gif', 'en': './assets/en/template/TEMPLATE_SIREN_Wakaba.gif', 'jp': './assets/jp/template/TEMPLATE_SIREN_Wakaba.gif', 'tw': './assets/tw/template/TEMPLATE_SIREN_Wakaba.gif'})