engine-bridge v3 live: /scene /camera /selection all real (#10)
Live end-to-end verification against EG-BIM Modeler succeeded on the
second attempt:
/health -> {"status":"ok","port":38080}
/scene -> {"object_count":4,"document_path":"NewSpace0"}
/camera -> {"eye":[192.97,-328.52,170.72],
"target":[33.03,-72.61,10.78],
"up":[0,0,1],"fov":45}
/selection -> {"selected_ids":["ac0380a2-...","d9a287ee-..."]}
1st attempt returned default-zero camera. Root cause: the viewport lambda
used EditorPlugin.View, which is only populated when the plugin is
actually Run() by a user trigger; our bridge plugin just boots an HTTP
server from its constructor and never runs a command, so View stayed
null. Space access worked because RootSpace goes through AppManager,
which is populated for the whole app.
Fix (HmEgBridgePlugin.BuildProvider):
Before: viewportProvider = () => View;
After: viewportProvider = () => {
var vm = AppManager?.ViewportManager;
if (vm is null) return null;
return vm.FocusedViewport ?? vm.Viewports.FirstOrDefault();
};
Confirmed against read-only view of
HmEGApplicationManagementLibrary/SubManager/ViewportManager.cs
which exposes FocusedViewport and Viewports. EGViewport : HmEGViewport
so the lambda matches the Func<HmEGViewport?> contract directly.
Plus: scripts/deploy-egbim-plugin.bat for one-click deploy. Checks for
a running SUT, builds Debug, purges the legacy Recordingtest.EgPlugin
folder, cleans the destination, copies 3 DLLs (+ PDBs) into
EG-BIM Modeler/Plugins/Recordingtest.Sut.EgBim.PluginHost/
and prints the curl commands for verification. HmEG.dll and the
Editor*.dll assemblies are deliberately NOT copied — the SUT already
supplies them.
PROGRESS.md: engine-bridge v3 row finalized; the long-running "라이브
검증 대기" item is done. PLAN.md P1 advances to the Runner <-> sidecar
integration (snapshot /scene /camera /selection at scenario end and
include in the golden baseline).
Follow-up (noted in history): document_path returned "NewSpace0" for
an unsaved scratch document — need to retest with a saved .hmeg file
to confirm the real FileManager.CurrentFile round-trip.
Ref: #10 follow-up, engine-bridge-v3 contract DoD D7 satisfied.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -73,14 +73,25 @@ public sealed class HmEgBridgePlugin : EditorPlugin, IDisposable
|
||||
|
||||
Func<HmEGViewport?> viewportProvider = () =>
|
||||
{
|
||||
// EditorPlugin.View is only populated when the plugin is actually
|
||||
// Run() by a user command; our bridge plugin lives as a long-
|
||||
// running HTTP server and never runs a trigger, so View stays
|
||||
// null. Instead pull the active viewport from the global
|
||||
// ViewportManager, preferring FocusedViewport, then falling back
|
||||
// to any registered viewport. EGViewport implements HmEGViewport.
|
||||
try
|
||||
{
|
||||
// EGViewport implements HmEGViewport; the base class exposes it
|
||||
// as EGViewport on the Obsolete View property. We catch and
|
||||
// return null to survive the obsolete warning at runtime.
|
||||
#pragma warning disable CS0618 // Obsolete API on EditorPlugin.View
|
||||
return View;
|
||||
#pragma warning restore CS0618
|
||||
var vm = AppManager?.ViewportManager;
|
||||
if (vm is null) return null;
|
||||
var focused = vm.FocusedViewport;
|
||||
if (focused is not null) return focused;
|
||||
var any = vm.Viewports;
|
||||
if (any is null) return null;
|
||||
foreach (var v in any)
|
||||
{
|
||||
if (v is not null) return v;
|
||||
}
|
||||
return null;
|
||||
}
|
||||
catch { return null; }
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user