"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":[]}