feat(battle-engine M13): M3 spell emits PlayActions headless via OperateMgr -> NetworkBattleSender (O1 read = GO)

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
gamer147
2026-06-06 12:23:51 -04:00
parent 25e9ae9573
commit ac0886389a
5 changed files with 179 additions and 11 deletions

View File

@@ -212,7 +212,10 @@ namespace Wizard.Battle.View {
ITurnEndButtonUI global::Wizard.Battle.View.IBattlePlayerView.TurnEndButtonUI { get => default!; }
GameObject global::Wizard.Battle.View.IBattlePlayerView.EpIcon { get => default!; }
bool global::Wizard.Battle.View.IBattlePlayerView.IsSelecting { get => default!; }
HandViewBase global::Wizard.Battle.View.IBattlePlayerView.HandView { get => default!; }
// HEADLESS FILL (M13): generator emitted `default!` (null); the OperateMgr emit path calls
// BattleView.HandView.RemoveCardFromView (BattlePlayerBase.cs:1422). Redirect to a shared no-op
// HandViewBase so the presentation call is a safe no-op (the played card is never in its list).
HandViewBase global::Wizard.Battle.View.IBattlePlayerView.HandView { get => global::Wizard.Battle.View.HeadlessHandViewStub.Instance; }
HandControl global::Wizard.Battle.View.IBattlePlayerView.HandControl { get => default!; }
BattleCardBase global::Wizard.Battle.View.IBattlePlayerView.SelectSkillActCard { get => default!; }
GameObject global::Wizard.Battle.View.IBattlePlayerView.TurnEndBtn { get => default!; }
@@ -312,7 +315,10 @@ namespace Wizard.Battle.View {
ITurnEndButtonUI global::Wizard.Battle.View.IBattlePlayerView.TurnEndButtonUI { get => default!; }
GameObject global::Wizard.Battle.View.IBattlePlayerView.EpIcon { get => default!; }
bool global::Wizard.Battle.View.IBattlePlayerView.IsSelecting { get => default!; }
HandViewBase global::Wizard.Battle.View.IBattlePlayerView.HandView { get => default!; }
// HEADLESS FILL (M13): generator emitted `default!` (null); the OperateMgr emit path calls
// BattleView.HandView.RemoveCardFromView (BattlePlayerBase.cs:1422). Redirect to a shared no-op
// HandViewBase so the presentation call is a safe no-op (the played card is never in its list).
HandViewBase global::Wizard.Battle.View.IBattlePlayerView.HandView { get => global::Wizard.Battle.View.HeadlessHandViewStub.Instance; }
HandControl global::Wizard.Battle.View.IBattlePlayerView.HandControl { get => default!; }
BattleCardBase global::Wizard.Battle.View.IBattlePlayerView.SelectSkillActCard { get => default!; }
GameObject global::Wizard.Battle.View.IBattlePlayerView.TurnEndBtn { get => default!; }

View File

@@ -0,0 +1,28 @@
// AUTHORED SHIM (not copied). A non-null no-op HandViewBase the headless emit path needs.
//
// On the OperateMgr/network play path, BattlePlayerBase.SetupActionProcessorEvent subscribes an
// OnBeforePlayCard handler that calls BattleView.HandView.RemoveCardFromView (BattlePlayerBase.cs:1422) —
// a pure presentation-layer hand-card removal. The m1_stub_gen generated IBattlePlayerView.HandView getter
// returns default! (null), so the call NREs. The direct-ActionProcessor solo oracles never hit this
// (SetupActionProcessorEvent is OperateMgr-only). Seed a single shared no-op HandViewBase whose card-view
// list is non-null so RemoveCardFromView is a safe no-op (the played card is never in this view's list, so
// the abstract RearrangeHand is never reached). Nothing here touches game state.
using UnityEngine;
namespace Wizard.Battle.View
{
public sealed class HeadlessHandViewStub : HandViewBase
{
// Shared instance the generated IBattlePlayerView.HandView getters return headless.
public static readonly HeadlessHandViewStub Instance = new HeadlessHandViewStub();
// Base param ctor initializes the protected _battleCardViewList (the default ctor leaves it null,
// which RemoveCardFromView would NRE on). CreateHandControl is overridden to a null no-op below.
public HeadlessHandViewStub() : base(null, null) { }
protected override void RearrangeHand(float rearrangeTime, bool isNewReplayMoveTurn = false) { }
protected override HandControl CreateHandControl(GameObject handGameObject, BattleCamera battleCamera) => null;
}
}