/**
* Tracker Editor Module
* Provides UI for customizing tracker configurations
*/
import { extensionSettings } from '../../core/state.js';
import { saveSettings } from '../../core/persistence.js';
import { renderUserStats } from '../rendering/userStats.js';
import { renderInfoBox } from '../rendering/infoBox.js';
import { renderThoughts } from '../rendering/thoughts.js';
let $editorModal = null;
let activeTab = 'userStats';
let tempConfig = null; // Temporary config for cancel functionality
/**
* Initialize the tracker editor modal
*/
export function initTrackerEditor() {
// Modal will be in template.html, just set up event listeners
$editorModal = $('#rpg-tracker-editor-popup');
if (!$editorModal.length) {
console.error('[RPG Companion] Tracker editor modal not found in template');
return;
}
// Tab switching
$(document).on('click', '.rpg-editor-tab', function() {
$('.rpg-editor-tab').removeClass('active');
$(this).addClass('active');
activeTab = $(this).data('tab');
$('.rpg-editor-tab-content').hide();
$(`#rpg-editor-tab-${activeTab}`).show();
});
// Save button
$(document).on('click', '#rpg-editor-save', function() {
applyTrackerConfig();
closeTrackerEditor();
});
// Cancel button
$(document).on('click', '#rpg-editor-cancel', function() {
closeTrackerEditor();
});
// Close X button
$(document).on('click', '#rpg-close-tracker-editor', function() {
closeTrackerEditor();
});
// Reset button
$(document).on('click', '#rpg-editor-reset', function() {
resetToDefaults();
renderEditorUI();
});
// Close on background click
$(document).on('click', '#rpg-tracker-editor-popup', function(e) {
if (e.target.id === 'rpg-tracker-editor-popup') {
closeTrackerEditor();
}
});
// Open button
$(document).on('click', '#rpg-open-tracker-editor', function() {
openTrackerEditor();
});
}
/**
* Open the tracker editor modal
*/
function openTrackerEditor() {
// Create temporary copy for cancel functionality
tempConfig = JSON.parse(JSON.stringify(extensionSettings.trackerConfig));
// Set theme to match current extension theme
const theme = extensionSettings.theme || 'modern';
$editorModal.attr('data-theme', theme);
renderEditorUI();
$editorModal.addClass('is-open').css('display', '');
}
/**
* Close the tracker editor modal
*/
function closeTrackerEditor() {
// Restore from temp if canceling
if (tempConfig) {
extensionSettings.trackerConfig = tempConfig;
tempConfig = null;
}
$editorModal.removeClass('is-open').addClass('is-closing');
setTimeout(() => {
$editorModal.removeClass('is-closing').hide();
}, 200);
}
/**
* Apply the tracker configuration and refresh all trackers
*/
function applyTrackerConfig() {
tempConfig = null; // Clear temp config
saveSettings();
// Re-render all trackers with new config (v1 system - backward compat)
renderUserStats();
renderInfoBox();
renderThoughts();
// Notify dashboard system of config changes (v2 system - reactive integration)
document.dispatchEvent(new CustomEvent('rpg:trackerConfigChanged', {
detail: {
config: extensionSettings.trackerConfig,
source: 'trackerEditor'
}
}));
console.log('[RPG Companion] Tracker config changed event dispatched');
}
/**
* Reset configuration to defaults
*/
function resetToDefaults() {
extensionSettings.trackerConfig = {
userStats: {
customStats: [
{ id: 'health', name: 'Health', enabled: true },
{ id: 'satiety', name: 'Satiety', enabled: true },
{ id: 'energy', name: 'Energy', enabled: true },
{ id: 'hygiene', name: 'Hygiene', enabled: true },
{ id: 'arousal', name: 'Arousal', enabled: true }
],
showRPGAttributes: true,
statusSection: {
enabled: true,
showMoodEmoji: true,
customFields: ['Conditions']
},
skillsSection: {
enabled: false,
label: 'Skills'
}
},
infoBox: {
widgets: {
date: { enabled: true, format: 'Weekday, Month, Year' },
weather: { enabled: true },
temperature: { enabled: true, unit: 'C' },
time: { enabled: true },
location: { enabled: true },
recentEvents: { enabled: true }
}
},
presentCharacters: {
showEmoji: true,
showName: true,
relationshipFields: ['Lover', 'Friend', 'Ally', 'Enemy', 'Neutral'],
relationshipEmojis: {
'Lover': '❤️',
'Friend': '⭐',
'Ally': '🤝',
'Enemy': '⚔️',
'Neutral': '⚖️'
},
customFields: [
{ id: 'appearance', name: 'Appearance', enabled: true, description: 'Visible physical appearance (clothing, hair, notable features)' },
{ id: 'demeanor', name: 'Demeanor', enabled: true, description: 'Observable demeanor or emotional state' }
],
thoughts: {
enabled: true,
name: 'Thoughts',
description: 'Internal monologue (in first person POV, up to three sentences long)'
},
characterStats: {
enabled: false,
customStats: [
{ id: 'health', name: 'Health', enabled: true, colorLow: '#ff4444', colorHigh: '#44ff44' },
{ id: 'energy', name: 'Energy', enabled: true, colorLow: '#ffaa00', colorHigh: '#44ffff' }
]
}
}
};
}
/**
* Render the editor UI based on current config
*/
function renderEditorUI() {
renderUserStatsTab();
renderInfoBoxTab();
renderPresentCharactersTab();
}
/**
* Render User Stats configuration tab
*/
function renderUserStatsTab() {
const config = extensionSettings.trackerConfig.userStats;
let html = '
';
// Custom Stats section
html += '
Custom Stats
';
html += '
';
html += '
';
// RPG Attributes toggle
html += '
';
html += ``;
html += '';
html += '
';
// Status Section
html += '
Status Section
';
html += '
';
html += ``;
html += '';
html += '
';
html += '
';
html += ``;
html += '';
html += '
';
html += '
';
html += `
`;
// Skills Section
html += '
Skills Section
';
html += '
';
html += ``;
html += '';
html += '
';
html += '
';
html += `
`;
html += '
';
$('#rpg-editor-tab-userStats').html(html);
setupUserStatsListeners();
}
/**
* Set up event listeners for User Stats tab
*/
function setupUserStatsListeners() {
// Add stat
$('#rpg-add-stat').off('click').on('click', function() {
const newId = 'custom_' + Date.now();
extensionSettings.trackerConfig.userStats.customStats.push({
id: newId,
name: 'New Stat',
enabled: true
});
// Initialize value if doesn't exist
if (extensionSettings.userStats[newId] === undefined) {
extensionSettings.userStats[newId] = 100;
}
renderUserStatsTab();
});
// Remove stat
$('.rpg-stat-remove').off('click').on('click', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.userStats.customStats.splice(index, 1);
renderUserStatsTab();
});
// Toggle stat
$('.rpg-stat-toggle').off('change').on('change', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.userStats.customStats[index].enabled = $(this).is(':checked');
});
// Rename stat
$('.rpg-stat-name').off('blur').on('blur', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.userStats.customStats[index].name = $(this).val();
});
// RPG attributes toggle
$('#rpg-show-rpg-attrs').off('change').on('change', function() {
extensionSettings.trackerConfig.userStats.showRPGAttributes = $(this).is(':checked');
});
// Status section toggles
$('#rpg-status-enabled').off('change').on('change', function() {
extensionSettings.trackerConfig.userStats.statusSection.enabled = $(this).is(':checked');
});
$('#rpg-mood-emoji').off('change').on('change', function() {
extensionSettings.trackerConfig.userStats.statusSection.showMoodEmoji = $(this).is(':checked');
});
$('#rpg-status-fields').off('blur').on('blur', function() {
const fields = $(this).val().split(',').map(f => f.trim()).filter(f => f);
extensionSettings.trackerConfig.userStats.statusSection.customFields = fields;
});
// Skills section toggles
$('#rpg-skills-enabled').off('change').on('change', function() {
extensionSettings.trackerConfig.userStats.skillsSection.enabled = $(this).is(':checked');
});
$('#rpg-skills-label').off('blur').on('blur', function() {
extensionSettings.trackerConfig.userStats.skillsSection.label = $(this).val();
});
}
/**
* Render Info Box configuration tab
*/
function renderInfoBoxTab() {
const config = extensionSettings.trackerConfig.infoBox;
let html = '';
$('#rpg-editor-tab-infoBox').html(html);
setupInfoBoxListeners();
}
/**
* Set up event listeners for Info Box tab
*/
function setupInfoBoxListeners() {
const widgets = extensionSettings.trackerConfig.infoBox.widgets;
$('#rpg-widget-date').off('change').on('change', function() {
widgets.date.enabled = $(this).is(':checked');
});
$('#rpg-date-format').off('change').on('change', function() {
widgets.date.format = $(this).val();
});
$('#rpg-widget-weather').off('change').on('change', function() {
widgets.weather.enabled = $(this).is(':checked');
});
$('#rpg-widget-temperature').off('change').on('change', function() {
widgets.temperature.enabled = $(this).is(':checked');
});
$('input[name="temp-unit"]').off('change').on('change', function() {
widgets.temperature.unit = $(this).val();
});
$('#rpg-widget-time').off('change').on('change', function() {
widgets.time.enabled = $(this).is(':checked');
});
$('#rpg-widget-location').off('change').on('change', function() {
widgets.location.enabled = $(this).is(':checked');
});
$('#rpg-widget-events').off('change').on('change', function() {
widgets.recentEvents.enabled = $(this).is(':checked');
});
}
/**
* Render Present Characters configuration tab
*/
function renderPresentCharactersTab() {
const config = extensionSettings.trackerConfig.presentCharacters;
let html = '';
// Relationship Fields Section
html += '
Relationship Status Fields
';
html += '
Define relationship types with corresponding emojis shown on character portraits
';
html += '
';
html += '
';
// Custom Fields Section
html += '
Appearance/Demeanor Fields
';
html += '
Fields shown below character name, separated by |
';
html += '
';
config.customFields.forEach((field, index) => {
html += `
`;
});
html += '
';
html += '
';
// Thoughts Section
html += '
Thoughts Configuration
';
html += '
';
html += ``;
html += '';
html += '
';
html += '
';
// Character Stats
html += '
Character Stats
';
html += '
';
html += ``;
html += '';
html += '
';
html += '
Create stats to track for each character (displayed as colored bars)
';
html += '
';
html += '
';
html += '
';
$('#rpg-editor-tab-presentCharacters').html(html);
setupPresentCharactersListeners();
}
/**
* Set up event listeners for Present Characters tab
*/
function setupPresentCharactersListeners() {
// Add new relationship
$('#rpg-add-relationship').off('click').on('click', function() {
if (!extensionSettings.trackerConfig.presentCharacters.relationshipEmojis) {
extensionSettings.trackerConfig.presentCharacters.relationshipEmojis = {};
}
extensionSettings.trackerConfig.presentCharacters.relationshipEmojis['New Relationship'] = '😊';
// Sync relationshipFields
extensionSettings.trackerConfig.presentCharacters.relationshipFields =
Object.keys(extensionSettings.trackerConfig.presentCharacters.relationshipEmojis);
renderPresentCharactersTab();
});
// Remove relationship
$('.rpg-remove-relationship').off('click').on('click', function() {
const relationship = $(this).data('relationship');
if (extensionSettings.trackerConfig.presentCharacters.relationshipEmojis) {
delete extensionSettings.trackerConfig.presentCharacters.relationshipEmojis[relationship];
}
// Sync relationshipFields
extensionSettings.trackerConfig.presentCharacters.relationshipFields =
Object.keys(extensionSettings.trackerConfig.presentCharacters.relationshipEmojis);
renderPresentCharactersTab();
});
// Update relationship name
$('.rpg-relationship-name').off('blur').on('blur', function() {
const newName = $(this).val();
const $item = $(this).closest('.rpg-relationship-item');
const emoji = $item.find('.rpg-relationship-emoji').val();
// Find the old name by matching the emoji
const oldName = Object.keys(extensionSettings.trackerConfig.presentCharacters.relationshipEmojis).find(
key => extensionSettings.trackerConfig.presentCharacters.relationshipEmojis[key] === emoji &&
key !== newName
);
if (oldName && oldName !== newName) {
delete extensionSettings.trackerConfig.presentCharacters.relationshipEmojis[oldName];
extensionSettings.trackerConfig.presentCharacters.relationshipEmojis[newName] = emoji;
// Sync relationshipFields
extensionSettings.trackerConfig.presentCharacters.relationshipFields =
Object.keys(extensionSettings.trackerConfig.presentCharacters.relationshipEmojis);
}
});
// Update relationship emoji
$('.rpg-relationship-emoji').off('blur').on('blur', function() {
const name = $(this).closest('.rpg-relationship-item').find('.rpg-relationship-name').val();
if (!extensionSettings.trackerConfig.presentCharacters.relationshipEmojis) {
extensionSettings.trackerConfig.presentCharacters.relationshipEmojis = {};
}
extensionSettings.trackerConfig.presentCharacters.relationshipEmojis[name] = $(this).val();
});
// Thoughts configuration
$('#rpg-thoughts-enabled').off('change').on('change', function() {
if (!extensionSettings.trackerConfig.presentCharacters.thoughts) {
extensionSettings.trackerConfig.presentCharacters.thoughts = {};
}
extensionSettings.trackerConfig.presentCharacters.thoughts.enabled = $(this).is(':checked');
});
$('#rpg-thoughts-name').off('blur').on('blur', function() {
if (!extensionSettings.trackerConfig.presentCharacters.thoughts) {
extensionSettings.trackerConfig.presentCharacters.thoughts = {};
}
extensionSettings.trackerConfig.presentCharacters.thoughts.name = $(this).val();
});
$('#rpg-thoughts-description').off('blur').on('blur', function() {
if (!extensionSettings.trackerConfig.presentCharacters.thoughts) {
extensionSettings.trackerConfig.presentCharacters.thoughts = {};
}
extensionSettings.trackerConfig.presentCharacters.thoughts.description = $(this).val();
});
// Add field
$('#rpg-add-field').off('click').on('click', function() {
extensionSettings.trackerConfig.presentCharacters.customFields.push({
id: 'custom_' + Date.now(),
name: 'New Field',
enabled: true,
description: 'Description for AI'
});
renderPresentCharactersTab();
});
// Remove field
$('.rpg-field-remove').off('click').on('click', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.customFields.splice(index, 1);
renderPresentCharactersTab();
});
// Move field up
$('.rpg-field-move-up').off('click').on('click', function() {
const index = $(this).data('index');
if (index > 0) {
const fields = extensionSettings.trackerConfig.presentCharacters.customFields;
[fields[index - 1], fields[index]] = [fields[index], fields[index - 1]];
renderPresentCharactersTab();
}
});
// Move field down
$('.rpg-field-move-down').off('click').on('click', function() {
const index = $(this).data('index');
const fields = extensionSettings.trackerConfig.presentCharacters.customFields;
if (index < fields.length - 1) {
[fields[index], fields[index + 1]] = [fields[index + 1], fields[index]];
renderPresentCharactersTab();
}
});
// Toggle field
$('.rpg-field-toggle').off('change').on('change', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.customFields[index].enabled = $(this).is(':checked');
});
// Rename field
$('.rpg-field-label').off('blur').on('blur', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.customFields[index].name = $(this).val();
});
// Update description
$('.rpg-field-placeholder').off('blur').on('blur', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.customFields[index].description = $(this).val();
});
// Character stats toggle
$('#rpg-char-stats-enabled').off('change').on('change', function() {
if (!extensionSettings.trackerConfig.presentCharacters.characterStats) {
extensionSettings.trackerConfig.presentCharacters.characterStats = { enabled: false, customStats: [] };
}
extensionSettings.trackerConfig.presentCharacters.characterStats.enabled = $(this).is(':checked');
});
// Add character stat
$('#rpg-add-char-stat').off('click').on('click', function() {
if (!extensionSettings.trackerConfig.presentCharacters.characterStats) {
extensionSettings.trackerConfig.presentCharacters.characterStats = { enabled: false, customStats: [] };
}
if (!extensionSettings.trackerConfig.presentCharacters.characterStats.customStats) {
extensionSettings.trackerConfig.presentCharacters.characterStats.customStats = [];
}
extensionSettings.trackerConfig.presentCharacters.characterStats.customStats.push({
id: `stat-${Date.now()}`,
name: 'New Stat',
enabled: true
});
renderPresentCharactersTab();
});
// Remove character stat
$('.rpg-char-stat-remove').off('click').on('click', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.characterStats.customStats.splice(index, 1);
renderPresentCharactersTab();
});
// Toggle character stat
$('.rpg-char-stat-toggle').off('change').on('change', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.characterStats.customStats[index].enabled = $(this).is(':checked');
});
// Rename character stat
$('.rpg-char-stat-label').off('blur').on('blur', function() {
const index = $(this).data('index');
extensionSettings.trackerConfig.presentCharacters.characterStats.customStats[index].name = $(this).val();
});
}