This commit is contained in:
Ocame
2026-03-02 12:02:33 +01:00
commit 41c3484904
7 changed files with 323 additions and 0 deletions

10
lang/de.json Normal file
View File

@ -0,0 +1,10 @@
{
"duneinichecker": {
"combat": {
"ToggleCombatantsTurnDone": "Toggle Turn Completed",
"CombatantsTurnDone": "Turn Completed",
"CombatantsTurnNotDone": "Turn Not Completed"
}
}
}

9
lang/en.json Normal file
View File

@ -0,0 +1,9 @@
{
"duneinichecker": {
"combat": {
"ToggleCombatantsTurnDone": "Toggle Turn Completed",
"CombatantsTurnDone": "Turn Completed",
"CombatantsTurnNotDone": "Turn Not Completed"
}
}
}

39
module.json Normal file
View File

@ -0,0 +1,39 @@
{
"id": "duneinichecker",
"title": "Dune - Ini Checker",
"version": "0.0.1",
"description": "",
"compatibility": {
"minimum": "13",
"verified": "13"
},
"flags": {
"canUpload": true
},
"relationships": {
"systems": [
{
"id": "dune",
"type": "system",
"compatibility": {}
}
]
},
"esmodules": [
"scripts/dune.js"
],
"languages": [
{
"lang": "en",
"name": "English",
"path": "lang/en.json",
"flags": {}
},
{
"lang": "de",
"name": "Deutsch",
"path": "lang/de.json",
"flags": {}
}
]
}

View File

@ -0,0 +1,87 @@
const KEY = "dune";
export default class Combat2d20 extends Combat {
get combatantsTurnDone() {
return this.getFlag(KEY, "combatantsTurnDone") ?? [];
}
get combatantsTurnsDoneThisRound() {
const combatantsTurnDone = this.combatantsTurnDone;
return combatantsTurnDone[this.round] ?? {};
}
async rollInitiative() {
return this;
}
async setTurn(newTurn) {
this.turn = newTurn;
// Update the document, passing data through a hook first
const updateData = {round: this.round, turn: newTurn};
const updateOptions = {advanceTime: CONFIG.time.turnTime, direction: 1};
Hooks.callAll("combatTurn", this, updateData, updateOptions);
return this.update(updateData, updateOptions);
}
setupTurns() {
// Determine the turn order and the current turn
const turns = this.combatants.contents;
// Sort alphabetically by name first
turns.sort((a, b) => {
if (a.name < b.name) {
return -1;
}
if (a.name > b.name) {
return 1;
}
return 0;
});
// Then make sure the player characters are first
turns.sort(
(a, b) => Number(b.hasPlayerOwner) - Number(a.hasPlayerOwner)
);
if (this.turn !== null) this.turn =
Math.clamp(this.turn, 0, turns.length - 1);
// Update state tracking
let c = turns[this.turn];
this.current = {
round: this.round,
turn: this.turn,
combatantId: c ? c.id : null,
tokenId: c ? c.tokenId : null,
};
// One-time initialization of the previous state
if (!this.previous) this.previous = this.current;
// Return the array of prepared turns
return this.turns = turns;
}
async toggleTurnDone(combatantId) {
if (!game.user.isGM) return;
if (!this.started) return;
const combatantsTurnsDoneThisRound = this.combatantsTurnsDoneThisRound;
const turnDone = !(combatantsTurnsDoneThisRound[combatantId] ?? false);
combatantsTurnsDoneThisRound[combatantId] = turnDone;
const combatantsTurnDone = this.combatantsTurnDone;
combatantsTurnDone[this.round] = combatantsTurnsDoneThisRound;
this.setFlag(KEY, "combatantsTurnDone", combatantsTurnDone);
}
}

View File

@ -0,0 +1,99 @@
export default class CombatTracker2d20V2
extends foundry.applications.sidebar.tabs.CombatTracker {
/** @inheritDoc */
static DEFAULT_OPTIONS = {
actions: {
toggleCombatantTurnDone: CombatTracker2d20V2._onDuneCombatantControl,
},
};
/** @override */
static PARTS = {
header: {
// We're still using the default Foundry template for this part
template: "templates/sidebar/tabs/combat/header.hbs",
},
tracker: {
template: "modules/duneinichecker/templates/tracker.hbs",
},
footer: {
// We're still using the default Foundry template for this part
template: "templates/sidebar/tabs/combat/footer.hbs",
},
};
_onCombatantMouseDown(event, target) {
super._onCombatantMouseDown(event, target);
const isInputElement = (event.target instanceof HTMLInputElement);
const isButtonElement = (event.target instanceof HTMLButtonElement);
if (isInputElement || isButtonElement) return;
if (game.user.isGM && this.viewed.started) {
const { combatantId } = target?.dataset ?? {};
const combat = this.viewed;
const currentTurn = combat.turn ?? -1;
let newTurn = currentTurn;
for (let [i, turn] of combat.turns.entries() ) {
if (turn.isDefeated) continue;
if (turn.id === combatantId) {
newTurn = i;
break;
}
}
if (newTurn !== currentTurn) {
combat.setTurn(newTurn);
}
}
}
static async _onDuneCombatantControl(event, target) {
event.preventDefault();
event.stopPropagation();
if (!game.user.isGM) return;
if (!this.viewed.started) {
ui.notifications.warn(
game.i18n.localize("COMBAT.NotStarted")
);
return;
}
const { combatantId } = target.closest("[data-combatant-id]")?.dataset ?? {};
const combatant = this.viewed?.combatants.get(combatantId);
if ( !combatant ) return;
if (combatant.isOwner) {
this.viewed.toggleTurnDone(combatant.id);
}
}
/**
* Prepare render context for the tracker part.
* @param {ApplicationRenderContext} context
* @param {HandlebarsRenderOptions} options
* @returns {Promise<void>}
* @protected
*/
async _prepareTrackerContext(context, options) {
await super._prepareTrackerContext(context, options);
const combat = this.viewed;
if ( !combat ) return;
const combatantsTurnDone = combat.combatantsTurnsDoneThisRound;
for (const turn of context.turns) {
turn.turnDone = combatantsTurnDone[turn.id] ?? false;
}
}
}

10
scripts/dune.js Normal file
View File

@ -0,0 +1,10 @@
// Wird ausgeführt, wenn Foundry geladen wurde, aber bevor das Spiel bereit ist
import Combat2d20 from "./combat/Combat2d20.js";
import CombatTracker2d20V2 from "./combat/CombatTracker2d20V2.js";
Hooks.once('init', () => {
CONFIG.ui.combat = CombatTracker2d20V2;
CONFIG.Combat.documentClass = Combat2d20;
});

69
templates/tracker.hbs Normal file
View File

@ -0,0 +1,69 @@
<ol class="combat-tracker plain">
{{#each turns}}
<li class="combatant {{ css }}" data-combatant-id="{{ id }}" data-action="activateCombatant">
{{!-- TODO: Targets --}}
{{!-- Image --}}
<img class="token-image" src="{{ img }}" alt="{{ name }}" loading="lazy">
{{!-- Name & Controls --}}
<div class="token-name">
<strong class="name">{{ name }}</strong>
<div class="combatant-controls">
{{#if @root.user.isGM}}
<button type="button" class="inline-control combatant-control icon fa-solid fa-eye-slash {{#if hidden}}active{{/if}}"
data-action="toggleHidden" data-tooltip aria-label="{{ localize "COMBAT.ToggleVis" }}"></button>
<button type="button" class="inline-control combatant-control icon fa-solid fa-skull {{#if isDefeated}}active{{/if}}"
data-action="toggleDefeated" data-tooltip
aria-label="{{ localize "COMBAT.ToggleDead" }}"></button>
{{/if}}
{{#if canPing}}
<button type="button" class="inline-control combatant-control icon fa-solid fa-bullseye-arrow"
data-action="pingCombatant" data-tooltip
aria-label="{{ localize "COMBAT.PingCombatant" }}"></button>
{{/if}}
{{#unless @root.user.isGM}}
<button type="button" class="inline-control combatant-control icon fa-solid fa-arrows-to-eye"
data-action="panToCombatant" data-tooltip
aria-label="{{ localize "COMBAT.PanToCombatant" }}"></button>
{{/unless}}
{{!-- TODO: Target Control --}}
<div class="token-effects" data-tooltip-html="{{ effects.tooltip }}">
{{#each effects.icons}}
<img class="token-effect" src="{{ img }}" alt="{{ name }}">
{{/each}}
</div>
</div>
</div>
{{!-- Resource --}}
{{#if resource includeZero=true}}
<div class="token-resource">
<span class="resource">{{ resource }}</span>
</div>
{{/if}}
{{!-- Turn Completed Toggle Button --}}
<div class="token-turn-completed">
<a
class="combatant-control"
{{#if turnDone}}
style="color: var(--color-text-subtle);"
{{/if}}
{{#if @root.user.isGM}}
data-tooltip="{{localize 'DUNE.combat.ToggleCombatantsTurnDone'}}"
{{else}}
{{#if turnDone}}
data-tooltip="{{localize 'DUNE.combat.CombatantsTurnDone'}}"
{{else}}
data-tooltip="{{localize 'DUNE.combat.CombatantsTurnNotDone'}}"
{{/if}}
{{/if}}
data-action="toggleCombatantTurnDone"
>
<i class="fa-solid fa-circle-check fa-xl"></i>
</a>
</div>
</li>
{{/each}}
</ol>