mirror of
https://github.com/Pax1601/DCSOlympus.git
synced 2025-10-29 16:56:34 +00:00
396 lines
15 KiB
TypeScript
396 lines
15 KiB
TypeScript
import { OlympusPlugin } from "interfaces";
|
|
import { OlympusApp } from "olympusapp";
|
|
import { Unit } from "unit/unit";
|
|
|
|
|
|
const SHOW_CONTROL_TIPS = "Show control tips"
|
|
|
|
export class ControlTipsPlugin implements OlympusPlugin {
|
|
#element: HTMLElement;
|
|
#app: any;
|
|
#shortcutManager: any;
|
|
#cursorIsHoveringOverUnit: boolean = false;
|
|
#cursorIsHoveringOverAirbase: boolean = false;
|
|
#mouseoverElement!: HTMLElement;
|
|
|
|
constructor() {
|
|
this.#element = document.createElement("div");
|
|
this.#element.id = "control-tips-panel";
|
|
document.body.appendChild(this.#element);
|
|
}
|
|
|
|
getName() {
|
|
return "Control Tips Plugin"
|
|
}
|
|
|
|
initialize(app: any) {
|
|
this.#app = app;
|
|
|
|
this.#shortcutManager = this.#app.getShortcutManager();
|
|
|
|
this.#shortcutManager.onKeyDown(() => {
|
|
this.#updateTips()
|
|
});
|
|
|
|
this.#shortcutManager.onKeyUp(() => {
|
|
this.#updateTips()
|
|
});
|
|
|
|
document.addEventListener("airbaseMouseover", (ev: CustomEventInit) => {
|
|
this.#cursorIsHoveringOverAirbase = true;
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("airbaseMouseout", (ev: CustomEventInit) => {
|
|
this.#cursorIsHoveringOverAirbase = false;
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("unitDeselection", (ev: CustomEventInit ) => {
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("unitMouseover", (ev: CustomEventInit) => {
|
|
this.#cursorIsHoveringOverUnit = true;
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("unitMouseout", (ev: CustomEventInit) => {
|
|
this.#cursorIsHoveringOverUnit = false;
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("unitsSelection", (ev: CustomEventInit ) => {
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener("mapOptionsChanged", () => {
|
|
this.toggle( !this.#app.getMap().getVisibilityOptions()[SHOW_CONTROL_TIPS] );
|
|
});
|
|
|
|
document.addEventListener( "mouseover", ( ev: MouseEvent ) => {
|
|
if ( ev.target instanceof HTMLElement ) {
|
|
this.#mouseoverElement = <HTMLElement>ev.target;
|
|
}
|
|
this.#updateTips();
|
|
});
|
|
|
|
document.addEventListener( "mouseup", ( ev: MouseEvent ) => {
|
|
this.#updateTips();
|
|
});
|
|
|
|
this.#updateTips();
|
|
|
|
this.#app.getMap().addVisibilityOption(SHOW_CONTROL_TIPS, true);
|
|
|
|
return true;
|
|
}
|
|
|
|
getElement() {
|
|
return this.#element;
|
|
}
|
|
|
|
toggle(bool?: boolean) {
|
|
this.getElement().classList.toggle("hide", bool);
|
|
}
|
|
|
|
#updateTips() {
|
|
const combos: Array<object> = [
|
|
{
|
|
"keys": [],
|
|
"tips": [
|
|
{
|
|
"key": `SHIFT`,
|
|
"action": `Box select`,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false,
|
|
"showIfUnitSelected": false
|
|
},
|
|
{
|
|
"key": `Mouse1`,
|
|
"action": `Deselect`,
|
|
"showIfUnitSelected": true
|
|
},
|
|
{
|
|
"key": `Mouse1+drag`,
|
|
"action": `Move map`,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false,
|
|
"showIfUnitSelected": false
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Spawn menu`,
|
|
"showIfUnitSelected": false,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Airbase menu`,
|
|
"showIfUnitSelected": false,
|
|
"showIfHoveringOverAirbase": true,
|
|
"showIfHoveringOverUnit": false
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Set first waypoint`,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfUnitSelected": true,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Mouse2 (hold)`,
|
|
"action": `Interact (ground)`,
|
|
"showIfUnitSelected": true,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Shift`,
|
|
"action": "<em> in formation...</em>",
|
|
"showIfUnitSelected": true,
|
|
"minSelectedUnits": 2
|
|
},
|
|
{
|
|
"key": "CTRL",
|
|
"action": "<em> ... more</em>",
|
|
"showIfUnitSelected": true,
|
|
"showIfHoveringOverAirbase": false,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": "CTRL",
|
|
"action": " Pin tool",
|
|
"showIfUnitSelected": false,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": "CTRL+Mouse2",
|
|
"action": " Airbase menu",
|
|
"showIfUnitSelected": true,
|
|
"showIfHoveringOverAirbase": true,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Mouse1`,
|
|
"action": "Toggle Blue/Red",
|
|
"mouseoverSelector": "#coalition-switch .ol-switch-fill"
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": "Set Neutral",
|
|
"mouseoverSelector": "#coalition-switch .ol-switch-fill"
|
|
},
|
|
{
|
|
"key": `Mouse1`,
|
|
"action": "Toggle time display",
|
|
"mouseoverSelector": "#connection-status-panel[data-is-connected] #connection-status-message abbr"
|
|
},
|
|
{
|
|
"key": `Mouse1 or Z`,
|
|
"action": "Change location system",
|
|
"mouseoverSelector": "#coordinates-tool, #coordinates-tool *"
|
|
},
|
|
{
|
|
"key": `Comma`,
|
|
"action": "Decrease precision",
|
|
"mouseoverSelector": `#coordinates-tool[data-location-system="MGRS"], #coordinates-tool[data-location-system="MGRS"] *`
|
|
},
|
|
{
|
|
"key": `Period`,
|
|
"action": "Increase precision",
|
|
"mouseoverSelector": `#coordinates-tool[data-location-system="MGRS"], #coordinates-tool[data-location-system="MGRS"] *`
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"keys": ["ControlLeft"],
|
|
"tips": [
|
|
{
|
|
"key": `Mouse1`,
|
|
"action": "Toggle pin",
|
|
"showIfUnitSelected": false,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false
|
|
},
|
|
{
|
|
"key": `Mouse1`,
|
|
"action": "Toggle selection",
|
|
"showIfUnitSelected": true,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": true
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Add waypoint`,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": false,
|
|
"showIfUnitSelected": true,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Interact (airbase)`,
|
|
"showIfHoveringOverAirbase": true,
|
|
"showIfUnitSelected": true,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": `Interact (unit)`,
|
|
"showIfHoveringOverAirbase": false,
|
|
"showIfHoveringOverUnit": true,
|
|
"showIfUnitSelected": true,
|
|
"unitsMustBeControlled": true
|
|
},
|
|
{
|
|
"key": `Shift`,
|
|
"action": "<em> in formation...</em>",
|
|
"showIfUnitSelected": true,
|
|
"minSelectedUnits": 2
|
|
},
|
|
{
|
|
"key": `[Num 1-9]`,
|
|
"action": "Set hotgroup",
|
|
"showIfUnitSelected": true
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"keys": ["ShiftLeft"],
|
|
"tips": [
|
|
{
|
|
"key": `Mouse1+drag`,
|
|
"action": "Box select",
|
|
"showIfUnitSelected": false
|
|
},
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": "Set first formation waypoint",
|
|
"showIfUnitSelected": true,
|
|
"minSelectedUnits": 2
|
|
},
|
|
{
|
|
"key": `[Num 1-9]`,
|
|
"action": "Add to hotgroup",
|
|
"showIfUnitSelected": true
|
|
},
|
|
{
|
|
"key": "CTRL",
|
|
"action": "<em> ... more</em>",
|
|
"minSelectedUnits": 2,
|
|
"showIfUnitSelected": true,
|
|
"showIfHoveringOverAirbase": false,
|
|
"unitsMustBeControlled": true
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"keys": ["ControlLeft", "ShiftLeft"],
|
|
"tips": [
|
|
{
|
|
"key": `Mouse2`,
|
|
"action": "Add formation waypoint",
|
|
"showIfUnitSelected": true,
|
|
"minSelectedUnits": 2,
|
|
"unitsMustBeControlled": true
|
|
}, {
|
|
"key": `[Num 1-9]`,
|
|
"action": "Add hotgroup to selection",
|
|
"callback": ( tip:object ) => {
|
|
return (Object.values<Unit>( this.#app.getUnitsManager().getUnits() ).some( ( unit:Unit ) => {
|
|
return unit.getAlive() && unit.getControlled() && unit.getHotgroup();
|
|
}));
|
|
},
|
|
"showIfUnitSelected": true,
|
|
"minSelectedUnits": 1
|
|
}
|
|
]
|
|
}
|
|
];
|
|
|
|
const currentCombo: any = combos.find((combo: any) => this.#shortcutManager.keyComboMatches(combo.keys)) || combos[0];
|
|
|
|
const element = this.getElement();
|
|
|
|
element.innerHTML = "";
|
|
|
|
let numSelectedUnits = 0;
|
|
let numSelectedControlledUnits = 0;
|
|
let unitSelectionContainsControlled = false;
|
|
|
|
if (this.#app.getUnitsManager()) {
|
|
let selectedUnits = Object.values(this.#app.getUnitsManager().getSelectedUnits());
|
|
numSelectedUnits = selectedUnits.length;
|
|
numSelectedControlledUnits = selectedUnits.filter((unit: any) => unit.getControlled()).length;
|
|
unitSelectionContainsControlled = numSelectedControlledUnits > 0;
|
|
}
|
|
|
|
const tipsIncludesActiveMouseover = ( currentCombo.tips.some( ( tip:any ) => {
|
|
if ( !tip.mouseoverSelector ) {
|
|
return false;
|
|
}
|
|
|
|
if ( this.#mouseoverElement instanceof HTMLElement === false ) {
|
|
return false;
|
|
}
|
|
|
|
if ( !this.#mouseoverElement.matches( tip.mouseoverSelector ) ) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}));
|
|
|
|
currentCombo.tips.filter((tip: any) => {
|
|
if (numSelectedUnits > 0) {
|
|
if (tip.showIfUnitSelected === false) {
|
|
return false;
|
|
}
|
|
|
|
if (tip.unitsMustBeControlled === true && unitSelectionContainsControlled === false) {
|
|
return false;
|
|
}
|
|
|
|
if ( typeof tip.minSelectedUnits === "number" && numSelectedControlledUnits < tip.minSelectedUnits ) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (numSelectedUnits === 0 && tip.showIfUnitSelected === true) {
|
|
return false;
|
|
}
|
|
|
|
if (typeof tip.showIfHoveringOverAirbase === "boolean") {
|
|
if (tip.showIfHoveringOverAirbase !== this.#cursorIsHoveringOverAirbase) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (typeof tip.showIfHoveringOverUnit === "boolean") {
|
|
if (tip.showIfHoveringOverUnit !== this.#cursorIsHoveringOverUnit) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if ( tipsIncludesActiveMouseover && ( typeof tip.mouseoverSelector !== "string" || !this.#mouseoverElement.matches( tip.mouseoverSelector ) ) ) {
|
|
return false;
|
|
}
|
|
|
|
if ( !tipsIncludesActiveMouseover && typeof tip.mouseoverSelector === "string" ) {
|
|
return false;
|
|
}
|
|
|
|
if ( typeof tip.callback === "function" && !tip.callback( tip ) ) {
|
|
return false;
|
|
}
|
|
|
|
element.innerHTML += `<div><span class="key">${tip.key}</span><span class="action">${tip.action}</span></div>`;
|
|
|
|
});
|
|
}
|
|
} |