Skip to content

2D Overlay System

hifihedgehog edited this page Mar 3, 2026 · 18 revisions

2D Overlay System

The 2D controller visualization uses PNG overlay images positioned on a Canvas to show controller state. Supports Xbox 360 and DualShock 4 layouts. For custom vJoy controllers (non-gamepad presets), a procedurally-generated schematic view is used instead.


Architecture Overview

overlay_positions.py          (SVG parsing + alpha-match refinement)
        |
        v
ControllerOverlayLayout.cs   (auto-generated C# layout data)
        |
        v
ControllerModel2DView.xaml.cs (2D PNG overlay view)

VJoySlotConfig.ComputeAxisLayout()
        |
        v
ControllerSchematicView.xaml.cs (procedural vJoy view)

When Each View Is Used

The PadPage.ApplyViewMode() method determines which view to show:

Condition View Toggle
Custom vJoy (VJoyConfig.IsGamepadPreset == false) ControllerSchematicView Always shown, 2D/3D toggle hidden
Xbox 360 / DS4 / vJoy gamepad preset, Use2DControllerView == true ControllerModel2DView 2D/3D toggle visible
Xbox 360 / DS4 / vJoy gamepad preset, Use2DControllerView == false ControllerModelView (3D) 2D/3D toggle visible

ControllerOverlayLayout.cs

File: PadForge.App/Models2D/ControllerOverlayLayout.cs

Auto-generated by tools/overlay_positions.py. Do not edit manually.

Types

public enum OverlayElementType
{
    Button,         // Show/hide on press
    Trigger,        // Clip-based fill level
    StickRing,      // Translates with stick input
    StickClick,     // Quadrant highlight for hover/flash
    FaceButtonGroup // Reserved (unused)
}

public record OverlayElement(
    string ImageFile,
    string TargetName,
    OverlayElementType ElementType,
    double X,
    double Y,
    double Width,
    double Height
);

Xbox360Layout

public static class Xbox360Layout
{
    public const int BaseWidth = 1545;
    public const int BaseHeight = 955;
    public const string BasePath = "2DModels/XBOX360/XB360_base.png";
    public const double StickMaxTravel = 30;
    public static readonly OverlayElement[] Overlays;  // 19 elements
}

Overlay Elements (19 total):

TargetName ElementType Image Position (X,Y) Size (WxH)
ButtonA Button XB360_A_Button.png 1178, 528 127x106
ButtonB Button XB360_B_Button.png 1312, 415 122x115
ButtonX Button XB360_X_Button.png 1058, 423 126x113
ButtonY Button XB360_Y_Button.png 1190, 314 129x118
LeftShoulder Button XB360_LeftBumper_Active.png 138, 134 312x141
RightShoulder Button XB360_RightBumper_Active.png 1125, 131 285x141
LeftTrigger Trigger XB360_LeftTrigger_Active.png 273, 26 143x152
RightTrigger Trigger XB360_RightTrigger_Active.png 1160, 26 143x152
ButtonBack Button XB360_BackButton.png 546, 439 92x65
ButtonStart Button XB360_StartButton.png 914, 440 92x65
ButtonGuide Button XB360_GuideButton.png 689, 414 171x139
LeftThumbRing StickRing XB360_LeftStick.png 192, 439 211x185
RightThumbRing StickRing XB360_RightStick.png 892, 677 211x178
LeftThumbButton StickClick XB360_LeftStick_Click.png 180, 434 233x195
RightThumbButton StickClick XB360_RightStick_Click.png 884, 662 232x191
DPadUp Button XB360_D-PAD_Up.png 484, 592 108x114
DPadDown Button XB360_D-PAD_Down.png 484, 729 108x114
DPadLeft Button XB360_D-PAD_Left.png 383, 664 134x108
DPadRight Button XB360_D-PAD_Right.png 558, 664 134x107

DS4Layout

public static class DS4Layout
{
    public const int BaseWidth = 1466;
    public const int BaseHeight = 783;
    public const string BasePath = "2DModels/DS4/DS4_V2_base.png";
    public const double StickMaxTravel = 25;
    public static readonly OverlayElement[] Overlays;  // 19 elements
}

DS4 reuses DS4_Face_Button.png for all four face buttons (same highlight overlay at different positions). Also reuses DS4_OptionsShare_Button.png for both Back and Start, and DS4_AnalogStick_Click.png for both stick clicks.


PNG Asset Structure

PadForge.App/2DModels/
  XBOX360/  (21 images)
    XB360_base.png                        (1545x955, base controller image)
    Xbox 360 Controller Overlay.png       (composite overlay for refinement tool)
    XB360_A_Button.png, XB360_B_Button.png, ...
    XB360_LeftBumper_Active.png, XB360_RightBumper_Active.png
    XB360_LeftTrigger_Active.png, XB360_RightTrigger_Active.png
    XB360_LeftStick.png, XB360_RightStick.png
    XB360_LeftStick_Click.png, XB360_RightStick_Click.png
    XB360_D-PAD_Up/Down/Left/Right.png
    XB360_BackButton.png, XB360_StartButton.png, XB360_GuideButton.png
  DS4/  (16 images)
    DS4_V2_base.png                       (1466x783, base controller image)
    DualShock 4 Controller V2 Model Overlay.png  (composite overlay for refinement tool)
    DS4_Face_Button.png                   (single image for all 4 face buttons)
    DS4_D-PAD_Up/Down/Left/Right.png
    DS4_L1-Active.png, DS4_R1-Active.png
    DS4_L2-Active.png, DS4_R2-Active.png
    DS4_OptionsShare_Button.png           (shared for Back and Start)
    DS4_Home_Button.png
    DS4_V2_LeftAnalogStick.png, DS4_V2_RightAnalogStick.png
    DS4_AnalogStick_Click.png             (shared for both stick clicks)

All PNG files are included as WPF Resource (not EmbeddedResource), loaded via pack:// URI:

<Resource Include="2DModels\**\*.png" />

Source artwork: Gamepad-Asset-Pack by AL2009man (MIT license).


ControllerModel2DView

File: PadForge.App/Views/ControllerModel2DView.xaml, ControllerModel2DView.xaml.cs

WPF UserControl with a Canvas inside a Viewbox for resolution-independent scaling. ~636 lines of code-behind.

XAML Structure

<Viewbox Stretch="Uniform">
    <Canvas x:Name="ModelCanvas" ClipToBounds="True" />
</Viewbox>

The Viewbox scales the canvas uniformly to fit the available space. The canvas dimensions are set to match the layout's BaseWidth/BaseHeight (e.g., 1545x955 for Xbox 360).

Events

public event EventHandler<string> ControllerElementRecordRequested;

Private State

Field Type Description
_vm PadViewModel Bound ViewModel
_loadedModel string "XBOX360" or "DS4"
_dirty bool Render-frame update flag
_baseImage Image Base controller image at Z=0
_overlayImages Dictionary<string, Image> Target name to overlay Image element
_stickTransforms Dictionary<string, TranslateTransform> Target name to stick translation transform
_triggerClips Dictionary<string, RectangleGeometry> Target name to trigger clip geometry
_elementTypes Dictionary<string, OverlayElementType> Target name to element type
_stickHighlights Dictionary<string, Image> Ring target to stick click overlay image (for quadrant highlights)
_stickMaxTravel double Maximum stick overlay travel in pixels
_flashTimer DispatcherTimer Flash animation timer (400ms)
_flashTarget string Resolved flash target (ring for axes)
_flashRawTarget string Original target before resolution (e.g., "LeftThumbAxisXNeg")
_flashStickClip Geometry Stored quadrant clip for stick flash
_flashOn bool Current flash toggle state
_hoverTarget string Currently hovered target

ViewModel Binding

public void Bind(PadViewModel vm)
public void Unbind()

Bind subscribes to PropertyChanged, hooks CompositionTarget.Rendering, calls EnsureModel(). Unbind stops flash, unhooks rendering, clears VM reference.

Model Selection

private void EnsureModel()

Same logic as the 3D view:

  • DualShock4 -> "DS4"
  • VJoy with Preset == DualShock4 -> "DS4"
  • All others -> "XBOX360"

If the model hasn't changed, returns immediately. Otherwise calls BuildCanvas().

BuildCanvas()

private void BuildCanvas(string modelName)

Clears the canvas and rebuilds from layout data. Creates four layers at different Z-indices:

Layer 0 -- Base Image:

_baseImage = CreateImage(basePath, 0, 0, baseW, baseH);

Layer 1 -- Overlay Images: For each OverlayElement in the layout:

ElementType Initial State Transform/Clip
StickRing Visible TranslateTransform stored in _stickTransforms
Trigger Visible, full clip RectangleGeometry clip stored in _triggerClips (starts at bottom = empty fill)
Button Collapsed None (toggled by SetOverlayVisible())
StickClick Collapsed None (used as source for quadrant highlights)

All overlay images have IsHitTestVisible = false -- clicks go through to hit-test rectangles.

Layer 10 -- Hit-Test Rectangles: Transparent Rectangle elements with Cursor = Hand and Tag = TargetName. Event handlers:

  • MouseLeftButtonDown -> HitArea_Click
  • MouseEnter -> HitArea_MouseEnter
  • MouseLeave -> HitArea_MouseLeave
  • MouseMove -> StickHitArea_MouseMove (stick rings only)

StickClick elements have no hit rect -- they're handled by the StickRing's center-click detection.

Layer 5 -- Stick Quadrant Highlights: Created from StickClick overlay images at 40% opacity. Initially collapsed. Used for hover and flash quadrant visualization.

Image Loading

private static Image CreateImage(string resourcePath, double x, double y, double w, double h)

Loads PNG via pack://application:,,,/{resourcePath} URI into a BitmapImage. Positions with Canvas.SetLeft/SetTop.

Per-Frame Updates

CompositionTarget.Rendering handler, gated by _dirty:

UpdateButtons()

Sets overlay Visibility for 15 button targets:

SetOverlayVisible("ButtonA", _vm.ButtonA);
SetOverlayVisible("ButtonB", _vm.ButtonB);
// ... 13 more

SetOverlayVisible() skips elements currently being flash-animated or hovered.

UpdateTriggers()

Adjusts RectangleGeometry clip to create a gas tank fill effect (fills from bottom to top):

double clipY = h * (1.0 - v);  // v = 0: empty (clip at bottom), v = 1: full (clip at top)
clip.Rect = new Rect(0, clipY, w, h - clipY);

UpdateSticks()

Updates TranslateTransform.X/Y from normalized stick values:

lt.X = lx * _stickMaxTravel;
lt.Y = -ly * _stickMaxTravel;  // Y inverted for screen coordinates

Raw short values (-32768..32767) are normalized to -1..1. StickMaxTravel is 30px for Xbox 360, 25px for DS4.

Click-to-Record

private void HitArea_Click(object sender, MouseButtonEventArgs e)

For non-stick elements, fires ControllerElementRecordRequested directly with the Tag value.

For stick rings, calls DetermineAxisFromQuadrant():

private static string DetermineAxisFromQuadrant(
    Point pos, double w, double h, string stickTarget)

Determines axis from click position relative to ring center:

  • Center (distance < 30% of radius): Returns LeftThumbButton or RightThumbButton.
  • Dominant X (|dx| >= |dy|): Returns LeftThumbAxisX or LeftThumbAxisXNeg.
  • Dominant Y: Returns LeftThumbAxisY or LeftThumbAxisYNeg.

Down = positive Y direction (screen coordinates). Step 3's NegateAxis inverts this so that screen-down maps to game-down.

Hover Highlight

Buttons: HitArea_MouseEnter shows overlay at 40% opacity. HitArea_MouseLeave triggers _dirty = true to restore proper state on next render frame.

Triggers: During hover, clip is opened to full image (Rect(0, 0, w, h)).

Stick Rings: StickHitArea_MouseMove tracks mouse position and creates a CombinedGeometry clip on the stick highlight image:

  1. Full ellipse (ring boundary)
  2. Minus center ellipse (30% radius -- this is the stick button area)
  3. Intersected with half-rectangle based on dominant axis direction

This produces a quadrant wedge that follows the mouse:

var quadrant = new CombinedGeometry(GeometryCombineMode.Intersect,
    fullEllipse, new RectangleGeometry(halfRect));
clip = new CombinedGeometry(GeometryCombineMode.Exclude,
    quadrant, centerEllipse);

For center hover (distance < 30%), shows just the center ellipse.

Flash Animation (Map All)

private void UpdateFlashTarget(string target)

Target resolution:

  • Stick axis targets resolve to ring: "LeftThumbAxisX" -> "LeftThumbRing".
  • Stick button targets also resolve to ring: "LeftThumbButton" -> "LeftThumbRing".
  • All others pass through.

Flash timer: 400ms DispatcherTimer toggles _flashOn.

Element Type Flash On Flash Off
Stick axes Quadrant highlight visible (with stored clip) Quadrant highlight collapsed
Stick ring Opacity 1.0 Opacity 0.2
Buttons Overlay visible, full opacity Overlay collapsed
Triggers Clip opened to full fill (restored by StopFlash)

Stick quadrant clip for flash:

private static Geometry GetStickQuadrantClip(string target, double w, double h)

Returns pre-computed clip geometry for the specified axis direction:

  • LeftThumbButton / RightThumbButton: Center ellipse (30% radius).
  • AxisX / AxisXNeg: Right or left half-ellipse minus center.
  • AxisY / AxisYNeg: Bottom or top half-ellipse minus center.

The clip is stored in _flashStickClip and re-applied on every tick to guard against it being cleared by other interactions.


ControllerSchematicView

File: PadForge.App/Views/ControllerSchematicView.xaml, ControllerSchematicView.xaml.cs

Procedurally generated view for custom vJoy controllers. Displays stick circles, trigger bars, POV compasses, and button grids. ~795 lines of code-behind.

XAML Structure

<Viewbox Stretch="Uniform">
    <Canvas x:Name="SchematicCanvas" ClipToBounds="True" />
</Viewbox>

Same Viewbox > Canvas pattern as the 2D overlay view. Canvas dimensions are computed dynamically to fit all widgets.

Events

public event EventHandler<string> ControllerElementRecordRequested;

Static Brushes

Name Color Usage
AccentBrush #0078D4 Pressed/active elements
FlashBrush #FFA500 Flash animation highlight (orange)
HoverBrush #40A0E0 Hover highlight
DimBrush #606060 Inactive borders
BgBrush #2D2D2D Widget backgrounds
LabelBrush #BBBBBB Text labels
DotBrush #FFFFFF Stick position dot

Layout Constants

const double StickSize = 100;     // Stick circle diameter
const double TriggerWidth = 24;   // Trigger bar width
const double TriggerHeight = 80;  // Trigger bar height
const double PovSize = 60;        // POV circle diameter
const double ButtonSize = 22;     // Button circle diameter
const double SectionGap = 24;     // Horizontal gap between widget sections
const double LabelHeight = 18;    // Space reserved for labels above widgets
const double Padding = 12;        // Canvas edge padding
const int ButtonsPerRow = 8;      // Button grid wrap count

Widget Structs

private struct StickWidget
{
    public int AxisXIndex, AxisYIndex;
    public Ellipse Dot;                    // Position indicator dot
    public Polygon DirectionArrow;         // Flash/hover direction arrow
    public Canvas ArrowCanvas;             // Arrow container for rotation
    public Ellipse OuterCircle;            // Outer boundary circle
    public double X, Y;                    // Canvas position
}

private struct TriggerWidget
{
    public int AxisIndex;
    public Rectangle Background;           // Border rectangle
    public Rectangle Fill;                 // Fill bar (grows from bottom)
    public double X, Y;
}

private struct PovWidget
{
    public int PovIndex;
    public Polygon Arrow;                  // Direction arrow polygon
    public Canvas ArrowCanvas;             // Arrow container for rotation
    public Ellipse Outer;                  // Outer boundary circle
    public double CenterX, CenterY;
}

private struct ButtonWidget
{
    public int ButtonIndex;
    public Ellipse Circle;                 // Button circle
}

ViewModel Binding

public void Bind(PadViewModel vm)
public void Unbind()

In addition to the standard Bind/Unbind pattern, subscribes to _vm.VJoyConfig.PropertyChanged so that layout rebuilds automatically when vJoy config properties (axis counts, button count) change.

Property change handling:

  • VJoyOutputSnapshot -> sets _dirty flag.
  • OutputType -> calls RebuildLayout().
  • CurrentRecordingTarget -> calls UpdateFlashTarget().
  • Any VJoyConfig property -> calls RebuildLayout().

RebuildLayout()

private void RebuildLayout()

Called on bind and when VJoyConfig properties change. Clears all widget lists and canvas children, then creates widgets.

Axis index assignment: Calls VJoySlotConfig.ComputeAxisLayout() to get:

  • stickAxisX[] / stickAxisY[]: HID axis indices for each stick pair.
  • triggerAxis[]: HID axis indices for standalone triggers.

Layout flow (left to right, starting at x = Padding):

+-- Sticks --+-- Triggers --+-- POVs --+
|            |               |          |
| Stick 1    | T1  T2        | D-Pad    |
| [circle]   | [bar][bar]    | [circle] |
|            |               |          |
+---- Buttons (wrapped grid, below main row) ----+
| [1][2][3][4][5][6][7][8]                        |
| [9][10][11]...                                  |
+------------------------------------------------+

Canvas dimensions are set to fit all widgets plus padding, enabling Viewbox auto-scaling.

Widget Creation Methods

CreateStickWidget

private StickWidget CreateStickWidget(int index, int axisXIdx, int axisYIdx, double x, double y)

Creates:

  1. Outer circle (Ellipse): Dim stroke, dark background, hand cursor.
  2. Crosshair lines (Line x2): Horizontal and vertical at 50% opacity.
  3. Position dot (Ellipse): 10px accent-colored dot at center.
  4. Direction arrow (Polygon inside Canvas): Hidden until flash/hover. Rotated via RotateTransform centered on the stick.
  5. Label (TextBlock): "Stick N" above the circle.

Hover: MouseMove on the outer circle determines quadrant (right/left/down/up) from mouse position, shows direction arrow with HoverBrush, highlights stroke.

Click-to-record: MouseLeftButtonDown uses quadrant detection:

  • |dx| > |dy| and dx > 0 -> VJoyAxis{axisXIdx} (positive X)
  • |dx| > |dy| and dx <= 0 -> VJoyAxis{axisXIdx}Neg (negative X)
  • |dy| > |dx| and dy > 0 -> VJoyAxis{axisYIdx} (positive Y = down)
  • |dy| > |dx| and dy <= 0 -> VJoyAxis{axisYIdx}Neg (negative Y = up)

CreateTriggerWidget

private TriggerWidget CreateTriggerWidget(int index, int axisIdx, double x, double y)

Creates:

  1. Background (Rectangle): Dim stroke, dark fill, rounded corners, hand cursor.
  2. Fill bar (Rectangle): Accent-colored, initially height 0. Grows from bottom.
  3. Label: "TN" above the bar.

Click-to-record: Fires VJoyAxis{axisIdx}.

CreatePovWidget

private PovWidget CreatePovWidget(int index, double x, double y)

Creates:

  1. Outer circle (Ellipse): Dim stroke, dark background.
  2. Arrow (Polygon inside Canvas): Triangular arrow, rotated around POV center. Initially hidden.
  3. Label: "D-Pad" (if 1 POV) or "POV N" (if multiple).

Hover: Shows direction arrow rotated to the mouse quadrant (0/90/180/270 degrees).

Click-to-record: Quadrant detection fires VJoyPov{index}Up, VJoyPov{index}Down, VJoyPov{index}Left, or VJoyPov{index}Right.

CreateButtonWidget

private ButtonWidget CreateButtonWidget(int index, double x, double y)

Creates:

  1. Circle (Ellipse): Dim stroke, dark background, hand cursor.
  2. Label (TextBlock): Button number (1-indexed) centered in circle.

Click-to-record: Fires VJoyBtn{index}.

Click-to-Record Target Names

Widget Target Format Quadrant-Based
Stick VJoyAxis{X} / VJoyAxis{X}Neg Yes (X vs Y, positive vs negative)
Trigger VJoyAxis{N} No
POV VJoyPov{N}Up / Down / Left / Right Yes (4 cardinal directions)
Button VJoyBtn{N} No

Per-Frame Rendering

CompositionTarget.Rendering handler reads VJoyOutputSnapshot from PadViewModel:

Sticks:

double nx = (raw.Axes[w.AxisXIndex] - (double)short.MinValue) / 65535.0;  // 0-1
double dotX = w.X + nx * (StickSize - 10);

Maps VJoyRawState.Axes[index] from signed short (-32768..32767) to 0-1 normalized. Positions the dot within the stick circle.

Triggers:

double value = (raw.Axes[w.AxisIndex] - (double)short.MinValue) / 65535.0;  // 0-1
double fillH = Math.Clamp(value, 0, 1) * (TriggerHeight - 4);
w.Fill.Height = fillH;
Canvas.SetTop(w.Fill, w.Y + TriggerHeight - 2 - fillH);  // Grows from bottom

POVs:

int povValue = raw.Povs[w.PovIndex];  // Centidegrees (0-35900) or -1
if (povValue >= 0 && povValue <= 36000)
{
    w.Arrow.Visibility = Visibility.Visible;
    w.ArrowCanvas.RenderTransform = new RotateTransform(povValue / 100.0, ...);
}

POV updates are skipped when the widget is hovered or flash-targeted to prevent visual flickering from competing rotations.

Buttons:

bool pressed = raw.IsButtonPressed(w.ButtonIndex);
w.Circle.Fill = pressed ? AccentBrush : BgBrush;

Flash Animation

private void UpdateFlashTarget(string target)
private void ApplyFlashState(bool highlight)

Flash timer at 170ms (~3Hz), faster than the 2D overlay's 400ms to make the schematic more visually responsive.

Strips "Neg" suffix for matching, then checks each widget list:

Widget Flash On Flash Off
Stick Orange stroke, direction arrow visible, arrow rotated to target axis Dim stroke, arrow hidden
Trigger Fill color = FlashBrush (orange) Fill color = AccentBrush (blue)
Button Stroke = FlashBrush, thicker Stroke = DimBrush, normal
POV Arrow visible, FlashBrush fill, rotated to target direction Arrow visible, AccentBrush fill

POV direction mapping:

string dir = target.Substring($"VJoyPov{w.PovIndex}".Length);  // "Up", "Down", "Left", "Right"
double angle = dir switch
{
    "Up" => 0,
    "Right" => 90,
    "Down" => 180,
    "Left" => 270,
    _ => 0
};

overlay_positions.py

File: tools/overlay_positions.py

Python tool that auto-generates ControllerOverlayLayout.cs from Gamepad-Asset-Pack SVG files.

Dependencies

pip install svgpathtools lxml opencv-python numpy

Process

  1. Parse SVG elements -- Reads labeled elements from Xbox 360 and DS4 SVG theme files using lxml. Computes cumulative SVG transforms (translate, scale, matrix) to get pixel-space bounding boxes.

  2. Center overlays -- For each element, loads the corresponding PNG overlay image and centers it on the SVG bounding box center.

  3. Alpha-channel refinement -- Uses OpenCV template matching (cv2.matchTemplate with TM_CCOEFF_NORMED) against a full composite overlay image to refine positions within a 40-pixel search radius. Only accepts matches with confidence > 0.3.

  4. Generate C# -- Outputs the ControllerOverlayLayout.cs file with layout constants, overlay element arrays, and stick travel values.

Key Functions

def parse_transform(transform_str) -> np.ndarray        # SVG transform -> 3x3 matrix
def get_cumulative_transform(elem) -> np.ndarray         # Walk ancestors for total transform
def element_bbox(elem) -> tuple                          # Single element bbox (path/ellipse/rect)
def group_bbox(group_elem) -> tuple                      # Combined bbox of group children
def get_element_pixel_bbox(root, label, scale) -> tuple  # Label lookup + pixel conversion
def center_overlay_on_bbox(bbox, overlay_path) -> tuple  # Center overlay image on bbox
def refine_with_composite(composite_path, results, search_radius=40) -> list  # Template matching
def process_xbox360() -> dict                            # Full Xbox 360 pipeline
def process_ds4() -> dict                                # Full DS4 pipeline
def generate_csharp(xbox_data, ds4_data, output_path)    # C# code generation

Usage

python tools/overlay_positions.py

Expects Gamepad-Asset-Pack/Controller Asset Pack/ to be a sibling of the PadForge repository directory.


How 2D and 3D Views Relate

Both the 2D overlay view and 3D model view:

  1. Share the same ViewModel interface: Both read the same PadViewModel properties (ButtonA, RawThumbLX, LeftTrigger, etc.) and fire the same ControllerElementRecordRequested event.

  2. Use the same Bind/Unbind pattern: PadPage.BindActiveModelView() calls Unbind() on all three views, then Bind(vm) on the active one. This ensures only one view is processing CompositionTarget.Rendering at a time.

  3. Use the same dirty-flag rendering: CompositionTarget.Rendering handler gated by _dirty flag, set by PropertyChanged.

  4. Support the same interactions: Click-to-record, hover highlight, flash animation (Map All). Same PadSetting target names for all.

  5. Use the same model selection logic: EnsureModel() in both views maps OutputType + VJoyConfig.Preset to "XBOX360" or "DS4".

The key differences:

Aspect 3D View 2D View Schematic View
Technology HelixToolkit.WPF Canvas + BitmapImage Canvas + WPF Shapes
Rotation Right-drag turntable None None
Quadrant detection 3D ray-cast + position 2D mouse position 2D mouse position
Flash rate 400ms 400ms 170ms
Custom vJoy Not used Not used Always used
Assets OBJ meshes (EmbeddedResource) PNG images (Resource) None (procedural)

Clone this wiki locally