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>
238 lines
31 KiB
JavaScript
Executable File
238 lines
31 KiB
JavaScript
Executable File
"use strict";
|
|
|
|
class Data_Table extends Component {
|
|
on_render() {
|
|
// Hide until data loads to prevent visual glitches
|
|
if (Object.keys(this.data).length === 0) {
|
|
this.$sid('footer').css('opacity', '0');
|
|
}
|
|
}
|
|
async on_load() {
|
|
// If data_source provided, fetch data
|
|
if (this.args.data_source) {
|
|
this.data = await this.fetch_data();
|
|
} else if (this.args.columns && this.args.rows) {
|
|
// Use provided static data
|
|
this.data = {
|
|
columns: this.args.columns,
|
|
rows: this.args.rows,
|
|
total: this.args.rows.length,
|
|
start: 1,
|
|
end: this.args.rows.length,
|
|
current_page: 1,
|
|
total_pages: 1
|
|
};
|
|
}
|
|
}
|
|
on_ready() {
|
|
// Show footer after render
|
|
this.$sid('footer').css('opacity', '1');
|
|
|
|
// Build column headers with sorting
|
|
if (this.data.columns) {
|
|
this.build_headers(this.data.columns);
|
|
}
|
|
|
|
// Setup search if enabled
|
|
if (this.args.searchable) {
|
|
this.setup_search();
|
|
}
|
|
|
|
// Setup column visibility toggle if enabled
|
|
if (this.args.column_toggle) {
|
|
this.setup_column_toggle();
|
|
}
|
|
|
|
// Setup bulk actions
|
|
if (this.args.bulk_actions) {
|
|
this.setup_bulk_actions();
|
|
}
|
|
|
|
// Attach row checkbox listeners
|
|
this.$.find('.row-checkbox').on('change', () => {
|
|
this.update_bulk_selection();
|
|
});
|
|
|
|
// Setup pagination click handlers
|
|
const $pagination = this.$sid('pagination');
|
|
$pagination.$.find('.page-link').on('click', e => {
|
|
e.preventDefault();
|
|
const page_text = $(e.target).text();
|
|
if (page_text === 'Previous') {
|
|
this.load_page(this.data.current_page - 1);
|
|
} else if (page_text === 'Next') {
|
|
this.load_page(this.data.current_page + 1);
|
|
} else {
|
|
const page = int(page_text);
|
|
if (!isNaN(page)) {
|
|
this.load_page(page);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
build_headers(columns) {
|
|
const $header_row = this.$sid('header_row');
|
|
|
|
// Skip first cell if bulk actions (already has Bulk_Selection)
|
|
const offset = this.args.bulk_actions ? 1 : 0;
|
|
columns.forEach((col, index) => {
|
|
const $th = $('<th>');
|
|
if (col.sortable !== false) {
|
|
// Create sortable column header
|
|
const $sortable = $('<div>').addClass('Sortable_Column_Header').attr({
|
|
'data-column': col.field,
|
|
'data-sort': 'none'
|
|
});
|
|
const $text = $('<span>').text(col.label || col.field);
|
|
const $icon = $('<span>').attr('data-id', 'sort_icon').html('<i class="text-muted">⇅</i>');
|
|
$sortable.append($text).append(' ').append($icon);
|
|
$sortable.css('cursor', 'pointer');
|
|
$sortable.on('click', () => {
|
|
this.handle_sort(col.field);
|
|
});
|
|
$th.append($sortable);
|
|
} else {
|
|
$th.text(col.label || col.field);
|
|
}
|
|
if (col.width) {
|
|
$th.css('width', col.width);
|
|
}
|
|
$header_row.append($th);
|
|
});
|
|
|
|
// Add actions column header if row_actions enabled
|
|
if (this.args.row_actions) {
|
|
const $th = $('<th>').text('Actions').css('width', '100px');
|
|
$header_row.append($th);
|
|
}
|
|
}
|
|
setup_search() {
|
|
const $container = this.$sid('search_container');
|
|
const $search = $('<input>').attr({
|
|
type: 'search',
|
|
placeholder: 'Search...',
|
|
class: 'form-control form-control-sm'
|
|
}).css('width', '200px');
|
|
$container.append($search);
|
|
let timeout;
|
|
$search.on('input', e => {
|
|
clearTimeout(timeout);
|
|
timeout = setTimeout(() => {
|
|
this.search_query = e.target.value;
|
|
this.reload_data();
|
|
}, 300);
|
|
});
|
|
}
|
|
setup_column_toggle() {
|
|
const $container = this.$sid('column_toggle_container');
|
|
const $toggle = $('<div>').addClass('Column_Visibility_Toggle');
|
|
$container.append($toggle);
|
|
|
|
// Initialize component manually
|
|
const toggle_component = $toggle.component();
|
|
if (toggle_component) {
|
|
toggle_component.args.columns = this.data.columns;
|
|
toggle_component.args.table = this.$sid('table').$;
|
|
toggle_component.build_menu(this.data.columns);
|
|
}
|
|
}
|
|
setup_bulk_actions() {
|
|
const $bulk_selection = this.$sid('bulk_selection');
|
|
$bulk_selection.$.find('input[type="checkbox"]').on('change', e => {
|
|
const checked = e.target.checked;
|
|
this.$.find('.row-checkbox').prop('checked', checked);
|
|
this.update_bulk_selection();
|
|
});
|
|
}
|
|
update_bulk_selection() {
|
|
const checked = this.$.find('.row-checkbox:checked').length;
|
|
const $bulk_bar = this.$sid('bulk_bar');
|
|
if (checked > 0) {
|
|
$bulk_bar.$.show();
|
|
$bulk_bar.set_count(checked);
|
|
} else {
|
|
$bulk_bar.$.hide();
|
|
}
|
|
}
|
|
async handle_sort(field) {
|
|
// Toggle sort direction
|
|
const current = this.sort_field === field ? this.sort_direction : 'none';
|
|
this.sort_direction = current === 'none' ? 'asc' : current === 'asc' ? 'desc' : 'asc';
|
|
this.sort_field = field;
|
|
|
|
// Update sort icon
|
|
this.$sid('header_row').find('[data-column]').each(function () {
|
|
const $sortable = $(this);
|
|
const col = $sortable.attr('data-column');
|
|
const $icon = $sortable.find('[data-id="sort_icon"]');
|
|
if (col === field) {
|
|
$sortable.attr('data-sort', this.sort_direction);
|
|
if (this.sort_direction === 'asc') {
|
|
$icon.html('<i class="text-primary">↑</i>');
|
|
} else if (this.sort_direction === 'desc') {
|
|
$icon.html('<i class="text-primary">↓</i>');
|
|
} else {
|
|
$icon.html('<i class="text-muted">⇅</i>');
|
|
}
|
|
} else {
|
|
$sortable.attr('data-sort', 'none');
|
|
$icon.html('<i class="text-muted">⇅</i>');
|
|
}
|
|
}.bind(this));
|
|
await this.reload_data();
|
|
}
|
|
async load_page(page) {
|
|
if (page < 1 || page > this.data.total_pages) return;
|
|
this.current_page = page;
|
|
await this.reload_data();
|
|
}
|
|
async fetch_data() {
|
|
const params = {
|
|
page: this.current_page || 1,
|
|
per_page: this.args.per_page || 20,
|
|
sort_field: this.sort_field,
|
|
sort_direction: this.sort_direction,
|
|
search: this.search_query
|
|
};
|
|
|
|
// Call data source (can be URL or function)
|
|
if (typeof this.args.data_source === 'function') {
|
|
return await this.args.data_source(params);
|
|
} else {
|
|
const url = new URL(this.args.data_source, window.location.origin);
|
|
Object.keys(params).forEach(key => {
|
|
if (params[key]) url.searchParams.append(key, params[key]);
|
|
});
|
|
const response = await fetch(url);
|
|
return await response.json();
|
|
}
|
|
}
|
|
get_selected_ids() {
|
|
const ids = [];
|
|
this.$.find('.row-checkbox:checked').each(function () {
|
|
ids.push($(this).val());
|
|
});
|
|
return ids;
|
|
}
|
|
async reload_data() {
|
|
// Show loading state
|
|
const $tbody = this.$sid('tbody');
|
|
$tbody.html(`
|
|
<tr>
|
|
<td colspan="100" class="text-center py-5">
|
|
<div class="spinner-border spinner-border-sm" role="status">
|
|
<span class="visually-hidden">Loading...</span>
|
|
</div>
|
|
<div class="mt-2 text-muted">Loading data...</div>
|
|
</td>
|
|
</tr>
|
|
`);
|
|
|
|
// Fetch new data
|
|
this.data = await this.fetch_data();
|
|
|
|
// Re-render entire component
|
|
this.render();
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Data_Table","Jqhtml_Component","on_render","Object","keys","data","length","$id","css","on_load","args","data_source","fetch_data","columns","rows","total","start","end","current_page","total_pages","on_ready","build_headers","searchable","setup_search","column_toggle","setup_column_toggle","bulk_actions","setup_bulk_actions","$","find","on","update_bulk_selection","$pagination","e","preventDefault","page_text","target","text","load_page","page","int","isNaN","$header_row","offset","forEach","col","index","$th","sortable","$sortable","addClass","attr","field","$text","label","$icon","html","append","handle_sort","width","row_actions","$container","$search","type","placeholder","class","timeout","clearTimeout","setTimeout","search_query","value","reload_data","$toggle","toggle_component","component","table","build_menu","$bulk_selection","checked","prop","$bulk_bar","show","set_count","hide","current","sort_field","sort_direction","each","bind","params","per_page","search","url","URL","window","location","origin","key","searchParams","response","fetch","json","get_selected_ids","ids","push","val","$tbody","render"],"sources":["rsx/app/frontend/contacts/_data_table_qq/Data_Table.js"],"sourcesContent":["class Data_Table extends Jqhtml_Component {\n    on_render() {\n        // Hide until data loads to prevent visual glitches\n        if (Object.keys(this.data).length === 0) {\n            this.$id('footer').css('opacity', '0');\n        }\n    }\n\n    async on_load() {\n        // If data_source provided, fetch data\n        if (this.args.data_source) {\n            this.data = await this.fetch_data();\n        } else if (this.args.columns && this.args.rows) {\n            // Use provided static data\n            this.data = {\n                columns: this.args.columns,\n                rows: this.args.rows,\n                total: this.args.rows.length,\n                start: 1,\n                end: this.args.rows.length,\n                current_page: 1,\n                total_pages: 1,\n            };\n        }\n    }\n\n    on_ready() {\n        // Show footer after render\n        this.$id('footer').css('opacity', '1');\n\n        // Build column headers with sorting\n        if (this.data.columns) {\n            this.build_headers(this.data.columns);\n        }\n\n        // Setup search if enabled\n        if (this.args.searchable) {\n            this.setup_search();\n        }\n\n        // Setup column visibility toggle if enabled\n        if (this.args.column_toggle) {\n            this.setup_column_toggle();\n        }\n\n        // Setup bulk actions\n        if (this.args.bulk_actions) {\n            this.setup_bulk_actions();\n        }\n\n        // Attach row checkbox listeners\n        this.$.find('.row-checkbox').on('change', () => {\n            this.update_bulk_selection();\n        });\n\n        // Setup pagination click handlers\n        const $pagination = this.$id('pagination');\n        $pagination.$.find('.page-link').on('click', (e) => {\n            e.preventDefault();\n            const page_text = $(e.target).text();\n            if (page_text === 'Previous') {\n                this.load_page(this.data.current_page - 1);\n            } else if (page_text === 'Next') {\n                this.load_page(this.data.current_page + 1);\n            } else {\n                const page = int(page_text);\n                if (!isNaN(page)) {\n                    this.load_page(page);\n                }\n            }\n        });\n    }\n\n    build_headers(columns) {\n        const $header_row = this.$id('header_row');\n\n        // Skip first cell if bulk actions (already has Bulk_Selection)\n        const offset = this.args.bulk_actions ? 1 : 0;\n\n        columns.forEach((col, index) => {\n            const $th = $('<th>');\n\n            if (col.sortable !== false) {\n                // Create sortable column header\n                const $sortable = $('<div>').addClass('Sortable_Column_Header').attr({\n                    'data-column': col.field,\n                    'data-sort': 'none',\n                });\n\n                const $text = $('<span>').text(col.label || col.field);\n                const $icon = $('<span>').attr('data-id', 'sort_icon').html('<i class=\"text-muted\">⇅</i>');\n\n                $sortable.append($text).append(' ').append($icon);\n                $sortable.css('cursor', 'pointer');\n\n                $sortable.on('click', () => {\n                    this.handle_sort(col.field);\n                });\n\n                $th.append($sortable);\n            } else {\n                $th.text(col.label || col.field);\n            }\n\n            if (col.width) {\n                $th.css('width', col.width);\n            }\n\n            $header_row.append($th);\n        });\n\n        // Add actions column header if row_actions enabled\n        if (this.args.row_actions) {\n            const $th = $('<th>').text('Actions').css('width', '100px');\n            $header_row.append($th);\n        }\n    }\n\n    setup_search() {\n        const $container = this.$id('search_container');\n        const $search = $('<input>')\n            .attr({\n                type: 'search',\n                placeholder: 'Search...',\n                class: 'form-control form-control-sm',\n            })\n            .css('width', '200px');\n\n        $container.append($search);\n\n        let timeout;\n        $search.on('input', (e) => {\n            clearTimeout(timeout);\n            timeout = setTimeout(() => {\n                this.search_query = e.target.value;\n                this.reload_data();\n            }, 300);\n        });\n    }\n\n    setup_column_toggle() {\n        const $container = this.$id('column_toggle_container');\n        const $toggle = $('<div>').addClass('Column_Visibility_Toggle');\n        $container.append($toggle);\n\n        // Initialize component manually\n        const toggle_component = $toggle.component();\n        if (toggle_component) {\n            toggle_component.args.columns = this.data.columns;\n            toggle_component.args.table = this.$id('table').$;\n            toggle_component.build_menu(this.data.columns);\n        }\n    }\n\n    setup_bulk_actions() {\n        const $bulk_selection = this.$id('bulk_selection');\n        $bulk_selection.$.find('input[type=\"checkbox\"]').on('change', (e) => {\n            const checked = e.target.checked;\n            this.$.find('.row-checkbox').prop('checked', checked);\n            this.update_bulk_selection();\n        });\n    }\n\n    update_bulk_selection() {\n        const checked = this.$.find('.row-checkbox:checked').length;\n        const $bulk_bar = this.$id('bulk_bar');\n\n        if (checked > 0) {\n            $bulk_bar.$.show();\n            $bulk_bar.set_count(checked);\n        } else {\n            $bulk_bar.$.hide();\n        }\n    }\n\n    async handle_sort(field) {\n        // Toggle sort direction\n        const current = this.sort_field === field ? this.sort_direction : 'none';\n        this.sort_direction = current === 'none' ? 'asc' : current === 'asc' ? 'desc' : 'asc';\n        this.sort_field = field;\n\n        // Update sort icon\n        this.$id('header_row')\n            .find('[data-column]')\n            .each(\n                function () {\n                    const $sortable = $(this);\n                    const col = $sortable.attr('data-column');\n                    const $icon = $sortable.find('[data-id=\"sort_icon\"]');\n\n                    if (col === field) {\n                        $sortable.attr('data-sort', this.sort_direction);\n                        if (this.sort_direction === 'asc') {\n                            $icon.html('<i class=\"text-primary\">↑</i>');\n                        } else if (this.sort_direction === 'desc') {\n                            $icon.html('<i class=\"text-primary\">↓</i>');\n                        } else {\n                            $icon.html('<i class=\"text-muted\">⇅</i>');\n                        }\n                    } else {\n                        $sortable.attr('data-sort', 'none');\n                        $icon.html('<i class=\"text-muted\">⇅</i>');\n                    }\n                }.bind(this)\n            );\n\n        await this.reload_data();\n    }\n\n    async load_page(page) {\n        if (page < 1 || page > this.data.total_pages) return;\n        this.current_page = page;\n        await this.reload_data();\n    }\n\n    async fetch_data() {\n        const params = {\n            page: this.current_page || 1,\n            per_page: this.args.per_page || 20,\n            sort_field: this.sort_field,\n            sort_direction: this.sort_direction,\n            search: this.search_query,\n        };\n\n        // Call data source (can be URL or function)\n        if (typeof this.args.data_source === 'function') {\n            return await this.args.data_source(params);\n        } else {\n            const url = new URL(this.args.data_source, window.location.origin);\n            Object.keys(params).forEach((key) => {\n                if (params[key]) url.searchParams.append(key, params[key]);\n            });\n\n            const response = await fetch(url);\n            return await response.json();\n        }\n    }\n\n    get_selected_ids() {\n        const ids = [];\n        this.$.find('.row-checkbox:checked').each(function () {\n            ids.push($(this).val());\n        });\n        return ids;\n    }\n\n    async reload_data() {\n        // Show loading state\n        const $tbody = this.$id('tbody');\n        $tbody.html(`\n            <tr>\n                <td colspan=\"100\" class=\"text-center py-5\">\n                    <div class=\"spinner-border spinner-border-sm\" role=\"status\">\n                        <span class=\"visually-hidden\">Loading...</span>\n                    </div>\n                    <div class=\"mt-2 text-muted\">Loading data...</div>\n                </td>\n            </tr>\n        `);\n\n        // Fetch new data\n        this.data = await this.fetch_data();\n\n        // Re-render entire component\n        this.render();\n    }\n}\n"],"mappings":";;AAAA,MAAMA,UAAU,SAASC,gBAAgB,CAAC;EACtCC,SAASA,CAAA,EAAG;IACR;IACA,IAAIC,MAAM,CAACC,IAAI,CAAC,IAAI,CAACC,IAAI,CAAC,CAACC,MAAM,KAAK,CAAC,EAAE;MACrC,IAAI,CAACC,GAAG,CAAC,QAAQ,CAAC,CAACC,GAAG,CAAC,SAAS,EAAE,GAAG,CAAC;IAC1C;EACJ;EAEA,MAAMC,OAAOA,CAAA,EAAG;IACZ;IACA,IAAI,IAAI,CAACC,IAAI,CAACC,WAAW,EAAE;MACvB,IAAI,CAACN,IAAI,GAAG,MAAM,IAAI,CAACO,UAAU,CAAC,CAAC;IACvC,CAAC,MAAM,IAAI,IAAI,CAACF,IAAI,CAACG,OAAO,IAAI,IAAI,CAACH,IAAI,CAACI,IAAI,EAAE;MAC5C;MACA,IAAI,CAACT,IAAI,GAAG;QACRQ,OAAO,EAAE,IAAI,CAACH,IAAI,CAACG,OAAO;QAC1BC,IAAI,EAAE,IAAI,CAACJ,IAAI,CAACI,IAAI;QACpBC,KAAK,EAAE,IAAI,CAACL,IAAI,CAACI,IAAI,CAACR,MAAM;QAC5BU,KAAK,EAAE,CAAC;QACRC,GAAG,EAAE,IAAI,CAACP,IAAI,CAACI,IAAI,CAACR,MAAM;QAC1BY,YAAY,EAAE,CAAC;QACfC,WAAW,EAAE;MACjB,CAAC;IACL;EACJ;EAEAC,QAAQA,CAAA,EAAG;IACP;IACA,IAAI,CAACb,GAAG,CAAC,QAAQ,CAAC,CAACC,GAAG,CAAC,SAAS,EAAE,GAAG,CAAC;;IAEtC;IACA,IAAI,IAAI,CAACH,IAAI,CAACQ,OAAO,EAAE;MACnB,IAAI,CAACQ,aAAa,CAAC,IAAI,CAAChB,IAAI,CAACQ,OAAO,CAAC;IACzC;;IAEA;IACA,IAAI,IAAI,CAACH,IAAI,CAACY,UAAU,EAAE;MACtB,IAAI,CAACC,YAAY,CAAC,CAAC;IACvB;;IAEA;IACA,IAAI,IAAI,CAACb,IAAI,CAACc,aAAa,EAAE;MACzB,IAAI,CAACC,mBAAmB,CAAC,CAAC;IAC9B;;IAEA;IACA,IAAI,IAAI,CAACf,IAAI,CAACgB,YAAY,EAAE;MACxB,IAAI,CAACC,kBAAkB,CAAC,CAAC;IAC7B;;IAEA;IACA,IAAI,CAACC,CAAC,CAACC,IAAI,CAAC,eAAe,CAAC,CAACC,EAAE,CAAC,QAAQ,EAAE,MAAM;MAC5C,IAAI,CAACC,qBAAqB,CAAC,CAAC;IAChC,CAAC,CAAC;;IAEF;IACA,MAAMC,WAAW,GAAG,IAAI,CAACzB,GAAG,CAAC,YAAY,CAAC;IAC1CyB,WAAW,CAACJ,CAAC,CAACC,IAAI,CAAC,YAAY,CAAC,CAACC,EAAE,CAAC,OAAO,EAAGG,CAAC,IAAK;MAChDA,CAAC,CAACC,cAAc,CAAC,CAAC;MAClB,MAAMC,SAAS,GAAGP,CAAC,CAACK,CAAC,CAACG,MAAM,CAAC,CAACC,IAAI,CAAC,CAAC;MACpC,IAAIF,SAAS,KAAK,UAAU,EAAE;QAC1B,IAAI,CAACG,SAAS,CAAC,IAAI,CAACjC,IAAI,CAACa,YAAY,GAAG,CAAC,CAAC;MAC9C,CAAC,MAAM,IAAIiB,SAAS,KAAK,MAAM,EAAE;QAC7B,IAAI,CAACG,SAAS,CAAC,IAAI,CAACjC,IAAI,CAACa,YAAY,GAAG,CAAC,CAAC;MAC9C,CAAC,MAAM;QACH,MAAMqB,IAAI,GAAGC,GAAG,CAACL,SAAS,CAAC;QAC3B,IAAI,CAACM,KAAK,CAACF,IAAI,CAAC,EAAE;UACd,IAAI,CAACD,SAAS,CAACC,IAAI,CAAC;QACxB;MACJ;IACJ,CAAC,CAAC;EACN;EAEAlB,aAAaA,CAACR,OAAO,EAAE;IACnB,MAAM6B,WAAW,GAAG,IAAI,CAACnC,GAAG,CAAC,YAAY,CAAC;;IAE1C;IACA,MAAMoC,MAAM,GAAG,IAAI,CAACjC,IAAI,CAACgB,YAAY,GAAG,CAAC,GAAG,CAAC;IAE7Cb,OAAO,CAAC+B,OAAO,CAAC,CAACC,GAAG,EAAEC,KAAK,KAAK;MAC5B,MAAMC,GAAG,GAAGnB,CAAC,CAAC,MAAM,CAAC;MAErB,IAAIiB,GAAG,CAACG,QAAQ,KAAK,KAAK,EAAE;QACxB;QACA,MAAMC,SAAS,GAAGrB,CAAC,CAAC,OAAO,CAAC,CAACsB,QAAQ,CAAC,wBAAwB,CAAC,CAACC,IAAI,CAAC;UACjE,aAAa,EAAEN,GAAG,CAACO,KAAK;UACxB,WAAW,EAAE;QACjB,CAAC,CAAC;QAEF,MAAMC,KAAK,GAAGzB,CAAC,CAAC,QAAQ,CAAC,CAACS,IAAI,CAACQ,GAAG,CAACS,KAAK,IAAIT,GAAG,CAACO,KAAK,CAAC;QACtD,MAAMG,KAAK,GAAG3B,CAAC,CAAC,QAAQ,CAAC,CAACuB,IAAI,CAAC,SAAS,EAAE,WAAW,CAAC,CAACK,IAAI,CAAC,6BAA6B,CAAC;QAE1FP,SAAS,CAACQ,MAAM,CAACJ,KAAK,CAAC,CAACI,MAAM,CAAC,GAAG,CAAC,CAACA,MAAM,CAACF,KAAK,CAAC;QACjDN,SAAS,CAACzC,GAAG,CAAC,QAAQ,EAAE,SAAS,CAAC;QAElCyC,SAAS,CAACnB,EAAE,CAAC,OAAO,EAAE,MAAM;UACxB,IAAI,CAAC4B,WAAW,CAACb,GAAG,CAACO,KAAK,CAAC;QAC/B,CAAC,CAAC;QAEFL,GAAG,CAACU,MAAM,CAACR,SAAS,CAAC;MACzB,CAAC,MAAM;QACHF,GAAG,CAACV,IAAI,CAACQ,GAAG,CAACS,KAAK,IAAIT,GAAG,CAACO,KAAK,CAAC;MACpC;MAEA,IAAIP,GAAG,CAACc,KAAK,EAAE;QACXZ,GAAG,CAACvC,GAAG,CAAC,OAAO,EAAEqC,GAAG,CAACc,KAAK,CAAC;MAC/B;MAEAjB,WAAW,CAACe,MAAM,CAACV,GAAG,CAAC;IAC3B,CAAC,CAAC;;IAEF;IACA,IAAI,IAAI,CAACrC,IAAI,CAACkD,WAAW,EAAE;MACvB,MAAMb,GAAG,GAAGnB,CAAC,CAAC,MAAM,CAAC,CAACS,IAAI,CAAC,SAAS,CAAC,CAAC7B,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC;MAC3DkC,WAAW,CAACe,MAAM,CAACV,GAAG,CAAC;IAC3B;EACJ;EAEAxB,YAAYA,CAAA,EAAG;IACX,MAAMsC,UAAU,GAAG,IAAI,CAACtD,GAAG,CAAC,kBAAkB,CAAC;IAC/C,MAAMuD,OAAO,GAAGlC,CAAC,CAAC,SAAS,CAAC,CACvBuB,IAAI,CAAC;MACFY,IAAI,EAAE,QAAQ;MACdC,WAAW,EAAE,WAAW;MACxBC,KAAK,EAAE;IACX,CAAC,CAAC,CACDzD,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC;IAE1BqD,UAAU,CAACJ,MAAM,CAACK,OAAO,CAAC;IAE1B,IAAII,OAAO;IACXJ,OAAO,CAAChC,EAAE,CAAC,OAAO,EAAGG,CAAC,IAAK;MACvBkC,YAAY,CAACD,OAAO,CAAC;MACrBA,OAAO,GAAGE,UAAU,CAAC,MAAM;QACvB,IAAI,CAACC,YAAY,GAAGpC,CAAC,CAACG,MAAM,CAACkC,KAAK;QAClC,IAAI,CAACC,WAAW,CAAC,CAAC;MACtB,CAAC,EAAE,GAAG,CAAC;IACX,CAAC,CAAC;EACN;EAEA9C,mBAAmBA,CAAA,EAAG;IAClB,MAAMoC,UAAU,GAAG,IAAI,CAACtD,GAAG,CAAC,yBAAyB,CAAC;IACtD,MAAMiE,OAAO,GAAG5C,CAAC,CAAC,OAAO,CAAC,CAACsB,QAAQ,CAAC,0BAA0B,CAAC;IAC/DW,UAAU,CAACJ,MAAM,CAACe,OAAO,CAAC;;IAE1B;IACA,MAAMC,gBAAgB,GAAGD,OAAO,CAACE,SAAS,CAAC,CAAC;IAC5C,IAAID,gBAAgB,EAAE;MAClBA,gBAAgB,CAAC/D,IAAI,CAACG,OAAO,GAAG,IAAI,CAACR,IAAI,CAACQ,OAAO;MACjD4D,gBAAgB,CAAC/D,IAAI,CAACiE,KAAK,GAAG,IAAI,CAACpE,GAAG,CAAC,OAAO,CAAC,CAACqB,CAAC;MACjD6C,gBAAgB,CAACG,UAAU,CAAC,IAAI,CAACvE,IAAI,CAACQ,OAAO,CAAC;IAClD;EACJ;EAEAc,kBAAkBA,CAAA,EAAG;IACjB,MAAMkD,eAAe,GAAG,IAAI,CAACtE,GAAG,CAAC,gBAAgB,CAAC;IAClDsE,eAAe,CAACjD,CAAC,CAACC,IAAI,CAAC,wBAAwB,CAAC,CAACC,EAAE,CAAC,QAAQ,EAAGG,CAAC,IAAK;MACjE,MAAM6C,OAAO,GAAG7C,CAAC,CAACG,MAAM,CAAC0C,OAAO;MAChC,IAAI,CAAClD,CAAC,CAACC,IAAI,CAAC,eAAe,CAAC,CAACkD,IAAI,CAAC,SAAS,EAAED,OAAO,CAAC;MACrD,IAAI,CAAC/C,qBAAqB,CAAC,CAAC;IAChC,CAAC,CAAC;EACN;EAEAA,qBAAqBA,CAAA,EAAG;IACpB,MAAM+C,OAAO,GAAG,IAAI,CAAClD,CAAC,CAACC,IAAI,CAAC,uBAAuB,CAAC,CAACvB,MAAM;IAC3D,MAAM0E,SAAS,GAAG,IAAI,CAACzE,GAAG,CAAC,UAAU,CAAC;IAEtC,IAAIuE,OAAO,GAAG,CAAC,EAAE;MACbE,SAAS,CAACpD,CAAC,CAACqD,IAAI,CAAC,CAAC;MAClBD,SAAS,CAACE,SAAS,CAACJ,OAAO,CAAC;IAChC,CAAC,MAAM;MACHE,SAAS,CAACpD,CAAC,CAACuD,IAAI,CAAC,CAAC;IACtB;EACJ;EAEA,MAAMzB,WAAWA,CAACN,KAAK,EAAE;IACrB;IACA,MAAMgC,OAAO,GAAG,IAAI,CAACC,UAAU,KAAKjC,KAAK,GAAG,IAAI,CAACkC,cAAc,GAAG,MAAM;IACxE,IAAI,CAACA,cAAc,GAAGF,OAAO,KAAK,MAAM,GAAG,KAAK,GAAGA,OAAO,KAAK,KAAK,GAAG,MAAM,GAAG,KAAK;IACrF,IAAI,CAACC,UAAU,GAAGjC,KAAK;;IAEvB;IACA,IAAI,CAAC7C,GAAG,CAAC,YAAY,CAAC,CACjBsB,IAAI,CAAC,eAAe,CAAC,CACrB0D,IAAI,CACD,YAAY;MACR,MAAMtC,SAAS,GAAGrB,CAAC,CAAC,IAAI,CAAC;MACzB,MAAMiB,GAAG,GAAGI,SAAS,CAACE,IAAI,CAAC,aAAa,CAAC;MACzC,MAAMI,KAAK,GAAGN,SAAS,CAACpB,IAAI,CAAC,uBAAuB,CAAC;MAErD,IAAIgB,GAAG,KAAKO,KAAK,EAAE;QACfH,SAAS,CAACE,IAAI,CAAC,WAAW,EAAE,IAAI,CAACmC,cAAc,CAAC;QAChD,IAAI,IAAI,CAACA,cAAc,KAAK,KAAK,EAAE;UAC/B/B,KAAK,CAACC,IAAI,CAAC,+BAA+B,CAAC;QAC/C,CAAC,MAAM,IAAI,IAAI,CAAC8B,cAAc,KAAK,MAAM,EAAE;UACvC/B,KAAK,CAACC,IAAI,CAAC,+BAA+B,CAAC;QAC/C,CAAC,MAAM;UACHD,KAAK,CAACC,IAAI,CAAC,6BAA6B,CAAC;QAC7C;MACJ,CAAC,MAAM;QACHP,SAAS,CAACE,IAAI,CAAC,WAAW,EAAE,MAAM,CAAC;QACnCI,KAAK,CAACC,IAAI,CAAC,6BAA6B,CAAC;MAC7C;IACJ,CAAC,CAACgC,IAAI,CAAC,IAAI,CACf,CAAC;IAEL,MAAM,IAAI,CAACjB,WAAW,CAAC,CAAC;EAC5B;EAEA,MAAMjC,SAASA,CAACC,IAAI,EAAE;IAClB,IAAIA,IAAI,GAAG,CAAC,IAAIA,IAAI,GAAG,IAAI,CAAClC,IAAI,CAACc,WAAW,EAAE;IAC9C,IAAI,CAACD,YAAY,GAAGqB,IAAI;IACxB,MAAM,IAAI,CAACgC,WAAW,CAAC,CAAC;EAC5B;EAEA,MAAM3D,UAAUA,CAAA,EAAG;IACf,MAAM6E,MAAM,GAAG;MACXlD,IAAI,EAAE,IAAI,CAACrB,YAAY,IAAI,CAAC;MAC5BwE,QAAQ,EAAE,IAAI,CAAChF,IAAI,CAACgF,QAAQ,IAAI,EAAE;MAClCL,UAAU,EAAE,IAAI,CAACA,UAAU;MAC3BC,cAAc,EAAE,IAAI,CAACA,cAAc;MACnCK,MAAM,EAAE,IAAI,CAACtB;IACjB,CAAC;;IAED;IACA,IAAI,OAAO,IAAI,CAAC3D,IAAI,CAACC,WAAW,KAAK,UAAU,EAAE;MAC7C,OAAO,MAAM,IAAI,CAACD,IAAI,CAACC,WAAW,CAAC8E,MAAM,CAAC;IAC9C,CAAC,MAAM;MACH,MAAMG,GAAG,GAAG,IAAIC,GAAG,CAAC,IAAI,CAACnF,IAAI,CAACC,WAAW,EAAEmF,MAAM,CAACC,QAAQ,CAACC,MAAM,CAAC;MAClE7F,MAAM,CAACC,IAAI,CAACqF,MAAM,CAAC,CAAC7C,OAAO,CAAEqD,GAAG,IAAK;QACjC,IAAIR,MAAM,CAACQ,GAAG,CAAC,EAAEL,GAAG,CAACM,YAAY,CAACzC,MAAM,CAACwC,GAAG,EAAER,MAAM,CAACQ,GAAG,CAAC,CAAC;MAC9D,CAAC,CAAC;MAEF,MAAME,QAAQ,GAAG,MAAMC,KAAK,CAACR,GAAG,CAAC;MACjC,OAAO,MAAMO,QAAQ,CAACE,IAAI,CAAC,CAAC;IAChC;EACJ;EAEAC,gBAAgBA,CAAA,EAAG;IACf,MAAMC,GAAG,GAAG,EAAE;IACd,IAAI,CAAC3E,CAAC,CAACC,IAAI,CAAC,uBAAuB,CAAC,CAAC0D,IAAI,CAAC,YAAY;MAClDgB,GAAG,CAACC,IAAI,CAAC5E,CAAC,CAAC,IAAI,CAAC,CAAC6E,GAAG,CAAC,CAAC,CAAC;IAC3B,CAAC,CAAC;IACF,OAAOF,GAAG;EACd;EAEA,MAAMhC,WAAWA,CAAA,EAAG;IAChB;IACA,MAAMmC,MAAM,GAAG,IAAI,CAACnG,GAAG,CAAC,OAAO,CAAC;IAChCmG,MAAM,CAAClD,IAAI,CAAC;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS,CAAC;;IAEF;IACA,IAAI,CAACnD,IAAI,GAAG,MAAM,IAAI,CAACO,UAAU,CAAC,CAAC;;IAEnC;IACA,IAAI,CAAC+F,MAAM,CAAC,CAAC;EACjB;AACJ","ignoreList":[]}
|