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>
111 lines
15 KiB
JavaScript
Executable File
111 lines
15 KiB
JavaScript
Executable File
"use strict";
|
|
|
|
class Multi_Select extends Component {
|
|
on_ready() {
|
|
this.selected_values = this.args.value || [];
|
|
this.all_options = this.args.options || [];
|
|
if (this.all_options.length > 0) {
|
|
this.render_options(this.all_options);
|
|
}
|
|
|
|
// Search functionality
|
|
const $search = this.$sid('search');
|
|
$search.on('input', e => {
|
|
this.filter_options(e.target.value);
|
|
});
|
|
|
|
// Prevent dropdown close when clicking inside
|
|
this.$sid('menu').on('click', e => {
|
|
e.stopPropagation();
|
|
});
|
|
|
|
// Clear/Select all buttons
|
|
this.$sid('clear_all').on('click', () => {
|
|
this.clear_all();
|
|
});
|
|
this.$sid('select_all').on('click', () => {
|
|
this.select_all();
|
|
});
|
|
|
|
// Update display
|
|
this.update_display();
|
|
}
|
|
render_options(options) {
|
|
const $container = this.$sid('options');
|
|
$container.empty();
|
|
if (options.length === 0) {
|
|
$container.append($('<div>').addClass('px-3 py-2 text-muted').text('No options found'));
|
|
return;
|
|
}
|
|
options.forEach(option => {
|
|
const $label = $('<label>').addClass('dropdown-item mb-0');
|
|
const $checkbox = $('<input>').attr('type', 'checkbox').addClass('form-check-input me-2').attr('data-value', option.value).prop('checked', this.selected_values.includes(option.value));
|
|
$checkbox.on('change', e => {
|
|
this.toggle_option(option.value, e.target.checked);
|
|
});
|
|
$label.append($checkbox).append(option.label);
|
|
$container.append($label);
|
|
});
|
|
}
|
|
filter_options(query) {
|
|
const filtered = this.all_options.filter(opt => opt.label.toLowerCase().includes(query.toLowerCase()));
|
|
this.render_options(filtered);
|
|
}
|
|
toggle_option(value, checked) {
|
|
if (checked) {
|
|
if (!this.selected_values.includes(value)) {
|
|
this.selected_values.push(value);
|
|
}
|
|
} else {
|
|
this.selected_values = this.selected_values.filter(v => v !== value);
|
|
}
|
|
this.update_display();
|
|
if (this.args.on_change) {
|
|
this.args.on_change(this.selected_values);
|
|
}
|
|
}
|
|
update_display() {
|
|
const count = this.selected_values.length;
|
|
const $text = this.$sid('selected_text');
|
|
if (count === 0) {
|
|
$text.text(this.args.placeholder || 'Select options');
|
|
} else if (count === 1) {
|
|
const option = this.all_options.find(opt => opt.value === this.selected_values[0]);
|
|
$text.text(option ? option.label : '1 selected');
|
|
} else {
|
|
$text.text(`${count} selected`);
|
|
}
|
|
}
|
|
clear_all() {
|
|
this.selected_values = [];
|
|
this.$sid('options').find('input[type="checkbox"]').prop('checked', false);
|
|
this.update_display();
|
|
if (this.args.on_change) {
|
|
this.args.on_change(this.selected_values);
|
|
}
|
|
}
|
|
select_all() {
|
|
this.selected_values = this.all_options.map(opt => opt.value);
|
|
this.$sid('options').find('input[type="checkbox"]').prop('checked', true);
|
|
this.update_display();
|
|
if (this.args.on_change) {
|
|
this.args.on_change(this.selected_values);
|
|
}
|
|
}
|
|
get_value() {
|
|
return this.selected_values;
|
|
}
|
|
set_value(values) {
|
|
this.selected_values = values;
|
|
this.$sid('options').find('input[type="checkbox"]').each((i, el) => {
|
|
const $el = $(el);
|
|
$el.prop('checked', values.includes($el.attr('data-value')));
|
|
});
|
|
this.update_display();
|
|
}
|
|
set_options(options) {
|
|
this.all_options = options;
|
|
this.render_options(options);
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Multi_Select","Jqhtml_Component","on_ready","selected_values","args","value","all_options","options","length","render_options","$search","$id","on","e","filter_options","target","stopPropagation","clear_all","select_all","update_display","$container","empty","append","$","addClass","text","forEach","option","$label","$checkbox","attr","prop","includes","toggle_option","checked","label","query","filtered","filter","opt","toLowerCase","push","v","on_change","count","$text","placeholder","find","map","get_value","set_value","values","each","i","el","$el","set_options"],"sources":["rsx/theme/components/_archived/unfinished/Multi_Select.js"],"sourcesContent":["class Multi_Select extends Jqhtml_Component {\n    on_ready() {\n        this.selected_values = this.args.value || [];\n        this.all_options = this.args.options || [];\n\n        if (this.all_options.length > 0) {\n            this.render_options(this.all_options);\n        }\n\n        // Search functionality\n        const $search = this.$id('search');\n        $search.on('input', (e) => {\n            this.filter_options(e.target.value);\n        });\n\n        // Prevent dropdown close when clicking inside\n        this.$id('menu').on('click', (e) => {\n            e.stopPropagation();\n        });\n\n        // Clear/Select all buttons\n        this.$id('clear_all').on('click', () => {\n            this.clear_all();\n        });\n\n        this.$id('select_all').on('click', () => {\n            this.select_all();\n        });\n\n        // Update display\n        this.update_display();\n    }\n\n    render_options(options) {\n        const $container = this.$id('options');\n        $container.empty();\n\n        if (options.length === 0) {\n            $container.append($('<div>').addClass('px-3 py-2 text-muted').text('No options found'));\n            return;\n        }\n\n        options.forEach(option => {\n            const $label = $('<label>').addClass('dropdown-item mb-0');\n            const $checkbox = $('<input>')\n                .attr('type', 'checkbox')\n                .addClass('form-check-input me-2')\n                .attr('data-value', option.value)\n                .prop('checked', this.selected_values.includes(option.value));\n\n            $checkbox.on('change', (e) => {\n                this.toggle_option(option.value, e.target.checked);\n            });\n\n            $label.append($checkbox).append(option.label);\n            $container.append($label);\n        });\n    }\n\n    filter_options(query) {\n        const filtered = this.all_options.filter(opt =>\n            opt.label.toLowerCase().includes(query.toLowerCase())\n        );\n        this.render_options(filtered);\n    }\n\n    toggle_option(value, checked) {\n        if (checked) {\n            if (!this.selected_values.includes(value)) {\n                this.selected_values.push(value);\n            }\n        } else {\n            this.selected_values = this.selected_values.filter(v => v !== value);\n        }\n\n        this.update_display();\n\n        if (this.args.on_change) {\n            this.args.on_change(this.selected_values);\n        }\n    }\n\n    update_display() {\n        const count = this.selected_values.length;\n        const $text = this.$id('selected_text');\n\n        if (count === 0) {\n            $text.text(this.args.placeholder || 'Select options');\n        } else if (count === 1) {\n            const option = this.all_options.find(opt => opt.value === this.selected_values[0]);\n            $text.text(option ? option.label : '1 selected');\n        } else {\n            $text.text(`${count} selected`);\n        }\n    }\n\n    clear_all() {\n        this.selected_values = [];\n        this.$id('options').find('input[type=\"checkbox\"]').prop('checked', false);\n        this.update_display();\n\n        if (this.args.on_change) {\n            this.args.on_change(this.selected_values);\n        }\n    }\n\n    select_all() {\n        this.selected_values = this.all_options.map(opt => opt.value);\n        this.$id('options').find('input[type=\"checkbox\"]').prop('checked', true);\n        this.update_display();\n\n        if (this.args.on_change) {\n            this.args.on_change(this.selected_values);\n        }\n    }\n\n    get_value() {\n        return this.selected_values;\n    }\n\n    set_value(values) {\n        this.selected_values = values;\n        this.$id('options').find('input[type=\"checkbox\"]').each((i, el) => {\n            const $el = $(el);\n            $el.prop('checked', values.includes($el.attr('data-value')));\n        });\n        this.update_display();\n    }\n\n    set_options(options) {\n        this.all_options = options;\n        this.render_options(options);\n    }\n}\n"],"mappings":";;AAAA,MAAMA,YAAY,SAASC,gBAAgB,CAAC;EACxCC,QAAQA,CAAA,EAAG;IACP,IAAI,CAACC,eAAe,GAAG,IAAI,CAACC,IAAI,CAACC,KAAK,IAAI,EAAE;IAC5C,IAAI,CAACC,WAAW,GAAG,IAAI,CAACF,IAAI,CAACG,OAAO,IAAI,EAAE;IAE1C,IAAI,IAAI,CAACD,WAAW,CAACE,MAAM,GAAG,CAAC,EAAE;MAC7B,IAAI,CAACC,cAAc,CAAC,IAAI,CAACH,WAAW,CAAC;IACzC;;IAEA;IACA,MAAMI,OAAO,GAAG,IAAI,CAACC,GAAG,CAAC,QAAQ,CAAC;IAClCD,OAAO,CAACE,EAAE,CAAC,OAAO,EAAGC,CAAC,IAAK;MACvB,IAAI,CAACC,cAAc,CAACD,CAAC,CAACE,MAAM,CAACV,KAAK,CAAC;IACvC,CAAC,CAAC;;IAEF;IACA,IAAI,CAACM,GAAG,CAAC,MAAM,CAAC,CAACC,EAAE,CAAC,OAAO,EAAGC,CAAC,IAAK;MAChCA,CAAC,CAACG,eAAe,CAAC,CAAC;IACvB,CAAC,CAAC;;IAEF;IACA,IAAI,CAACL,GAAG,CAAC,WAAW,CAAC,CAACC,EAAE,CAAC,OAAO,EAAE,MAAM;MACpC,IAAI,CAACK,SAAS,CAAC,CAAC;IACpB,CAAC,CAAC;IAEF,IAAI,CAACN,GAAG,CAAC,YAAY,CAAC,CAACC,EAAE,CAAC,OAAO,EAAE,MAAM;MACrC,IAAI,CAACM,UAAU,CAAC,CAAC;IACrB,CAAC,CAAC;;IAEF;IACA,IAAI,CAACC,cAAc,CAAC,CAAC;EACzB;EAEAV,cAAcA,CAACF,OAAO,EAAE;IACpB,MAAMa,UAAU,GAAG,IAAI,CAACT,GAAG,CAAC,SAAS,CAAC;IACtCS,UAAU,CAACC,KAAK,CAAC,CAAC;IAElB,IAAId,OAAO,CAACC,MAAM,KAAK,CAAC,EAAE;MACtBY,UAAU,CAACE,MAAM,CAACC,CAAC,CAAC,OAAO,CAAC,CAACC,QAAQ,CAAC,sBAAsB,CAAC,CAACC,IAAI,CAAC,kBAAkB,CAAC,CAAC;MACvF;IACJ;IAEAlB,OAAO,CAACmB,OAAO,CAACC,MAAM,IAAI;MACtB,MAAMC,MAAM,GAAGL,CAAC,CAAC,SAAS,CAAC,CAACC,QAAQ,CAAC,oBAAoB,CAAC;MAC1D,MAAMK,SAAS,GAAGN,CAAC,CAAC,SAAS,CAAC,CACzBO,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,CACxBN,QAAQ,CAAC,uBAAuB,CAAC,CACjCM,IAAI,CAAC,YAAY,EAAEH,MAAM,CAACtB,KAAK,CAAC,CAChC0B,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC5B,eAAe,CAAC6B,QAAQ,CAACL,MAAM,CAACtB,KAAK,CAAC,CAAC;MAEjEwB,SAAS,CAACjB,EAAE,CAAC,QAAQ,EAAGC,CAAC,IAAK;QAC1B,IAAI,CAACoB,aAAa,CAACN,MAAM,CAACtB,KAAK,EAAEQ,CAAC,CAACE,MAAM,CAACmB,OAAO,CAAC;MACtD,CAAC,CAAC;MAEFN,MAAM,CAACN,MAAM,CAACO,SAAS,CAAC,CAACP,MAAM,CAACK,MAAM,CAACQ,KAAK,CAAC;MAC7Cf,UAAU,CAACE,MAAM,CAACM,MAAM,CAAC;IAC7B,CAAC,CAAC;EACN;EAEAd,cAAcA,CAACsB,KAAK,EAAE;IAClB,MAAMC,QAAQ,GAAG,IAAI,CAAC/B,WAAW,CAACgC,MAAM,CAACC,GAAG,IACxCA,GAAG,CAACJ,KAAK,CAACK,WAAW,CAAC,CAAC,CAACR,QAAQ,CAACI,KAAK,CAACI,WAAW,CAAC,CAAC,CACxD,CAAC;IACD,IAAI,CAAC/B,cAAc,CAAC4B,QAAQ,CAAC;EACjC;EAEAJ,aAAaA,CAAC5B,KAAK,EAAE6B,OAAO,EAAE;IAC1B,IAAIA,OAAO,EAAE;MACT,IAAI,CAAC,IAAI,CAAC/B,eAAe,CAAC6B,QAAQ,CAAC3B,KAAK,CAAC,EAAE;QACvC,IAAI,CAACF,eAAe,CAACsC,IAAI,CAACpC,KAAK,CAAC;MACpC;IACJ,CAAC,MAAM;MACH,IAAI,CAACF,eAAe,GAAG,IAAI,CAACA,eAAe,CAACmC,MAAM,CAACI,CAAC,IAAIA,CAAC,KAAKrC,KAAK,CAAC;IACxE;IAEA,IAAI,CAACc,cAAc,CAAC,CAAC;IAErB,IAAI,IAAI,CAACf,IAAI,CAACuC,SAAS,EAAE;MACrB,IAAI,CAACvC,IAAI,CAACuC,SAAS,CAAC,IAAI,CAACxC,eAAe,CAAC;IAC7C;EACJ;EAEAgB,cAAcA,CAAA,EAAG;IACb,MAAMyB,KAAK,GAAG,IAAI,CAACzC,eAAe,CAACK,MAAM;IACzC,MAAMqC,KAAK,GAAG,IAAI,CAAClC,GAAG,CAAC,eAAe,CAAC;IAEvC,IAAIiC,KAAK,KAAK,CAAC,EAAE;MACbC,KAAK,CAACpB,IAAI,CAAC,IAAI,CAACrB,IAAI,CAAC0C,WAAW,IAAI,gBAAgB,CAAC;IACzD,CAAC,MAAM,IAAIF,KAAK,KAAK,CAAC,EAAE;MACpB,MAAMjB,MAAM,GAAG,IAAI,CAACrB,WAAW,CAACyC,IAAI,CAACR,GAAG,IAAIA,GAAG,CAAClC,KAAK,KAAK,IAAI,CAACF,eAAe,CAAC,CAAC,CAAC,CAAC;MAClF0C,KAAK,CAACpB,IAAI,CAACE,MAAM,GAAGA,MAAM,CAACQ,KAAK,GAAG,YAAY,CAAC;IACpD,CAAC,MAAM;MACHU,KAAK,CAACpB,IAAI,CAAC,GAAGmB,KAAK,WAAW,CAAC;IACnC;EACJ;EAEA3B,SAASA,CAAA,EAAG;IACR,IAAI,CAACd,eAAe,GAAG,EAAE;IACzB,IAAI,CAACQ,GAAG,CAAC,SAAS,CAAC,CAACoC,IAAI,CAAC,wBAAwB,CAAC,CAAChB,IAAI,CAAC,SAAS,EAAE,KAAK,CAAC;IACzE,IAAI,CAACZ,cAAc,CAAC,CAAC;IAErB,IAAI,IAAI,CAACf,IAAI,CAACuC,SAAS,EAAE;MACrB,IAAI,CAACvC,IAAI,CAACuC,SAAS,CAAC,IAAI,CAACxC,eAAe,CAAC;IAC7C;EACJ;EAEAe,UAAUA,CAAA,EAAG;IACT,IAAI,CAACf,eAAe,GAAG,IAAI,CAACG,WAAW,CAAC0C,GAAG,CAACT,GAAG,IAAIA,GAAG,CAAClC,KAAK,CAAC;IAC7D,IAAI,CAACM,GAAG,CAAC,SAAS,CAAC,CAACoC,IAAI,CAAC,wBAAwB,CAAC,CAAChB,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC;IACxE,IAAI,CAACZ,cAAc,CAAC,CAAC;IAErB,IAAI,IAAI,CAACf,IAAI,CAACuC,SAAS,EAAE;MACrB,IAAI,CAACvC,IAAI,CAACuC,SAAS,CAAC,IAAI,CAACxC,eAAe,CAAC;IAC7C;EACJ;EAEA8C,SAASA,CAAA,EAAG;IACR,OAAO,IAAI,CAAC9C,eAAe;EAC/B;EAEA+C,SAASA,CAACC,MAAM,EAAE;IACd,IAAI,CAAChD,eAAe,GAAGgD,MAAM;IAC7B,IAAI,CAACxC,GAAG,CAAC,SAAS,CAAC,CAACoC,IAAI,CAAC,wBAAwB,CAAC,CAACK,IAAI,CAAC,CAACC,CAAC,EAAEC,EAAE,KAAK;MAC/D,MAAMC,GAAG,GAAGhC,CAAC,CAAC+B,EAAE,CAAC;MACjBC,GAAG,CAACxB,IAAI,CAAC,SAAS,EAAEoB,MAAM,CAACnB,QAAQ,CAACuB,GAAG,CAACzB,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;IAChE,CAAC,CAAC;IACF,IAAI,CAACX,cAAc,CAAC,CAAC;EACzB;EAEAqC,WAAWA,CAACjD,OAAO,EAAE;IACjB,IAAI,CAACD,WAAW,GAAGC,OAAO;IAC1B,IAAI,CAACE,cAAc,CAACF,OAAO,CAAC;EAChC;AACJ","ignoreList":[]}
|