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>
343 lines
40 KiB
JavaScript
Executable File
343 lines
40 KiB
JavaScript
Executable File
"use strict";
|
|
|
|
/**
|
|
* Rsx_Modal Component
|
|
*
|
|
* Instance of a modal dialog. Handles lifecycle, sizing, and user interaction.
|
|
* Typically created and managed by the Modal static API class.
|
|
*/
|
|
class Rsx_Modal extends Component {
|
|
on_create() {
|
|
this.data.title = '';
|
|
this.data.body_content = null;
|
|
this.data.buttons = [];
|
|
this.data.closable = true;
|
|
this.data.max_width = 800;
|
|
this.data.close_on_submit = true;
|
|
this.data.is_visible = false;
|
|
this.data.result_promise = null;
|
|
this.data.resolve_fn = null;
|
|
|
|
// Store reference to bootstrap modal instance
|
|
this._bs_modal = null;
|
|
this._resize_handler = null;
|
|
}
|
|
on_ready() {
|
|
const that = this;
|
|
|
|
// Set up close button handler
|
|
this.$sid('close_btn').on('click', function (e) {
|
|
e.preventDefault();
|
|
if (that.data.closable) {
|
|
that.close(false);
|
|
}
|
|
});
|
|
|
|
// Set up backdrop click handler
|
|
this.$sid('backdrop').on('click', function (e) {
|
|
if (that.data.closable && e.target === this) {
|
|
that.close(false);
|
|
}
|
|
});
|
|
|
|
// Set up ESC key handler
|
|
$(document).on('keydown.rsx_modal_' + this._cid, function (e) {
|
|
if (e.key === 'Escape' && that.data.closable && that.data.is_visible) {
|
|
that.close(false);
|
|
}
|
|
});
|
|
|
|
// Set up resize handler
|
|
this._resize_handler = debounce(() => {
|
|
if (that.data.is_visible) {
|
|
that._apply_sizing();
|
|
}
|
|
}, 100);
|
|
$(window).on('resize.rsx_modal_' + this._cid, this._resize_handler);
|
|
}
|
|
|
|
/**
|
|
* Configure and show the modal
|
|
* @param {Object} options - Modal options (title, body, buttons, etc.)
|
|
* @param {Object} internal_options - Internal options (skip_backdrop, animate)
|
|
*/
|
|
async show(options) {
|
|
let internal_options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
|
|
const that = this;
|
|
const skip_backdrop = internal_options.skip_backdrop || false;
|
|
const should_animate = internal_options.animate || false;
|
|
console.log('[Rsx_Modal] show() called with options:', options);
|
|
|
|
// Store options
|
|
this.data.title = options.title || '';
|
|
this.data.closable = options.closable !== undefined ? options.closable : true;
|
|
this.data.max_width = options.max_width || 800;
|
|
this.data.close_on_submit = options.close_on_submit !== undefined ? options.close_on_submit : true;
|
|
this.data.buttons = options.buttons || [];
|
|
this.data.skip_backdrop = skip_backdrop;
|
|
this.data.icon = options.icon || null;
|
|
console.log('[Rsx_Modal] Setting title to:', this.data.title);
|
|
console.log('[Rsx_Modal] Title element:', this.$sid('title'));
|
|
|
|
// Set title
|
|
this.$sid('title').text(this.data.title);
|
|
|
|
// Show/hide close button based on closable
|
|
if (this.data.closable) {
|
|
this.$sid('close_btn').show();
|
|
} else {
|
|
this.$sid('close_btn').hide();
|
|
}
|
|
|
|
// Set body content (with optional icon)
|
|
this._set_body_content(options.body, this.data.icon);
|
|
|
|
// Set buttons
|
|
this._set_buttons();
|
|
|
|
// Create promise that will resolve when modal closes
|
|
const result_promise = new Promise(resolve => {
|
|
that.data.resolve_fn = resolve;
|
|
});
|
|
|
|
// Show modal and backdrop
|
|
this.data.is_visible = true;
|
|
|
|
// Append to body so it's on top (don't append backdrop if using shared)
|
|
if (!skip_backdrop) {
|
|
$('body').append(this.$sid('backdrop'));
|
|
}
|
|
$('body').append(this.$);
|
|
|
|
// Apply sizing before showing
|
|
this._apply_sizing();
|
|
|
|
// Fade in modal (and backdrop if not using shared)
|
|
await this._fade_in(should_animate);
|
|
|
|
// Auto-focus first input element
|
|
this._focus_first_input();
|
|
return result_promise;
|
|
}
|
|
|
|
/**
|
|
* Set body content with optional icon
|
|
*/
|
|
_set_body_content(body, icon) {
|
|
const $body = this.$sid('body');
|
|
$body.empty();
|
|
|
|
// If icon provided, add it
|
|
if (icon) {
|
|
const $icon = $(`<i class="bi bi-${icon} modal-icon"></i>`);
|
|
$body.append($icon);
|
|
$body.addClass('has-icon');
|
|
} else {
|
|
$body.removeClass('has-icon');
|
|
}
|
|
|
|
// Get or create body content wrapper
|
|
let $content = this.$sid('body_content');
|
|
if (!$content.exists()) {
|
|
$content = $('<div class="modal-body-content"></div>');
|
|
$body.append($content);
|
|
}
|
|
if (typeof body === 'string') {
|
|
// Text content - escape and convert newlines
|
|
const escaped = $('<div>').text(body).html().replace(/\n/g, '<br>');
|
|
$content.html(escaped);
|
|
} else if (body instanceof jQuery) {
|
|
// jQuery element
|
|
$content.append(body);
|
|
} else if (body && typeof body === 'object') {
|
|
// Assume it's a jqhtml component instance
|
|
$content.append(body.$);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set buttons in footer
|
|
*/
|
|
_set_buttons() {
|
|
const that = this;
|
|
const $footer = this.$sid('footer');
|
|
$footer.empty();
|
|
if (this.data.buttons.length === 0) {
|
|
$footer.hide();
|
|
return;
|
|
}
|
|
$footer.show();
|
|
for (let button_def of this.data.buttons) {
|
|
const $button = $('<button>').attr('type', 'button').addClass('btn').addClass(button_def.class || 'btn-secondary').text(button_def.label || 'Button');
|
|
$button.on('click', async function () {
|
|
let result = button_def.value;
|
|
let had_callback = false;
|
|
|
|
// If button has a callback, call it and use return value as result
|
|
if (button_def.callback && typeof button_def.callback === 'function') {
|
|
had_callback = true;
|
|
result = await button_def.callback();
|
|
}
|
|
|
|
// If callback returned false, keep modal open (but not if just button value is false)
|
|
if (result === false && had_callback) {
|
|
return;
|
|
}
|
|
|
|
// Close modal with result
|
|
that.close(result);
|
|
});
|
|
$footer.append($button);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Calculate and apply responsive sizing
|
|
*/
|
|
_apply_sizing() {
|
|
const viewport_width = $(window).width();
|
|
const viewport_height = $(window).height();
|
|
const is_mobile = viewport_width < 768;
|
|
|
|
// Calculate max width based on viewport
|
|
let max_width = this.data.max_width;
|
|
const viewport_limit = is_mobile ? viewport_width * 0.9 : viewport_width * 0.8;
|
|
max_width = Math.min(max_width, viewport_limit);
|
|
|
|
// Try to constrain to 60% width for better proportions on desktop
|
|
if (!is_mobile) {
|
|
const preferred_width = viewport_width * 0.6;
|
|
if (preferred_width < max_width) {
|
|
max_width = preferred_width;
|
|
}
|
|
}
|
|
|
|
// Apply width
|
|
this.$sid('dialog').css('max-width', max_width + 'px');
|
|
|
|
// Check if content exceeds 80% height
|
|
const content_height = this.$sid('dialog').outerHeight();
|
|
const max_height = viewport_height * 0.8;
|
|
if (content_height > max_height) {
|
|
// Enable scrolling
|
|
this.$sid('dialog').css('max-height', max_height + 'px');
|
|
this.$sid('body').css({
|
|
'overflow-y': 'auto',
|
|
'max-height': max_height - 150 + 'px' // Account for header/footer
|
|
});
|
|
} else {
|
|
// Reset scrolling
|
|
this.$sid('dialog').css('max-height', '');
|
|
this.$sid('body').css({
|
|
'overflow-y': '',
|
|
'max-height': ''
|
|
});
|
|
}
|
|
|
|
// Mobile edge spacing
|
|
if (is_mobile) {
|
|
this.$sid('dialog').css('margin', '5%');
|
|
} else {
|
|
this.$sid('dialog').css('margin', '0');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Show animation (instant or with fly-in)
|
|
* @param {boolean} animate - Whether to animate the modal entrance
|
|
*/
|
|
async _fade_in() {
|
|
let animate = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;
|
|
if (animate) {
|
|
// Initial state: modal positioned above final position
|
|
this.$.css('display', 'flex').css('opacity', '0');
|
|
this.$sid('modal').css({
|
|
'transform': 'translate(0, -50px)',
|
|
'opacity': '0'
|
|
});
|
|
this.$sid('backdrop').css('display', 'block').addClass('show');
|
|
|
|
// Force reflow
|
|
this.$sid('modal')[0].offsetHeight;
|
|
|
|
// Trigger animation
|
|
this.$sid('modal').addClass('show').css({
|
|
'transform': 'translate(0, 0)',
|
|
'opacity': '1'
|
|
});
|
|
this.$.css('opacity', '1');
|
|
|
|
// Wait for animation to complete
|
|
await new Promise(resolve => setTimeout(resolve, 150));
|
|
} else {
|
|
// Disable transitions temporarily for instant display
|
|
this.$sid('dialog').css('transition', 'none');
|
|
|
|
// Show modal and backdrop instantly
|
|
this.$.css('display', 'flex').css('opacity', '1');
|
|
this.$sid('modal').addClass('show').css('opacity', '1');
|
|
this.$sid('backdrop').css('display', 'block').addClass('show');
|
|
|
|
// Force reflow to apply the no-transition state
|
|
this.$sid('dialog')[0].offsetHeight;
|
|
|
|
// Re-enable transitions for future animations
|
|
this.$sid('dialog').css('transition', '');
|
|
}
|
|
return Promise.resolve();
|
|
}
|
|
|
|
/**
|
|
* Focus the first input element in the modal
|
|
*/
|
|
_focus_first_input() {
|
|
// Find first input/textarea/select in modal body
|
|
const $first_input = this.$sid('body').find('input:not([type="hidden"]), textarea, select').first();
|
|
if ($first_input.exists()) {
|
|
requestAnimationFrame(() => {
|
|
$first_input.focus();
|
|
// Select text if it's an input with existing value
|
|
if ($first_input.is('input[type="text"], input[type="email"]') && $first_input.val()) {
|
|
$first_input.select();
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Close the modal instantly
|
|
*/
|
|
async close(result) {
|
|
const that = this;
|
|
|
|
// Mark as not visible
|
|
this.data.is_visible = false;
|
|
|
|
// Remove event listeners
|
|
$(document).off('keydown.rsx_modal_' + this._cid);
|
|
$(window).off('resize.rsx_modal_' + this._cid);
|
|
|
|
// Hide instantly (no fade out)
|
|
this.$.hide();
|
|
this.$sid('backdrop').hide();
|
|
|
|
// Remove from DOM
|
|
this.$.remove();
|
|
this.$sid('backdrop').remove();
|
|
|
|
// Resolve promise
|
|
if (this.data.resolve_fn) {
|
|
this.data.resolve_fn(result);
|
|
this.data.resolve_fn = null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Apply validation errors to form fields in modal body
|
|
*/
|
|
apply_errors(errors) {
|
|
// Use Form_Utils to apply errors to elements within modal body
|
|
Form_Utils.apply_form_errors(this.$sid('body'), errors);
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Rsx_Modal","Jqhtml_Component","on_create","data","title","body_content","buttons","closable","max_width","close_on_submit","is_visible","result_promise","resolve_fn","_bs_modal","_resize_handler","on_ready","that","$id","on","e","preventDefault","close","target","$","document","_cid","key","debounce","_apply_sizing","window","show","options","internal_options","arguments","length","undefined","skip_backdrop","should_animate","animate","console","log","icon","text","hide","_set_body_content","body","_set_buttons","Promise","resolve","append","_fade_in","_focus_first_input","$body","empty","$icon","addClass","removeClass","$content","exists","escaped","html","replace","jQuery","$footer","button_def","$button","attr","class","label","result","value","had_callback","callback","viewport_width","width","viewport_height","height","is_mobile","viewport_limit","Math","min","preferred_width","css","content_height","outerHeight","max_height","offsetHeight","setTimeout","$first_input","find","first","requestAnimationFrame","focus","is","val","select","off","remove","apply_errors","errors","Form_Utils","apply_form_errors"],"sources":["rsx/theme/components/modal/rsx_modal.js"],"sourcesContent":["/**\n * Rsx_Modal Component\n *\n * Instance of a modal dialog. Handles lifecycle, sizing, and user interaction.\n * Typically created and managed by the Modal static API class.\n */\nclass Rsx_Modal extends Jqhtml_Component {\n    on_create() {\n        this.data.title = '';\n        this.data.body_content = null;\n        this.data.buttons = [];\n        this.data.closable = true;\n        this.data.max_width = 800;\n        this.data.close_on_submit = true;\n        this.data.is_visible = false;\n        this.data.result_promise = null;\n        this.data.resolve_fn = null;\n\n        // Store reference to bootstrap modal instance\n        this._bs_modal = null;\n        this._resize_handler = null;\n    }\n\n    on_ready() {\n        const that = this;\n\n        // Set up close button handler\n        this.$id('close_btn').on('click', function (e) {\n            e.preventDefault();\n            if (that.data.closable) {\n                that.close(false);\n            }\n        });\n\n        // Set up backdrop click handler\n        this.$id('backdrop').on('click', function (e) {\n            if (that.data.closable && e.target === this) {\n                that.close(false);\n            }\n        });\n\n        // Set up ESC key handler\n        $(document).on('keydown.rsx_modal_' + this._cid, function (e) {\n            if (e.key === 'Escape' && that.data.closable && that.data.is_visible) {\n                that.close(false);\n            }\n        });\n\n        // Set up resize handler\n        this._resize_handler = debounce(() => {\n            if (that.data.is_visible) {\n                that._apply_sizing();\n            }\n        }, 100);\n\n        $(window).on('resize.rsx_modal_' + this._cid, this._resize_handler);\n    }\n\n    /**\n     * Configure and show the modal\n     * @param {Object} options - Modal options (title, body, buttons, etc.)\n     * @param {Object} internal_options - Internal options (skip_backdrop, animate)\n     */\n    async show(options, internal_options = {}) {\n        const that = this;\n        const skip_backdrop = internal_options.skip_backdrop || false;\n        const should_animate = internal_options.animate || false;\n\n        console.log('[Rsx_Modal] show() called with options:', options);\n\n        // Store options\n        this.data.title = options.title || '';\n        this.data.closable = options.closable !== undefined ? options.closable : true;\n        this.data.max_width = options.max_width || 800;\n        this.data.close_on_submit = options.close_on_submit !== undefined ? options.close_on_submit : true;\n        this.data.buttons = options.buttons || [];\n        this.data.skip_backdrop = skip_backdrop;\n        this.data.icon = options.icon || null;\n\n        console.log('[Rsx_Modal] Setting title to:', this.data.title);\n        console.log('[Rsx_Modal] Title element:', this.$id('title'));\n\n        // Set title\n        this.$id('title').text(this.data.title);\n\n        // Show/hide close button based on closable\n        if (this.data.closable) {\n            this.$id('close_btn').show();\n        } else {\n            this.$id('close_btn').hide();\n        }\n\n        // Set body content (with optional icon)\n        this._set_body_content(options.body, this.data.icon);\n\n        // Set buttons\n        this._set_buttons();\n\n        // Create promise that will resolve when modal closes\n        const result_promise = new Promise((resolve) => {\n            that.data.resolve_fn = resolve;\n        });\n\n        // Show modal and backdrop\n        this.data.is_visible = true;\n\n        // Append to body so it's on top (don't append backdrop if using shared)\n        if (!skip_backdrop) {\n            $('body').append(this.$id('backdrop'));\n        }\n        $('body').append(this.$);\n\n        // Apply sizing before showing\n        this._apply_sizing();\n\n        // Fade in modal (and backdrop if not using shared)\n        await this._fade_in(should_animate);\n\n        // Auto-focus first input element\n        this._focus_first_input();\n\n        return result_promise;\n    }\n\n    /**\n     * Set body content with optional icon\n     */\n    _set_body_content(body, icon) {\n        const $body = this.$id('body');\n        $body.empty();\n\n        // If icon provided, add it\n        if (icon) {\n            const $icon = $(`<i class=\"bi bi-${icon} modal-icon\"></i>`);\n            $body.append($icon);\n            $body.addClass('has-icon');\n        } else {\n            $body.removeClass('has-icon');\n        }\n\n        // Get or create body content wrapper\n        let $content = this.$id('body_content');\n        if (!$content.exists()) {\n            $content = $('<div class=\"modal-body-content\"></div>');\n            $body.append($content);\n        }\n\n        if (typeof body === 'string') {\n            // Text content - escape and convert newlines\n            const escaped = $('<div>').text(body).html().replace(/\\n/g, '<br>');\n            $content.html(escaped);\n        } else if (body instanceof jQuery) {\n            // jQuery element\n            $content.append(body);\n        } else if (body && typeof body === 'object') {\n            // Assume it's a jqhtml component instance\n            $content.append(body.$);\n        }\n    }\n\n    /**\n     * Set buttons in footer\n     */\n    _set_buttons() {\n        const that = this;\n        const $footer = this.$id('footer');\n        $footer.empty();\n\n        if (this.data.buttons.length === 0) {\n            $footer.hide();\n            return;\n        }\n\n        $footer.show();\n\n        for (let button_def of this.data.buttons) {\n            const $button = $('<button>')\n                .attr('type', 'button')\n                .addClass('btn')\n                .addClass(button_def.class || 'btn-secondary')\n                .text(button_def.label || 'Button');\n\n            $button.on('click', async function () {\n                let result = button_def.value;\n                let had_callback = false;\n\n                // If button has a callback, call it and use return value as result\n                if (button_def.callback && typeof button_def.callback === 'function') {\n                    had_callback = true;\n                    result = await button_def.callback();\n                }\n\n                // If callback returned false, keep modal open (but not if just button value is false)\n                if (result === false && had_callback) {\n                    return;\n                }\n\n                // Close modal with result\n                that.close(result);\n            });\n\n            $footer.append($button);\n        }\n    }\n\n    /**\n     * Calculate and apply responsive sizing\n     */\n    _apply_sizing() {\n        const viewport_width = $(window).width();\n        const viewport_height = $(window).height();\n        const is_mobile = viewport_width < 768;\n\n        // Calculate max width based on viewport\n        let max_width = this.data.max_width;\n        const viewport_limit = is_mobile ? viewport_width * 0.9 : viewport_width * 0.8;\n\n        max_width = Math.min(max_width, viewport_limit);\n\n        // Try to constrain to 60% width for better proportions on desktop\n        if (!is_mobile) {\n            const preferred_width = viewport_width * 0.6;\n            if (preferred_width < max_width) {\n                max_width = preferred_width;\n            }\n        }\n\n        // Apply width\n        this.$id('dialog').css('max-width', max_width + 'px');\n\n        // Check if content exceeds 80% height\n        const content_height = this.$id('dialog').outerHeight();\n        const max_height = viewport_height * 0.8;\n\n        if (content_height > max_height) {\n            // Enable scrolling\n            this.$id('dialog').css('max-height', max_height + 'px');\n            this.$id('body').css({\n                'overflow-y': 'auto',\n                'max-height': max_height - 150 + 'px', // Account for header/footer\n            });\n        } else {\n            // Reset scrolling\n            this.$id('dialog').css('max-height', '');\n            this.$id('body').css({\n                'overflow-y': '',\n                'max-height': '',\n            });\n        }\n\n        // Mobile edge spacing\n        if (is_mobile) {\n            this.$id('dialog').css('margin', '5%');\n        } else {\n            this.$id('dialog').css('margin', '0');\n        }\n    }\n\n    /**\n     * Show animation (instant or with fly-in)\n     * @param {boolean} animate - Whether to animate the modal entrance\n     */\n    async _fade_in(animate = false) {\n        if (animate) {\n            // Initial state: modal positioned above final position\n            this.$.css('display', 'flex').css('opacity', '0');\n            this.$id('modal').css({\n                'transform': 'translate(0, -50px)',\n                'opacity': '0'\n            });\n            this.$id('backdrop').css('display', 'block').addClass('show');\n\n            // Force reflow\n            this.$id('modal')[0].offsetHeight;\n\n            // Trigger animation\n            this.$id('modal').addClass('show').css({\n                'transform': 'translate(0, 0)',\n                'opacity': '1'\n            });\n            this.$.css('opacity', '1');\n\n            // Wait for animation to complete\n            await new Promise(resolve => setTimeout(resolve, 150));\n        } else {\n            // Disable transitions temporarily for instant display\n            this.$id('dialog').css('transition', 'none');\n\n            // Show modal and backdrop instantly\n            this.$.css('display', 'flex').css('opacity', '1');\n            this.$id('modal').addClass('show').css('opacity', '1');\n            this.$id('backdrop').css('display', 'block').addClass('show');\n\n            // Force reflow to apply the no-transition state\n            this.$id('dialog')[0].offsetHeight;\n\n            // Re-enable transitions for future animations\n            this.$id('dialog').css('transition', '');\n        }\n\n        return Promise.resolve();\n    }\n\n    /**\n     * Focus the first input element in the modal\n     */\n    _focus_first_input() {\n        // Find first input/textarea/select in modal body\n        const $first_input = this.$id('body').find('input:not([type=\"hidden\"]), textarea, select').first();\n        if ($first_input.exists()) {\n            requestAnimationFrame(() => {\n                $first_input.focus();\n                // Select text if it's an input with existing value\n                if ($first_input.is('input[type=\"text\"], input[type=\"email\"]') && $first_input.val()) {\n                    $first_input.select();\n                }\n            });\n        }\n    }\n\n    /**\n     * Close the modal instantly\n     */\n    async close(result) {\n        const that = this;\n\n        // Mark as not visible\n        this.data.is_visible = false;\n\n        // Remove event listeners\n        $(document).off('keydown.rsx_modal_' + this._cid);\n        $(window).off('resize.rsx_modal_' + this._cid);\n\n        // Hide instantly (no fade out)\n        this.$.hide();\n        this.$id('backdrop').hide();\n\n        // Remove from DOM\n        this.$.remove();\n        this.$id('backdrop').remove();\n\n        // Resolve promise\n        if (this.data.resolve_fn) {\n            this.data.resolve_fn(result);\n            this.data.resolve_fn = null;\n        }\n    }\n\n    /**\n     * Apply validation errors to form fields in modal body\n     */\n    apply_errors(errors) {\n        // Use Form_Utils to apply errors to elements within modal body\n        Form_Utils.apply_form_errors(this.$id('body'), errors);\n    }\n}\n"],"mappings":"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","ignoreList":[]}
|