Standardize settings file naming and relocate documentation files Fix code quality violations from rsx:check Reorganize user_management directory into logical subdirectories Move Quill Bundle to core and align with Tom Select pattern Simplify Site Settings page to focus on core site information Complete Phase 5: Multi-tenant authentication with login flow and site selection Add route query parameter rule and synchronize filename validation logic Fix critical bug in UpdateNpmCommand causing missing JavaScript stubs Implement filename convention rule and resolve VS Code auto-rename conflict Implement js-sanitizer RPC server to eliminate 900+ Node.js process spawns Implement RPC server architecture for JavaScript parsing WIP: Add RPC server infrastructure for JS parsing (partial implementation) Update jqhtml terminology from destroy to stop, fix datagrid DOM preservation Add JQHTML-CLASS-01 rule and fix redundant class names Improve code quality rules and resolve violations Remove legacy fatal error format in favor of unified 'fatal' error type Filter internal keys from window.rsxapp output Update button styling and comprehensive form/modal documentation Add conditional fly-in animation for modals Fix non-deterministic bundle compilation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
242 lines
25 KiB
JavaScript
Executable File
242 lines
25 KiB
JavaScript
Executable File
/* Compiled from: rsx/theme/components/datagrid/datagrid_abstract.jqhtml */ (function() {
|
|
'use strict';
|
|
|
|
const template_DataGrid_Abstract = {
|
|
_jqhtml_version: '2.2.185',
|
|
name: 'DataGrid_Abstract',
|
|
tag: 'div',
|
|
defaultAttributes: {"class": "card DataGrid"},
|
|
render: function render(data, args, content, jqhtml) { let _output = []; const _cid = this._cid; const that = this;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_output.push(" ");
|
|
_output.push({comp: ["Card_Header", {}, function(Card_Header) { let _output = [];
|
|
_output.push(" "); (() => { const result = content('DG_Card_Header');; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })();
|
|
_output.push(" "); return [_output, this]; }.bind(this)]});
|
|
|
|
_output.push({tag: ["div", {"class": "card-body p-0"}, false]});
|
|
_output.push(" ");_output.push({tag: ["div", {"class": "table-responsive"}, false]});
|
|
_output.push(" ");_output.push({tag: ["table", {"class": "table table-hover mb-0"}, false]});
|
|
_output.push(" "); _output.push({comp: ["Redrawable", {"id": "datagrid_table_header" + ":" + this._cid, "data-id": "datagrid_table_header", "_tag": "thead"}, function(Redrawable) { let _output = [];
|
|
_output.push(" "); (() => { const result = content('DG_Table_Header');; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })();
|
|
_output.push(" "); return [_output, this]; }.bind(this)]});
|
|
_output.push({comp: ["Redrawable", {"id": "datagrid_table_body" + ":" + this._cid, "data-id": "datagrid_table_body", "class": ((this.data.loading && this.data.rows.length === 0) ? 'is-loading' : (this.data.is_empty ? 'is-empty' : '')), "_tag": "tbody"}, function(Redrawable) { let _output = [];
|
|
_output.push(" "); if (this.data.loading && this.data.rows.length === 0) {
|
|
_output.push({tag: ["tr", {"class": "loading-row"}, false]}); _output.push(" "); _output.push({tag: ["td", {"colspan": "999", "class": "text-center py-5"}, false]}); _output.push(" "); _output.push({tag: ["div", {"class": "spinner-border text-primary mb-3", "role": "status"}, false]}); _output.push(" "); _output.push({tag: ["span", {"class": "visually-hidden"}, false]}); _output.push("Loading..."); _output.push("</span>"); _output.push(" "); _output.push("</div>"); _output.push({tag: ["p", {"class": "text-muted"}, false]}); _output.push("Loading..."); _output.push("</p>"); _output.push(" "); _output.push("</td>"); _output.push("</tr>");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} else if (this.data.is_empty) {
|
|
_output.push({tag: ["tr", {"class": "empty-row"}, false]}); _output.push(" "); _output.push({tag: ["td", {"colspan": "999", "class": "text-center py-5"}, false]}); _output.push(" "); _output.push({tag: ["i", {"class": "bi bi-inbox", "style": "font-size: 3rem; color: #6c757d;"}, false]}); _output.push("</i>"); _output.push(" "); if (this.data.filter) { _output.push({tag: ["p", {"class": "text-muted mt-3"}, false]}); _output.push("No results found for \""); (() => { const result = this.data.filter ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push("\""); _output.push("</p>"); _output.push(" "); _output.push({tag: ["button", {"id": "clear_filter_btn" + ":" + this._cid, "data-id": "clear_filter_btn", "class": "btn btn-sm btn-secondary mt-2"}, false]}); _output.push("Clear Filter"); _output.push("</button>"); _output.push(" "); } else { _output.push({tag: ["p", {"class": "text-muted mt-3"}, false]}); _output.push("No results found"); _output.push("</p>"); _output.push(" "); } _output.push("</td>"); _output.push("</tr>");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} else {
|
|
for(let row of this.data.rows) {
|
|
(() => { const result = content('row', row);; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })();
|
|
_output.push(" "); }
|
|
}
|
|
return [_output, this]; }.bind(this)]});
|
|
_output.push("</table>");
|
|
_output.push("</div>");
|
|
_output.push("</div>");
|
|
|
|
_output.push({comp: ["Card_Footer", {}, function(Card_Footer) { let _output = [];
|
|
_output.push(" "); _output.push({tag: ["div", {"class": "d-flex gap-2 align-items-center"}, false]}); _output.push(" "); _output.push({tag: ["div", {"class": "dropdown"}, false]}); _output.push(" "); _output.push({tag: ["button", {"class": "btn btn-secondary btn-sm dropdown-toggle", "type": "button", "data-bs-toggle": "dropdown"}, false]}); _output.push(" "); _output.push(" Actions "); _output.push("</button>"); _output.push(" "); _output.push({tag: ["ul", {"class": "dropdown-menu"}, false]}); _output.push(" "); _output.push({tag: ["li", {}, false]}); _output.push({tag: ["a", {"class": "dropdown-item", "href": "#"}, false]}); _output.push("Export Selected"); _output.push("</a>"); _output.push("</li>"); _output.push(" "); _output.push({tag: ["li", {}, false]}); _output.push({tag: ["a", {"class": "dropdown-item", "href": "#"}, false]}); _output.push("Delete Selected"); _output.push("</a>"); _output.push("</li>"); _output.push(" "); _output.push({tag: ["li", {}, false]}); _output.push(" "); _output.push({tag: ["hr", {"class": "dropdown-divider"}, true]}); _output.push(" "); _output.push("</li>"); _output.push({tag: ["li", {}, false]}); _output.push({tag: ["a", {"class": "dropdown-item", "href": "#"}, false]}); _output.push("Mark as Active"); _output.push("</a>"); _output.push("</li>"); _output.push(" "); _output.push("</ul>"); _output.push("</div>"); _output.push({comp: ["Pagination_Info", {"id": "pagination_info" + ":" + this._cid, "data-id": "pagination_info"}, function(Pagination_Info) {
|
|
const _output = [];
|
|
_output.push(" "); if (this.data.total && this.data.page && this.data.per_page) { /* empty line */
|
|
const start = ((this.data.page - 1) * this.data.per_page) + 1;
|
|
const end = Math.min(this.data.page * this.data.per_page, this.data.total); _output.push(" Showing "); (() => { const result = start ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push(" to "); (() => { const result = end ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push(" of "); (() => { const result = this.data.total ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push(" entries "); } _output.push(" ");
|
|
return [_output, this];
|
|
}.bind(this)]}); _output.push("</div>");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_output.push({comp: ["Pagination_Controls", {"id": "pagination_controls" + ":" + this._cid, "data-id": "pagination_controls"}, function(Pagination_Controls) {
|
|
const _output = [];
|
|
_output.push(" "); if (this.data.total_pages && this.data.total_pages > 1) { /* empty line */
|
|
const currentPage = this.data.page || 1;
|
|
const totalPages = this.data.total_pages;
|
|
const maxVisible = 7; // Maximum page numbers to show
|
|
/* empty line */
|
|
// Calculate page range to display
|
|
let startPage, endPage;
|
|
/* empty line */
|
|
if (totalPages <= maxVisible) {
|
|
// Show all pages if total is less than max
|
|
startPage = 1;
|
|
endPage = totalPages;
|
|
} else {
|
|
// Calculate range with current page near center
|
|
const maxPagesBeforeCurrentPage = Math.floor(maxVisible / 2);
|
|
const maxPagesAfterCurrentPage = Math.ceil(maxVisible / 2) - 1;
|
|
/* empty line */
|
|
if (currentPage <= maxPagesBeforeCurrentPage) {
|
|
// Near the start
|
|
startPage = 1;
|
|
endPage = maxVisible;
|
|
} else if (currentPage + maxPagesAfterCurrentPage >= totalPages) {
|
|
// Near the end
|
|
startPage = totalPages - maxVisible + 1;
|
|
endPage = totalPages;
|
|
} else {
|
|
// In the middle
|
|
startPage = currentPage - maxPagesBeforeCurrentPage;
|
|
endPage = currentPage + maxPagesAfterCurrentPage;
|
|
}
|
|
}
|
|
/* empty line */
|
|
// Generate page numbers array
|
|
const pages = [];
|
|
for (let i = startPage; i <= endPage; i++) {
|
|
pages.push(i);
|
|
}
|
|
/* empty line */
|
|
const showStartEllipsis = startPage > 1;
|
|
const showEndEllipsis = endPage < totalPages; _output.push({tag: ["li", {"class": "page-item" + (currentPage === 1 ? 'disabled' : '')}, false]}); _output.push(" "); _output.push({tag: ["a", {"class": "page-link", "href": "#", "data-page": (currentPage - 1), "tabindex": (currentPage === 1 ? '-1' : '0')}, false]}); _output.push(" "); _output.push(" Previous "); _output.push("</a>"); _output.push(" "); _output.push("</li>"); if (showStartEllipsis) { _output.push({tag: ["li", {"class": "page-item"}, false]}); _output.push(" "); _output.push({tag: ["a", {"class": "page-link", "href": "#", "data-page": "1"}, false]}); _output.push("1"); _output.push("</a>"); _output.push(" "); _output.push("</li>"); _output.push({tag: ["li", {"class": "page-item disabled"}, false]}); _output.push(" "); _output.push({tag: ["span", {"class": "page-link"}, false]}); _output.push("..."); _output.push("</span>"); _output.push(" "); _output.push("</li>"); } for (let pageNum of pages) { console.log("Loop check", pageNum, currentPage); _output.push({tag: ["li", {"class": "page-item" + (pageNum === currentPage ? 'active' : '')}, false]}); _output.push(" "); _output.push({tag: ["a", {"class": "page-link", "href": "#", "data-page": (pageNum)}, false]}); _output.push(" "); (() => { const result = pageNum ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push(" "); _output.push("</a>"); _output.push("</li>"); } if (showEndEllipsis) { _output.push({tag: ["li", {"class": "page-item disabled"}, false]}); _output.push(" "); _output.push({tag: ["span", {"class": "page-link"}, false]}); _output.push("..."); _output.push("</span>"); _output.push(" "); _output.push("</li>"); _output.push({tag: ["li", {"class": "page-item"}, false]}); _output.push(" "); _output.push({tag: ["a", {"class": "page-link", "href": "#", "data-page": (totalPages)}, false]}); (() => { const result = totalPages ; if (Array.isArray(result)) { if (result.length === 2 && Array.isArray(result[0])) { _output.push(...result[0]); } else { _output.push(...result); } } else { _output.push(jqhtml.escape_html(result)); } })(); _output.push("</a>"); _output.push(" "); _output.push("</li>"); } _output.push({tag: ["li", {"class": "page-item" + (currentPage === totalPages ? 'disabled' : '')}, false]}); _output.push(" "); _output.push({tag: ["a", {"class": "page-link", "href": "#", "data-page": (currentPage + 1)}, false]}); _output.push(" "); _output.push(" Next "); _output.push("</a>"); _output.push(" "); _output.push("</li>"); }
|
|
return [_output, this];
|
|
}.bind(this)]});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return [_output, this]; }.bind(this)]});
|
|
|
|
return [_output, this]; },
|
|
dependencies: []
|
|
};
|
|
|
|
// Self-register with jqhtml runtime
|
|
// Must use window.jqhtml since we're in bundle scope
|
|
if (!window.jqhtml) {
|
|
throw new Error('FATAL: window.jqhtml is not defined. The jqhtml runtime must be loaded before registering templates.');
|
|
}
|
|
|
|
// Auto-register following standard jqhtml pattern
|
|
window.jqhtml.register_template(template_DataGrid_Abstract);
|
|
})();
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["datagrid_abstract.jqhtml"],"sourcesContent":["<!--\nContacts_DataGrid\n\nCustom datagrid for contacts with specialized row rendering.\nWraps base DataGrid component.\n\nARGUMENTS:\n    $api - API controller class\n-->\n<Define:DataGrid_Abstract class=\"card DataGrid\">\n\n    <Card_Header>\n        <%= content('DG_Card_Header'); %>\n    </Card_Header>\n\n    <div class=\"card-body p-0\">\n        <div class=\"table-responsive\">\n            <table class=\"table table-hover mb-0\">\n                <thead $redrawable $id=\"datagrid_table_header\">\n                    <%= content('DG_Table_Header'); %>\n                </thead>\n                <tbody $redrawable $id=\"datagrid_table_body\" class=\"<%= (this.data.loading && this.data.rows.length === 0) ? 'is-loading' : (this.data.is_empty ? 'is-empty' : '') %>\">\n                    <% if (this.data.loading && this.data.rows.length === 0) { %>\n                        <tr class=\"loading-row\">\n                            <td colspan=\"999\" class=\"text-center py-5\">\n                                <div class=\"spinner-border text-primary mb-3\" role=\"status\">\n                                    <span class=\"visually-hidden\">Loading...</span>\n                                </div>\n                                <p class=\"text-muted\">Loading...</p>\n                            </td>\n                        </tr>\n                    <% } else if (this.data.is_empty) { %>\n                        <tr class=\"empty-row\">\n                            <td colspan=\"999\" class=\"text-center py-5\">\n                                <i class=\"bi bi-inbox\" style=\"font-size: 3rem; color: #6c757d;\"></i>\n                                <% if (this.data.filter) { %>\n                                    <p class=\"text-muted mt-3\">No results found for \"<%= this.data.filter %>\"</p>\n                                    <button $id=\"clear_filter_btn\" class=\"btn btn-sm btn-secondary mt-2\">Clear Filter</button>\n                                <% } else { %>\n                                    <p class=\"text-muted mt-3\">No results found</p>\n                                <% } %>\n                            </td>\n                        </tr>\n                    <% } else { %>\n                        <% for(let row of this.data.rows) { %>\n                            <%= content('row', row); %>\n                        <% } %>\n                    <% } %>\n                </tbody>\n            </table>\n        </div>\n    </div>\n\n    <Card_Footer>\n        <div class=\"d-flex gap-2 align-items-center\">\n            <div class=\"dropdown\">\n                <button class=\"btn btn-secondary btn-sm dropdown-toggle\" type=\"button\" data-bs-toggle=\"dropdown\">\n                    Actions\n                </button>\n                <ul class=\"dropdown-menu\">\n                    <li><a class=\"dropdown-item\" href=\"#\">Export Selected</a></li>\n                    <li><a class=\"dropdown-item\" href=\"#\">Delete Selected</a></li>\n                    <li>\n                        <hr class=\"dropdown-divider\">\n                    </li>\n                    <li><a class=\"dropdown-item\" href=\"#\">Mark as Active</a></li>\n                </ul>\n            </div>\n            <Pagination_Info $id=\"pagination_info\">\n                <% if (this.data.total && this.data.page && this.data.per_page) { %>\n                    <%\n                        const start = ((this.data.page - 1) * this.data.per_page) + 1;\n                        const end = Math.min(this.data.page * this.data.per_page, this.data.total);\n                    %>\n                    Showing <%= start %> to <%= end %> of <%= this.data.total %> entries\n                <% } %>\n            </Pagination_Info>\n        </div>\n\n        <Pagination_Controls $id=\"pagination_controls\">\n            <% if (this.data.total_pages && this.data.total_pages > 1) { %>\n                <%\n                    const currentPage = this.data.page || 1;\n                    const totalPages = this.data.total_pages;\n                    const maxVisible = 7; // Maximum page numbers to show\n\n                    // Calculate page range to display\n                    let startPage, endPage;\n\n                    if (totalPages <= maxVisible) {\n                        // Show all pages if total is less than max\n                        startPage = 1;\n                        endPage = totalPages;\n                    } else {\n                        // Calculate range with current page near center\n                        const maxPagesBeforeCurrentPage = Math.floor(maxVisible / 2);\n                        const maxPagesAfterCurrentPage = Math.ceil(maxVisible / 2) - 1;\n\n                        if (currentPage <= maxPagesBeforeCurrentPage) {\n                            // Near the start\n                            startPage = 1;\n                            endPage = maxVisible;\n                        } else if (currentPage + maxPagesAfterCurrentPage >= totalPages) {\n                            // Near the end\n                            startPage = totalPages - maxVisible + 1;\n                            endPage = totalPages;\n                        } else {\n                            // In the middle\n                            startPage = currentPage - maxPagesBeforeCurrentPage;\n                            endPage = currentPage + maxPagesAfterCurrentPage;\n                        }\n                    }\n\n                    // Generate page numbers array\n                    const pages = [];\n                    for (let i = startPage; i <= endPage; i++) {\n                        pages.push(i);\n                    }\n\n                    const showStartEllipsis = startPage > 1;\n                    const showEndEllipsis = endPage < totalPages;\n                %>\n\n                <%-- Previous button --%>\n                <li class=\"page-item <%= currentPage === 1 ? 'disabled' : '' %>\">\n                    <a class=\"page-link\" href=\"#\" data-page=\"<%= currentPage - 1 %>\" tabindex=\"<%= currentPage === 1 ? '-1' : '0' %>\">\n                        Previous\n                    </a>\n                </li>\n\n                <%-- First page + ellipsis --%>\n                <% if (showStartEllipsis) { %>\n                    <li class=\"page-item\">\n                        <a class=\"page-link\" href=\"#\" data-page=\"1\">1</a>\n                    </li>\n                    <li class=\"page-item disabled\">\n                        <span class=\"page-link\">...</span>\n                    </li>\n                <% } %>\n\n                <%-- Page numbers --%>\n                <% for (let pageNum of pages) { %>\n                    <% console.log(\"Loop check\", pageNum, currentPage); %>\n                    <li class=\"page-item <%= pageNum === currentPage ? 'active' : '' %>\">\n                        <a class=\"page-link\" href=\"#\" data-page=\"<%= pageNum %>\">\n                            <%= pageNum %>\n                        </a>\n                    </li>\n                <% } %>\n\n                <%-- Ellipsis + last page --%>\n                <% if (showEndEllipsis) { %>\n                    <li class=\"page-item disabled\">\n                        <span class=\"page-link\">...</span>\n                    </li>\n                    <li class=\"page-item\">\n                        <a class=\"page-link\" href=\"#\" data-page=\"<%= totalPages %>\"><%= totalPages %></a>\n                    </li>\n                <% } %>\n\n                <%-- Next button --%>\n                <li class=\"page-item <%= currentPage === totalPages ? 'disabled' : '' %>\">\n                    <a class=\"page-link\" href=\"#\" data-page=\"<%= currentPage + 1 %>\">\n                        Next\n                    </a>\n                </li>\n            <% } %>\n        </Pagination_Controls>\n    </Card_Footer>\n\n</Define:DataGrid_Abstract>"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA","names":[]}
|