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>
704 lines
76 KiB
JavaScript
Executable File
704 lines
76 KiB
JavaScript
Executable File
"use strict";
|
|
|
|
function _d1f5a3cb_defineProperty(e, r, t) { return (r = _d1f5a3cb_toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
|
|
function _d1f5a3cb_toPropertyKey(t) { var i = _d1f5a3cb_toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
|
|
function _d1f5a3cb_toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
|
/**
|
|
* Modal Static API
|
|
*
|
|
* Primary interface for displaying modals throughout the application.
|
|
* Provides simple methods for common dialogs and flexible options for custom modals.
|
|
*
|
|
* Usage:
|
|
* await Modal.alert("File saved")
|
|
* if (await Modal.confirm("Delete?")) { ... }
|
|
* let name = await Modal.prompt("Enter name:")
|
|
* let result = await Modal.show({ title, body, buttons })
|
|
*/
|
|
class Modal {
|
|
/**
|
|
* Initialize global handlers (called automatically on first modal)
|
|
* @private
|
|
*/
|
|
static _init() {
|
|
if (this._initialized) return;
|
|
this._initialized = true;
|
|
|
|
// Create shared backdrop element
|
|
this._backdrop = $('<div class="modal-backdrop fade"></div>');
|
|
$('body').append(this._backdrop);
|
|
}
|
|
|
|
/**
|
|
* Calculate scrollbar width
|
|
* @private
|
|
* @returns {number}
|
|
*/
|
|
static _get_scrollbar_width() {
|
|
// Create temporary element to measure scrollbar width
|
|
const $outer = $('<div>').css({
|
|
visibility: 'hidden',
|
|
overflow: 'scroll',
|
|
width: '100px',
|
|
position: 'absolute',
|
|
top: '-9999px'
|
|
});
|
|
$('body').append($outer);
|
|
const width_with_scrollbar = $outer[0].offsetWidth;
|
|
const $inner = $('<div>').css('width', '100%');
|
|
$outer.append($inner);
|
|
const width_without_scrollbar = $inner[0].offsetWidth;
|
|
$outer.remove();
|
|
return width_with_scrollbar - width_without_scrollbar;
|
|
}
|
|
|
|
/**
|
|
* Lock body scroll and compensate for scrollbar width
|
|
* Only locks if we haven't already saved the original state (first modal in chain)
|
|
* @private
|
|
*/
|
|
static _lock_body_scroll() {
|
|
// Cancel any pending unlock timeout
|
|
if (this._unlock_timeout) {
|
|
clearTimeout(this._unlock_timeout);
|
|
this._unlock_timeout = null;
|
|
}
|
|
|
|
// Only lock scroll if we haven't already saved state (first modal)
|
|
// This is the true indicator - not backdrop visibility which can be transitional
|
|
if (this._original_body_overflow === null) {
|
|
const $body = $('body');
|
|
|
|
// Store original values
|
|
this._original_body_overflow = $body.css('overflow');
|
|
this._original_body_padding = $body.css('padding-right');
|
|
|
|
// Check if body currently has vertical scroll
|
|
const has_scrollbar = document.body.scrollHeight > window.innerHeight;
|
|
|
|
// If there's a scrollbar, add padding to compensate for its removal
|
|
if (has_scrollbar) {
|
|
const scrollbar_width = this._get_scrollbar_width();
|
|
const current_padding = int(this._original_body_padding) || 0;
|
|
$body.css('padding-right', current_padding + scrollbar_width + 'px');
|
|
}
|
|
|
|
// Lock scroll
|
|
$body.css('overflow', 'hidden');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Unlock body scroll and restore original state
|
|
* Uses delayed check to ensure no other modals are opening
|
|
* @private
|
|
*/
|
|
static _unlock_body_scroll() {
|
|
// Clear any existing timeout
|
|
if (this._unlock_timeout) {
|
|
clearTimeout(this._unlock_timeout);
|
|
}
|
|
|
|
// Minimal delay before unlocking
|
|
this._unlock_timeout = setTimeout(() => {
|
|
// Double-check no modal is currently open and queue is empty
|
|
if (!this._current && this._queue.length === 0) {
|
|
const $body = $('body');
|
|
|
|
// Restore original values
|
|
if (this._original_body_overflow !== null) {
|
|
$body.css('overflow', this._original_body_overflow);
|
|
this._original_body_overflow = null;
|
|
}
|
|
if (this._original_body_padding !== null) {
|
|
$body.css('padding-right', this._original_body_padding);
|
|
this._original_body_padding = null;
|
|
}
|
|
}
|
|
this._unlock_timeout = null;
|
|
}, 50); // Minimal safety buffer
|
|
}
|
|
|
|
/**
|
|
* Show the shared backdrop (instant - no animation)
|
|
* @private
|
|
*/
|
|
static async _show_backdrop() {
|
|
if (!this._backdrop.hasClass('show')) {
|
|
// Lock body scroll before showing backdrop
|
|
this._lock_body_scroll();
|
|
this._backdrop.css('display', 'block').addClass('show');
|
|
// No delay - return immediately
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Hide the shared backdrop (instant - no animation)
|
|
* @private
|
|
*/
|
|
static async _hide_backdrop() {
|
|
this._backdrop.removeClass('show').css('display', 'none');
|
|
|
|
// Unlock body scroll after backdrop is hidden
|
|
this._unlock_body_scroll();
|
|
}
|
|
|
|
/**
|
|
* Create a new Rsx_Modal instance
|
|
* @private
|
|
*/
|
|
static async _create_modal() {
|
|
// Create modal component using jQuery plugin
|
|
const $modal_element = $('<div>');
|
|
|
|
// Create component instance directly (returns the component)
|
|
const modal_instance = $modal_element.component('Rsx_Modal', {});
|
|
|
|
// Wait for component to be fully ready (DOM elements queryable)
|
|
await new Promise(resolve => {
|
|
modal_instance.on('ready', () => {
|
|
console.log('[Modal] Component ready, elements:', {
|
|
title: modal_instance.$sid('title').length,
|
|
body: modal_instance.$sid('body').length,
|
|
footer: modal_instance.$sid('footer').length
|
|
});
|
|
resolve();
|
|
});
|
|
});
|
|
return modal_instance;
|
|
}
|
|
|
|
/**
|
|
* Show a modal and manage queue
|
|
* @private
|
|
*/
|
|
static async _show_modal(options) {
|
|
return new Promise(resolve => {
|
|
this._queue.push({
|
|
options,
|
|
resolve
|
|
});
|
|
|
|
// Process queue if no modal currently showing
|
|
if (!this._current) {
|
|
this._process_queue();
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Process the modal queue
|
|
* @private
|
|
*/
|
|
static async _process_queue() {
|
|
if (this._queue.length === 0) {
|
|
this._current = null;
|
|
// Hide backdrop when queue is empty
|
|
await this._hide_backdrop();
|
|
return;
|
|
}
|
|
const {
|
|
options,
|
|
resolve
|
|
} = this._queue.shift();
|
|
|
|
// Ensure initialized
|
|
this._init();
|
|
|
|
// Show backdrop if not already visible (instant - no delay between modals)
|
|
const backdrop_visible = this._backdrop.hasClass('show');
|
|
if (!backdrop_visible) {
|
|
await this._show_backdrop();
|
|
}
|
|
// No delay between sequential modals - immediate transition
|
|
|
|
// Create modal instance
|
|
const modal_instance = await this._create_modal();
|
|
this._current = modal_instance;
|
|
|
|
// Determine if we should animate based on:
|
|
// 1. Desktop viewport (>= 1000px)
|
|
// 2. More than 1 second since last modal closed
|
|
const viewport_width = $(window).width();
|
|
const is_desktop = viewport_width >= 1000;
|
|
const time_since_last_close = Date.now() - this._last_close_timestamp;
|
|
const should_animate = is_desktop && time_since_last_close > 1000;
|
|
|
|
// Show modal and wait for result (modal won't create its own backdrop)
|
|
const result = await modal_instance.show(options, {
|
|
skip_backdrop: true,
|
|
animate: should_animate
|
|
});
|
|
|
|
// Record close timestamp BEFORE resolving (ensures it's set before next modal can start)
|
|
this._last_close_timestamp = Date.now();
|
|
|
|
// Resolve the promise with the result
|
|
resolve(result);
|
|
|
|
// Clear current and process next
|
|
this._current = null;
|
|
this._process_queue();
|
|
}
|
|
|
|
// ================================================================================
|
|
// State Management Methods
|
|
// ================================================================================
|
|
|
|
/**
|
|
* Check if a modal is currently open
|
|
* @returns {boolean}
|
|
*/
|
|
static is_open() {
|
|
return this._current !== null;
|
|
}
|
|
|
|
/**
|
|
* Get the currently open modal instance
|
|
* @returns {Rsx_Modal|null}
|
|
*/
|
|
static get_current() {
|
|
return this._current;
|
|
}
|
|
|
|
/**
|
|
* Force close the current modal
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async close() {
|
|
if (this._current) {
|
|
await this._current.close(false);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Apply validation errors to the current modal
|
|
* @param {Object} errors - Error object {field: message}
|
|
*/
|
|
static apply_errors(errors) {
|
|
if (this._current) {
|
|
this._current.apply_errors(errors);
|
|
}
|
|
}
|
|
|
|
// ================================================================================
|
|
// Simple Dialog Methods
|
|
// ================================================================================
|
|
|
|
/**
|
|
* Show an alert dialog
|
|
* @param {string|jQuery} title_or_body - Message (if only 1 arg) or Title (if 2 args). Can be string or jQuery element.
|
|
* @param {string|jQuery} body - Message body (if 2 args). Can be string or jQuery element.
|
|
* @param {string} button_label - Button text (default: "OK")
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async alert(title_or_body) {
|
|
let body = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null;
|
|
let button_label = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 'OK';
|
|
let title = 'Notice';
|
|
let message = title_or_body;
|
|
if (body !== null) {
|
|
title = title_or_body;
|
|
message = body;
|
|
}
|
|
await this._show_modal({
|
|
title: title,
|
|
body: message,
|
|
buttons: [{
|
|
label: button_label,
|
|
value: true,
|
|
class: 'btn-primary',
|
|
default: true
|
|
}],
|
|
closable: true,
|
|
close_on_submit: true
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Show a confirmation dialog
|
|
* @param {string|jQuery} title_or_body - Message (if 1-2 args) or Title (if 3-4 args). Can be string or jQuery element.
|
|
* @param {string|jQuery} body - Message body (optional). Can be string or jQuery element.
|
|
* @param {string} confirm_label - Confirm button text (default: "Confirm")
|
|
* @param {string} cancel_label - Cancel button text (default: "Cancel")
|
|
* @returns {Promise<boolean>}
|
|
*/
|
|
static async confirm(title_or_body) {
|
|
let body = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null;
|
|
let confirm_label = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 'Confirm';
|
|
let cancel_label = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'Cancel';
|
|
let title = 'Confirm';
|
|
let message = title_or_body;
|
|
if (body !== null) {
|
|
title = title_or_body;
|
|
message = body;
|
|
}
|
|
const result = await this._show_modal({
|
|
title: title,
|
|
body: message,
|
|
buttons: [{
|
|
label: cancel_label,
|
|
value: false,
|
|
class: 'btn-secondary'
|
|
}, {
|
|
label: confirm_label,
|
|
value: true,
|
|
class: 'btn-primary',
|
|
default: true
|
|
}],
|
|
closable: true,
|
|
close_on_submit: true
|
|
});
|
|
return result === true;
|
|
}
|
|
|
|
/**
|
|
* Show a prompt dialog for text input
|
|
* @param {string|jQuery} title_or_body - Message (if 1-3 args) or Title (if 4 args). Can be string or jQuery element.
|
|
* @param {string|jQuery} body - Message body (optional). Can be string or jQuery element.
|
|
* @param {string} default_value - Default input value
|
|
* @param {boolean} multiline - Show textarea instead of input
|
|
* @param {string} error - Optional error message to display as validation feedback
|
|
* @returns {Promise<string|false>}
|
|
*/
|
|
static async prompt(title_or_body) {
|
|
let body = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null;
|
|
let default_value = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
|
|
let multiline = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : false;
|
|
let error = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : null;
|
|
let title = 'Input';
|
|
let message = title_or_body;
|
|
|
|
// Handle overloaded arguments
|
|
if (typeof body === 'string' && body !== '') {
|
|
title = title_or_body;
|
|
message = body;
|
|
}
|
|
|
|
// Create input element with minimum width constraints
|
|
const $input = multiline ? $('<textarea class="form-control" rows="4" style="min-width: 315px;"></textarea>') : $('<input type="text" class="form-control" style="min-width: 245px;">');
|
|
$input.val(default_value);
|
|
|
|
// Mark as invalid if there's an error
|
|
if (error) {
|
|
$input.addClass('is-invalid');
|
|
}
|
|
|
|
// Create body with message and input
|
|
let $body;
|
|
if (message instanceof jQuery) {
|
|
// If message is a jQuery element, use it as the container and append input
|
|
$body = message.append($input);
|
|
} else {
|
|
// If message is a string, create wrapper with text and input (36px spacing)
|
|
$body = $('<div class="form-group">').append($('<div style="margin-bottom: 36px;">').text(message)).append($input);
|
|
}
|
|
|
|
// Add error message if provided
|
|
if (error) {
|
|
const $error = $('<div class="invalid-feedback d-block"></div>').text(error);
|
|
$body.append($error);
|
|
}
|
|
const result = await this._show_modal({
|
|
title: title,
|
|
body: $body,
|
|
buttons: [{
|
|
label: 'Cancel',
|
|
value: false,
|
|
class: 'btn-secondary'
|
|
}, {
|
|
label: 'Submit',
|
|
value: null,
|
|
// Will be replaced by callback
|
|
class: 'btn-primary',
|
|
default: true,
|
|
callback: function () {
|
|
return $input.val();
|
|
}
|
|
}],
|
|
closable: true,
|
|
close_on_submit: true,
|
|
max_width: 500
|
|
});
|
|
|
|
// Focus and select input after modal shows
|
|
requestAnimationFrame(() => {
|
|
$input.focus();
|
|
if (!multiline) {
|
|
$input.select();
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Show an error dialog with red alert styling
|
|
*
|
|
* Can appear over other modals to show critical uncaught exceptions.
|
|
* Used primarily for Ajax errors that weren't caught by application code.
|
|
*
|
|
* @param {string|Error|Object} error - Error message string, Error object, or structured error
|
|
* @param {string} title - Modal title (default: "Error")
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async error(error) {
|
|
let title = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'Error';
|
|
let message = '';
|
|
|
|
// Handle different error types
|
|
if (typeof error === 'string') {
|
|
message = error;
|
|
} else if (error instanceof Error) {
|
|
message = error.message || error.toString();
|
|
} else if (error && error.message) {
|
|
message = error.message;
|
|
} else if (error && error.error) {
|
|
// Fatal error with details
|
|
const details = error.error;
|
|
if (details.file && details.line) {
|
|
message = `Uncaught Fatal Error in ${details.file}:${details.line}:\n\n${details.error}`;
|
|
} else {
|
|
message = details.error || 'An unknown error occurred';
|
|
}
|
|
} else {
|
|
message = 'An unknown error occurred';
|
|
}
|
|
|
|
// Create error body with red alert styling
|
|
const $body = $('<div class="alert alert-danger mb-0" role="alert">').append($('<pre class="mb-0" style="white-space: pre-wrap; word-wrap: break-word; font-family: monospace; font-size: 0.9em;">').text(message));
|
|
await this._show_modal({
|
|
title: title,
|
|
body: $body,
|
|
buttons: [{
|
|
label: 'Close',
|
|
value: true,
|
|
class: 'btn-danger',
|
|
default: true
|
|
}],
|
|
closable: true,
|
|
close_on_submit: true,
|
|
max_width: 600
|
|
});
|
|
}
|
|
|
|
// ================================================================================
|
|
// Custom Modal Methods
|
|
// ================================================================================
|
|
|
|
/**
|
|
* Show a custom modal with specified content and buttons
|
|
* @param {Object} options
|
|
* @returns {Promise<*>}
|
|
*/
|
|
static async show(options) {
|
|
const defaults = {
|
|
title: 'Modal',
|
|
body: '',
|
|
buttons: [],
|
|
max_width: 800,
|
|
closable: true,
|
|
close_on_submit: true
|
|
};
|
|
const final_options = Object.assign({}, defaults, options);
|
|
return await this._show_modal(final_options);
|
|
}
|
|
|
|
/**
|
|
* Show a modal with a jqhtml form component
|
|
* @param {Object} options
|
|
* @param {string} options.component - Component class name
|
|
* @param {Object} options.component_args - Arguments to pass to component
|
|
* @param {Function} options.on_submit - Callback function called on submit. Receives form component instance.
|
|
* Return false to keep modal open, or return data to close and resolve.
|
|
* @returns {Promise<Object|false>}
|
|
*/
|
|
static async form(options) {
|
|
const defaults = {
|
|
title: 'Form',
|
|
component: null,
|
|
component_args: {},
|
|
max_width: 800,
|
|
closable: true,
|
|
submit_label: 'Submit',
|
|
cancel_label: 'Cancel',
|
|
on_submit: null
|
|
};
|
|
const final_options = Object.assign({}, defaults, options);
|
|
if (!final_options.component) {
|
|
console.error('Modal.form() requires a component');
|
|
return false;
|
|
}
|
|
|
|
// Create component instance
|
|
let $component_container = $('<div>');
|
|
let component_instance = $component_container.component(final_options.component, final_options.component_args);
|
|
|
|
// Wait for component to be ready
|
|
await new Promise(resolve => {
|
|
component_instance.on('ready', () => resolve());
|
|
});
|
|
|
|
// Find a form instance if component instance doesnt have .vals()
|
|
if (!component_instance.vals) {
|
|
let $form = component_instance.$.find('.Rsx_Form');
|
|
if ($form.exists()) {
|
|
component_instance = $form.component();
|
|
}
|
|
}
|
|
|
|
// Create buttons
|
|
const buttons = [{
|
|
label: final_options.cancel_label,
|
|
value: false,
|
|
class: 'btn-secondary'
|
|
}, {
|
|
label: final_options.submit_label,
|
|
value: null,
|
|
class: 'btn-primary',
|
|
default: true,
|
|
callback: async function () {
|
|
// If on_submit callback provided, use it
|
|
if (final_options.on_submit && typeof final_options.on_submit === 'function') {
|
|
const result = await final_options.on_submit(component_instance);
|
|
// If callback returns null/undefined, keep modal open
|
|
if (result === null || result === undefined) {
|
|
return false;
|
|
}
|
|
// Otherwise (including false), return the result to close modal
|
|
return result;
|
|
}
|
|
|
|
// No on_submit callback - get form data and close modal
|
|
if (component_instance.submit && typeof component_instance.submit === 'function') {
|
|
return await component_instance.submit();
|
|
} else if (component_instance.vals && typeof component_instance.vals === 'function') {
|
|
return component_instance.vals();
|
|
} else {
|
|
console.warn('Form component has no submit() or vals() method');
|
|
return true;
|
|
}
|
|
}
|
|
}];
|
|
return await this._show_modal({
|
|
title: final_options.title,
|
|
body: component_instance.$,
|
|
buttons: buttons,
|
|
max_width: final_options.max_width,
|
|
closable: final_options.closable
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Show an unclosable modal
|
|
* @param {string} title_or_body
|
|
* @param {string} body
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async unclosable(title_or_body) {
|
|
let body = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null;
|
|
let title = 'Please Wait';
|
|
let message = title_or_body;
|
|
if (body !== null) {
|
|
title = title_or_body;
|
|
message = body;
|
|
}
|
|
|
|
// Don't wait for this promise - it never resolves until closed manually
|
|
this._show_modal({
|
|
title: title,
|
|
body: message,
|
|
buttons: [],
|
|
// No buttons
|
|
closable: false,
|
|
// Can't close
|
|
close_on_submit: false
|
|
});
|
|
|
|
// Wait for next animation frame for modal to render
|
|
await new Promise(resolve => requestAnimationFrame(resolve));
|
|
}
|
|
|
|
/**
|
|
* Show a modal with custom jQuery content
|
|
* @param {Object} options
|
|
* @returns {Promise<*>}
|
|
*/
|
|
static async custom(options) {
|
|
// Alias for show() - same functionality
|
|
return await this.show(options);
|
|
}
|
|
|
|
// ================================================================================
|
|
// Helper Methods
|
|
// ================================================================================
|
|
|
|
/**
|
|
* Show an error alert
|
|
* @param {*} errors
|
|
* @param {string} title
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async error(errors) {
|
|
let title = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'Error';
|
|
let message = 'An error occurred';
|
|
|
|
// Handle various error formats
|
|
if (typeof errors === 'string') {
|
|
message = errors;
|
|
} else if (errors && 'responseJSON' in errors && 'message' in errors.responseJSON) {
|
|
message = errors.responseJSON.message;
|
|
} else if (errors && 'message' in errors) {
|
|
message = errors.message;
|
|
} else if (errors && typeof errors === 'object') {
|
|
// Try to format error object
|
|
const error_messages = [];
|
|
for (const key in errors) {
|
|
if (is_array(errors[key])) {
|
|
error_messages.push(errors[key][0]);
|
|
} else {
|
|
error_messages.push(errors[key]);
|
|
}
|
|
}
|
|
if (error_messages.length > 0) {
|
|
message = error_messages.join('\n');
|
|
}
|
|
}
|
|
await this._show_modal({
|
|
title: title,
|
|
body: message,
|
|
icon: 'exclamation-circle',
|
|
buttons: [{
|
|
label: 'OK',
|
|
value: true,
|
|
class: 'btn-danger',
|
|
default: true
|
|
}],
|
|
closable: true,
|
|
close_on_submit: true
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Reopen current modal with validation errors
|
|
* @param {Object} errors
|
|
* @returns {Promise<void>}
|
|
*/
|
|
static async reopen_with_errors(errors) {
|
|
if (this._current) {
|
|
// Modal is still open, just apply errors
|
|
this.apply_errors(errors);
|
|
} else {
|
|
console.warn('No modal open to apply errors to');
|
|
}
|
|
}
|
|
}
|
|
// Internal state
|
|
_d1f5a3cb_defineProperty(Modal, "_queue", []);
|
|
_d1f5a3cb_defineProperty(Modal, "_current", null);
|
|
_d1f5a3cb_defineProperty(Modal, "_initialized", false);
|
|
_d1f5a3cb_defineProperty(Modal, "_backdrop", null);
|
|
_d1f5a3cb_defineProperty(Modal, "_original_body_overflow", null);
|
|
_d1f5a3cb_defineProperty(Modal, "_original_body_padding", null);
|
|
_d1f5a3cb_defineProperty(Modal, "_unlock_timeout", null);
|
|
_d1f5a3cb_defineProperty(Modal, "_last_close_timestamp", 0);
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Modal","_init","_initialized","_backdrop","$","append","_get_scrollbar_width","$outer","css","visibility","overflow","width","position","top","width_with_scrollbar","offsetWidth","$inner","width_without_scrollbar","remove","_lock_body_scroll","_unlock_timeout","clearTimeout","_original_body_overflow","$body","_original_body_padding","has_scrollbar","document","body","scrollHeight","window","innerHeight","scrollbar_width","current_padding","int","_unlock_body_scroll","setTimeout","_current","_queue","length","_show_backdrop","hasClass","addClass","_hide_backdrop","removeClass","_create_modal","$modal_element","modal_instance","component","Promise","resolve","on","console","log","title","$id","footer","_show_modal","options","push","_process_queue","shift","backdrop_visible","viewport_width","is_desktop","time_since_last_close","Date","now","_last_close_timestamp","should_animate","result","show","skip_backdrop","animate","is_open","get_current","close","apply_errors","errors","alert","title_or_body","arguments","undefined","button_label","message","buttons","label","value","class","default","closable","close_on_submit","confirm","confirm_label","cancel_label","prompt","default_value","multiline","error","$input","val","jQuery","text","$error","callback","max_width","requestAnimationFrame","focus","select","Error","toString","details","file","line","defaults","final_options","Object","assign","form","component_args","submit_label","on_submit","$component_container","component_instance","vals","$form","find","exists","submit","warn","unclosable","custom","responseJSON","error_messages","key","is_array","join","icon","reopen_with_errors","_d1f5a3cb_defineProperty"],"sources":["rsx/theme/components/modal/Modal.js"],"sourcesContent":["/**\n * Modal Static API\n *\n * Primary interface for displaying modals throughout the application.\n * Provides simple methods for common dialogs and flexible options for custom modals.\n *\n * Usage:\n *   await Modal.alert(\"File saved\")\n *   if (await Modal.confirm(\"Delete?\")) { ... }\n *   let name = await Modal.prompt(\"Enter name:\")\n *   let result = await Modal.show({ title, body, buttons })\n */\nclass Modal {\n    // Internal state\n    static _queue = [];\n    static _current = null;\n    static _initialized = false;\n    static _backdrop = null;\n    static _original_body_overflow = null;\n    static _original_body_padding = null;\n    static _unlock_timeout = null;\n    static _last_close_timestamp = 0;\n\n    /**\n     * Initialize global handlers (called automatically on first modal)\n     * @private\n     */\n    static _init() {\n        if (this._initialized) return;\n        this._initialized = true;\n\n        // Create shared backdrop element\n        this._backdrop = $('<div class=\"modal-backdrop fade\"></div>');\n        $('body').append(this._backdrop);\n    }\n\n    /**\n     * Calculate scrollbar width\n     * @private\n     * @returns {number}\n     */\n    static _get_scrollbar_width() {\n        // Create temporary element to measure scrollbar width\n        const $outer = $('<div>').css({\n            visibility: 'hidden',\n            overflow: 'scroll',\n            width: '100px',\n            position: 'absolute',\n            top: '-9999px',\n        });\n        $('body').append($outer);\n\n        const width_with_scrollbar = $outer[0].offsetWidth;\n\n        const $inner = $('<div>').css('width', '100%');\n        $outer.append($inner);\n\n        const width_without_scrollbar = $inner[0].offsetWidth;\n\n        $outer.remove();\n\n        return width_with_scrollbar - width_without_scrollbar;\n    }\n\n    /**\n     * Lock body scroll and compensate for scrollbar width\n     * Only locks if we haven't already saved the original state (first modal in chain)\n     * @private\n     */\n    static _lock_body_scroll() {\n        // Cancel any pending unlock timeout\n        if (this._unlock_timeout) {\n            clearTimeout(this._unlock_timeout);\n            this._unlock_timeout = null;\n        }\n\n        // Only lock scroll if we haven't already saved state (first modal)\n        // This is the true indicator - not backdrop visibility which can be transitional\n        if (this._original_body_overflow === null) {\n            const $body = $('body');\n\n            // Store original values\n            this._original_body_overflow = $body.css('overflow');\n            this._original_body_padding = $body.css('padding-right');\n\n            // Check if body currently has vertical scroll\n            const has_scrollbar = document.body.scrollHeight > window.innerHeight;\n\n            // If there's a scrollbar, add padding to compensate for its removal\n            if (has_scrollbar) {\n                const scrollbar_width = this._get_scrollbar_width();\n                const current_padding = int(this._original_body_padding) || 0;\n                $body.css('padding-right', current_padding + scrollbar_width + 'px');\n            }\n\n            // Lock scroll\n            $body.css('overflow', 'hidden');\n        }\n    }\n\n    /**\n     * Unlock body scroll and restore original state\n     * Uses delayed check to ensure no other modals are opening\n     * @private\n     */\n    static _unlock_body_scroll() {\n        // Clear any existing timeout\n        if (this._unlock_timeout) {\n            clearTimeout(this._unlock_timeout);\n        }\n\n        // Minimal delay before unlocking\n        this._unlock_timeout = setTimeout(() => {\n            // Double-check no modal is currently open and queue is empty\n            if (!this._current && this._queue.length === 0) {\n                const $body = $('body');\n\n                // Restore original values\n                if (this._original_body_overflow !== null) {\n                    $body.css('overflow', this._original_body_overflow);\n                    this._original_body_overflow = null;\n                }\n\n                if (this._original_body_padding !== null) {\n                    $body.css('padding-right', this._original_body_padding);\n                    this._original_body_padding = null;\n                }\n            }\n\n            this._unlock_timeout = null;\n        }, 50); // Minimal safety buffer\n    }\n\n    /**\n     * Show the shared backdrop (instant - no animation)\n     * @private\n     */\n    static async _show_backdrop() {\n        if (!this._backdrop.hasClass('show')) {\n            // Lock body scroll before showing backdrop\n            this._lock_body_scroll();\n\n            this._backdrop.css('display', 'block').addClass('show');\n            // No delay - return immediately\n        }\n    }\n\n    /**\n     * Hide the shared backdrop (instant - no animation)\n     * @private\n     */\n    static async _hide_backdrop() {\n        this._backdrop.removeClass('show').css('display', 'none');\n\n        // Unlock body scroll after backdrop is hidden\n        this._unlock_body_scroll();\n    }\n\n    /**\n     * Create a new Rsx_Modal instance\n     * @private\n     */\n    static async _create_modal() {\n        // Create modal component using jQuery plugin\n        const $modal_element = $('<div>');\n\n        // Create component instance directly (returns the component)\n        const modal_instance = $modal_element.component('Rsx_Modal', {});\n\n        // Wait for component to be fully ready (DOM elements queryable)\n        await new Promise((resolve) => {\n            modal_instance.on('ready', () => {\n                console.log('[Modal] Component ready, elements:', {\n                    title: modal_instance.$id('title').length,\n                    body: modal_instance.$id('body').length,\n                    footer: modal_instance.$id('footer').length,\n                });\n                resolve();\n            });\n        });\n\n        return modal_instance;\n    }\n\n    /**\n     * Show a modal and manage queue\n     * @private\n     */\n    static async _show_modal(options) {\n        return new Promise((resolve) => {\n            this._queue.push({ options, resolve });\n\n            // Process queue if no modal currently showing\n            if (!this._current) {\n                this._process_queue();\n            }\n        });\n    }\n\n    /**\n     * Process the modal queue\n     * @private\n     */\n    static async _process_queue() {\n        if (this._queue.length === 0) {\n            this._current = null;\n            // Hide backdrop when queue is empty\n            await this._hide_backdrop();\n            return;\n        }\n\n        const { options, resolve } = this._queue.shift();\n\n        // Ensure initialized\n        this._init();\n\n        // Show backdrop if not already visible (instant - no delay between modals)\n        const backdrop_visible = this._backdrop.hasClass('show');\n        if (!backdrop_visible) {\n            await this._show_backdrop();\n        }\n        // No delay between sequential modals - immediate transition\n\n        // Create modal instance\n        const modal_instance = await this._create_modal();\n        this._current = modal_instance;\n\n        // Determine if we should animate based on:\n        // 1. Desktop viewport (>= 1000px)\n        // 2. More than 1 second since last modal closed\n        const viewport_width = $(window).width();\n        const is_desktop = viewport_width >= 1000;\n        const time_since_last_close = Date.now() - this._last_close_timestamp;\n        const should_animate = is_desktop && time_since_last_close > 1000;\n\n        // Show modal and wait for result (modal won't create its own backdrop)\n        const result = await modal_instance.show(options, { skip_backdrop: true, animate: should_animate });\n\n        // Record close timestamp BEFORE resolving (ensures it's set before next modal can start)\n        this._last_close_timestamp = Date.now();\n\n        // Resolve the promise with the result\n        resolve(result);\n\n        // Clear current and process next\n        this._current = null;\n        this._process_queue();\n    }\n\n    // ================================================================================\n    // State Management Methods\n    // ================================================================================\n\n    /**\n     * Check if a modal is currently open\n     * @returns {boolean}\n     */\n    static is_open() {\n        return this._current !== null;\n    }\n\n    /**\n     * Get the currently open modal instance\n     * @returns {Rsx_Modal|null}\n     */\n    static get_current() {\n        return this._current;\n    }\n\n    /**\n     * Force close the current modal\n     * @returns {Promise<void>}\n     */\n    static async close() {\n        if (this._current) {\n            await this._current.close(false);\n        }\n    }\n\n    /**\n     * Apply validation errors to the current modal\n     * @param {Object} errors - Error object {field: message}\n     */\n    static apply_errors(errors) {\n        if (this._current) {\n            this._current.apply_errors(errors);\n        }\n    }\n\n    // ================================================================================\n    // Simple Dialog Methods\n    // ================================================================================\n\n    /**\n     * Show an alert dialog\n     * @param {string|jQuery} title_or_body - Message (if only 1 arg) or Title (if 2 args). Can be string or jQuery element.\n     * @param {string|jQuery} body - Message body (if 2 args). Can be string or jQuery element.\n     * @param {string} button_label - Button text (default: \"OK\")\n     * @returns {Promise<void>}\n     */\n    static async alert(title_or_body, body = null, button_label = 'OK') {\n        let title = 'Notice';\n        let message = title_or_body;\n\n        if (body !== null) {\n            title = title_or_body;\n            message = body;\n        }\n\n        await this._show_modal({\n            title: title,\n            body: message,\n            buttons: [\n                {\n                    label: button_label,\n                    value: true,\n                    class: 'btn-primary',\n                    default: true,\n                },\n            ],\n            closable: true,\n            close_on_submit: true,\n        });\n    }\n\n    /**\n     * Show a confirmation dialog\n     * @param {string|jQuery} title_or_body - Message (if 1-2 args) or Title (if 3-4 args). Can be string or jQuery element.\n     * @param {string|jQuery} body - Message body (optional). Can be string or jQuery element.\n     * @param {string} confirm_label - Confirm button text (default: \"Confirm\")\n     * @param {string} cancel_label - Cancel button text (default: \"Cancel\")\n     * @returns {Promise<boolean>}\n     */\n    static async confirm(title_or_body, body = null, confirm_label = 'Confirm', cancel_label = 'Cancel') {\n        let title = 'Confirm';\n        let message = title_or_body;\n\n        if (body !== null) {\n            title = title_or_body;\n            message = body;\n        }\n\n        const result = await this._show_modal({\n            title: title,\n            body: message,\n            buttons: [\n                {\n                    label: cancel_label,\n                    value: false,\n                    class: 'btn-secondary',\n                },\n                {\n                    label: confirm_label,\n                    value: true,\n                    class: 'btn-primary',\n                    default: true,\n                },\n            ],\n            closable: true,\n            close_on_submit: true,\n        });\n\n        return result === true;\n    }\n\n    /**\n     * Show a prompt dialog for text input\n     * @param {string|jQuery} title_or_body - Message (if 1-3 args) or Title (if 4 args). Can be string or jQuery element.\n     * @param {string|jQuery} body - Message body (optional). Can be string or jQuery element.\n     * @param {string} default_value - Default input value\n     * @param {boolean} multiline - Show textarea instead of input\n     * @param {string} error - Optional error message to display as validation feedback\n     * @returns {Promise<string|false>}\n     */\n    static async prompt(title_or_body, body = null, default_value = '', multiline = false, error = null) {\n        let title = 'Input';\n        let message = title_or_body;\n\n        // Handle overloaded arguments\n        if (typeof body === 'string' && body !== '') {\n            title = title_or_body;\n            message = body;\n        }\n\n        // Create input element with minimum width constraints\n        const $input = multiline\n            ? $('<textarea class=\"form-control\" rows=\"4\" style=\"min-width: 315px;\"></textarea>')\n            : $('<input type=\"text\" class=\"form-control\" style=\"min-width: 245px;\">');\n\n        $input.val(default_value);\n\n        // Mark as invalid if there's an error\n        if (error) {\n            $input.addClass('is-invalid');\n        }\n\n        // Create body with message and input\n        let $body;\n        if (message instanceof jQuery) {\n            // If message is a jQuery element, use it as the container and append input\n            $body = message.append($input);\n        } else {\n            // If message is a string, create wrapper with text and input (36px spacing)\n            $body = $('<div class=\"form-group\">').append($('<div style=\"margin-bottom: 36px;\">').text(message)).append($input);\n        }\n\n        // Add error message if provided\n        if (error) {\n            const $error = $('<div class=\"invalid-feedback d-block\"></div>').text(error);\n            $body.append($error);\n        }\n\n        const result = await this._show_modal({\n            title: title,\n            body: $body,\n            buttons: [\n                {\n                    label: 'Cancel',\n                    value: false,\n                    class: 'btn-secondary',\n                },\n                {\n                    label: 'Submit',\n                    value: null, // Will be replaced by callback\n                    class: 'btn-primary',\n                    default: true,\n                    callback: function () {\n                        return $input.val();\n                    },\n                },\n            ],\n            closable: true,\n            close_on_submit: true,\n            max_width: 500,\n        });\n\n        // Focus and select input after modal shows\n        requestAnimationFrame(() => {\n            $input.focus();\n            if (!multiline) {\n                $input.select();\n            }\n        });\n\n        return result;\n    }\n\n    /**\n     * Show an error dialog with red alert styling\n     *\n     * Can appear over other modals to show critical uncaught exceptions.\n     * Used primarily for Ajax errors that weren't caught by application code.\n     *\n     * @param {string|Error|Object} error - Error message string, Error object, or structured error\n     * @param {string} title - Modal title (default: \"Error\")\n     * @returns {Promise<void>}\n     */\n    static async error(error, title = 'Error') {\n        let message = '';\n\n        // Handle different error types\n        if (typeof error === 'string') {\n            message = error;\n        } else if (error instanceof Error) {\n            message = error.message || error.toString();\n        } else if (error && error.message) {\n            message = error.message;\n        } else if (error && error.error) {\n            // Fatal error with details\n            const details = error.error;\n            if (details.file && details.line) {\n                message = `Uncaught Fatal Error in ${details.file}:${details.line}:\\n\\n${details.error}`;\n            } else {\n                message = details.error || 'An unknown error occurred';\n            }\n        } else {\n            message = 'An unknown error occurred';\n        }\n\n        // Create error body with red alert styling\n        const $body = $('<div class=\"alert alert-danger mb-0\" role=\"alert\">').append(\n            $('<pre class=\"mb-0\" style=\"white-space: pre-wrap; word-wrap: break-word; font-family: monospace; font-size: 0.9em;\">').text(\n                message\n            )\n        );\n\n        await this._show_modal({\n            title: title,\n            body: $body,\n            buttons: [\n                {\n                    label: 'Close',\n                    value: true,\n                    class: 'btn-danger',\n                    default: true,\n                },\n            ],\n            closable: true,\n            close_on_submit: true,\n            max_width: 600,\n        });\n    }\n\n    // ================================================================================\n    // Custom Modal Methods\n    // ================================================================================\n\n    /**\n     * Show a custom modal with specified content and buttons\n     * @param {Object} options\n     * @returns {Promise<*>}\n     */\n    static async show(options) {\n        const defaults = {\n            title: 'Modal',\n            body: '',\n            buttons: [],\n            max_width: 800,\n            closable: true,\n            close_on_submit: true,\n        };\n\n        const final_options = Object.assign({}, defaults, options);\n\n        return await this._show_modal(final_options);\n    }\n\n    /**\n     * Show a modal with a jqhtml form component\n     * @param {Object} options\n     * @param {string} options.component - Component class name\n     * @param {Object} options.component_args - Arguments to pass to component\n     * @param {Function} options.on_submit - Callback function called on submit. Receives form component instance.\n     *                                        Return false to keep modal open, or return data to close and resolve.\n     * @returns {Promise<Object|false>}\n     */\n    static async form(options) {\n        const defaults = {\n            title: 'Form',\n            component: null,\n            component_args: {},\n            max_width: 800,\n            closable: true,\n            submit_label: 'Submit',\n            cancel_label: 'Cancel',\n            on_submit: null,\n        };\n\n        const final_options = Object.assign({}, defaults, options);\n\n        if (!final_options.component) {\n            console.error('Modal.form() requires a component');\n            return false;\n        }\n\n        // Create component instance\n        let $component_container = $('<div>');\n        let component_instance = $component_container.component(final_options.component, final_options.component_args);\n\n        // Wait for component to be ready\n        await new Promise((resolve) => {\n            component_instance.on('ready', () => resolve());\n        });\n\n        // Find a form instance if component instance doesnt have .vals()\n        if (!component_instance.vals) {\n            let $form = component_instance.$.find('.Rsx_Form');\n            if ($form.exists()) {\n                component_instance = $form.component();\n            }\n        }\n\n        // Create buttons\n        const buttons = [\n            {\n                label: final_options.cancel_label,\n                value: false,\n                class: 'btn-secondary',\n            },\n            {\n                label: final_options.submit_label,\n                value: null,\n                class: 'btn-primary',\n                default: true,\n                callback: async function () {\n                    // If on_submit callback provided, use it\n                    if (final_options.on_submit && typeof final_options.on_submit === 'function') {\n                        const result = await final_options.on_submit(component_instance);\n                        // If callback returns null/undefined, keep modal open\n                        if (result === null || result === undefined) {\n                            return false;\n                        }\n                        // Otherwise (including false), return the result to close modal\n                        return result;\n                    }\n\n                    // No on_submit callback - get form data and close modal\n                    if (component_instance.submit && typeof component_instance.submit === 'function') {\n                        return await component_instance.submit();\n                    } else if (component_instance.vals && typeof component_instance.vals === 'function') {\n                        return component_instance.vals();\n                    } else {\n                        console.warn('Form component has no submit() or vals() method');\n                        return true;\n                    }\n                },\n            },\n        ];\n\n        return await this._show_modal({\n            title: final_options.title,\n            body: component_instance.$,\n            buttons: buttons,\n            max_width: final_options.max_width,\n            closable: final_options.closable,\n        });\n    }\n\n    /**\n     * Show an unclosable modal\n     * @param {string} title_or_body\n     * @param {string} body\n     * @returns {Promise<void>}\n     */\n    static async unclosable(title_or_body, body = null) {\n        let title = 'Please Wait';\n        let message = title_or_body;\n\n        if (body !== null) {\n            title = title_or_body;\n            message = body;\n        }\n\n        // Don't wait for this promise - it never resolves until closed manually\n        this._show_modal({\n            title: title,\n            body: message,\n            buttons: [], // No buttons\n            closable: false, // Can't close\n            close_on_submit: false,\n        });\n\n        // Wait for next animation frame for modal to render\n        await new Promise(resolve => requestAnimationFrame(resolve));\n    }\n\n    /**\n     * Show a modal with custom jQuery content\n     * @param {Object} options\n     * @returns {Promise<*>}\n     */\n    static async custom(options) {\n        // Alias for show() - same functionality\n        return await this.show(options);\n    }\n\n    // ================================================================================\n    // Helper Methods\n    // ================================================================================\n\n    /**\n     * Show an error alert\n     * @param {*} errors\n     * @param {string} title\n     * @returns {Promise<void>}\n     */\n    static async error(errors, title = 'Error') {\n        let message = 'An error occurred';\n\n        // Handle various error formats\n        if (typeof errors === 'string') {\n            message = errors;\n        } else if (errors && 'responseJSON' in errors && 'message' in errors.responseJSON) {\n            message = errors.responseJSON.message;\n        } else if (errors && 'message' in errors) {\n            message = errors.message;\n        } else if (errors && typeof errors === 'object') {\n            // Try to format error object\n            const error_messages = [];\n            for (const key in errors) {\n                if (is_array(errors[key])) {\n                    error_messages.push(errors[key][0]);\n                } else {\n                    error_messages.push(errors[key]);\n                }\n            }\n            if (error_messages.length > 0) {\n                message = error_messages.join('\\n');\n            }\n        }\n\n        await this._show_modal({\n            title: title,\n            body: message,\n            icon: 'exclamation-circle',\n            buttons: [\n                {\n                    label: 'OK',\n                    value: true,\n                    class: 'btn-danger',\n                    default: true,\n                },\n            ],\n            closable: true,\n            close_on_submit: true,\n        });\n    }\n\n    /**\n     * Reopen current modal with validation errors\n     * @param {Object} errors\n     * @returns {Promise<void>}\n     */\n    static async reopen_with_errors(errors) {\n        if (this._current) {\n            // Modal is still open, just apply errors\n            this.apply_errors(errors);\n        } else {\n            console.warn('No modal open to apply errors to');\n        }\n    }\n}\n"],"mappings":"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","ignoreList":[]}
|