Files
rspade_system/storage-broken/rsx-tmp/babel_88541583ec1c5f505db952afb075ee53.js
root 78553d4edf Fix code quality violations for publish
Remove unused blade settings pages not linked from UI
Convert remaining frontend pages to SPA actions
Convert settings user_settings and general to SPA actions
Convert settings profile pages to SPA actions
Convert contacts and projects add/edit pages to SPA actions
Convert clients add/edit page to SPA action with loading pattern
Refactor component scoped IDs from $id to $sid
Fix jqhtml comment syntax and implement universal error component system
Update all application code to use new unified error system
Remove all backwards compatibility - unified error system complete
Phase 5: Remove old response classes
Phase 3-4: Ajax response handler sends new format, old helpers deprecated
Phase 2: Add client-side unified error foundation
Phase 1: Add server-side unified error foundation
Add unified Ajax error response system with constants

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-21 04:35:01 +00:00

118 lines
15 KiB
JavaScript
Executable File

"use strict";
/**
* Rsx_Tabs
*
* Tab container component with form-aware error handling integration.
* See rsx_tabs.jqhtml for full documentation.
*
* JavaScript Responsibilities:
* - Builds tab navigation dynamically from registered Rsx_Tab children
* - Manages tab activation and switching behavior
* - Persists active tab to URL hash for bookmarking
* - Integrates with form validation to show error badges on tabs
* - Auto-switches to first tab with errors on validation failure
* - Provides API for parent forms to report validation errors
*/
class Rsx_Tabs extends Component {
on_create() {
this.tabs = []; // Registered Rsx_Tab components
this.active_tab_id = null;
this.form = null;
}
on_ready() {
// Find parent form if it exists
this.form = this.closest('.Rsx_Form');
// Build tab navigation from registered tabs
this._build_nav();
// Restore active tab from URL hash or activate first tab
const hash = window.location.hash;
if (hash) {
const tab_id = hash.substring(1);
this.activate_tab(tab_id);
} else if (this.tabs.length > 0) {
this.activate_tab(this.tabs[0].args.id);
}
// Persist active tab to URL hash
const that = this;
this.$sid('nav').on('click', 'a[data-bs-toggle="tab"]', function (e) {
const tab_id = $(e.currentTarget).data('tab-id');
window.location.hash = '#' + tab_id;
});
}
register_tab(tab_component) {
this.tabs.push(tab_component);
}
_build_nav() {
const $nav = this.$sid('nav');
$nav.empty();
for (let i = 0; i < this.tabs.length; i++) {
const tab = this.tabs[i];
const is_active = i === 0 ? 'active' : '';
const $li = $(`
<li class="nav-item" role="presentation">
<a class="nav-link ${is_active}"
data-bs-toggle="tab"
href="#${tab.args.id}"
data-tab-id="${tab.args.id}"
aria-selected="${i === 0 ? 'true' : 'false'}"
role="tab">
${tab.args.icon ? `<i class="${tab.args.icon}"></i> ` : ''}
${tab.args.label}
<span class="badge bg-danger ms-2" style="display: none;" data-error-badge="${tab.args.id}">0</span>
</a>
</li>
`);
$nav.append($li);
}
}
activate_tab(tab_id) {
// Find the tab
const tab = this.tabs.find(t => t.args.id === tab_id);
if (!tab) return;
// Remove active show from all tab panes
for (let t of this.tabs) {
t.$.removeClass('active show');
}
// Add active show to the selected tab pane
tab.$.addClass('active show');
// Update Bootstrap tab navigation
this.$sid('nav').find('a[data-bs-toggle="tab"]').removeClass('active').attr('aria-selected', 'false');
this.$sid('nav').find('a[data-tab-id="' + tab_id + '"]').addClass('active').attr('aria-selected', 'true');
this.active_tab_id = tab_id;
}
handle_validation_errors(errors) {
// Count errors per tab
const tab_errors = {};
for (let tab of this.tabs) {
const error_count = tab.count_errors(errors);
tab_errors[tab.args.id] = error_count;
// Update badge
const $badge = this.$sid('nav').find(`[data-error-badge="${tab.args.id}"]`);
if (error_count > 0) {
$badge.text(error_count).show();
} else {
$badge.hide();
}
}
// Find first tab with errors
const first_errored_tab = this.tabs.find(t => tab_errors[t.args.id] > 0);
// Switch to first errored tab if not currently on an errored tab
if (first_errored_tab && tab_errors[this.active_tab_id] === 0) {
this.activate_tab(first_errored_tab.args.id);
}
}
clear_error_badges() {
this.$sid('nav').find('[data-error-badge]').hide();
}
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Rsx_Tabs","Jqhtml_Component","on_create","tabs","active_tab_id","form","on_ready","closest","_build_nav","hash","window","location","tab_id","substring","activate_tab","length","args","id","that","$id","on","e","$","currentTarget","data","register_tab","tab_component","push","$nav","empty","i","tab","is_active","$li","icon","label","append","find","t","removeClass","addClass","attr","handle_validation_errors","errors","tab_errors","error_count","count_errors","$badge","text","show","hide","first_errored_tab","clear_error_badges"],"sources":["rsx/theme/components/forms/rsx_tabs.js"],"sourcesContent":["/**\n * Rsx_Tabs\n *\n * Tab container component with form-aware error handling integration.\n * See rsx_tabs.jqhtml for full documentation.\n *\n * JavaScript Responsibilities:\n * - Builds tab navigation dynamically from registered Rsx_Tab children\n * - Manages tab activation and switching behavior\n * - Persists active tab to URL hash for bookmarking\n * - Integrates with form validation to show error badges on tabs\n * - Auto-switches to first tab with errors on validation failure\n * - Provides API for parent forms to report validation errors\n */\nclass Rsx_Tabs extends Jqhtml_Component {\n    on_create() {\n        this.tabs = []; // Registered Rsx_Tab components\n        this.active_tab_id = null;\n        this.form = null;\n    }\n\n    on_ready() {\n        // Find parent form if it exists\n        this.form = this.closest('.Rsx_Form');\n\n        // Build tab navigation from registered tabs\n        this._build_nav();\n\n        // Restore active tab from URL hash or activate first tab\n        const hash = window.location.hash;\n        if (hash) {\n            const tab_id = hash.substring(1);\n            this.activate_tab(tab_id);\n        } else if (this.tabs.length > 0) {\n            this.activate_tab(this.tabs[0].args.id);\n        }\n\n        // Persist active tab to URL hash\n        const that = this;\n        this.$id('nav').on('click', 'a[data-bs-toggle=\"tab\"]', function (e) {\n            const tab_id = $(e.currentTarget).data('tab-id');\n            window.location.hash = '#' + tab_id;\n        });\n    }\n\n    register_tab(tab_component) {\n        this.tabs.push(tab_component);\n    }\n\n    _build_nav() {\n        const $nav = this.$id('nav');\n        $nav.empty();\n\n        for (let i = 0; i < this.tabs.length; i++) {\n            const tab = this.tabs[i];\n            const is_active = i === 0 ? 'active' : '';\n\n            const $li = $(`\n                <li class=\"nav-item\" role=\"presentation\">\n                    <a class=\"nav-link ${is_active}\"\n                       data-bs-toggle=\"tab\"\n                       href=\"#${tab.args.id}\"\n                       data-tab-id=\"${tab.args.id}\"\n                       aria-selected=\"${i === 0 ? 'true' : 'false'}\"\n                       role=\"tab\">\n                        ${tab.args.icon ? `<i class=\"${tab.args.icon}\"></i> ` : ''}\n                        ${tab.args.label}\n                        <span class=\"badge bg-danger ms-2\" style=\"display: none;\" data-error-badge=\"${tab.args.id}\">0</span>\n                    </a>\n                </li>\n            `);\n\n            $nav.append($li);\n        }\n    }\n\n    activate_tab(tab_id) {\n        // Find the tab\n        const tab = this.tabs.find((t) => t.args.id === tab_id);\n        if (!tab) return;\n\n        // Remove active show from all tab panes\n        for (let t of this.tabs) {\n            t.$.removeClass('active show');\n        }\n\n        // Add active show to the selected tab pane\n        tab.$.addClass('active show');\n\n        // Update Bootstrap tab navigation\n        this.$id('nav').find('a[data-bs-toggle=\"tab\"]').removeClass('active').attr('aria-selected', 'false');\n        this.$id('nav')\n            .find('a[data-tab-id=\"' + tab_id + '\"]')\n            .addClass('active')\n            .attr('aria-selected', 'true');\n\n        this.active_tab_id = tab_id;\n    }\n\n    handle_validation_errors(errors) {\n        // Count errors per tab\n        const tab_errors = {};\n\n        for (let tab of this.tabs) {\n            const error_count = tab.count_errors(errors);\n            tab_errors[tab.args.id] = error_count;\n\n            // Update badge\n            const $badge = this.$id('nav').find(`[data-error-badge=\"${tab.args.id}\"]`);\n            if (error_count > 0) {\n                $badge.text(error_count).show();\n            } else {\n                $badge.hide();\n            }\n        }\n\n        // Find first tab with errors\n        const first_errored_tab = this.tabs.find((t) => tab_errors[t.args.id] > 0);\n\n        // Switch to first errored tab if not currently on an errored tab\n        if (first_errored_tab && tab_errors[this.active_tab_id] === 0) {\n            this.activate_tab(first_errored_tab.args.id);\n        }\n    }\n\n    clear_error_badges() {\n        this.$id('nav').find('[data-error-badge]').hide();\n    }\n}\n"],"mappings":"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","ignoreList":[]}