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>
145 lines
17 KiB
JavaScript
Executable File
145 lines
17 KiB
JavaScript
Executable File
"use strict";
|
|
|
|
/**
|
|
* Profile_Photo_Input
|
|
*
|
|
* Profile photo upload widget with thumbnail display and upload handling.
|
|
* See profile_photo_input.jqhtml for full documentation.
|
|
*
|
|
* JavaScript Responsibilities:
|
|
* - Handle file selection and upload
|
|
* - Update thumbnail on successful upload
|
|
* - Manage loading state with spinner
|
|
* - Provide val() getter/setter for attachment key
|
|
* - Handle remove button functionality
|
|
*/
|
|
class Profile_Photo_Input extends Form_Input_Abstract {
|
|
on_create() {
|
|
// Initialize data
|
|
this.data.attachment_key = '';
|
|
this.data.thumbnail_url = '';
|
|
}
|
|
on_render() {
|
|
// Handle upload button click - trigger hidden file input
|
|
this.$sid('upload_btn').on('click', () => {
|
|
this.$sid('file_input').click();
|
|
});
|
|
|
|
// Handle file selection
|
|
this.$sid('file_input').on('change', () => {
|
|
const file = this.$sid('file_input')[0].files[0];
|
|
if (!file) return;
|
|
this.upload_photo(file);
|
|
});
|
|
|
|
// Handle remove button
|
|
if (this.args.show_remove) {
|
|
this.$sid('remove_btn').on('click', () => {
|
|
this.remove_photo();
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* val() - Get or set the attachment key
|
|
* @param {string} [key] - If provided, sets the attachment key and updates thumbnail
|
|
* @returns {string} The current attachment key when called as getter
|
|
*/
|
|
val(key) {
|
|
if (arguments.length === 0) {
|
|
// Getter - return attachment key
|
|
return this.data.attachment_key || '';
|
|
} else {
|
|
// Setter - set attachment key and update thumbnail
|
|
this.data.attachment_key = key || '';
|
|
if (this.data.attachment_key) {
|
|
// Generate thumbnail URL from attachment key
|
|
const width = this.args.width || 96;
|
|
const height = this.args.height || 96;
|
|
this.data.thumbnail_url = `/_thumbnail/${this.data.attachment_key}/cover/${width}/${height}`;
|
|
} else {
|
|
// No key - clear thumbnail
|
|
this.data.thumbnail_url = '';
|
|
}
|
|
console.log('Rerender');
|
|
// Re-render to switch between icon and image
|
|
this.render();
|
|
}
|
|
}
|
|
upload_photo(file) {
|
|
// Validate file size
|
|
const max_size = (this.args.max_size || 2) * 1024 * 1024; // Convert MB to bytes
|
|
if (file.size > max_size) {
|
|
alert(`File size must be less than ${this.args.max_size || 2}MB`);
|
|
this.$sid('file_input').val(''); // Clear selection
|
|
return;
|
|
}
|
|
|
|
// Show spinner, dim image
|
|
this.$sid('spinner').removeClass('d-none');
|
|
this.$sid('photo').css('opacity', '0.3');
|
|
|
|
// Create FormData for file upload
|
|
const form_data = new FormData();
|
|
form_data.append('file', file);
|
|
form_data.append('site_id', '1'); // TODO: Get from session/config
|
|
// Do NOT set fileable_type/fileable_category - file uploads unattached
|
|
// The parent form will assign it via attach_to() on save
|
|
|
|
// Upload file via AJAX
|
|
$.ajax({
|
|
url: '/_upload',
|
|
type: 'POST',
|
|
data: form_data,
|
|
processData: false,
|
|
contentType: false,
|
|
success: response => {
|
|
console.log('Profile photo upload successful:', response);
|
|
|
|
// Update attachment key (this will also update thumbnail)
|
|
this.val(response.attachment.key);
|
|
|
|
// Hide spinner, restore opacity
|
|
this.$sid('spinner').addClass('d-none');
|
|
this.$sid('photo').css('opacity', '1');
|
|
|
|
// Clear file input for future uploads
|
|
this.$sid('file_input').val('');
|
|
|
|
// Trigger change event for form tracking
|
|
this.$.trigger('change');
|
|
},
|
|
error: (xhr, status, error) => {
|
|
var _xhr$responseJSON;
|
|
console.error('Profile photo upload failed:', error);
|
|
console.error('Response:', xhr.responseJSON);
|
|
|
|
// Hide spinner, restore opacity
|
|
this.$sid('spinner').addClass('d-none');
|
|
this.$sid('photo').css('opacity', '1');
|
|
|
|
// Clear file input
|
|
this.$sid('file_input').val('');
|
|
|
|
// Show error to user
|
|
alert('Upload failed: ' + (((_xhr$responseJSON = xhr.responseJSON) === null || _xhr$responseJSON === void 0 ? void 0 : _xhr$responseJSON.error) || error));
|
|
}
|
|
});
|
|
}
|
|
update_photo() {
|
|
// <% if (this.args.show_remove && this.data.attachment_key) { %>
|
|
}
|
|
remove_photo() {
|
|
// Clear attachment key (sets to placeholder)
|
|
this.val('');
|
|
|
|
// Trigger change event for form tracking
|
|
this.$.trigger('change');
|
|
}
|
|
async seed() {
|
|
// For testing - set a placeholder key
|
|
// In production, this would use actual test data
|
|
this.val('');
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Profile_Photo_Input","Form_Input_Abstract","on_create","data","attachment_key","thumbnail_url","on_render","$id","on","click","file","files","upload_photo","args","show_remove","remove_photo","val","key","arguments","length","width","height","console","log","render","max_size","size","alert","removeClass","css","form_data","FormData","append","$","ajax","url","type","processData","contentType","success","response","attachment","addClass","trigger","error","xhr","status","_xhr$responseJSON","responseJSON","update_photo","seed"],"sources":["rsx/theme/components/inputs/profile_photo_input.js"],"sourcesContent":["/**\n * Profile_Photo_Input\n *\n * Profile photo upload widget with thumbnail display and upload handling.\n * See profile_photo_input.jqhtml for full documentation.\n *\n * JavaScript Responsibilities:\n * - Handle file selection and upload\n * - Update thumbnail on successful upload\n * - Manage loading state with spinner\n * - Provide val() getter/setter for attachment key\n * - Handle remove button functionality\n */\nclass Profile_Photo_Input extends Form_Input_Abstract {\n    on_create() {\n        // Initialize data\n        this.data.attachment_key = '';\n        this.data.thumbnail_url = '';\n    }\n\n    on_render() {\n        // Handle upload button click - trigger hidden file input\n        this.$id('upload_btn').on('click', () => {\n            this.$id('file_input').click();\n        });\n\n        // Handle file selection\n        this.$id('file_input').on('change', () => {\n            const file = this.$id('file_input')[0].files[0];\n            if (!file) return;\n\n            this.upload_photo(file);\n        });\n\n        // Handle remove button\n        if (this.args.show_remove) {\n            this.$id('remove_btn').on('click', () => {\n                this.remove_photo();\n            });\n        }\n    }\n\n    /**\n     * val() - Get or set the attachment key\n     * @param {string} [key] - If provided, sets the attachment key and updates thumbnail\n     * @returns {string} The current attachment key when called as getter\n     */\n    val(key) {\n        if (arguments.length === 0) {\n            // Getter - return attachment key\n            return this.data.attachment_key || '';\n        } else {\n            // Setter - set attachment key and update thumbnail\n            this.data.attachment_key = key || '';\n\n            if (this.data.attachment_key) {\n                // Generate thumbnail URL from attachment key\n                const width = this.args.width || 96;\n                const height = this.args.height || 96;\n                this.data.thumbnail_url = `/_thumbnail/${this.data.attachment_key}/cover/${width}/${height}`;\n            } else {\n                // No key - clear thumbnail\n                this.data.thumbnail_url = '';\n            }\n\n            console.log('Rerender');\n            // Re-render to switch between icon and image\n            this.render();\n        }\n    }\n\n    upload_photo(file) {\n        // Validate file size\n        const max_size = (this.args.max_size || 2) * 1024 * 1024; // Convert MB to bytes\n        if (file.size > max_size) {\n            alert(`File size must be less than ${this.args.max_size || 2}MB`);\n            this.$id('file_input').val(''); // Clear selection\n            return;\n        }\n\n        // Show spinner, dim image\n        this.$id('spinner').removeClass('d-none');\n        this.$id('photo').css('opacity', '0.3');\n\n        // Create FormData for file upload\n        const form_data = new FormData();\n        form_data.append('file', file);\n        form_data.append('site_id', '1'); // TODO: Get from session/config\n        // Do NOT set fileable_type/fileable_category - file uploads unattached\n        // The parent form will assign it via attach_to() on save\n\n        // Upload file via AJAX\n        $.ajax({\n            url: '/_upload',\n            type: 'POST',\n            data: form_data,\n            processData: false,\n            contentType: false,\n            success: (response) => {\n                console.log('Profile photo upload successful:', response);\n\n                // Update attachment key (this will also update thumbnail)\n                this.val(response.attachment.key);\n\n                // Hide spinner, restore opacity\n                this.$id('spinner').addClass('d-none');\n                this.$id('photo').css('opacity', '1');\n\n                // Clear file input for future uploads\n                this.$id('file_input').val('');\n\n                // Trigger change event for form tracking\n                this.$.trigger('change');\n            },\n            error: (xhr, status, error) => {\n                console.error('Profile photo upload failed:', error);\n                console.error('Response:', xhr.responseJSON);\n\n                // Hide spinner, restore opacity\n                this.$id('spinner').addClass('d-none');\n                this.$id('photo').css('opacity', '1');\n\n                // Clear file input\n                this.$id('file_input').val('');\n\n                // Show error to user\n                alert('Upload failed: ' + (xhr.responseJSON?.error || error));\n            },\n        });\n    }\n\n    update_photo() {\n        // <% if (this.args.show_remove && this.data.attachment_key) { %>\n    }\n\n    remove_photo() {\n        // Clear attachment key (sets to placeholder)\n        this.val('');\n\n        // Trigger change event for form tracking\n        this.$.trigger('change');\n    }\n\n    async seed() {\n        // For testing - set a placeholder key\n        // In production, this would use actual test data\n        this.val('');\n    }\n}\n"],"mappings":"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","ignoreList":[]}
|