├── Api ├── Data │ ├── GridDataInterface.php │ └── GridInterface.php ├── DataProcessorInterface.php └── GridDataInterface.php ├── Block ├── GenericGrid.php └── Grid │ └── Filters.php ├── Controller └── Adminhtml │ └── Grid │ ├── Data.php │ ├── Filter.php │ └── Index.php ├── Model ├── DataProcessor │ ├── ChainProcessor.php │ ├── DateProcessor.php │ ├── DefaultProcessor.php │ ├── IdProcessor.php │ ├── ObfuscateProcessor.php │ ├── PriceProcessor.php │ └── StatusProcessor.php ├── Fields │ ├── DataSourceInterface.php │ └── DefaultDataSource.php ├── Grid.php └── GridData.php ├── Plugin ├── Admin │ └── SecretKeyValidator.php └── WebapiAuthorizationPlugin.php ├── README.md ├── ViewModel ├── GenericViewModelGrid.php └── Grid.php ├── composer.json ├── etc ├── acl.xml ├── adminhtml │ ├── default.xml │ ├── menu.xml │ └── routes.xml ├── di.xml ├── module.xml └── webapi.xml ├── registration.php └── view └── adminhtml ├── layout ├── adminhtml_dashboard_index.xml ├── default.xml ├── grid_grid_data.xml ├── grid_grid_index.xml └── mage_grid_grid_index.xml ├── require-config.js ├── templates ├── grid │ ├── additional-html.phtml │ ├── filters.phtml │ └── grid-component.phtml └── gridjs.phtml └── web ├── css ├── DataTable.css └── gridjs.css ├── images ├── flag32.png └── flags32.png └── js ├── DataTable.js ├── gridjs.js └── preact.js /Api/Data/GridDataInterface.php: -------------------------------------------------------------------------------- 1 | labels 32 | */ 33 | protected $fields; 34 | 35 | /** 36 | * @var string|null The database table name (if using SQL-based grid) 37 | */ 38 | protected $tableName; 39 | 40 | /** 41 | * @var string|object|null The collection class (if using collection-based grid) 42 | */ 43 | protected $collectionClass; 44 | 45 | /** 46 | * @var ManagerInterface For displaying admin error messages 47 | */ 48 | protected $messageManager; 49 | 50 | /** 51 | * @var string The main grid template file 52 | */ 53 | protected $_template = 'Mage_Grid::grid/grid-component.phtml'; 54 | 55 | /** 56 | * @var array List of additional HTML/JS templates to render after the grid 57 | */ 58 | protected $additionalHtmlTemplates = ['Mage_Grid::grid/additional-html.phtml']; 59 | 60 | /** 61 | * Constructor 62 | * 63 | * @param BackendContext $backendContext 64 | * @param GenericViewModelGrid $defaultViewModel 65 | * @param ManagerInterface $messageManager 66 | * @param array $data 67 | * 68 | * Initializes the grid block, sets up fields, data sources, and error handling. 69 | */ 70 | public function __construct( 71 | BackendContext $backendContext, 72 | GenericViewModelGrid $defaultViewModel, 73 | ManagerInterface $messageManager, 74 | array $data = [] 75 | ) { 76 | parent::__construct($backendContext, $data); 77 | $this->messageManager = $messageManager; 78 | // Use provided ViewModel or fallback to default 79 | $this->viewModel = $this->getData('viewModel') ?: $defaultViewModel; 80 | $dataProcessors = $this->getData('dataProcessors') ?: []; 81 | foreach ($dataProcessors as $field => $processor) { 82 | $this->viewModel->addDataProcessor($field, $processor); 83 | } 84 | 85 | // Set up fields and field names 86 | $this->fields = $this->getData('fields') ?: ['id' => 'ID']; 87 | $this->viewModel->setFields($this->fields); 88 | $this->viewModel->setFieldsNames($this->fields); 89 | $this->viewModel->setTableName($this->tableName); 90 | } 91 | 92 | /** 93 | * Get the ViewModel instance 94 | * @return GenericViewModelGrid 95 | */ 96 | public function getViewModel() 97 | { 98 | return $this->viewModel; 99 | } 100 | 101 | /** 102 | * Get the configured fields (associative array) 103 | * @return array 104 | */ 105 | public function getFields() 106 | { 107 | return $this->fields; 108 | } 109 | 110 | /** 111 | * Get the field keys (for use in data arrays) 112 | * @return array 113 | */ 114 | public function getFieldsNames() 115 | { 116 | return $this->viewModel->getFieldsNames(); 117 | } 118 | 119 | /** 120 | * Get the configured table name (if any) 121 | * @return string|null 122 | */ 123 | public function getTableName() 124 | { 125 | return $this->tableName; 126 | } 127 | 128 | /** 129 | * Get the configured fields config 130 | * Must be called after getGridJsonData because it depends on the table name 131 | * @return array 132 | */ 133 | public function getFieldsConfig() 134 | { 135 | $fieldsConfig = $this->viewModel->getFieldsConfig(); 136 | $table = $this->viewModel->getCollection()->getMainTable(); 137 | 138 | foreach ($fieldsConfig as $key => $value) { 139 | foreach ($value as $key2 => $value2) { 140 | if ($key2 === 'source_model' && is_object($value2)) { 141 | $value2->setTableName($table); 142 | $fieldsConfig[$key]['data'] = $value2->getValues($key); 143 | } 144 | } 145 | } 146 | return $fieldsConfig; 147 | } 148 | 149 | public function getFieldsHtml() 150 | { 151 | return $this->viewModel->getFieldsHtml(); 152 | } 153 | 154 | /** 155 | * Get the processed filter fields 156 | * @param array $fields 157 | * @param array $fieldsConfig 158 | * @param array $filters 159 | * @return array 160 | */ 161 | public function getProcessedFields($fields, $fieldsConfig, $filters) 162 | { 163 | foreach ($fields as $field) { 164 | $processedFields[$field] = [ 165 | 'config' => isset($fieldsConfig[$field]) ? $fieldsConfig[$field] : [], 166 | 'hidden' => isset($fieldsConfig[$field]['hidden']) 167 | ? $fieldsConfig[$field]['hidden'] 168 | : false, 169 | 'label' => isset($fieldsConfig[$field]['label']) 170 | ? $fieldsConfig[$field]['label'] 171 | : ucwords(str_replace('_', ' ', $field)), 172 | 'element' => isset($fieldsConfig[$field]['element']) 173 | ? $fieldsConfig[$field]['element'] 174 | : 'text', 175 | 'data' => isset($fieldsConfig[$field]['data']) 176 | ? $fieldsConfig[$field]['data'] 177 | : [], 178 | 'filter_value' => isset($filters[$field]) 179 | ? (is_array($filters[$field]) ? $filters[$field] : (string) $filters[$field]) 180 | : (isset($fieldsConfig[$field]['element']) && 181 | in_array($fieldsConfig[$field]['element'], ['select', 'multiselect']) ? [] : '') 182 | ]; 183 | } 184 | return $processedFields; 185 | } 186 | 187 | /** 188 | * Lazy load the collection class 189 | */ 190 | function lazyLoadCollectionClass() 191 | { 192 | $this->collectionClass = $this->getData('collectionClass'); 193 | 194 | $this->tableName = $this->getData('tableName'); 195 | // If a collection class is set, ignore table name 196 | if ($this->collectionClass !== 'none') { 197 | $this->tableName = null; 198 | } 199 | // Set collection class in ViewModel if provided 200 | if ($this->collectionClass !== 'none') { 201 | $this->viewModel->setCollectionClass($this->collectionClass); 202 | } 203 | // Require at least one data source 204 | if ($this->tableName === 'none' && $this->collectionClass === 'none') { 205 | $this->messageManager->addErrorMessage('Mage Grid: Collection class or table name is required'); 206 | throw new \Exception('Mage Grid: Collection class or table name is required'); 207 | } 208 | } 209 | 210 | /** 211 | * Get the grid data as JSON (for Grid.js) 212 | * @return string JSON-encoded data 213 | */ 214 | public function getGridJsonData() 215 | { 216 | $this->lazyLoadCollectionClass(); 217 | 218 | return $this->viewModel->getJsonGridData(array_keys($this->getFieldsNames())); 219 | } 220 | 221 | /** 222 | * Render all configured additional HTML/JS templates after the grid 223 | * 224 | * @return string Rendered HTML/JS (with error messages for missing templates) 225 | * 226 | * Usage: Place getAditionalHTML() ?> in your grid template. 227 | * Configure templates via layout XML with the 'additional_html_templates' argument. 228 | */ 229 | public function getAditionalHTML() 230 | { 231 | $templates = $this->getData('additional_html_templates') ?: $this->additionalHtmlTemplates; 232 | if (!is_array($templates)) { 233 | $templates = [$templates]; 234 | } 235 | $output = ''; 236 | foreach ($templates as $template) { 237 | $templateFile = $this->getTemplateFile($template); 238 | if (!is_file($templateFile)) { 239 | $output .= '
Error: Additional HTML template not found: ' . htmlspecialchars($template) . '
'; 240 | continue; 241 | } 242 | $output .= '' . $this->fetchView($templateFile); 243 | } 244 | return $output; 245 | } 246 | 247 | /** 248 | * Get filter value for a specific field 249 | * 250 | * @param string $field 251 | * @param array $filters 252 | * @return string|array 253 | */ 254 | protected function getFilterValue($field, $filters) 255 | { 256 | $fieldsConfig = $this->getFieldsConfig(); 257 | 258 | if (isset($fieldsConfig[$field]['element']) && 259 | in_array($fieldsConfig[$field]['element'], ['select', 'multiselect'])) { 260 | // For select/multiselect, keep as array 261 | $filterValue = isset($filters[$field]) ? $filters[$field] : []; 262 | if (!is_array($filterValue)) { 263 | $filterValue = $filterValue ? [$filterValue] : []; 264 | } 265 | } else { 266 | // For text inputs, ensure string 267 | $filterValue = isset($filters[$field]) ? (string) $filters[$field] : ''; 268 | } 269 | 270 | return $filterValue; 271 | } 272 | 273 | /** 274 | * Get current filter values 275 | * 276 | * @return array 277 | */ 278 | public function getCurrentFilters() 279 | { 280 | return $this->getRequest()->getParam('filter', []); 281 | } 282 | 283 | /** 284 | * Get rendered filters HTML 285 | * 286 | * @return string 287 | */ 288 | public function getFiltersHtml($filterData): string 289 | { 290 | return $this 291 | ->getLayout() 292 | ->createBlock(\Mage\Grid\Block\Grid\Filters::class) 293 | ->setFilterData($filterData) 294 | ->toHtml(); 295 | } 296 | } 297 | -------------------------------------------------------------------------------- /Block/Grid/Filters.php: -------------------------------------------------------------------------------- 1 | filterData = $data; 27 | return $this; 28 | } 29 | 30 | /** 31 | * Get filter data 32 | * 33 | * @return array 34 | */ 35 | public function getFilterData(): array 36 | { 37 | return $this->filterData; 38 | } 39 | 40 | /** 41 | * Get fields data 42 | * 43 | * @return array 44 | */ 45 | public function getFields(): array 46 | { 47 | return $this->filterData['fields'] ?? []; 48 | } 49 | 50 | /** 51 | * Get filters data 52 | * 53 | * @return array 54 | */ 55 | public function getFilters(): array 56 | { 57 | return $this->filterData['filters'] ?? []; 58 | } 59 | } -------------------------------------------------------------------------------- /Controller/Adminhtml/Grid/Data.php: -------------------------------------------------------------------------------- 1 | gridViewModel = $gridViewModel; 47 | $this->layoutFactory = $layoutFactory; 48 | $this->resultPageFactory = $resultPageFactory; 49 | } 50 | 51 | /** 52 | * Check admin permissions 53 | * 54 | * @return bool 55 | */ 56 | protected function _isAllowed() 57 | { 58 | return $this->_authorization->isAllowed('Magento_Sales::sales'); 59 | } 60 | 61 | /** 62 | * Get grid data via API 63 | * 64 | * @return ResponseInterface 65 | */ 66 | public function execute() 67 | { 68 | try { 69 | // Get the layout and block 70 | $layout = $this->resultPageFactory->create()->getLayout(); 71 | $request = $this->getRequest(); 72 | $request->setParams(['data' => 'false']); 73 | 74 | $layout->getUpdate()->load(); // collect all handles (default, route, etc.) 75 | $layout->generateXml(); // merge the XML 76 | $layout->generateElements(); // turn tags into PHP objects 77 | 78 | $this->gridBlock = $layout->getBlock('grid_generic_grid'); 79 | 80 | if (!$this->gridBlock) { 81 | throw new LocalizedException(__('Grid block not found')); 82 | } 83 | 84 | $fields = array_keys($this->gridBlock->getFieldsNames()); // array: ['id', 'order_number', ...] 85 | $fieldsFull = $this->gridBlock->getFields(); // associative array: ['id' => 'ID', ...] 86 | $jsonGridData = $this->gridBlock->getGridJsonData(); // JSON-encoded grid data 87 | // dd($jsonGridData); 88 | 89 | // Return JSON response 90 | $this->getResponse()->representJson( 91 | $jsonGridData 92 | ); 93 | } catch (LocalizedException $e) { 94 | throw new \Exception('Error: ' . $e->getMessage(), 400); 95 | } catch (\Exception $e) { 96 | throw new \Exception('An error occurred while fetching grid data:' . $e->getMessage(), 500); 97 | } 98 | } 99 | } 100 | -------------------------------------------------------------------------------- /Controller/Adminhtml/Grid/Filter.php: -------------------------------------------------------------------------------- 1 | resultJsonFactory = $resultJsonFactory; 24 | $this->om = $om; 25 | } 26 | 27 | protected function _isAllowed() 28 | { 29 | return true; // Or set proper ACL permission 30 | } 31 | 32 | public function execute(): ResultInterface 33 | { 34 | // dd($this->getRequest()->getParams()); 35 | try { 36 | $query = $this->getRequest()->getParam('query'); 37 | if (!$query) { 38 | throw new \Exception(__('No query provided')); 39 | } 40 | 41 | try { 42 | $this->chatQuery = $this->om->get(ChatQuery::class); 43 | } catch (\Exception $e) { 44 | $this->logger->error('Error getting Chat AI extension: ' . $e->getMessage()); 45 | return $this->resultJsonFactory->create()->setData([ 46 | 'success' => false, 47 | 'message' => __('An error occurred while processing your request. Please try again.') 48 | ]); 49 | } 50 | 51 | // Prepare context for grid filtering 52 | $storeContext = [ 53 | 'DataAPIUrl' => $this->getRequest()->getParam('DataUrl', ''), 54 | 'store_id' => $this->getRequest()->getParam('store', 0), 55 | 'website_id' => $this->getRequest()->getParam('website', 0) 56 | ]; 57 | 58 | // Get AI response 59 | $aiResponse = $this->getAiResponse($query, $storeContext); 60 | 61 | $errorMessage = ''; 62 | if (isset($aiResponse['success']) && $aiResponse['success'] === false) { 63 | if (isset($aiResponse['message'])) { 64 | $errorMessage = $aiResponse['message']; 65 | } 66 | throw new \Exception($errorMessage); 67 | } 68 | 69 | 70 | $groupByFields = json_decode($aiResponse, true)['groupByFields']; 71 | $urlParams = json_decode($aiResponse, true)['filterUrl']; 72 | $explanation = json_decode($aiResponse, true)['explanation']; 73 | $confidence = json_decode($aiResponse, true)['confidence']; 74 | 75 | // Parse AI response to get filter suggestions 76 | $filters = $this->parseAiResponse($aiResponse); 77 | 78 | return $this->resultJsonFactory->create()->setData([ 79 | 'success' => true, 80 | 'suggestion' => [ 81 | 'filterUrl' => $urlParams ?? '', 82 | 'groupByFields' => $groupByFields ?? '', 83 | 'explanation' => $explanation ?? '', 84 | 'confidence' => $confidence ?? '', 85 | 'field' => $filters[0]['field'] ?? 'entity_id', 86 | 'value' => $filters[0]['value'] ?? $query, 87 | 'filters' => $filters 88 | ], 89 | 'raw' => $aiResponse 90 | ]); 91 | } catch (\Exception $e) { 92 | return $this->resultJsonFactory->create()->setData([ 93 | 'success' => false, 94 | 'message' => $e->getMessage() 95 | ]); 96 | } 97 | } 98 | 99 | /** 100 | * Get AI response for a query 101 | */ 102 | private function getAiResponse($query, $storeContext) 103 | { 104 | try { 105 | // Get available grid fields and their types 106 | $gridFields = $this->getGridFields(); 107 | 108 | // Get current filters from URL 109 | $currentFilters = $this->getCurrentFilters(); 110 | 111 | // Get pagination info 112 | $pagination = $this->getPaginationInfo(); 113 | 114 | // Get sorting info 115 | $sorting = $this->getSortingInfo(); 116 | 117 | // Build enhanced system prompt 118 | $systemPrompt = 'You are a Magento Grid Filter Assistant. ' 119 | . 'Your task is to analyze user queries and suggest appropriate filters for the grid.' 120 | . "\n\nRules:\n1. Only suggest filters that exist in the grid you you don't know ask for more context\n" 121 | . "2. Consider current filters and pagination\n" 122 | . "3. Provide multiple filter options when appropriate\n" 123 | . "4. If unsure, suggest multiple ask more context from user\n" 124 | . "5. Always respond in JSON format or ask more context from user\n" 125 | . "6. Respect field examples provided\n" 126 | . "7. Available Fields:\n" . json_encode($gridFields, JSON_PRETTY_PRINT) . "\n\nCurrent Context:\n" . json_encode([ 127 | 'current_filters' => [], 128 | 'pagination' => $pagination, 129 | 'sorting' => $sorting, 130 | 'store_context' => $storeContext 131 | ], JSON_PRETTY_PRINT) 132 | . "\n\n8. Make sure you are having valid url for filterUrl \n\n" 133 | . "8.1 add new filters to the end of the url get parameters \n\n" 134 | . "8.2 add groupByFields if such is needed and allowed by the field config 'groupByField' => true, to the end of the url get parameters groupByFields=field1,field2 \n\n" 135 | . "9. make sure us are using correct domain and url path from the DatAPIurl . Thanks \n\n" 136 | . "9.1 url like key/gdgdgf/filter[entity_id]=value?page=1&pageSize=40 is not valid \n\n" 137 | . "9.2 url like key/gdgdgf/?filter[entity_id]=value&page=1&pageSize=40 is valid \n\n" 138 | . "9.3 if you have multiple filters like key/gdgdgf/?filter[entity_id]=value&filter[entity_id]=value2 make them like filter[entity_id][]=value \n\n" 139 | . "\n\nResponse Format:\n{\n \"filterUrl\": url with all filter parameters,\n \"groupByFields\": \"value\", \"filters\": [\n {\n \"field\": \"field_name\",\n \"value\": \"filter_value\",\n \"description\": \"Optional description of the filter\"\n }\n ],\n \"explanation\": \"Optional explanation of the suggested filters\",\n \"confidence\": \"high|medium|low\",\n \"alternative_suggestions\": [\n {\n \"filters\": [...],\n \"reason\": \"Why this alternative might be useful\"\n }\n ]\n}\n"; 140 | 141 | // Use AgentoAI Chat Query to get response 142 | $response = $this->chatQuery->getAiResponse( 143 | $query, 144 | $storeContext, 145 | [], // No conversation history needed for grid filtering 146 | $systemPrompt 147 | ); 148 | 149 | return $response; 150 | } catch (\Exception $e) { 151 | throw new \Exception(__('Error processing AI request: %1', $e->getMessage())); 152 | } 153 | } 154 | 155 | /** 156 | * Get available grid fields and their types 157 | */ 158 | private function getGridFields() 159 | { 160 | $fields = [ 161 | 'entity_id' => [ 162 | 'type' => 'number', 163 | 'label' => 'Order ID', 164 | 'multiple' => false, 165 | 'alias' => 'entity id, order id , id', 166 | 'filter_example_get_url_parameter' => 'filter[entity_id][]=value', 167 | 'description' => 'Unique identifier for the record' 168 | ], 169 | 'increment_id' => [ 170 | 'type' => 'string', 171 | 'label' => 'Increment ID', 172 | 'multiple' => false, 173 | 'filter_example_get_url_parameter' => 'filter[increment_id][]=value', 174 | 'description' => 'Order number or unique identifier' 175 | ], 176 | 'status' => [ 177 | 'type' => 'select', 178 | 'label' => 'Status', 179 | 'multiple' => true, 180 | 'filter_example_get_url_parameter' => 'filter[status][]=value', 181 | 'options' => [ 182 | 'pending' => 'Pending', 183 | 'processing' => 'Processing', 184 | 'complete' => 'Complete', 185 | 'canceled' => 'Canceled', 186 | 'holded' => 'Holded', 187 | 'rejected' => 'Rejected' 188 | ], 189 | 'description' => 'Current status of the record' 190 | ], 191 | 'created_at' => [ 192 | 'multiple' => false, 193 | 'type' => 'date', 194 | 'label' => 'Created At', 195 | 'filter_example_get_url_parameter' => 'filter[created_at]=value', 196 | 'description' => 'Date when the record was created' 197 | ], 198 | 'grand_total' => [ 199 | 'multiple' => false, 200 | 'type' => 'number', 201 | 'label' => 'Grand Total', 202 | 'alias' => 'grand_total, amount, sum, total', 203 | 'filter_example_get_url_parameter' => 'filter[grand_total]=value', 204 | 'description' => 'Total amount of the order' 205 | ], 206 | 'customer_email' => [ 207 | 'groupByField' => true, 208 | 'multiple' => false, 209 | 'type' => 'string', 210 | 'label' => 'Customer Email', 211 | 'alias' => 'customer_email, email, user email, @email, @', 212 | 'filter_example_get_url_parameter' => 'filter[customer_email][]=value', 213 | 'description' => 'Customer email address' 214 | ], 215 | 'updated_at' => [ 216 | 'multiple' => false, 217 | 'type' => 'date', 218 | 'label' => 'Updated At', 219 | 'filter_example_get_url_parameter' => 'filter[updated_at]=value', 220 | 'description' => 'Date when the record was last updated' 221 | ] 222 | ]; 223 | 224 | return $fields; 225 | } 226 | 227 | /** 228 | * Get current filters from URL 229 | */ 230 | private function getCurrentFilters() 231 | { 232 | $filters = []; 233 | $urlParams = $this->getRequest()->getParams(); 234 | unset($urlParams['query']); 235 | foreach ($urlParams as $key => $value) { 236 | $field = str_replace('filter[', '', $key); 237 | $field = str_replace(']', '', $field); 238 | $filters[$field] = $value; 239 | } 240 | 241 | return $filters; 242 | } 243 | 244 | /** 245 | * Get pagination info 246 | */ 247 | private function getPaginationInfo() 248 | { 249 | return [ 250 | 'current_page' => $this->getRequest()->getParam('page', 1), 251 | 'page_size' => $this->getRequest()->getParam('pageSize', 20), 252 | 'total_records' => $this->getRequest()->getParam('totalRecords', 0) 253 | ]; 254 | } 255 | 256 | /** 257 | * Get sorting info 258 | */ 259 | private function getSortingInfo() 260 | { 261 | return [ 262 | 'sort_field' => $this->getRequest()->getParam('sortField', null), 263 | 'sort_order' => $this->getRequest()->getParam('sortOrder', null) 264 | ]; 265 | } 266 | 267 | /** 268 | * Parse AI response to get filter suggestions 269 | */ 270 | private function parseAiResponse($response) 271 | { 272 | try { 273 | $data = json_decode($response, true); 274 | if (isset($data['filters']) && is_array($data['filters'])) { 275 | return $data['filters']; 276 | } 277 | throw new \Exception(__('Invalid AI response format')); 278 | } catch (\Exception $e) { 279 | // If parsing fails, create a basic filter 280 | return [ 281 | [ 282 | 'filterUrl' => '', 283 | 'field' => 'entity_id', 284 | 'operator' => 'like', 285 | 'value' => '%' . $response . '%' 286 | ] 287 | ]; 288 | } 289 | } 290 | } 291 | -------------------------------------------------------------------------------- /Controller/Adminhtml/Grid/Index.php: -------------------------------------------------------------------------------- 1 | resultPageFactory = $resultPageFactory; 17 | } 18 | 19 | /** 20 | * Check admin permissions 21 | * 22 | * @return bool 23 | */ 24 | protected function _isAllowed() 25 | { 26 | return $this->_authorization->isAllowed('Magento_Sales::sales'); 27 | } 28 | 29 | public function execute() 30 | { 31 | $resultPage = $this->resultPageFactory->create(); 32 | $resultPage->setActiveMenu('Mage_Grid::grid'); 33 | $resultPage->getConfig()->getTitle()->prepend(__('Orders')); 34 | return $resultPage; 35 | } 36 | } -------------------------------------------------------------------------------- /Model/DataProcessor/ChainProcessor.php: -------------------------------------------------------------------------------- 1 | addProcessor($processor); 17 | } 18 | } 19 | 20 | /** 21 | * Add a processor to the chain 22 | * 23 | * @param DataProcessorInterface $processor 24 | * @return $this 25 | */ 26 | public function addProcessor(DataProcessorInterface $processor) 27 | { 28 | $this->processors[] = $processor; 29 | return $this; 30 | } 31 | 32 | /** 33 | * @inheritDoc 34 | */ 35 | public function process(string $field, $value, array $row = []): string 36 | { 37 | if (empty($this->processors)) { 38 | return $value; 39 | } 40 | usort($this->processors, function ($a, $b) { 41 | return $a->getOrder() <=> $b->getOrder(); 42 | }); 43 | 44 | $result = $value; 45 | foreach ($this->processors as $processor) { 46 | $result = $processor->process($field, $result, $row); 47 | } 48 | return $result; 49 | } 50 | 51 | public function getOrder(): int 52 | { 53 | return 0; 54 | } 55 | } -------------------------------------------------------------------------------- /Model/DataProcessor/DateProcessor.php: -------------------------------------------------------------------------------- 1 | order = $order ?? 10; 13 | } 14 | /** 15 | * @inheritDoc 16 | */ 17 | public function process(string $field, $value, array $row = []): string 18 | { 19 | if (empty($value)) { 20 | return ''; 21 | } 22 | 23 | try { 24 | $date = new \DateTime($value); 25 | return $date->format('Y-m-d H:i:s'); 26 | } catch (\Exception $e) { 27 | return $value; 28 | } 29 | } 30 | } -------------------------------------------------------------------------------- /Model/DataProcessor/DefaultProcessor.php: -------------------------------------------------------------------------------- 1 | order = $order ?? $this->order; 14 | } 15 | 16 | /** 17 | * @inheritDoc 18 | */ 19 | public function process(string $field, $value, array $row = []): string 20 | { 21 | return htmlspecialchars($value ?? '', ENT_QUOTES, 'UTF-8'); 22 | } 23 | 24 | public function getOrder(): int 25 | { 26 | return $this->order; 27 | } 28 | } -------------------------------------------------------------------------------- /Model/DataProcessor/IdProcessor.php: -------------------------------------------------------------------------------- 1 | order); 16 | } 17 | 18 | /** 19 | * @inheritDoc 20 | */ 21 | public function process(string $field, $value, array $row = []): string 22 | { 23 | if (!is_numeric($value)) { 24 | return $value; 25 | } 26 | 27 | // Get the order URL using the increment ID 28 | $orderUrl = $this->urlBuilder->getUrl( 29 | 'sales/order/view', 30 | ['order_id' => $value] 31 | ); 32 | 33 | return sprintf( 34 | ' %s', 35 | $orderUrl, 36 | $value 37 | ); 38 | } 39 | } -------------------------------------------------------------------------------- /Model/DataProcessor/ObfuscateProcessor.php: -------------------------------------------------------------------------------- 1 | $%s', 21 | number_format((float)$value, 2) 22 | ); 23 | } 24 | } -------------------------------------------------------------------------------- /Model/DataProcessor/StatusProcessor.php: -------------------------------------------------------------------------------- 1 | 42 | %s 43 | %s 44 | ', 45 | $color, 46 | $color, 47 | $color, 48 | $rowJson, 49 | $icon, 50 | htmlspecialchars($value, ENT_QUOTES, 'UTF-8') 51 | ); 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /Model/Fields/DataSourceInterface.php: -------------------------------------------------------------------------------- 1 | resource = $resource; 43 | $this->cache = $cache; 44 | } 45 | 46 | public function setFields(array|string $fields) 47 | { 48 | if (is_string($fields)) { 49 | $this->fields[] = $fields; 50 | } else { 51 | $this->fields = $fields; 52 | } 53 | } 54 | 55 | /** 56 | * Set the table name 57 | * @param string $tableName 58 | */ 59 | public function setTableName($tableName) 60 | { 61 | $this->tableName = $tableName; 62 | } 63 | 64 | /** 65 | * Get cache key for field values 66 | * 67 | * @param string $field 68 | * @return string 69 | */ 70 | protected function getCacheKey($field) 71 | { 72 | return 'grid_field_values_' . $this->tableName . '_' . $field; 73 | } 74 | 75 | /** 76 | * Get all distinct values for a given field 77 | * 78 | * @param string $field 79 | * @return array 80 | */ 81 | public function getValues($field) 82 | { 83 | $cacheKey = $this->getCacheKey($field); 84 | $cachedData = $this->cache->load($cacheKey); 85 | 86 | if ($cachedData !== false) { 87 | return json_decode($cachedData, true); 88 | } 89 | 90 | $connection = $this->resource->getConnection(); 91 | $select = $connection->select() 92 | ->from($this->tableName, [$field]) 93 | ->distinct(); 94 | $results = $connection->fetchCol($select); 95 | 96 | // Cache the results 97 | $this->cache->save( 98 | json_encode($results), 99 | $cacheKey, 100 | [self::CACHE_TAG], 101 | self::CACHE_LIFETIME 102 | ); 103 | 104 | return $results; 105 | } 106 | 107 | /** 108 | * Get all distinct values for multiple fields 109 | * 110 | * @param array $fields 111 | * @return array 112 | */ 113 | public function getAllValues(array $fields) 114 | { 115 | $fields = array_merge($this->fields, $fields); 116 | $result = []; 117 | 118 | foreach ($fields as $field) { 119 | $result[$field] = $this->getValues($field); 120 | } 121 | 122 | return $result; 123 | } 124 | } 125 | -------------------------------------------------------------------------------- /Model/Grid.php: -------------------------------------------------------------------------------- 1 | getData('data') ?? []; 17 | } 18 | 19 | /** 20 | * @inheritDoc 21 | */ 22 | public function setGridData(array $data): self 23 | { 24 | return $this->setData('data', $data); 25 | } 26 | 27 | /** 28 | * @inheritDoc 29 | */ 30 | public function getData($key = '', $index = null): array 31 | { 32 | return $this->getGridData($key, $index); 33 | } 34 | 35 | /** 36 | * @inheritDoc 37 | */ 38 | public function getTotalCount(): int 39 | { 40 | return (int)($this->getData('total_count') ?? 0); 41 | } 42 | 43 | /** 44 | * @inheritDoc 45 | */ 46 | public function setTotalCount(int $totalCount): self 47 | { 48 | return $this->setData('total_count', $totalCount); 49 | } 50 | 51 | /** 52 | * @inheritDoc 53 | */ 54 | public function getConfig(): array 55 | { 56 | return $this->getData('config') ?? []; 57 | } 58 | 59 | /** 60 | * @inheritDoc 61 | */ 62 | public function setConfig(array $config): self 63 | { 64 | return $this->setData('config', $config); 65 | } 66 | 67 | /** 68 | * @inheritDoc 69 | */ 70 | public function getFields(): array 71 | { 72 | return $this->getData('fields') ?? []; 73 | } 74 | 75 | /** 76 | * @inheritDoc 77 | */ 78 | public function setFields(array $fields): self 79 | { 80 | return $this->setData('fields', $fields); 81 | } 82 | 83 | /** 84 | * @inheritDoc 85 | */ 86 | public function getFilters(): array 87 | { 88 | return $this->getData('filters') ?? []; 89 | } 90 | 91 | /** 92 | * @inheritDoc 93 | */ 94 | public function setFilters(array $filters): self 95 | { 96 | return $this->setData('filters', $filters); 97 | } 98 | 99 | /** 100 | * @inheritDoc 101 | */ 102 | public function getPerformanceMetrics(): array 103 | { 104 | return $this->getData('performance_metrics') ?? []; 105 | } 106 | 107 | /** 108 | * @inheritDoc 109 | */ 110 | public function setPerformanceMetrics(array $metrics): self 111 | { 112 | return $this->setData('performance_metrics', $metrics); 113 | } 114 | 115 | /** 116 | * Get grid data 117 | * 118 | * @param string $gridId 119 | * @param array $filters 120 | * @param int $page 121 | * @param int $pageSize 122 | * @return array 123 | */ 124 | public function getGrid( 125 | string $gridId, 126 | array $filters = [], 127 | int $page = 1, 128 | int $pageSize = 20 129 | ): array { 130 | // Example data structure 131 | return [ 132 | 'data' => $this->getData('data') ?? [], 133 | 'total' => $this->getData('total') ?? 0, 134 | 'page' => $page, 135 | 'pageSize' => $pageSize 136 | ]; 137 | } 138 | } -------------------------------------------------------------------------------- /Model/GridData.php: -------------------------------------------------------------------------------- 1 | viewModel = $viewModel; 25 | } 26 | 27 | /** 28 | * @inheritDoc 29 | */ 30 | public function getGridData( 31 | string $gridId, 32 | array $filters = [], 33 | int $page = 1, 34 | int $pageSize = 20 35 | ): array { 36 | try { 37 | // Start timing 38 | $startTime = microtime(true); 39 | 40 | // Get grid data 41 | $data = $this->viewModel->getGridData($filters, $page, $pageSize); 42 | $totalCount = $this->viewModel->getTotalCount(); 43 | 44 | // Calculate performance metrics 45 | $endTime = microtime(true); 46 | $executionTime = ($endTime - $startTime) * 1000; // Convert to milliseconds 47 | 48 | // Return response data 49 | return [ 50 | 'data' => $data, 51 | 'total' => $totalCount, 52 | 'page' => $page, 53 | 'pageSize' => $pageSize, 54 | 'performance' => [ 55 | 'execution_time' => $executionTime, 56 | 'sql_time' => $this->viewModel->getSqlTime(), 57 | 'count_time' => $this->viewModel->getCountTime() 58 | ] 59 | ]; 60 | } catch (\Exception $e) { 61 | throw new LocalizedException(__('Error retrieving grid data: %1', $e->getMessage())); 62 | } 63 | } 64 | } -------------------------------------------------------------------------------- /Plugin/Admin/SecretKeyValidator.php: -------------------------------------------------------------------------------- 1 | request = $request; 24 | } 25 | 26 | /** 27 | * Modify useSecretKey to return false for our specific action 28 | */ 29 | public function aroundUseSecretKey( 30 | Url $subject, 31 | \Closure $proceed 32 | ): bool { 33 | if (strpos($this->request->getUri()->getPath(), 'grid/data') || 34 | strpos($this->request->getUri()->getPath(), 'grid/filter')) { 35 | return false; // disable secret key for this action 36 | } 37 | 38 | // For all other actions, use the default behavior 39 | return $proceed(); 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /Plugin/WebapiAuthorizationPlugin.php: -------------------------------------------------------------------------------- 1 | adminSession = $adminSession; 31 | $this->request = $request; 32 | } 33 | 34 | /** 35 | * Allow access if user has admin session only for grid API 36 | * 37 | * @param Authorization $subject 38 | * @param bool $result 39 | * @return bool 40 | */ 41 | public function afterIsAllowed(Authorization $subject, bool $result): bool 42 | { 43 | // If already authorized, return true 44 | if ($result) { 45 | return true; 46 | } 47 | 48 | // Check if this is our grid API route 49 | $path = $this->request->getPathInfo(); 50 | if (strpos($path, '/V1/grid/') !== 0) { 51 | return $result; 52 | } 53 | 54 | // For grid API, check if user has admin session 55 | return $this->adminSession->isLoggedIn(); 56 | } 57 | } -------------------------------------------------------------------------------- /ViewModel/GenericViewModelGrid.php: -------------------------------------------------------------------------------- 1 | DataProcessorInterface mapping 68 | */ 69 | protected $dataProcessors = []; 70 | 71 | /** 72 | * @var DefaultProcessor The default processor for fields 73 | */ 74 | protected $defaultProcessor; 75 | 76 | /** 77 | * @var ChainProcessor The chain processor for layered processing 78 | */ 79 | protected $chainProcessor; 80 | 81 | /** 82 | * @var string|null The SQL table name (if using SQL mode) 83 | */ 84 | public $tableName; 85 | 86 | /** 87 | * @var string|null The raw SQL query (if using SQL mode) 88 | */ 89 | public $sqlQuery = null; 90 | 91 | /** 92 | * @var string|null The SQL query for total count (if using SQL mode) 93 | */ 94 | public $totalSqlQuery; 95 | 96 | /** 97 | * @var array List of field keys for the grid 98 | */ 99 | public $fields = []; 100 | 101 | /** 102 | * @var array List of field keys for the grid that should be rendered as HTML 103 | */ 104 | public $fieldsHtml = []; 105 | 106 | /** 107 | * @var array List of field config for the grid 108 | * 109 | * example: [ 110 | * "customer_group_id" => [ 111 | * "label" => "Customer Group ID", 112 | * "element" => "select", 113 | * "source_model" => "Mage\Grid\Model\Fields\DefaultDataSource" 114 | * ] 115 | * ] 116 | */ 117 | public $fieldsConfig = []; 118 | 119 | /** 120 | * @var array List of field labels for the grid 121 | */ 122 | public $fieldsNames = []; 123 | 124 | /** 125 | * @var array Current filter values 126 | */ 127 | public $filters = []; 128 | 129 | /** 130 | * @var array Fields to group by 131 | */ 132 | protected $groupByFields = []; 133 | 134 | /** 135 | * @var string|object|null The collection class (if using collection mode) 136 | */ 137 | public string|object|null $collectionClass = null; 138 | 139 | /** 140 | * @var int The maximum number of rows to return (for SQL mode) 141 | */ 142 | public $LIMIT = 1000; 143 | 144 | /** 145 | * @var int The default page size for pagination 146 | */ 147 | public $DEFAULT_LIMIT = 40; 148 | 149 | /** 150 | * @var string|null The raw SQL (if set directly) 151 | */ 152 | protected $sql; 153 | 154 | /** 155 | * @var LayoutInterface For rendering templates (if needed) 156 | */ 157 | protected $layout; 158 | 159 | /** 160 | * @var ResourceConnection 161 | */ 162 | protected $resource; 163 | 164 | /** 165 | * @var bool Whether to use AJAX for data loading 166 | */ 167 | public $isAjax = true; 168 | 169 | /** 170 | * @var CacheInterface 171 | */ 172 | private $cache; 173 | 174 | /** 175 | * Constructor 176 | * 177 | * @param ObjectManagerInterface $objectManager 178 | * @param RequestInterface $request 179 | * @param LoggerInterface $logger 180 | * @param LayoutInterface $layout 181 | * @param DefaultProcessor $defaultProcessor 182 | * @param ChainProcessor $chainProcessor 183 | * @param ResourceConnection $resource 184 | * @param CacheInterface $cache 185 | * @param array $dataProcessors 186 | * 187 | * All dependencies are injected for maximum flexibility and testability. 188 | */ 189 | public function __construct( 190 | ObjectManagerInterface $objectManager, 191 | RequestInterface $request, 192 | LoggerInterface $logger, 193 | LayoutInterface $layout, 194 | DefaultProcessor $defaultProcessor, 195 | ChainProcessor $chainProcessor, 196 | ResourceConnection $resource, 197 | CacheInterface $cache, 198 | array $dataProcessors = [] 199 | ) { 200 | $this->objectManager = $objectManager; 201 | $this->request = $request; 202 | $this->logger = $logger; 203 | $this->layout = $layout; 204 | $this->defaultProcessor = $defaultProcessor; 205 | $this->chainProcessor = $chainProcessor; 206 | $this->resource = $resource; 207 | $this->cache = $cache; 208 | $this->dataProcessors = $dataProcessors; 209 | } 210 | 211 | /** 212 | * Set the collection class (for collection-based grids) 213 | * @param string|object $collectionClass 214 | * @return $this 215 | */ 216 | public function setCollectionClass($collectionClass) 217 | { 218 | $this->collectionClass = $collectionClass; 219 | return $this; 220 | } 221 | 222 | /** 223 | * Build and return the SQL query for SQL-based grids 224 | * @param array $fields 225 | * @param array $filters 226 | * @return string|false 227 | */ 228 | public function getSqlQuery($fields, $filters) 229 | { 230 | if ($this->getTableName() && $this->sqlQuery === null && $this->collectionClass === null) { 231 | $sql = 'SELECT ' . implode(', ', $fields) . ' FROM ' . $this->getTableName() . ' WHERE 1=1 '; 232 | foreach ($filters as $field => $value) { 233 | $sql .= ' AND ' . $field . ' LIKE :' . $field; 234 | } 235 | $this->sqlQuery = $sql; 236 | return $this->sqlQuery; 237 | } else { 238 | return false; 239 | } 240 | } 241 | 242 | /** 243 | * Get the data for the grid (from collection or SQL) 244 | * Handles filtering, pagination, and sorting. 245 | * 246 | * @param array $fields 247 | * @param array $filters 248 | * @return array 249 | */ 250 | public function getData(array $fields = [], array $filters = []) 251 | { 252 | try { 253 | // If using SQL mode, execute the SQL query 254 | if ($this->getSqlQuery($fields, $filters)) { 255 | return $this->executeSqlQuery(); 256 | } 257 | 258 | /* @var $collection AbstractCollection */ 259 | $collection = $this->getCollection(); 260 | 261 | $fields = array_merge($fields, $this->getFields()); 262 | //dd($fields); 263 | $this->setFields($fields); 264 | 265 | // Select fields dynamically 266 | //$collection->addFieldToSelect($fields); 267 | 268 | $this->setGroupByFields(explode(',', $this->request->getParam('groupByFields', ''))); 269 | 270 | // Handle filters 271 | // $filters = $this->getFilters(); 272 | 273 | // Build the select fields with appropriate aggregation 274 | $selectFields = []; 275 | foreach ($fields as $field) { 276 | if (in_array($field, $this->groupByFields)) { 277 | // If it's a group by field, use it as is 278 | $selectFields[$field] = $field; 279 | } else { 280 | // For all other fields, use SUM by default when grouping 281 | if (!empty($this->groupByFields) && $this->strposx($field, ["total", "amount", "price", "qty", "count"]) !== false) { 282 | $selectFields[$field] = sprintf('SUM(%s) as %s', $field, $field); 283 | } elseif (!empty($this->groupByFields) && $this->strposx($field, ["entity_id"]) !== false) { 284 | //dd($field); 285 | $selectFields[$field] = sprintf('COUNT(%s) as entity_id', $field, $field); 286 | } 287 | else { 288 | // If not grouping, use the field as is 289 | $selectFields[$field] = $field; 290 | } 291 | } 292 | } 293 | 294 | // Select fields dynamically 295 | $collection->getSelect()->columns($selectFields); 296 | 297 | if(count($filters) === 0) $filters = $this->getFilters(); 298 | //dd($filters); 299 | 300 | foreach ($filters as $field => $value) { 301 | if (in_array($field, $fields)) { 302 | if (is_array($value)) { 303 | $collection->addFieldToFilter($field, ['in' => $value]); 304 | } else { 305 | if (isset($this->fieldsConfig[$field]['element']) && $this->fieldsConfig[$field]['element'] === 'select') { 306 | $collection->addFieldToFilter($field, ['in' => $value]); 307 | } else { 308 | $collection->addFieldToFilter($field, ['like' => $value . '%']); 309 | } 310 | } 311 | } 312 | } 313 | 314 | // Add GROUP BY clause if needed 315 | if (!empty($this->groupByFields)) { 316 | $collection->getSelect()->group($this->groupByFields); 317 | } 318 | 319 | // Handle pagination 320 | $page = (int) $this->request->getParam('page', 1); 321 | $limit = (int) $this->request->getParam('pageSize', $this->DEFAULT_LIMIT); 322 | 323 | // $page = $offset / $limit + 1; 324 | $collection->setPageSize($limit); 325 | $collection->setCurPage($page); 326 | 327 | // Handle sorting 328 | $sortField = $this->request->getParam('sortField', null); 329 | $sortOrder = $this->request->getParam('sortOrder', null); 330 | if ($sortField && $sortOrder) { 331 | $collection->setOrder($sortField, $sortOrder); 332 | } 333 | 334 | // Log the SQL query (uncomment for debugging) 335 | //dd('SQL Query: ' . $collection->getSelect()->__toString()); 336 | 337 | return $collection->getData(); 338 | } catch (\Exception $e) { 339 | $this->logger->error('Error fetching data: ' . $e->getMessage()); 340 | return [ 341 | 'error' => true, 342 | 'message' => __('Error fetching data: %1', $e->getMessage()) 343 | ]; 344 | } 345 | } 346 | 347 | /** 348 | * Find the position of any needle in a haystack 349 | * 350 | * @param string $haystack The string to search in 351 | * @param array|string $needles The needle(s) to search for 352 | * @return int|false Returns the position of the first needle found, or false if none found 353 | */ 354 | protected function strposx($haystack, $needles) 355 | { 356 | if (!is_array($needles)) { 357 | return strpos($haystack, $needles); 358 | } 359 | 360 | foreach ($needles as $needle) { 361 | $pos = strpos($haystack, $needle); 362 | if ($pos !== false) { 363 | return true; 364 | } 365 | } 366 | 367 | return false; 368 | } 369 | 370 | 371 | /** 372 | * Get the grid data as JSON (for AJAX/Grid.js) 373 | * 374 | * @param array $fields 375 | * @param array $filters 376 | * @return string JSON-encoded data 377 | */ 378 | public function getJsonGridData(array $fields = [], array $filters = [], bool $return = true) 379 | { 380 | $result = []; 381 | 382 | $filters = array_merge($this->request->getParam('filter', []), $filters); 383 | $this->setFilters($filters); 384 | 385 | // If AJAX request, output JSON and exit 386 | if (strtolower($this->request->getParam('data', 'false')) === 'true') { 387 | try { 388 | $result = $this->getGridData($fields, $filters); 389 | 390 | header('Content-Type: application/json'); 391 | echo json_encode($result); 392 | exit; 393 | } catch (\Throwable $e) { 394 | header('Content-Type: application/json'); 395 | echo json_encode(['error' => true, 'message' => $e->getMessage()]); 396 | exit; 397 | } 398 | } 399 | if ($return) { 400 | $result = $this->getGridData($fields, $filters); 401 | } 402 | 403 | return json_encode($result); 404 | } 405 | 406 | /** 407 | * Get the grid data as an array (with processors applied) 408 | * 409 | * @param array $fields 410 | * @param array $filters 411 | * @return array 412 | */ 413 | public function getGridData(array $fields = [], array $filters = []) 414 | { 415 | $timeStart = microtime(true); 416 | $data = $this->getData($fields); 417 | if (isset($data['error'])) { 418 | return json_encode(['error' => true, 'message' => $data['message']]); 419 | } 420 | $timeEnd = microtime(true); 421 | $sqlTime = $timeEnd - $timeStart; 422 | // $item is the row data 423 | $jsonData = array_map(function ($item) use ($fields) { 424 | return array_map(function ($field) use ($item) { 425 | $value = $item[$field] ?? ''; 426 | $processor = $this->getDataProcessor($field); 427 | if ($processor) { 428 | $value = $processor->process($field, $value, $item); 429 | } 430 | // Always run the chain processor last 431 | $value = $this->chainProcessor->process($field, $value, $item); 432 | return $value; 433 | }, $fields); 434 | }, $data); 435 | 436 | $timeStart = microtime(true); 437 | $totalCount = $this->getTotalCount(); 438 | $timeEnd = microtime(true); 439 | $timeCount = $timeEnd - $timeStart; 440 | $result = ['data' => $jsonData, 'total' => $totalCount, 'timeCount' => $timeCount, 'timeSql' => $sqlTime]; 441 | return $result; 442 | } 443 | 444 | /** 445 | * Set the fields for the grid coming from the block .xml configuration 446 | * @param array $fields 447 | * @return $this example [ 448 | * "order_id" => "order_id", 449 | * "customer_group_id" => "customer_group_id", 450 | * "customer_email" => "customer_email", 451 | * "created_at" => "created_at", 452 | * "grand_total" => "grand_total", 453 | * "status" => "status", 454 | * "payment_method" => "payment_method" 455 | * ] 456 | */ 457 | function setFields(array $fields) 458 | { 459 | foreach ($fields as $key => $field) { 460 | // dump($field); 461 | if (is_string($field)) { 462 | $this->fields[$key] = $key; 463 | } else if (is_array($field)) { 464 | foreach ($field as $key2 => $value) { 465 | if ($key2 === 'source_model') { 466 | $this->fields[$key] = $key; 467 | try { 468 | $this->fieldsConfig[$key][$key2] = $this->objectManager->create($value); 469 | } catch (\Exception $e) { 470 | $this->fieldsConfig[$key][$key2] = $e->getMessage(); 471 | $this->logger->error('Error creating data source model: ' . $e->getMessage()); 472 | } 473 | } else { 474 | $this->fields[$key] = $key; 475 | $this->fieldsConfig[$key][$key2] = $value; 476 | } 477 | if ($key2 === 'html') { 478 | $this->fieldsHtml[] = $key; 479 | } 480 | } 481 | } 482 | } 483 | // dd($this->fieldsConfig); 484 | return $this; 485 | } 486 | 487 | /** 488 | * Set fields to group by checking if they exist in the fields array 489 | * 490 | * @param array $fields Array of field names to group by 491 | * @return $this 492 | */ 493 | public function setGroupByFields(array $fields) 494 | { 495 | foreach ($fields as $field) { 496 | if (in_array($field, $this->fields)) { 497 | $this->groupByFields[] = $field; 498 | } 499 | } 500 | 501 | return $this; 502 | } 503 | 504 | /** 505 | * Get the fields that should be rendered as HTML 506 | * @return array 507 | */ 508 | public function getFieldsHtml() 509 | { 510 | return $this->fieldsHtml; 511 | } 512 | 513 | /** 514 | * Set the raw SQL query (for SQL mode) 515 | * @param string|null $sql 516 | * @return $this 517 | */ 518 | function setSqlQuery($sql = null) 519 | { 520 | $this->sqlQuery = $sql; 521 | return $this; 522 | } 523 | 524 | /** 525 | * Execute the raw SQL query (for SQL mode) 526 | * @return array|false 527 | */ 528 | function executeSqlQuery() 529 | { 530 | if (!$this->sqlQuery) { 531 | return false; 532 | } 533 | $connection = $this->getCollection()->getConnection(); 534 | return $connection->fetchAll($this->sqlQuery); 535 | } 536 | 537 | /** 538 | * Set the table name (for SQL mode) 539 | * @param string|null $tableName 540 | * @return $this 541 | */ 542 | function setTableName(null|string $tableName) 543 | { 544 | $this->tableName = $tableName; 545 | return $this; 546 | } 547 | 548 | /** 549 | * Get the table name (for SQL mode) 550 | * @return string|null 551 | */ 552 | function getTableName() 553 | { 554 | return $this->tableName; 555 | } 556 | 557 | /** 558 | * Get the configured fields 559 | * @return array 560 | */ 561 | function getFields() 562 | { 563 | return $this->fields; 564 | } 565 | 566 | function getFieldsConfig() 567 | { 568 | return $this->fieldsConfig; 569 | } 570 | 571 | /** 572 | * Set the field labels (for display) 573 | * @param array $fields 574 | * @return $this 575 | */ 576 | function setFieldsNames(array $fields) 577 | { 578 | foreach ($fields as $key => $field) { 579 | if (is_string($field)) { 580 | $this->fieldsNames[$key] = $field; 581 | } else if (is_array($field)) { 582 | $this->fieldsNames[$key] = $field['label']; 583 | } 584 | } 585 | return $this; 586 | } 587 | 588 | /** 589 | * Get the field labels (for display) 590 | * @return array 591 | */ 592 | function getFieldsNames() 593 | { 594 | return $this->fieldsNames; 595 | } 596 | 597 | /** 598 | * Set the filters for the grid 599 | * @param array $filters 600 | * @return $this 601 | */ 602 | function setFilters(array $filters) 603 | { 604 | $this->filters = $filters; 605 | return $this; 606 | } 607 | 608 | /** 609 | * Get the current filters (merged from request and internal state) 610 | * @return array 611 | */ 612 | function getFilters() 613 | { 614 | // Handle filters from request and internal state 615 | $filters = $this->request->getParam('filter', []); 616 | $filters = array_merge($filters, $this->filters); 617 | $this->setFilters($filters); 618 | return $this->filters; 619 | } 620 | 621 | /** 622 | * Get the collection instance (collection mode) 623 | * @return AbstractCollection|null 624 | */ 625 | public function getCollection() 626 | { 627 | if (!$this->collection) { 628 | try { 629 | if ($this->collectionClass !== null) { 630 | if (is_string($this->collectionClass)) { 631 | $this->collection = $this->objectManager->create($this->collectionClass); 632 | } else if (is_object($this->collectionClass)) { 633 | $this->collection = $this->collectionClass; 634 | } else { 635 | throw new \Exception('Grid Collection class issue: collectionClass parameter is not a string or object'); 636 | } 637 | } else { 638 | throw new \Exception('Grid Collection class not set: collectionClass parameter is required'); 639 | } 640 | $this->setTableName($this->collection->getMainTable()); 641 | } catch (\Exception $e) { 642 | $this->logger->error('Error creating collection: ' . $e->getMessage()); 643 | } 644 | } 645 | return $this->collection; 646 | } 647 | 648 | /** 649 | * Get cache key for total count 650 | * 651 | * @param string $tableName 652 | * @param array $filters 653 | * @return string 654 | */ 655 | protected function getTotalCountCacheKey($tableName, $filters = []) 656 | { 657 | $filterKey = empty($filters) ? 'no_filters' : md5(json_encode($filters)); 658 | return 'grid_total_count_' . $tableName . '_' . $filterKey; 659 | } 660 | 661 | /** 662 | * Get the total count of records (for pagination) 663 | * 664 | * @param string|null $tableName 665 | * @return int 666 | */ 667 | public function getTotalCount($tableName = null) 668 | { 669 | if ($this->sqlQuery) { 670 | return $this->executeSqlQuery(); 671 | } 672 | 673 | try { 674 | // Get the database connection 675 | $connection = $this->resource->getConnection(); 676 | $tableName = $tableName ?: $this->tableName; 677 | 678 | // Build the query 679 | $select = $connection->select()->from($tableName, 'COUNT(*)'); 680 | 681 | // Apply filters if they exist 682 | if (!empty($this->filters)) { 683 | foreach ($this->filters as $field => $value) { 684 | if (is_array($value)) { 685 | $select->where($field . ' IN (?)', $value); 686 | } else { 687 | $select->where($field . ' LIKE ?', $value . '%'); 688 | } 689 | } 690 | } 691 | 692 | // Add GROUP BY clause if needed 693 | if (!empty($this->groupByFields)) { 694 | $select->group($this->groupByFields); 695 | } 696 | 697 | // Check cache first 698 | $cacheKey = $this->getTotalCountCacheKey($tableName, $this->filters); 699 | $cachedCount = $this->cache->load($cacheKey); 700 | 701 | if ($cachedCount !== false) { 702 | $count = (int) $cachedCount; 703 | // If cached count is > 100, use it 704 | if ($count > 100) { 705 | return $count; 706 | } 707 | } 708 | 709 | // Hard limit to optimise huge table performace 710 | $select->limit(999); 711 | 712 | // Execute the query 713 | $count = (int) $connection->fetchOne($select); 714 | 715 | // Cache only if count is greater than 100 to improve performance 716 | if ($count > 100) { 717 | $this->cache->save( 718 | (string) $count, 719 | $cacheKey, 720 | [self::CACHE_TAG], 721 | self::CACHE_LIFETIME 722 | ); 723 | } 724 | } catch (\Exception $e) { 725 | $this->logger->error('Error getting total count: ' . $e->getMessage()); 726 | throw $e; 727 | } 728 | 729 | return $count; 730 | } 731 | 732 | /** 733 | * Get a request parameter (helper) 734 | * @param string $param 735 | * @param string $default 736 | * @return string 737 | */ 738 | public function getRequestParam($param, $default = '') 739 | { 740 | return $this->request->getParam($param, $default); 741 | } 742 | 743 | /** 744 | * Render the grid as HTML (using the main template) 745 | * @param string $blockName 746 | * @return string 747 | */ 748 | public function getGridHtml() 749 | { 750 | $fields = $this->getFields(); 751 | $jsonGridData = $this->getJsonGridData($fields); 752 | 753 | if (isset($jsonGridData['error']) && $jsonGridData['error']) { 754 | return '
' . $jsonGridData['message'] . '
'; 755 | } 756 | 757 | $gridHtml = $this 758 | ->layout 759 | ->createBlock('Magento\Framework\View\Element\Template') 760 | ->setTemplate('Mage_Grid::grid/grid-component.phtml') 761 | ->setData('jsonGridData', $jsonGridData) 762 | ->setData('fields', $fields) 763 | ->toHtml(); 764 | 765 | return $gridHtml; 766 | } 767 | 768 | /** 769 | * Add a data processor for a specific field (for custom formatting) 770 | * @param string $field 771 | * @param DataProcessorInterface $processor 772 | * @return $this 773 | */ 774 | public function addDataProcessor(string $field, DataProcessorInterface $processor) 775 | { 776 | $this->dataProcessors[$field] = $processor; 777 | return $this; 778 | } 779 | 780 | /** 781 | * Get the data processor for a specific field (if any) 782 | * @param string $field 783 | * @return DataProcessorInterface|null 784 | */ 785 | protected function getDataProcessor(string $field): DataProcessorInterface|null 786 | { 787 | // Get the processor for the field, or default 788 | $processor = $this->dataProcessors[$field] ?? null; 789 | 790 | return $processor; 791 | } 792 | } 793 | -------------------------------------------------------------------------------- /ViewModel/Grid.php: -------------------------------------------------------------------------------- 1 | grid = $grid; 23 | } 24 | 25 | /** 26 | * Get grid instance 27 | * 28 | * @return Grid 29 | */ 30 | public function getGrid(): Grid 31 | { 32 | return $this->grid; 33 | } 34 | } -------------------------------------------------------------------------------- /composer.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "mage/grid", 3 | "description": "Magento 2 custom React GridJS integration ", 4 | "type": "magento2-module", 5 | "license": "MIT", 6 | "authors": [ 7 | { 8 | "name": "Yehor Shytikov", 9 | "email": "egorshitikov@gmail.com" 10 | } 11 | ], 12 | "minimum-stability": "dev", 13 | "autoload": { 14 | "psr-4": { 15 | "Mage\\Grid\\": "" 16 | }, 17 | "files": [ 18 | "registration.php" 19 | ] 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /etc/acl.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /etc/adminhtml/default.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | Mage\Grid\ViewModel\GenericViewModelGrid 9 | none 10 | 11 | ID 12 | 13 | 14 | none 15 | 20 16 | true 17 | 18 | Mage_Grid::grid/additional-html.phtml 19 | 20 | 21 | 22 | 23 | -------------------------------------------------------------------------------- /etc/adminhtml/menu.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 12 | 13 | -------------------------------------------------------------------------------- /etc/adminhtml/routes.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /etc/di.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | Mage\Grid\ViewModel\GenericViewModelGrid 6 | 7 | 8 | 9 | 10 | Mage\Grid\Model\DataProcessor\DefaultProcessor 11 | 12 | Mage\Grid\Model\DataProcessor\ObfuscateProcessor 13 | Mage\Grid\Model\DataProcessor\IdProcessor 14 | Mage\Grid\Model\DataProcessor\ObfuscateProcessor 15 | Mage\Grid\Model\DataProcessor\DateProcessor 16 | Mage\Grid\Model\DataProcessor\DateProcessor 17 | Mage\Grid\Model\DataProcessor\PriceProcessor 18 | Mage\Grid\Model\DataProcessor\PriceProcessor 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | -------------------------------------------------------------------------------- /etc/module.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /etc/webapi.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /registration.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /view/adminhtml/layout/default.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 7 | 8 | 9 | 12 | 13 | Mage\Grid\ViewModel\GenericViewModelGrid 14 | Mage_Grid::grid/additional-html.phtml 15 | none 16 | 17 | ID 18 | 19 | 20 | none 21 | 20 22 | true 23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /view/adminhtml/layout/grid_grid_data.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 8 | 9 | Mage\Grid\ViewModel\GenericViewModelGrid 10 | Magento\Sales\Model\ResourceModel\Order\Grid\Collection 11 | 12 | ID 13 | Increment ID 14 | 15 | Status 16 | select 17 | true 18 | Mage\Grid\Model\Fields\DefaultDataSource 19 | 20 | 21 | Store 22 | true 23 | multiselect 24 | Mage\Grid\Model\Fields\DefaultDataSource 25 | 26 | Customer Email 27 | Created At 28 | 29 | Grand Total 30 | true 31 | 32 | 33 | Payment Method 34 | multiselect 35 | Mage\Grid\Model\Fields\DefaultDataSource 36 | 37 | 38 | 39 | Mage\Grid\Model\DataProcessor\StatusProcessor 40 | 41 | 20 42 | true 43 | 44 | 45 | 46 | 47 | 48 | 49 | -------------------------------------------------------------------------------- /view/adminhtml/layout/grid_grid_index.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 8 | 9 | Mage\Grid\ViewModel\GenericViewModelGrid 10 | Magento\Sales\Model\ResourceModel\Order\Grid\Collection 11 | 12 | 13 | ID 14 | true 15 | 16 | Increment ID 17 | 18 | Status 19 | select 20 | true 21 | Mage\Grid\Model\Fields\DefaultDataSource 22 | 23 | 24 | Store 25 | true 26 | multiselect 27 | Mage\Grid\Model\Fields\DefaultDataSource 28 | 29 | Customer Email 30 | Created At 31 | 32 | Grand Total 33 | true 34 | 35 | 36 | Payment Method 37 | multiselect 38 | Mage\Grid\Model\Fields\DefaultDataSource 39 | 40 | 41 | 42 | Mage\Grid\Model\DataProcessor\StatusProcessor 43 | 44 | 20 45 | true 46 | 47 | 48 | 49 | 50 | -------------------------------------------------------------------------------- /view/adminhtml/layout/mage_grid_grid_index.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | Mage\Grid\ViewModel\GridViewModel 7 | 8 | 9 | 10 | -------------------------------------------------------------------------------- /view/adminhtml/require-config.js: -------------------------------------------------------------------------------- 1 | 2 | var config = { 3 | paths: { 4 | gridjs: 'Mage_Gridjs/js/gridjs', 5 | DataTable: 'Mage_Gridjs/js/DataTable' 6 | //preact: 'Mage_Gridjs/js/preact', 7 | }, 8 | shim: { 9 | slick: { 10 | deps: ['jquery', 'jquery/ui'] 11 | } 12 | } 13 | }; 14 | -------------------------------------------------------------------------------- /view/adminhtml/templates/grid/additional-html.phtml: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /view/adminhtml/templates/grid/grid-component.phtml: -------------------------------------------------------------------------------- 1 | getAditionalHTML()) 12 | // 13 | // For full Grid.js documentation and advanced configuration, see: 14 | // https://gridjs.io/docs 15 | 16 | $fields = array_keys($block->getFieldsNames()); // array: ['id', 'order_number', ...] 17 | $fieldsFull = $block->getFields(); // associative array: ['id' => 'ID', ...] 18 | $block->lazyLoadCollectionClass(); 19 | // $jsonGridData = $block->getGridJsonData(); // JSON-encoded grid data 20 | $fieldsConfig = $block->getFieldsConfig(); 21 | $tableName = $block->getTableName(); 22 | $fieldsNames = $block->getFieldsNames(); 23 | $aditionalHTML = $block->getAditionalHTML(); 24 | $fieldsHtml = $block->getFieldsHtml(); 25 | // Get filter values from request 26 | $filters = $this->getRequest()->getParam('filter', []); 27 | 28 | // Process field configurations and filter values 29 | $processedFields = $block->getProcessedFields($fields, $fieldsConfig, $filters); 30 | 31 | $filterData = [ 32 | 'fields' => $processedFields, 33 | 'filters' => $filters 34 | ]; 35 | // Now start rendering the template 36 | ?> 37 |
38 | 39 | getFiltersHtml($filterData) ?> 40 | 41 | 42 |
43 | 44 | 45 | 46 | 47 | 48 |
49 | 50 | 56 | 57 |
58 |
59 | 60 | 61 | 62 |
63 |
64 | Server SQL Query time: 65 | - 66 |
67 |
68 | Server Count time: 69 | - 70 |
71 |
72 | 73 | 307 | 308 | 345 | 346 | 347 | getAditionalHTML() ?> 348 | -------------------------------------------------------------------------------- /view/adminhtml/templates/gridjs.phtml: -------------------------------------------------------------------------------- 1 | 4 | 5 |

Adobe and Partners stop business with Russia and Belarus!

6 |

Grid JS

7 | 8 | 15 | 16 | 43 |
44 | 45 |

DataTable

46 | 136 | 137 | 138 |
139 | -------------------------------------------------------------------------------- /view/adminhtml/web/css/DataTable.css: -------------------------------------------------------------------------------- 1 | @charset "UTF-8"; 2 | :root { 3 | --dt-row-selected: 13, 110, 253; 4 | --dt-row-selected-text: 255, 255, 255; 5 | --dt-row-selected-link: 9, 10, 11; 6 | --dt-row-stripe: 0, 0, 0; 7 | --dt-row-hover: 0, 0, 0; 8 | --dt-column-ordering: 0, 0, 0; 9 | --dt-html-background: white; 10 | } 11 | :root.dark { 12 | --dt-html-background: rgb(33, 37, 41); 13 | } 14 | 15 | table.dataTable td.dt-control { 16 | text-align: center; 17 | cursor: pointer; 18 | } 19 | table.dataTable td.dt-control:before { 20 | display: inline-block; 21 | box-sizing: border-box; 22 | content: ""; 23 | border-top: 5px solid transparent; 24 | border-left: 10px solid rgba(0, 0, 0, 0.5); 25 | border-bottom: 5px solid transparent; 26 | border-right: 0px solid transparent; 27 | } 28 | table.dataTable tr.dt-hasChild td.dt-control:before { 29 | border-top: 10px solid rgba(0, 0, 0, 0.5); 30 | border-left: 5px solid transparent; 31 | border-bottom: 0px solid transparent; 32 | border-right: 5px solid transparent; 33 | } 34 | 35 | html.dark table.dataTable td.dt-control:before, 36 | :root[data-bs-theme=dark] table.dataTable td.dt-control:before { 37 | border-left-color: rgba(255, 255, 255, 0.5); 38 | } 39 | html.dark table.dataTable tr.dt-hasChild td.dt-control:before, 40 | :root[data-bs-theme=dark] table.dataTable tr.dt-hasChild td.dt-control:before { 41 | border-top-color: rgba(255, 255, 255, 0.5); 42 | border-left-color: transparent; 43 | } 44 | 45 | div.dt-scroll-body thead tr, 46 | div.dt-scroll-body tfoot tr { 47 | height: 0; 48 | } 49 | div.dt-scroll-body thead tr th, div.dt-scroll-body thead tr td, 50 | div.dt-scroll-body tfoot tr th, 51 | div.dt-scroll-body tfoot tr td { 52 | height: 0 !important; 53 | padding-top: 0px !important; 54 | padding-bottom: 0px !important; 55 | border-top-width: 0px !important; 56 | border-bottom-width: 0px !important; 57 | } 58 | div.dt-scroll-body thead tr th div.dt-scroll-sizing, div.dt-scroll-body thead tr td div.dt-scroll-sizing, 59 | div.dt-scroll-body tfoot tr th div.dt-scroll-sizing, 60 | div.dt-scroll-body tfoot tr td div.dt-scroll-sizing { 61 | height: 0 !important; 62 | overflow: hidden !important; 63 | } 64 | 65 | table.dataTable thead > tr > th:active, 66 | table.dataTable thead > tr > td:active { 67 | outline: none; 68 | } 69 | table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, 70 | table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:before, 71 | table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before { 72 | position: absolute; 73 | display: block; 74 | bottom: 50%; 75 | content: "▲"; 76 | content: "▲"/""; 77 | } 78 | table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, 79 | table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:after, 80 | table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { 81 | position: absolute; 82 | display: block; 83 | top: 50%; 84 | content: "▼"; 85 | content: "▼"/""; 86 | } 87 | table.dataTable thead > tr > th.dt-orderable-asc, table.dataTable thead > tr > th.dt-orderable-desc, table.dataTable thead > tr > th.dt-ordering-asc, table.dataTable thead > tr > th.dt-ordering-desc, 88 | table.dataTable thead > tr > td.dt-orderable-asc, 89 | table.dataTable thead > tr > td.dt-orderable-desc, 90 | table.dataTable thead > tr > td.dt-ordering-asc, 91 | table.dataTable thead > tr > td.dt-ordering-desc { 92 | position: relative; 93 | padding-right: 30px; 94 | } 95 | table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order, 96 | table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order, 97 | table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order, 98 | table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order, 99 | table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order { 100 | position: absolute; 101 | right: 12px; 102 | top: 0; 103 | bottom: 0; 104 | width: 12px; 105 | } 106 | table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-orderable-asc span.dt-column-order:after, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:before, table.dataTable thead > tr > th.dt-orderable-desc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:after, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, 107 | table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:before, 108 | table.dataTable thead > tr > td.dt-orderable-asc span.dt-column-order:after, 109 | table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:before, 110 | table.dataTable thead > tr > td.dt-orderable-desc span.dt-column-order:after, 111 | table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before, 112 | table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:after, 113 | table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:before, 114 | table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { 115 | left: 0; 116 | opacity: 0.125; 117 | line-height: 9px; 118 | font-size: 0.8em; 119 | } 120 | table.dataTable thead > tr > th.dt-orderable-asc, table.dataTable thead > tr > th.dt-orderable-desc, 121 | table.dataTable thead > tr > td.dt-orderable-asc, 122 | table.dataTable thead > tr > td.dt-orderable-desc { 123 | cursor: pointer; 124 | } 125 | table.dataTable thead > tr > th.dt-orderable-asc:hover, table.dataTable thead > tr > th.dt-orderable-desc:hover, 126 | table.dataTable thead > tr > td.dt-orderable-asc:hover, 127 | table.dataTable thead > tr > td.dt-orderable-desc:hover { 128 | outline: 2px solid rgba(0, 0, 0, 0.05); 129 | outline-offset: -2px; 130 | } 131 | table.dataTable thead > tr > th.dt-ordering-asc span.dt-column-order:before, table.dataTable thead > tr > th.dt-ordering-desc span.dt-column-order:after, 132 | table.dataTable thead > tr > td.dt-ordering-asc span.dt-column-order:before, 133 | table.dataTable thead > tr > td.dt-ordering-desc span.dt-column-order:after { 134 | opacity: 0.6; 135 | } 136 | table.dataTable thead > tr > th.sorting_desc_disabled span.dt-column-order:after, table.dataTable thead > tr > th.sorting_asc_disabled span.dt-column-order:before, 137 | table.dataTable thead > tr > td.sorting_desc_disabled span.dt-column-order:after, 138 | table.dataTable thead > tr > td.sorting_asc_disabled span.dt-column-order:before { 139 | display: none; 140 | } 141 | table.dataTable thead > tr > th:active, 142 | table.dataTable thead > tr > td:active { 143 | outline: none; 144 | } 145 | 146 | div.dt-scroll-body > table.dataTable > thead > tr > th, 147 | div.dt-scroll-body > table.dataTable > thead > tr > td { 148 | overflow: hidden; 149 | } 150 | 151 | :root.dark table.dataTable thead > tr > th.dt-orderable-asc:hover, :root.dark table.dataTable thead > tr > th.dt-orderable-desc:hover, 152 | :root.dark table.dataTable thead > tr > td.dt-orderable-asc:hover, 153 | :root.dark table.dataTable thead > tr > td.dt-orderable-desc:hover, 154 | :root[data-bs-theme=dark] table.dataTable thead > tr > th.dt-orderable-asc:hover, 155 | :root[data-bs-theme=dark] table.dataTable thead > tr > th.dt-orderable-desc:hover, 156 | :root[data-bs-theme=dark] table.dataTable thead > tr > td.dt-orderable-asc:hover, 157 | :root[data-bs-theme=dark] table.dataTable thead > tr > td.dt-orderable-desc:hover { 158 | outline: 2px solid rgba(255, 255, 255, 0.05); 159 | } 160 | 161 | div.dt-processing { 162 | position: absolute; 163 | top: 50%; 164 | left: 50%; 165 | width: 200px; 166 | margin-left: -100px; 167 | margin-top: -22px; 168 | text-align: center; 169 | padding: 2px; 170 | z-index: 10; 171 | } 172 | div.dt-processing > div:last-child { 173 | position: relative; 174 | width: 80px; 175 | height: 15px; 176 | margin: 1em auto; 177 | } 178 | div.dt-processing > div:last-child > div { 179 | position: absolute; 180 | top: 0; 181 | width: 13px; 182 | height: 13px; 183 | border-radius: 50%; 184 | background: rgb(13, 110, 253); 185 | background: rgb(var(--dt-row-selected)); 186 | animation-timing-function: cubic-bezier(0, 1, 1, 0); 187 | } 188 | div.dt-processing > div:last-child > div:nth-child(1) { 189 | left: 8px; 190 | animation: datatables-loader-1 0.6s infinite; 191 | } 192 | div.dt-processing > div:last-child > div:nth-child(2) { 193 | left: 8px; 194 | animation: datatables-loader-2 0.6s infinite; 195 | } 196 | div.dt-processing > div:last-child > div:nth-child(3) { 197 | left: 32px; 198 | animation: datatables-loader-2 0.6s infinite; 199 | } 200 | div.dt-processing > div:last-child > div:nth-child(4) { 201 | left: 56px; 202 | animation: datatables-loader-3 0.6s infinite; 203 | } 204 | 205 | @keyframes datatables-loader-1 { 206 | 0% { 207 | transform: scale(0); 208 | } 209 | 100% { 210 | transform: scale(1); 211 | } 212 | } 213 | @keyframes datatables-loader-3 { 214 | 0% { 215 | transform: scale(1); 216 | } 217 | 100% { 218 | transform: scale(0); 219 | } 220 | } 221 | @keyframes datatables-loader-2 { 222 | 0% { 223 | transform: translate(0, 0); 224 | } 225 | 100% { 226 | transform: translate(24px, 0); 227 | } 228 | } 229 | table.dataTable.nowrap th, table.dataTable.nowrap td { 230 | white-space: nowrap; 231 | } 232 | table.dataTable th, 233 | table.dataTable td { 234 | box-sizing: border-box; 235 | } 236 | table.dataTable th.dt-left, 237 | table.dataTable td.dt-left { 238 | text-align: left; 239 | } 240 | table.dataTable th.dt-center, 241 | table.dataTable td.dt-center { 242 | text-align: center; 243 | } 244 | table.dataTable th.dt-right, 245 | table.dataTable td.dt-right { 246 | text-align: right; 247 | } 248 | table.dataTable th.dt-justify, 249 | table.dataTable td.dt-justify { 250 | text-align: justify; 251 | } 252 | table.dataTable th.dt-nowrap, 253 | table.dataTable td.dt-nowrap { 254 | white-space: nowrap; 255 | } 256 | table.dataTable th.dt-empty, 257 | table.dataTable td.dt-empty { 258 | text-align: center; 259 | vertical-align: top; 260 | } 261 | table.dataTable th.dt-type-numeric, table.dataTable th.dt-type-date, 262 | table.dataTable td.dt-type-numeric, 263 | table.dataTable td.dt-type-date { 264 | text-align: right; 265 | } 266 | table.dataTable thead th, 267 | table.dataTable thead td, 268 | table.dataTable tfoot th, 269 | table.dataTable tfoot td { 270 | text-align: left; 271 | } 272 | table.dataTable thead th.dt-head-left, 273 | table.dataTable thead td.dt-head-left, 274 | table.dataTable tfoot th.dt-head-left, 275 | table.dataTable tfoot td.dt-head-left { 276 | text-align: left; 277 | } 278 | table.dataTable thead th.dt-head-center, 279 | table.dataTable thead td.dt-head-center, 280 | table.dataTable tfoot th.dt-head-center, 281 | table.dataTable tfoot td.dt-head-center { 282 | text-align: center; 283 | } 284 | table.dataTable thead th.dt-head-right, 285 | table.dataTable thead td.dt-head-right, 286 | table.dataTable tfoot th.dt-head-right, 287 | table.dataTable tfoot td.dt-head-right { 288 | text-align: right; 289 | } 290 | table.dataTable thead th.dt-head-justify, 291 | table.dataTable thead td.dt-head-justify, 292 | table.dataTable tfoot th.dt-head-justify, 293 | table.dataTable tfoot td.dt-head-justify { 294 | text-align: justify; 295 | } 296 | table.dataTable thead th.dt-head-nowrap, 297 | table.dataTable thead td.dt-head-nowrap, 298 | table.dataTable tfoot th.dt-head-nowrap, 299 | table.dataTable tfoot td.dt-head-nowrap { 300 | white-space: nowrap; 301 | } 302 | table.dataTable tbody th.dt-body-left, 303 | table.dataTable tbody td.dt-body-left { 304 | text-align: left; 305 | } 306 | table.dataTable tbody th.dt-body-center, 307 | table.dataTable tbody td.dt-body-center { 308 | text-align: center; 309 | } 310 | table.dataTable tbody th.dt-body-right, 311 | table.dataTable tbody td.dt-body-right { 312 | text-align: right; 313 | } 314 | table.dataTable tbody th.dt-body-justify, 315 | table.dataTable tbody td.dt-body-justify { 316 | text-align: justify; 317 | } 318 | table.dataTable tbody th.dt-body-nowrap, 319 | table.dataTable tbody td.dt-body-nowrap { 320 | white-space: nowrap; 321 | } 322 | 323 | /* 324 | * Table styles 325 | */ 326 | table.dataTable { 327 | width: 100%; 328 | margin: 0 auto; 329 | border-spacing: 0; 330 | /* 331 | * Header and footer styles 332 | */ 333 | /* 334 | * Body styles 335 | */ 336 | } 337 | table.dataTable thead th, 338 | table.dataTable tfoot th { 339 | font-weight: bold; 340 | } 341 | table.dataTable > thead > tr > th, 342 | table.dataTable > thead > tr > td { 343 | padding: 10px; 344 | border-bottom: 1px solid rgba(0, 0, 0, 0.3); 345 | } 346 | table.dataTable > thead > tr > th:active, 347 | table.dataTable > thead > tr > td:active { 348 | outline: none; 349 | } 350 | table.dataTable > tfoot > tr > th, 351 | table.dataTable > tfoot > tr > td { 352 | border-top: 1px solid rgba(0, 0, 0, 0.3); 353 | padding: 10px 10px 6px 10px; 354 | } 355 | table.dataTable > tbody > tr { 356 | background-color: transparent; 357 | } 358 | table.dataTable > tbody > tr:first-child > * { 359 | border-top: none; 360 | } 361 | table.dataTable > tbody > tr:last-child > * { 362 | border-bottom: none; 363 | } 364 | table.dataTable > tbody > tr.selected > * { 365 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.9); 366 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.9); 367 | color: rgb(255, 255, 255); 368 | color: rgb(var(--dt-row-selected-text)); 369 | } 370 | table.dataTable > tbody > tr.selected a { 371 | color: rgb(9, 10, 11); 372 | color: rgb(var(--dt-row-selected-link)); 373 | } 374 | table.dataTable > tbody > tr > th, 375 | table.dataTable > tbody > tr > td { 376 | padding: 8px 10px; 377 | } 378 | table.dataTable.row-border > tbody > tr > *, table.dataTable.display > tbody > tr > * { 379 | border-top: 1px solid rgba(0, 0, 0, 0.15); 380 | } 381 | table.dataTable.row-border > tbody > tr:first-child > *, table.dataTable.display > tbody > tr:first-child > * { 382 | border-top: none; 383 | } 384 | table.dataTable.row-border > tbody > tr.selected + tr.selected > td, table.dataTable.display > tbody > tr.selected + tr.selected > td { 385 | border-top-color: rgba(13, 110, 253, 0.65); 386 | border-top-color: rgba(var(--dt-row-selected), 0.65); 387 | } 388 | table.dataTable.cell-border > tbody > tr > * { 389 | border-top: 1px solid rgba(0, 0, 0, 0.15); 390 | border-right: 1px solid rgba(0, 0, 0, 0.15); 391 | } 392 | table.dataTable.cell-border > tbody > tr > *:first-child { 393 | border-left: 1px solid rgba(0, 0, 0, 0.15); 394 | } 395 | table.dataTable.cell-border > tbody > tr:first-child > * { 396 | border-top: 1px solid rgba(0, 0, 0, 0.3); 397 | } 398 | table.dataTable.stripe > tbody > tr:nth-child(odd) > *, table.dataTable.display > tbody > tr:nth-child(odd) > * { 399 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.023); 400 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-stripe), 0.023); 401 | } 402 | table.dataTable.stripe > tbody > tr:nth-child(odd).selected > *, table.dataTable.display > tbody > tr:nth-child(odd).selected > * { 403 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.923); 404 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.923); 405 | } 406 | table.dataTable.hover > tbody > tr:hover > *, table.dataTable.display > tbody > tr:hover > * { 407 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.035); 408 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.035); 409 | } 410 | table.dataTable.hover > tbody > tr.selected:hover > *, table.dataTable.display > tbody > tr.selected:hover > * { 411 | box-shadow: inset 0 0 0 9999px #0d6efd !important; 412 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 1) !important; 413 | } 414 | table.dataTable.order-column > tbody tr > .sorting_1, 415 | table.dataTable.order-column > tbody tr > .sorting_2, 416 | table.dataTable.order-column > tbody tr > .sorting_3, table.dataTable.display > tbody tr > .sorting_1, 417 | table.dataTable.display > tbody tr > .sorting_2, 418 | table.dataTable.display > tbody tr > .sorting_3 { 419 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.019); 420 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019); 421 | } 422 | table.dataTable.order-column > tbody tr.selected > .sorting_1, 423 | table.dataTable.order-column > tbody tr.selected > .sorting_2, 424 | table.dataTable.order-column > tbody tr.selected > .sorting_3, table.dataTable.display > tbody tr.selected > .sorting_1, 425 | table.dataTable.display > tbody tr.selected > .sorting_2, 426 | table.dataTable.display > tbody tr.selected > .sorting_3 { 427 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.919); 428 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919); 429 | } 430 | table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_1, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_1 { 431 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.054); 432 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.054); 433 | } 434 | table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_2, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_2 { 435 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.047); 436 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.047); 437 | } 438 | table.dataTable.display > tbody > tr:nth-child(odd) > .sorting_3, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd) > .sorting_3 { 439 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.039); 440 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.039); 441 | } 442 | table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_1, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_1 { 443 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.954); 444 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.954); 445 | } 446 | table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_2, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_2 { 447 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.947); 448 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.947); 449 | } 450 | table.dataTable.display > tbody > tr:nth-child(odd).selected > .sorting_3, table.dataTable.order-column.stripe > tbody > tr:nth-child(odd).selected > .sorting_3 { 451 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.939); 452 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.939); 453 | } 454 | table.dataTable.display > tbody > tr.even > .sorting_1, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_1 { 455 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.019); 456 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.019); 457 | } 458 | table.dataTable.display > tbody > tr.even > .sorting_2, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_2 { 459 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.011); 460 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.011); 461 | } 462 | table.dataTable.display > tbody > tr.even > .sorting_3, table.dataTable.order-column.stripe > tbody > tr.even > .sorting_3 { 463 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.003); 464 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-column-ordering), 0.003); 465 | } 466 | table.dataTable.display > tbody > tr.even.selected > .sorting_1, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_1 { 467 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.919); 468 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.919); 469 | } 470 | table.dataTable.display > tbody > tr.even.selected > .sorting_2, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_2 { 471 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.911); 472 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.911); 473 | } 474 | table.dataTable.display > tbody > tr.even.selected > .sorting_3, table.dataTable.order-column.stripe > tbody > tr.even.selected > .sorting_3 { 475 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.903); 476 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.903); 477 | } 478 | table.dataTable.display tbody tr:hover > .sorting_1, table.dataTable.order-column.hover tbody tr:hover > .sorting_1 { 479 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.082); 480 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.082); 481 | } 482 | table.dataTable.display tbody tr:hover > .sorting_2, table.dataTable.order-column.hover tbody tr:hover > .sorting_2 { 483 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.074); 484 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.074); 485 | } 486 | table.dataTable.display tbody tr:hover > .sorting_3, table.dataTable.order-column.hover tbody tr:hover > .sorting_3 { 487 | box-shadow: inset 0 0 0 9999px rgba(0, 0, 0, 0.062); 488 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-hover), 0.062); 489 | } 490 | table.dataTable.display tbody tr:hover.selected > .sorting_1, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_1 { 491 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.982); 492 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.982); 493 | } 494 | table.dataTable.display tbody tr:hover.selected > .sorting_2, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_2 { 495 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.974); 496 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.974); 497 | } 498 | table.dataTable.display tbody tr:hover.selected > .sorting_3, table.dataTable.order-column.hover tbody tr:hover.selected > .sorting_3 { 499 | box-shadow: inset 0 0 0 9999px rgba(13, 110, 253, 0.962); 500 | box-shadow: inset 0 0 0 9999px rgba(var(--dt-row-selected), 0.962); 501 | } 502 | table.dataTable.compact thead th, 503 | table.dataTable.compact thead td, 504 | table.dataTable.compact tfoot th, 505 | table.dataTable.compact tfoot td, 506 | table.dataTable.compact tbody th, 507 | table.dataTable.compact tbody td { 508 | padding: 4px; 509 | } 510 | 511 | /* 512 | * Control feature layout 513 | */ 514 | div.dt-container { 515 | position: relative; 516 | clear: both; 517 | } 518 | div.dt-container div.dt-layout-row { 519 | display: table; 520 | clear: both; 521 | width: 100%; 522 | } 523 | div.dt-container div.dt-layout-row.dt-layout-table { 524 | display: block; 525 | } 526 | div.dt-container div.dt-layout-row.dt-layout-table div.dt-layout-cell { 527 | display: block; 528 | } 529 | div.dt-container div.dt-layout-cell { 530 | display: table-cell; 531 | vertical-align: middle; 532 | padding: 5px 0; 533 | } 534 | div.dt-container div.dt-layout-cell.dt-full { 535 | text-align: center; 536 | } 537 | div.dt-container div.dt-layout-cell.dt-start { 538 | text-align: left; 539 | } 540 | div.dt-container div.dt-layout-cell.dt-end { 541 | text-align: right; 542 | } 543 | div.dt-container div.dt-layout-cell:empty { 544 | display: none; 545 | } 546 | div.dt-container .dt-search input { 547 | border: 1px solid #aaa; 548 | border-radius: 3px; 549 | padding: 5px; 550 | background-color: transparent; 551 | color: inherit; 552 | margin-left: 3px; 553 | } 554 | div.dt-container .dt-input { 555 | border: 1px solid #aaa; 556 | border-radius: 3px; 557 | padding: 5px; 558 | background-color: transparent; 559 | color: inherit; 560 | } 561 | div.dt-container select.dt-input { 562 | padding: 4px; 563 | } 564 | div.dt-container .dt-paging .dt-paging-button { 565 | box-sizing: border-box; 566 | display: inline-block; 567 | min-width: 1.5em; 568 | padding: 0.5em 1em; 569 | margin-left: 2px; 570 | text-align: center; 571 | text-decoration: none !important; 572 | cursor: pointer; 573 | color: inherit !important; 574 | border: 1px solid transparent; 575 | border-radius: 2px; 576 | background: transparent; 577 | } 578 | div.dt-container .dt-paging .dt-paging-button.current, div.dt-container .dt-paging .dt-paging-button.current:hover { 579 | color: inherit !important; 580 | border: 1px solid rgba(0, 0, 0, 0.3); 581 | background-color: rgba(0, 0, 0, 0.05); 582 | background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(230, 230, 230, 0.05)), color-stop(100%, rgba(0, 0, 0, 0.05))); /* Chrome,Safari4+ */ 583 | background: -webkit-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* Chrome10+,Safari5.1+ */ 584 | background: -moz-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* FF3.6+ */ 585 | background: -ms-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* IE10+ */ 586 | background: -o-linear-gradient(top, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* Opera 11.10+ */ 587 | background: linear-gradient(to bottom, rgba(230, 230, 230, 0.05) 0%, rgba(0, 0, 0, 0.05) 100%); /* W3C */ 588 | } 589 | div.dt-container .dt-paging .dt-paging-button.disabled, div.dt-container .dt-paging .dt-paging-button.disabled:hover, div.dt-container .dt-paging .dt-paging-button.disabled:active { 590 | cursor: default; 591 | color: rgba(0, 0, 0, 0.5) !important; 592 | border: 1px solid transparent; 593 | background: transparent; 594 | box-shadow: none; 595 | } 596 | div.dt-container .dt-paging .dt-paging-button:hover { 597 | color: white !important; 598 | border: 1px solid #111; 599 | background-color: #111; 600 | background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #585858), color-stop(100%, #111)); /* Chrome,Safari4+ */ 601 | background: -webkit-linear-gradient(top, #585858 0%, #111 100%); /* Chrome10+,Safari5.1+ */ 602 | background: -moz-linear-gradient(top, #585858 0%, #111 100%); /* FF3.6+ */ 603 | background: -ms-linear-gradient(top, #585858 0%, #111 100%); /* IE10+ */ 604 | background: -o-linear-gradient(top, #585858 0%, #111 100%); /* Opera 11.10+ */ 605 | background: linear-gradient(to bottom, #585858 0%, #111 100%); /* W3C */ 606 | } 607 | div.dt-container .dt-paging .dt-paging-button:active { 608 | outline: none; 609 | background-color: #0c0c0c; 610 | background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #2b2b2b), color-stop(100%, #0c0c0c)); /* Chrome,Safari4+ */ 611 | background: -webkit-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* Chrome10+,Safari5.1+ */ 612 | background: -moz-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* FF3.6+ */ 613 | background: -ms-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* IE10+ */ 614 | background: -o-linear-gradient(top, #2b2b2b 0%, #0c0c0c 100%); /* Opera 11.10+ */ 615 | background: linear-gradient(to bottom, #2b2b2b 0%, #0c0c0c 100%); /* W3C */ 616 | box-shadow: inset 0 0 3px #111; 617 | } 618 | div.dt-container .dt-paging .ellipsis { 619 | padding: 0 1em; 620 | } 621 | div.dt-container .dt-length, 622 | div.dt-container .dt-search, 623 | div.dt-container .dt-info, 624 | div.dt-container .dt-processing, 625 | div.dt-container .dt-paging { 626 | color: inherit; 627 | } 628 | div.dt-container .dataTables_scroll { 629 | clear: both; 630 | } 631 | div.dt-container .dataTables_scroll div.dt-scroll-body { 632 | -webkit-overflow-scrolling: touch; 633 | } 634 | div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > th, div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > td, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > th, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > td { 635 | vertical-align: middle; 636 | } 637 | div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > th > div.dataTables_sizing, 638 | div.dt-container .dataTables_scroll div.dt-scroll-body > table > thead > tr > td > div.dataTables_sizing, div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > th > div.dataTables_sizing, 639 | div.dt-container .dataTables_scroll div.dt-scroll-body > table > tbody > tr > td > div.dataTables_sizing { 640 | height: 0; 641 | overflow: hidden; 642 | margin: 0 !important; 643 | padding: 0 !important; 644 | } 645 | div.dt-container.dt-empty-footer tbody > tr:last-child > * { 646 | border-bottom: 1px solid rgba(0, 0, 0, 0.3); 647 | } 648 | div.dt-container.dt-empty-footer .dt-scroll-body { 649 | border-bottom: 1px solid rgba(0, 0, 0, 0.3); 650 | } 651 | div.dt-container.dt-empty-footer .dt-scroll-body tbody > tr:last-child > * { 652 | border-bottom: none; 653 | } 654 | 655 | @media screen and (max-width: 767px) { 656 | div.dt-container div.dt-layout-row { 657 | display: block; 658 | } 659 | div.dt-container div.dt-layout-cell { 660 | display: block; 661 | } 662 | div.dt-container div.dt-layout-cell.dt-full, div.dt-container div.dt-layout-cell.dt-start, div.dt-container div.dt-layout-cell.dt-end { 663 | text-align: center; 664 | } 665 | } 666 | @media screen and (max-width: 640px) { 667 | .dt-container .dt-length, 668 | .dt-container .dt-search { 669 | float: none; 670 | text-align: center; 671 | } 672 | .dt-container .dt-search { 673 | margin-top: 0.5em; 674 | } 675 | } 676 | html.dark { 677 | --dt-row-hover: 255, 255, 255; 678 | --dt-row-stripe: 255, 255, 255; 679 | --dt-column-ordering: 255, 255, 255; 680 | } 681 | html.dark table.dataTable > thead > tr > th, 682 | html.dark table.dataTable > thead > tr > td { 683 | border-bottom: 1px solid rgb(89, 91, 94); 684 | } 685 | html.dark table.dataTable > thead > tr > th:active, 686 | html.dark table.dataTable > thead > tr > td:active { 687 | outline: none; 688 | } 689 | html.dark table.dataTable > tfoot > tr > th, 690 | html.dark table.dataTable > tfoot > tr > td { 691 | border-top: 1px solid rgb(89, 91, 94); 692 | } 693 | html.dark table.dataTable.row-border > tbody > tr > *, html.dark table.dataTable.display > tbody > tr > * { 694 | border-top: 1px solid rgb(64, 67, 70); 695 | } 696 | html.dark table.dataTable.row-border > tbody > tr:first-child > *, html.dark table.dataTable.display > tbody > tr:first-child > * { 697 | border-top: none; 698 | } 699 | html.dark table.dataTable.row-border > tbody > tr.selected + tr.selected > td, html.dark table.dataTable.display > tbody > tr.selected + tr.selected > td { 700 | border-top-color: rgba(13, 110, 253, 0.65); 701 | border-top-color: rgba(var(--dt-row-selected), 0.65); 702 | } 703 | html.dark table.dataTable.cell-border > tbody > tr > th, 704 | html.dark table.dataTable.cell-border > tbody > tr > td { 705 | border-top: 1px solid rgb(64, 67, 70); 706 | border-right: 1px solid rgb(64, 67, 70); 707 | } 708 | html.dark table.dataTable.cell-border > tbody > tr > th:first-child, 709 | html.dark table.dataTable.cell-border > tbody > tr > td:first-child { 710 | border-left: 1px solid rgb(64, 67, 70); 711 | } 712 | html.dark .dt-container.dt-empty-footer table.dataTable { 713 | border-bottom: 1px solid rgb(89, 91, 94); 714 | } 715 | html.dark .dt-container .dt-search input, 716 | html.dark .dt-container .dt-length select { 717 | border: 1px solid rgba(255, 255, 255, 0.2); 718 | background-color: var(--dt-html-background); 719 | } 720 | html.dark .dt-container .dt-paging .dt-paging-button.current, html.dark .dt-container .dt-paging .dt-paging-button.current:hover { 721 | border: 1px solid rgb(89, 91, 94); 722 | background: rgba(255, 255, 255, 0.15); 723 | } 724 | html.dark .dt-container .dt-paging .dt-paging-button.disabled, html.dark .dt-container .dt-paging .dt-paging-button.disabled:hover, html.dark .dt-container .dt-paging .dt-paging-button.disabled:active { 725 | color: #666 !important; 726 | } 727 | html.dark .dt-container .dt-paging .dt-paging-button:hover { 728 | border: 1px solid rgb(53, 53, 53); 729 | background: rgb(53, 53, 53); 730 | } 731 | html.dark .dt-container .dt-paging .dt-paging-button:active { 732 | background: #3a3a3a; 733 | } 734 | 735 | /* 736 | * Overrides for RTL support 737 | */ 738 | *[dir=rtl] table.dataTable thead th, 739 | *[dir=rtl] table.dataTable thead td, 740 | *[dir=rtl] table.dataTable tfoot th, 741 | *[dir=rtl] table.dataTable tfoot td { 742 | text-align: right; 743 | } 744 | *[dir=rtl] table.dataTable th.dt-type-numeric, *[dir=rtl] table.dataTable th.dt-type-date, 745 | *[dir=rtl] table.dataTable td.dt-type-numeric, 746 | *[dir=rtl] table.dataTable td.dt-type-date { 747 | text-align: left; 748 | } 749 | *[dir=rtl] div.dt-container div.dt-layout-cell.dt-start { 750 | text-align: right; 751 | } 752 | *[dir=rtl] div.dt-container div.dt-layout-cell.dt-end { 753 | text-align: left; 754 | } 755 | *[dir=rtl] div.dt-container div.dt-search input { 756 | margin: 0 3px 0 0; 757 | } -------------------------------------------------------------------------------- /view/adminhtml/web/css/gridjs.css: -------------------------------------------------------------------------------- 1 | .gridjs-footer button,.gridjs-head button{background-color:transparent;background-image:none;border:none;cursor:pointer;margin:0;outline:none;padding:0}.gridjs-temp{position:relative}.gridjs-head{margin-bottom:5px;padding:5px 1px;width:100%}.gridjs-head:after{clear:both;content:"";display:block}.gridjs-head:empty{border:none;padding:0}.gridjs-container{color:#000;display:inline-block;overflow:hidden;padding:2px;position:relative;z-index:0}.gridjs-footer{background-color:#fff;border-bottom-width:1px;border-color:#e5e7eb;border-radius:0 0 8px 8px;border-top:1px solid #e5e7eb;box-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px 0 rgba(0,0,0,.26);display:block;padding:12px 24px;position:relative;width:100%;z-index:5}.gridjs-footer:empty{border:none;padding:0}input.gridjs-input{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:#fff;border:1px solid #d2d6dc;border-radius:5px;font-size:14px;line-height:1.45;outline:none;padding:10px 13px}input.gridjs-input:focus{border-color:#9bc2f7;box-shadow:0 0 0 3px rgba(149,189,243,.5)}.gridjs-pagination{color:#3d4044}.gridjs-pagination:after{clear:both;content:"";display:block}.gridjs-pagination .gridjs-summary{float:left;margin-top:5px}.gridjs-pagination .gridjs-pages{float:right}.gridjs-pagination .gridjs-pages button{background-color:#fff;border:1px solid #d2d6dc;border-right:none;outline:none;padding:5px 14px;-webkit-user-select:none;-moz-user-select:none;user-select:none}.gridjs-pagination .gridjs-pages button:focus{border-right:1px solid #d2d6dc;box-shadow:0 0 0 2px rgba(149,189,243,.5);margin-right:-1px;position:relative}.gridjs-pagination .gridjs-pages button:hover{background-color:#f7f7f7;color:#3c4257;outline:none}.gridjs-pagination .gridjs-pages button:disabled,.gridjs-pagination .gridjs-pages button:hover:disabled,.gridjs-pagination .gridjs-pages button[disabled]{background-color:#fff;color:#6b7280;cursor:default}.gridjs-pagination .gridjs-pages button.gridjs-spread{background-color:#fff;box-shadow:none;cursor:default}.gridjs-pagination .gridjs-pages button.gridjs-currentPage{background-color:#f7f7f7;font-weight:700}.gridjs-pagination .gridjs-pages button:last-child{border-bottom-right-radius:6px;border-right:1px solid #d2d6dc;border-top-right-radius:6px}.gridjs-pagination .gridjs-pages button:first-child{border-bottom-left-radius:6px;border-top-left-radius:6px}.gridjs-pagination .gridjs-pages button:last-child:focus{margin-right:0}button.gridjs-sort{background-color:transparent;background-position-x:center;background-repeat:no-repeat;background-size:contain;border:none;cursor:pointer;float:right;height:24px;margin:0;outline:none;padding:0;width:13px}button.gridjs-sort-neutral{background-image:url("");background-position-y:center;opacity:.3}button.gridjs-sort-asc{background-image:url("");background-position-y:35%;background-size:10px}button.gridjs-sort-desc{background-image:url("");background-position-y:65%;background-size:10px}button.gridjs-sort:focus{outline:none}table.gridjs-table{border-collapse:collapse;display:table;margin:0;max-width:100%;overflow:auto;padding:0;table-layout:fixed;text-align:left;width:100%}.gridjs-tbody,td.gridjs-td{background-color:#fff}td.gridjs-td{border:1px solid #e5e7eb;box-sizing:content-box;padding:12px 24px}td.gridjs-td:first-child{border-left:none}td.gridjs-td:last-child{border-right:none}td.gridjs-message{text-align:center}th.gridjs-th{background-color:#f9fafb;border:1px solid #e5e7eb;border-top:none;box-sizing:border-box;color:#6b7280;outline:none;padding:14px 24px;position:relative;-webkit-user-select:none;-moz-user-select:none;user-select:none;vertical-align:middle;white-space:nowrap}th.gridjs-th .gridjs-th-content{float:left;overflow:hidden;text-overflow:ellipsis;width:100%}th.gridjs-th-sort{cursor:pointer}th.gridjs-th-sort .gridjs-th-content{width:calc(100% - 15px)}th.gridjs-th-sort:focus,th.gridjs-th-sort:hover{background-color:#e5e7eb}th.gridjs-th-fixed{box-shadow:0 1px 0 0 #e5e7eb;position:sticky}@supports (-moz-appearance:none){th.gridjs-th-fixed{box-shadow:0 0 0 1px #e5e7eb}}th.gridjs-th:first-child{border-left:none}th.gridjs-th:last-child{border-right:none}.gridjs-tr{border:none}.gridjs-tr-selected td{background-color:#ebf5ff}.gridjs-tr:last-child td{border-bottom:0}.gridjs *,.gridjs :after,.gridjs :before{box-sizing:border-box}.gridjs-wrapper{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;border-color:#e5e7eb;border-radius:8px 8px 0 0;border-top-width:1px;box-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px 0 rgba(0,0,0,.26);display:block;overflow:auto;position:relative;width:100%;z-index:1}.gridjs-wrapper:nth-last-of-type(2){border-bottom-width:1px;border-radius:8px}.gridjs-search{float:left}.gridjs-search-input{width:250px}.gridjs-loading-bar{background-color:#fff;opacity:.5;z-index:10}.gridjs-loading-bar,.gridjs-loading-bar:after{bottom:0;left:0;position:absolute;right:0;top:0}.gridjs-loading-bar:after{animation:shimmer 2s infinite;background-image:linear-gradient(90deg,hsla(0,0%,80%,0),hsla(0,0%,80%,.2) 20%,hsla(0,0%,80%,.5) 60%,hsla(0,0%,80%,0));content:"";transform:translateX(-100%)}@keyframes shimmer{to{transform:translateX(100%)}}.gridjs-td .gridjs-checkbox{cursor:pointer;display:block;margin:auto}.gridjs-resizable{bottom:0;position:absolute;right:0;top:0;width:5px}.gridjs-resizable:hover{background-color:#9bc2f7;cursor:ew-resize} 2 | 3 | .gridjs-footer button,.gridjs-head button{background-color:transparent;background-image:none;border:none;cursor:pointer;margin:0;outline:none;padding:0}.gridjs-temp{position:relative}.gridjs-head{margin-bottom:5px;padding:5px 1px;width:100%}.gridjs-head:after{clear:both;content:"";display:block}.gridjs-head:empty{border:none;padding:0}.gridjs-container{color:#000;display:inline-block;overflow:hidden;padding:2px;position:relative;z-index:0}.gridjs-footer{background-color:#fff;border-bottom-width:1px;border-color:#e5e7eb;border-radius:0 0 8px 8px;border-top:1px solid #e5e7eb;box-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px 0 rgba(0,0,0,.26);display:block;padding:12px 24px;position:relative;width:100%;z-index:5}.gridjs-footer:empty{border:none;padding:0}input.gridjs-input{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:#fff;border:1px solid #d2d6dc;border-radius:5px;font-size:14px;line-height:1.45;outline:none;padding:10px 13px}input.gridjs-input:focus{border-color:#9bc2f7;box-shadow:0 0 0 3px rgba(149,189,243,.5)}.gridjs-pagination{color:#3d4044}.gridjs-pagination:after{clear:both;content:"";display:block}.gridjs-pagination .gridjs-summary{float:left;margin-top:5px}.gridjs-pagination .gridjs-pages{float:right}.gridjs-pagination .gridjs-pages button{background-color:#fff;border:1px solid #d2d6dc;border-right:none;outline:none;padding:5px 14px;-webkit-user-select:none;-moz-user-select:none;user-select:none}.gridjs-pagination .gridjs-pages button:focus{border-right:1px solid #d2d6dc;box-shadow:0 0 0 2px rgba(149,189,243,.5);margin-right:-1px;position:relative}.gridjs-pagination .gridjs-pages button:hover{background-color:#f7f7f7;color:#3c4257;outline:none}.gridjs-pagination .gridjs-pages button:disabled,.gridjs-pagination .gridjs-pages button:hover:disabled,.gridjs-pagination .gridjs-pages button[disabled]{background-color:#fff;color:#6b7280;cursor:default}.gridjs-pagination .gridjs-pages button.gridjs-spread{background-color:#fff;box-shadow:none;cursor:default}.gridjs-pagination .gridjs-pages button.gridjs-currentPage{background-color:#f7f7f7;font-weight:700}.gridjs-pagination .gridjs-pages button:last-child{border-bottom-right-radius:6px;border-right:1px solid #d2d6dc;border-top-right-radius:6px}.gridjs-pagination .gridjs-pages button:first-child{border-bottom-left-radius:6px;border-top-left-radius:6px}.gridjs-pagination .gridjs-pages button:last-child:focus{margin-right:0}button.gridjs-sort{background-color:transparent;background-position-x:center;background-repeat:no-repeat;background-size:contain;border:none;cursor:pointer;float:right;height:24px;margin:0;outline:none;padding:0;width:13px}button.gridjs-sort-neutral{background-image:url("");background-position-y:center;opacity:.3}button.gridjs-sort-asc{background-image:url("");background-position-y:35%;background-size:10px}button.gridjs-sort-desc{background-image:url("");background-position-y:65%;background-size:10px}button.gridjs-sort:focus{outline:none}table.gridjs-table{border-collapse:collapse;display:table;margin:0;max-width:100%;overflow:auto;padding:0;table-layout:fixed;text-align:left;width:100%}.gridjs-tbody,td.gridjs-td{background-color:#fff}td.gridjs-td{border:1px solid #e5e7eb;box-sizing:content-box;padding:12px 24px}td.gridjs-td:first-child{border-left:none}td.gridjs-td:last-child{border-right:none}td.gridjs-message{text-align:center}th.gridjs-th{background-color:#f9fafb;border:1px solid #e5e7eb;border-top:none;box-sizing:border-box;color:#6b7280;outline:none;padding:14px 24px;position:relative;-webkit-user-select:none;-moz-user-select:none;user-select:none;vertical-align:middle;white-space:nowrap}th.gridjs-th .gridjs-th-content{float:left;overflow:hidden;text-overflow:ellipsis;width:100%}th.gridjs-th-sort{cursor:pointer}th.gridjs-th-sort .gridjs-th-content{width:calc(100% - 15px)}th.gridjs-th-sort:focus,th.gridjs-th-sort:hover{background-color:#e5e7eb}th.gridjs-th-fixed{box-shadow:0 1px 0 0 #e5e7eb;position:sticky}@supports (-moz-appearance:none){th.gridjs-th-fixed{box-shadow:0 0 0 1px #e5e7eb}}th.gridjs-th:first-child{border-left:none}th.gridjs-th:last-child{border-right:none}.gridjs-tr{border:none}.gridjs-tr-selected td{background-color:#ebf5ff}.gridjs-tr:last-child td{border-bottom:0}.gridjs *,.gridjs :after,.gridjs :before{box-sizing:border-box}.gridjs-wrapper{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;border-color:#e5e7eb;border-radius:8px 8px 0 0;border-top-width:1px;box-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px 0 rgba(0,0,0,.26);display:block;overflow:auto;position:relative;width:100%;z-index:1}.gridjs-wrapper:nth-last-of-type(2){border-bottom-width:1px;border-radius:8px}.gridjs-search{float:left}.gridjs-search-input{width:250px}.gridjs-loading-bar{background-color:#fff;opacity:.5;z-index:10}.gridjs-loading-bar,.gridjs-loading-bar:after{bottom:0;left:0;position:absolute;right:0;top:0}.gridjs-loading-bar:after{animation:shimmer 2s infinite;background-image:linear-gradient(90deg,hsla(0,0%,80%,0),hsla(0,0%,80%,.2) 20%,hsla(0,0%,80%,.5) 60%,hsla(0,0%,80%,0));content:"";transform:translateX(-100%)}@keyframes shimmer{to{transform:translateX(100%)}}.gridjs-td .gridjs-checkbox{cursor:pointer;display:block;margin:auto}.gridjs-resizable{bottom:0;position:absolute;right:0;top:0;width:5px}.gridjs-resizable:hover{background-color:#9bc2f7;cursor:ew-resize} 4 | 5 | 6 | .f32 .flag{display:inline-block;height:32px;width:32px;vertical-align:text-top;line-height:32px;background:url(../images/flags32.png) no-repeat;} 7 | .f32 ._African_Union{background-position:0 -32px;} 8 | .f32 ._Arab_League{background-position:0 -64px;} 9 | .f32 ._ASEAN{background-position:0 -96px;} 10 | .f32 ._CARICOM{background-position:0 -128px;} 11 | .f32 ._CIS{background-position:0 -160px;} 12 | .f32 ._Commonwealth{background-position:0 -192px;} 13 | .f32 ._England{background-position:0 -224px;} 14 | .f32 ._European_Union, .f32 .eu{background-position:0 -256px;} 15 | .f32 ._Islamic_Conference{background-position:0 -288px;} 16 | .f32 ._Kosovo{background-position:0 -320px;} 17 | .f32 ._NATO{background-position:0 -352px;} 18 | .f32 ._Northern_Cyprus{background-position:0 -384px;} 19 | .f32 ._Northern_Ireland{background-position:0 -416px;} 20 | .f32 ._Olimpic_Movement{background-position:0 -448px;} 21 | .f32 ._OPEC{background-position:0 -480px;} 22 | .f32 ._Red_Cross{background-position:0 -512px;} 23 | .f32 ._Scotland{background-position:0 -544px;} 24 | .f32 ._Somaliland{background-position:0 -576px;} 25 | .f32 ._Tibet{background-position:0 -608px;} 26 | .f32 ._United_Nations{background-position:0 -640px;} 27 | .f32 ._Wales{background-position:0 -672px;} 28 | .f32 .ad, .f32 .and {background-position: 0 -704px;} 29 | .f32 .ae, .f32 .are {background-position: 0 -736px;} 30 | .f32 .af, .f32 .afg {background-position: 0 -768px;} 31 | .f32 .ag, .f32 .atg {background-position: 0 -800px;} 32 | .f32 .ai, .f32 .aia {background-position: 0 -832px;} 33 | .f32 .al, .f32 .alb {background-position: 0 -864px;} 34 | .f32 .am, .f32 .arm {background-position: 0 -896px;} 35 | .f32 .ao, .f32 .ago {background-position: 0 -928px;} 36 | .f32 .aq, .f32 .ata {background-position: 0 -960px;} 37 | .f32 .ar, .f32 .arg {background-position: 0 -992px;} 38 | .f32 .as, .f32 .asm {background-position: 0 -1024px;} 39 | .f32 .at, .f32 .aut {background-position: 0 -1056px;} 40 | .f32 .au, .f32 .aus {background-position: 0 -1088px;} 41 | .f32 .aw, .f32 .abw {background-position: 0 -1120px;} 42 | .f32 .ax, .f32 .ala {background-position: 0 -1152px;} 43 | .f32 .az, .f32 .aze {background-position: 0 -1184px;} 44 | .f32 .ba, .f32 .bih {background-position: 0 -1216px;} 45 | .f32 .bb, .f32 .brb {background-position: 0 -1248px;} 46 | .f32 .bd, .f32 .bgd {background-position: 0 -1280px;} 47 | .f32 .be, .f32 .bel {background-position: 0 -1312px;} 48 | .f32 .bf, .f32 .bfa {background-position: 0 -1344px;} 49 | .f32 .bg, .f32 .bgr {background-position: 0 -1376px;} 50 | .f32 .bh, .f32 .bhr {background-position: 0 -1408px;} 51 | .f32 .bi, .f32 .bdi {background-position: 0 -1440px;} 52 | .f32 .bj, .f32 .ben {background-position: 0 -1472px;} 53 | .f32 .bm, .f32 .bmu {background-position: 0 -1504px;} 54 | .f32 .bn, .f32 .brn {background-position: 0 -1536px;} 55 | .f32 .bo, .f32 .bol {background-position: 0 -1568px;} 56 | .f32 .br, .f32 .bra {background-position: 0 -1600px;} 57 | .f32 .bs, .f32 .bhs {background-position: 0 -1632px;} 58 | .f32 .bt, .f32 .btn {background-position: 0 -1664px;} 59 | .f32 .bw, .f32 .bwa {background-position: 0 -1696px;} 60 | .f32 .by, .f32 .blr {background-position: 0 -1728px;} 61 | .f32 .bz, .f32 .blz {background-position: 0 -1760px;} 62 | .f32 .ca, .f32 .can {background-position: 0 -1792px;} 63 | .f32 .cd, .f32 .cod {background-position: 0 -1824px;} 64 | .f32 .cf, .f32 .rca {background-position: 0 -1856px;} 65 | .f32 .cg, .f32 .cog {background-position: 0 -1888px;} 66 | .f32 .ch, .f32 .che {background-position: 0 -1920px;} 67 | .f32 .ci, .f32 .civ {background-position: 0 -1952px;} 68 | .f32 .ck, .f32 .cok {background-position: 0 -1984px;} 69 | .f32 .cl, .f32 .chl {background-position: 0 -2016px;} 70 | .f32 .cm, .f32 .cmr {background-position: 0 -2048px;} 71 | .f32 .cn, .f32 .chn {background-position: 0 -2080px;} 72 | .f32 .co, .f32 .col {background-position: 0 -2112px;} 73 | .f32 .cr, .f32 .cri {background-position: 0 -2144px;} 74 | .f32 .cu, .f32 .cub {background-position: 0 -2176px;} 75 | .f32 .cv, .f32 .cpv {background-position: 0 -2208px;} 76 | .f32 .cy, .f32 .cyp {background-position: 0 -2240px;} 77 | .f32 .cz, .f32 .cze {background-position: 0 -2272px;} 78 | .f32 .de, .f32 .deu {background-position: 0 -2304px;} 79 | .f32 .dj, .f32 .dji {background-position: 0 -2336px;} 80 | .f32 .dk, .f32 .dnk {background-position: 0 -2368px;} 81 | .f32 .dm, .f32 .dma {background-position: 0 -2400px;} 82 | .f32 .do, .f32 .dom {background-position: 0 -2432px;} 83 | .f32 .dz, .f32 .dza {background-position: 0 -2464px;} 84 | .f32 .ec, .f32 .ecu {background-position: 0 -2496px;} 85 | .f32 .ee, .f32 .est {background-position: 0 -2528px;} 86 | .f32 .eg, .f32 .egy {background-position: 0 -2560px;} 87 | .f32 .eh, .f32 .esh {background-position: 0 -2592px;} 88 | .f32 .er, .f32 .eri {background-position: 0 -2624px;} 89 | .f32 .es, .f32 .esp {background-position: 0 -2656px;} 90 | .f32 .et, .f32 .eth {background-position: 0 -2688px;} 91 | .f32 .fi, .f32 .fin {background-position: 0 -2720px;} 92 | .f32 .fj, .f32 .fji {background-position: 0 -2752px;} 93 | .f32 .fm, .f32 .fsm {background-position: 0 -2784px;} 94 | .f32 .fo, .f32 .fro {background-position: 0 -2816px;} 95 | .f32 .fr, .f32 .fra {background-position: 0 -2848px;} 96 | .f32 .bl, .f32 .blm {background-position: 0 -2848px;} 97 | .f32 .cp, .f32 .cpt {background-position: 0 -2848px;} 98 | .f32 .mf, .f32 .maf {background-position: 0 -2848px;} 99 | .f32 .yt, .f32 .myt {background-position: 0 -2848px;} 100 | .f32 .ga, .f32 .gab {background-position: 0 -2880px;} 101 | .f32 .gb, .f32 .gbr {background-position: 0 -2912px;} 102 | .f32 .sh, .f32 .shn {background-position: 0 -2912px;} 103 | .f32 .gd, .f32 .grd {background-position: 0 -2944px;} 104 | .f32 .ge, .f32 .geo {background-position: 0 -2976px;} 105 | .f32 .gg, .f32 .ggy {background-position: 0 -3008px;} 106 | .f32 .gh, .f32 .gha {background-position: 0 -3040px;} 107 | .f32 .gi, .f32 .gib {background-position: 0 -3072px;} 108 | .f32 .gl, .f32 .grl {background-position: 0 -3104px;} 109 | .f32 .gm, .f32 .gmb {background-position: 0 -3136px;} 110 | .f32 .gn, .f32 .gin {background-position: 0 -3168px;} 111 | .f32 .gp, .f32 .glp {background-position: 0 -3200px;} 112 | .f32 .gq, .f32 .gnq {background-position: 0 -3232px;} 113 | .f32 .gr, .f32 .grc {background-position: 0 -3264px;} 114 | .f32 .gt, .f32 .gtm {background-position: 0 -3296px;} 115 | .f32 .gu, .f32 .gum {background-position: 0 -3328px;} 116 | .f32 .gw, .f32 .gnb {background-position: 0 -3360px;} 117 | .f32 .gy, .f32 .guy {background-position: 0 -3392px;} 118 | .f32 .hk, .f32 .hkg {background-position: 0 -3424px;} 119 | .f32 .hn, .f32 .hnd {background-position: 0 -3456px;} 120 | .f32 .hr, .f32 .hrv {background-position: 0 -3488px;} 121 | .f32 .ht, .f32 .hti {background-position: 0 -3520px;} 122 | .f32 .hu, .f32 .hun {background-position: 0 -3552px;} 123 | .f32 .id, .f32 .idn {background-position: 0 -3584px;} 124 | .f32 .mc, .f32 .mco {background-position: 0 -3584px;} 125 | .f32 .ie {background-position: 0 -3616px;} 126 | .f32 .il, .f32 .isr {background-position: 0 -3648px;} 127 | .f32 .im, .f32 .imn {background-position: 0 -3680px;} 128 | .f32 .in, .f32 .ind {background-position: 0 -3712px;} 129 | .f32 .iq, .f32 .irq {background-position: 0 -3744px;} 130 | .f32 .ir, .f32 .irn {background-position: 0 -3776px;} 131 | .f32 .is, .f32 .isl {background-position: 0 -3808px;} 132 | .f32 .it, .f32 .ita {background-position: 0 -3840px;} 133 | .f32 .je, .f32 .jey {background-position: 0 -3872px;} 134 | .f32 .jm, .f32 .jam {background-position: 0 -3904px;} 135 | .f32 .jo, .f32 .jor {background-position: 0 -3936px;} 136 | .f32 .jp, .f32 .jpn {background-position: 0 -3968px;} 137 | .f32 .ke, .f32 .ken {background-position: 0 -4000px;} 138 | .f32 .kg, .f32 .kgz {background-position: 0 -4032px;} 139 | .f32 .kh, .f32 .khm {background-position: 0 -4064px;} 140 | .f32 .ki, .f32 .kir {background-position: 0 -4096px;} 141 | .f32 .km, .f32 .com {background-position: 0 -4128px;} 142 | .f32 .kn, .f32 .kna {background-position: 0 -4160px;} 143 | .f32 .kp, .f32 .prk {background-position: 0 -4192px;} 144 | .f32 .kr, .f32 .kor {background-position: 0 -4224px;} 145 | .f32 .kw, .f32 .kwt {background-position: 0 -4256px;} 146 | .f32 .ky, .f32 .cym {background-position: 0 -4288px;} 147 | .f32 .kz, .f32 .kaz {background-position: 0 -4320px;} 148 | .f32 .la, .f32 .lao {background-position: 0 -4352px;} 149 | .f32 .lb, .f32 .lbn {background-position: 0 -4384px;} 150 | .f32 .lc, .f32 .lca {background-position: 0 -4416px;} 151 | .f32 .li, .f32 .lie {background-position: 0 -4448px;} 152 | .f32 .lk, .f32 .lka {background-position: 0 -4480px;} 153 | .f32 .lr, .f32 .lbr {background-position: 0 -4512px;} 154 | .f32 .ls, .f32 .lso {background-position: 0 -4544px;} 155 | .f32 .lt, .f32 .ltu {background-position: 0 -4576px;} 156 | .f32 .lu, .f32 .lux {background-position: 0 -4608px;} 157 | .f32 .lv, .f32 .lva {background-position: 0 -4640px;} 158 | .f32 .ly, .f32 .lby {background-position: 0 -4672px;} 159 | .f32 .ma, .f32 .mar {background-position: 0 -4704px;} 160 | .f32 .md, .f32 .mda {background-position: 0 -4736px;} 161 | .f32 .me, .f32 .mne {background-position: 0 -4768px;} 162 | .f32 .mg, .f32 .mdg {background-position: 0 -4800px;} 163 | .f32 .mh, .f32 .mhl {background-position: 0 -4832px;} 164 | .f32 .mk, .f32 .mkd {background-position: 0 -4864px;} 165 | .f32 .ml, .f32 .mli {background-position: 0 -4896px;} 166 | .f32 .mm, .f32 .mmr {background-position: 0 -4928px;} 167 | .f32 .mn, .f32 .mng {background-position: 0 -4960px;} 168 | .f32 .mo, .f32 .mac {background-position: 0 -4992px;} 169 | .f32 .mq, .f32 .mtq {background-position: 0 -5024px;} 170 | .f32 .mr, .f32 .mrt {background-position: 0 -5056px;} 171 | .f32 .ms, .f32 .msr {background-position: 0 -5088px;} 172 | .f32 .mt, .f32 .mlt {background-position: 0 -5120px;} 173 | .f32 .mu, .f32 .mus {background-position: 0 -5152px;} 174 | .f32 .mv, .f32 .mdv {background-position: 0 -5184px;} 175 | .f32 .mw, .f32 .mwi {background-position: 0 -5216px;} 176 | .f32 .mx, .f32 .mex {background-position: 0 -5248px;} 177 | .f32 .my, .f32 .mys {background-position: 0 -5280px;} 178 | .f32 .mz, .f32 .moz {background-position: 0 -5312px;} 179 | .f32 .na, .f32 .nam {background-position: 0 -5344px;} 180 | .f32 .nc, .f32 .ncl {background-position: 0 -5376px;} 181 | .f32 .ne, .f32 .ner {background-position: 0 -5408px;} 182 | .f32 .ng, .f32 .nga {background-position: 0 -5440px;} 183 | .f32 .ni, .f32 .nic {background-position: 0 -5472px;} 184 | .f32 .nl, .f32 .nld {background-position: 0 -5504px;} 185 | .f32 .nl{background-position:0 -5504px;} .f32 .bq{background-position:0 -5504px;}.f32 .no{background-position:0 -5536px;} .f32 .bv, .f32 .nq, .f32 .sj{background-position:0 -5536px;} 186 | .f32 .bq, .f32 .bes {background-position: 0 -5504px;} 187 | .f32 .no, .f32 .nor {background-position: 0 -5536px;} 188 | .f32 .bv, .f32 .bvt {background-position: 0 -5536px;} 189 | .f32 .nq, .f32 .atn {background-position: 0 -5536px;} 190 | .f32 .sj, .f32 .sjm {background-position: 0 -5536px;} 191 | .f32 .np, .f32 .npl {background-position: 0 -5568px;} 192 | .f32 .nr, .f32 .nru {background-position: 0 -5600px;} 193 | .f32 .nz, .f32 .nzl {background-position: 0 -5632px;} 194 | .f32 .om, .f32 .omn {background-position: 0 -5664px;} 195 | .f32 .pa, .f32 .pan {background-position: 0 -5696px;} 196 | .f32 .pe, .f32 .per {background-position: 0 -5728px;} 197 | .f32 .pf, .f32 .pyf {background-position: 0 -5760px;} 198 | .f32 .pg, .f32 .png {background-position: 0 -5792px;} 199 | .f32 .ph, .f32 .phl {background-position: 0 -5824px;} 200 | .f32 .pk, .f32 .pak {background-position: 0 -5856px;} 201 | .f32 .pl, .f32 .pol {background-position: 0 -5888px;} 202 | .f32 .pr, .f32 .pri {background-position: 0 -5920px;} 203 | .f32 .ps, .f32 .pse {background-position: 0 -5952px;} 204 | .f32 .pt, .f32 .prt {background-position: 0 -5984px;} 205 | .f32 .pw, .f32 .plw {background-position: 0 -6016px;} 206 | .f32 .py, .f32 .pry {background-position: 0 -6048px;} 207 | .f32 .qa, .f32 .qat {background-position: 0 -6080px;} 208 | .f32 .re, .f32 .reu {background-position: 0 -6112px;} 209 | .f32 .ro, .f32 .rou {background-position: 0 -6144px;} 210 | .f32 .rs, .f32 .srb {background-position: 0 -6176px;} 211 | .f32 .ru, .f32 .rus {background-position: 0 -6208px;} 212 | .f32 .rw, .f32 .rwa {background-position: 0 -6240px;} 213 | .f32 .sa, .f32 .sau {background-position: 0 -6272px;} 214 | .f32 .sb, .f32 .slb {background-position: 0 -6304px;} 215 | .f32 .sc, .f32 .syc {background-position: 0 -6336px;} 216 | .f32 .sd, .f32 .sdn {background-position: 0 -6368px;} 217 | .f32 .se, .f32 .swe {background-position: 0 -6400px;} 218 | .f32 .sg, .f32 .sgp {background-position: 0 -6432px;} 219 | .f32 .si, .f32 .svn {background-position: 0 -6464px;} 220 | .f32 .sk, .f32 .svk {background-position: 0 -6496px;} 221 | .f32 .sl, .f32 .sle {background-position: 0 -6528px;} 222 | .f32 .sm, .f32 .smr {background-position: 0 -6560px;} 223 | .f32 .sn, .f32 .sen {background-position: 0 -6592px;} 224 | .f32 .so, .f32 .som {background-position: 0 -6624px;} 225 | .f32 .sr, .f32 .sur {background-position: 0 -6656px;} 226 | .f32 .st, .f32 .stp {background-position: 0 -6688px;} 227 | .f32 .sv, .f32 .slv {background-position: 0 -6720px;} 228 | .f32 .sy, .f32 .syr {background-position: 0 -6752px;} 229 | .f32 .sz, .f32 .swz {background-position: 0 -6784px;} 230 | .f32 .tc, .f32 .tca {background-position: 0 -6816px;} 231 | .f32 .td, .f32 .tcd {background-position: 0 -6848px;} 232 | .f32 .tg, .f32 .tgo {background-position: 0 -6880px;} 233 | .f32 .th, .f32 .tha {background-position: 0 -6912px;} 234 | .f32 .tj, .f32 .tjk {background-position: 0 -6944px;} 235 | .f32 .tl, .f32 .tls {background-position: 0 -6976px;} 236 | .f32 .tm, .f32 .tkm {background-position: 0 -7008px;} 237 | .f32 .tn, .f32 .tun {background-position: 0 -7040px;} 238 | .f32 .to, .f32 .ton {background-position: 0 -7072px;} 239 | .f32 .tr, .f32 .tur {background-position: 0 -7104px;} 240 | .f32 .tt, .f32 .tto {background-position: 0 -7136px;} 241 | .f32 .tv, .f32 .tuv {background-position: 0 -7168px;} 242 | .f32 .tw, .f32 .twn {background-position: 0 -7200px;} 243 | .f32 .tz, .f32 .tza {background-position: 0 -7232px;} 244 | .f32 .ua, .f32 .ukr {background-position: 0 -7264px;} 245 | .f32 .ug, .f32 .uga {background-position: 0 -7296px;} 246 | .f32 .us, .f32 .usa {background-position: 0 -7328px;} 247 | .f32 .uy, .f32 .ury {background-position: 0 -7360px;} 248 | .f32 .uz, .f32 .uzb {background-position: 0 -7392px;} 249 | .f32 .va, .f32 .vat {background-position: 0 -7424px;} 250 | .f32 .vc, .f32 .vct {background-position: 0 -7456px;} 251 | .f32 .ve, .f32 .ven {background-position: 0 -7488px;} 252 | .f32 .vg, .f32 .vgb {background-position: 0 -7520px;} 253 | .f32 .vi, .f32 .vir {background-position: 0 -7552px;} 254 | .f32 .vn, .f32 .vnm {background-position: 0 -7584px;} 255 | .f32 .vu, .f32 .vut {background-position: 0 -7616px;} 256 | .f32 .ws, .f32 .wsm {background-position: 0 -7648px;} 257 | .f32 .ye, .f32 .yem {background-position: 0 -7680px;} 258 | .f32 .za, .f32 .zaf {background-position: 0 -7712px;} 259 | .f32 .zm, .f32 .zmb {background-position: 0 -7744px;} 260 | .f32 .zw, .f32 .zwe {background-position: 0 -7776px;} 261 | .f32 .sx, .f32 .sxm {background-position: 0 -7808px;} 262 | .f32 .cw, .f32 .cuw {background-position: 0 -7840px;} 263 | .f32 .ss, .f32 .ssd {background-position: 0 -7872px;} 264 | .f32 .nu, .f32 .niu {background-position: 0 -7904px;} 265 | -------------------------------------------------------------------------------- /view/adminhtml/web/images/flag32.png: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /view/adminhtml/web/images/flags32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Genaker/magento_gridjs/b5c1f352cb0ca7116134ab3335d4dff6139e6863/view/adminhtml/web/images/flags32.png -------------------------------------------------------------------------------- /view/adminhtml/web/js/preact.js: -------------------------------------------------------------------------------- 1 | /* esm.sh - esbuild bundle(htm@3.1.1/preact/standalone) es2022 production */ 2 | var N,f,en,D,tn,B,_n,F={},rn=[],yn=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i;function C(e,n){for(var t in n)e[t]=n[t];return e}function on(e){var n=e.parentNode;n&&n.removeChild(e)}function ln(e,n,t){var _,l,r,u={};for(r in n)r=="key"?_=n[r]:r=="ref"?l=n[r]:u[r]=n[r];if(arguments.length>2&&(u.children=arguments.length>3?N.call(arguments,2):t),typeof e=="function"&&e.defaultProps!=null)for(r in e.defaultProps)u[r]===void 0&&(u[r]=e.defaultProps[r]);return U(e,u,_,l,null)}function U(e,n,t,_,l){var r={type:e,props:n,key:t,ref:_,__k:null,__:null,__b:0,__e:null,__d:void 0,__c:null,__h:null,constructor:void 0,__v:l??++en};return f.vnode!=null&&f.vnode(r),r}function W(e){return e.children}function A(e,n){this.props=e,this.context=n}function E(e,n){if(n==null)return e.__?E(e.__,e.__.__k.indexOf(e)+1):null;for(var t;n0?U(i.type,i.props,i.key,null,i.__v):i)!=null){if(i.__=t,i.__b=t.__b+1,(c=y[o])===null||c&&i.key==c.key&&i.type===c.type)y[o]=void 0;else for(m=0;m=t.__.length&&t.__.push({}),t.__[e]}function bn(e){return S=1,Cn(vn,e)}function Cn(e,n,t){var _=P(x++,2);return _.t=e,_.__c||(_.__=[t?t(n):vn(void 0,n),function(l){var r=_.t(_.__[0],l);_.__[0]!==r&&(_.__=[r,_.__[1]],_.__c.setState({}))}],_.__c=g),_.__}function Pn(e,n){var t=P(x++,3);!f.__s&&q(t.__H,n)&&(t.__=e,t.__H=n,g.__H.__h.push(t))}function xn(e,n){var t=P(x++,4);!f.__s&&q(t.__H,n)&&(t.__=e,t.__H=n,g.__h.push(t))}function Dn(e){return S=5,dn(function(){return{current:e}},[])}function wn(e,n,t){S=6,xn(function(){typeof e=="function"?e(n()):e&&(e.current=n())},t==null?t:t.concat(e))}function dn(e,n){var t=P(x++,7);return q(t.__H,n)&&(t.__=e(),t.__H=n,t.__h=e),t.__}function Tn(e,n){return S=8,dn(function(){return e},n)}function Un(e){var n=g.context[e.__c],t=P(x++,9);return t.c=e,n?(t.__==null&&(t.__=!0,n.sub(g)),n.props.value):e.__}function An(e,n){f.useDebugValue&&f.useDebugValue(n?n(e):e)}function Mn(e){var n=P(x++,10),t=bn();return n.__=e,g.componentDidCatch||(g.componentDidCatch=function(_){n.__&&n.__(_),t[1](_)}),[t[0],function(){t[1](void 0)}]}function Hn(){I.forEach(function(e){if(e.__P)try{e.__H.__h.forEach(M),e.__H.__h.forEach(O),e.__H.__h=[]}catch(n){e.__H.__h=[],f.__e(n,e.__v)}}),I=[]}f.__b=function(e){g=null,J&&J(e)},f.__r=function(e){K&&K(e),x=0;var n=(g=e.__c).__H;n&&(n.__h.forEach(M),n.__h.forEach(O),n.__h=[])},f.diffed=function(e){Q&&Q(e);var n=e.__c;n&&n.__H&&n.__H.__h.length&&(I.push(n)!==1&&z===f.requestAnimationFrame||((z=f.requestAnimationFrame)||function(t){var _,l=function(){clearTimeout(r),Z&&cancelAnimationFrame(_),setTimeout(t)},r=setTimeout(l,100);Z&&(_=requestAnimationFrame(l))})(Hn)),g=void 0},f.__c=function(e,n){n.some(function(t){try{t.__h.forEach(M),t.__h=t.__h.filter(function(_){return!_.__||O(_)})}catch(_){n.some(function(l){l.__h&&(l.__h=[])}),n=[],f.__e(_,t.__v)}}),X&&X(e,n)},f.unmount=function(e){Y&&Y(e);var n=e.__c;if(n&&n.__H)try{n.__H.__.forEach(M)}catch(t){f.__e(t,n.__v)}};var Z=typeof requestAnimationFrame=="function";function M(e){var n=g;typeof e.__c=="function"&&e.__c(),g=n}function O(e){var n=g;e.__c=e.__(),g=n}function q(e,n){return!e||e.length!==n.length||n.some(function(t,_){return t!==e[_]})}function vn(e,n){return typeof n=="function"?n(e):n}var mn=function(e,n,t,_){var l;n[0]=0;for(var r=1;r=5&&((u||!c&&r===5)&&(s.push(r,0,u,l),r=6),c&&(s.push(r,c,0,l),r=6)),u=""},o=0;o"?(r=1,u=""):u=_+u[0]:a?_===a?a="":u+=_:_==='"'||_==="'"?a=_:_===">"?(h(),r=1):r&&(_==="="?(r=5,l=u,u=""):_==="/"&&(r<5||t[o][m+1]===">")?(h(),r===3&&(s=s[0]),r=s,(s=s[0]).push(2,0,r),r=0):_===" "||_===" "||_===` 3 | `||_==="\r"?(h(),r=2):u+=_),r===3&&u==="!--"&&(r=4,s=s[0])}return h(),s}(e)),n),arguments,[])).length>1?n:n[0]}.bind(ln);export{A as Component,Sn as createContext,ln as h,Fn as html,En as render,Tn as useCallback,Un as useContext,An as useDebugValue,Pn as useEffect,Mn as useErrorBoundary,wn as useImperativeHandle,xn as useLayoutEffect,dn as useMemo,Cn as useReducer,Dn as useRef,bn as useState}; 4 | //# sourceMappingURL=standalone.js.map --------------------------------------------------------------------------------