mirror of
https://github.com/wintercms/winter.git
synced 2024-06-28 05:33:29 +02:00
318 lines
9.1 KiB
PHP
318 lines
9.1 KiB
PHP
<?php namespace Backend\Widgets;
|
|
|
|
use Lang;
|
|
use Input;
|
|
use Request;
|
|
use Backend\Classes\WidgetBase;
|
|
use October\Rain\Html\Helper as HtmlHelper;
|
|
use SystemException;
|
|
|
|
/**
|
|
* Table Widget.
|
|
*
|
|
* Represents an editable tabular control.
|
|
*
|
|
* @package october\backend
|
|
* @author Alexey Bobkov, Samuel Georges
|
|
*/
|
|
class Table extends WidgetBase
|
|
{
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
protected $defaultAlias = 'table';
|
|
|
|
/**
|
|
* @var array Table columns
|
|
*/
|
|
protected $columns = [];
|
|
|
|
/**
|
|
* @var boolean Show data table header
|
|
*/
|
|
protected $showHeader = true;
|
|
|
|
/**
|
|
* @var Backend\Widgets\Table\DatasourceBase
|
|
*/
|
|
protected $dataSource;
|
|
|
|
/**
|
|
* @var string Field name used for request data.
|
|
*/
|
|
protected $fieldName;
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $recordsKeyFrom;
|
|
|
|
protected $dataSourceAliases = [
|
|
'client' => '\Backend\Widgets\Table\ClientMemoryDataSource',
|
|
'server' => '\Backend\Widgets\Table\ServerEventDataSource'
|
|
];
|
|
|
|
/**
|
|
* Initialize the widget, called by the constructor and free from its parameters.
|
|
*/
|
|
public function init()
|
|
{
|
|
$this->columns = $this->getConfig('columns', []);
|
|
|
|
$this->fieldName = $this->getConfig('fieldName', $this->alias);
|
|
|
|
$this->recordsKeyFrom = $this->getConfig('keyFrom', 'id');
|
|
|
|
$dataSourceClass = $this->getConfig('dataSource');
|
|
if (!strlen($dataSourceClass)) {
|
|
throw new SystemException('The Table widget data source is not specified in the configuration.');
|
|
}
|
|
|
|
if (array_key_exists($dataSourceClass, $this->dataSourceAliases)) {
|
|
$dataSourceClass = $this->dataSourceAliases[$dataSourceClass];
|
|
}
|
|
|
|
if (!class_exists($dataSourceClass)) {
|
|
throw new SystemException(sprintf('The Table widget data source class "%s" is could not be found.', $dataSourceClass));
|
|
}
|
|
|
|
$this->dataSource = new $dataSourceClass($this->recordsKeyFrom);
|
|
|
|
if (Request::method() == 'POST' && $this->isClientDataSource()) {
|
|
if (strpos($this->fieldName, '[') === false) {
|
|
$requestDataField = $this->fieldName . 'TableData';
|
|
} else {
|
|
$requestDataField = $this->fieldName . '[TableData]';
|
|
}
|
|
|
|
// Use dot notation for request data field
|
|
$requestDataField = implode('.', HtmlHelper::nameToArray($requestDataField));
|
|
|
|
if (Request::exists($requestDataField)) {
|
|
// Load data into the client memory data source on POST
|
|
$this->dataSource->purge();
|
|
$this->dataSource->initRecords(Request::input($requestDataField));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the data source object.
|
|
* @return \Backend\Widgets\Table\DataSourceBase
|
|
*/
|
|
public function getDataSource()
|
|
{
|
|
return $this->dataSource;
|
|
}
|
|
|
|
/**
|
|
* Renders the widget.
|
|
*/
|
|
public function render()
|
|
{
|
|
$this->prepareVars();
|
|
return $this->makePartial('table');
|
|
}
|
|
|
|
/**
|
|
* Prepares the view data
|
|
*/
|
|
public function prepareVars()
|
|
{
|
|
$this->vars['columns'] = $this->prepareColumnsArray();
|
|
$this->vars['recordsKeyFrom'] = $this->recordsKeyFrom;
|
|
|
|
$this->vars['recordsPerPage'] = $this->getConfig('recordsPerPage', false) ?: 'false';
|
|
$this->vars['postbackHandlerName'] = $this->getConfig('postbackHandlerName', 'onSave');
|
|
$this->vars['searching'] = $this->getConfig('searching', false);
|
|
$this->vars['adding'] = $this->getConfig('adding', true);
|
|
$this->vars['deleting'] = $this->getConfig('deleting', true);
|
|
$this->vars['toolbar'] = $this->getConfig('toolbar', true);
|
|
$this->vars['height'] = $this->getConfig('height', false) ?: 'false';
|
|
$this->vars['dynamicHeight'] = $this->getConfig('dynamicHeight', false) ?: 'false';
|
|
|
|
$this->vars['btnAddRowLabel'] = Lang::get($this->getConfig('btnAddRowLabel', 'backend::lang.form.insert_row'));
|
|
$this->vars['btnAddRowBelowLabel'] = Lang::get($this->getConfig('btnAddRowBelowLabel', 'backend::lang.form.insert_row_below'));
|
|
$this->vars['btnDeleteRowLabel'] = Lang::get($this->getConfig('btnDeleteRowLabel', 'backend::lang.form.delete_row'));
|
|
|
|
$isClientDataSource = $this->isClientDataSource();
|
|
|
|
$this->vars['clientDataSourceClass'] = $isClientDataSource ? 'client' : 'server';
|
|
$this->vars['data'] = json_encode($isClientDataSource
|
|
? $this->dataSource->getAllRecords()
|
|
: []
|
|
);
|
|
}
|
|
|
|
//
|
|
// Internals
|
|
//
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
protected function loadAssets()
|
|
{
|
|
$this->addCss('css/table.css', 'core');
|
|
$this->addJs('js/build-min.js', 'core');
|
|
}
|
|
|
|
/**
|
|
* Converts the columns associative array to a regular array and translates column headers and drop-down options.
|
|
* Working with regular arrays is much faster in JavaScript.
|
|
* References:
|
|
* - http://www.smashingmagazine.com/2012/11/05/writing-fast-memory-efficient-javascript/
|
|
* - http://jsperf.com/performance-of-array-vs-object/3
|
|
*/
|
|
protected function prepareColumnsArray()
|
|
{
|
|
$result = [];
|
|
|
|
foreach ($this->columns as $key=>$data) {
|
|
$data['key'] = $key;
|
|
|
|
if (isset($data['title']))
|
|
$data['title'] = trans($data['title']);
|
|
|
|
if (isset($data['options'])) {
|
|
foreach ($data['options'] as &$option)
|
|
$option = trans($option);
|
|
}
|
|
|
|
if (isset($data['validation'])) {
|
|
foreach ($data['validation'] as &$validation) {
|
|
if (isset($validation['message'])) {
|
|
$validation['message'] = trans($validation['message']);
|
|
}
|
|
}
|
|
}
|
|
|
|
$result[] = $data;
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
protected function isClientDataSource()
|
|
{
|
|
return $this->dataSource instanceof \Backend\Widgets\Table\ClientMemoryDataSource;
|
|
}
|
|
|
|
//
|
|
// Event handlers
|
|
//
|
|
|
|
public function onServerGetRecords()
|
|
{
|
|
// Disable asset broadcasting
|
|
$this->controller->flushAssets();
|
|
|
|
if ($this->isClientDataSource()) {
|
|
throw new SystemException('The Table widget is not configured to use the server data source.');
|
|
}
|
|
|
|
$count = post('count');
|
|
|
|
// Oddly, JS may pass false as a string (@todo)
|
|
if ($count === 'false') {
|
|
$count = false;
|
|
}
|
|
|
|
return [
|
|
'records' => $this->dataSource->getRecords(post('offset'), $count),
|
|
'count' => $this->dataSource->getCount()
|
|
];
|
|
}
|
|
|
|
public function onServerSearchRecords()
|
|
{
|
|
// Disable asset broadcasting
|
|
$this->controller->flushAssets();
|
|
|
|
if ($this->isClientDataSource()) {
|
|
throw new SystemException('The Table widget is not configured to use the server data source.');
|
|
}
|
|
|
|
$count = post('count');
|
|
|
|
// Oddly, JS may pass false as a string (@todo)
|
|
if ($count === 'false') {
|
|
$count = false;
|
|
}
|
|
|
|
return [
|
|
'records' => $this->dataSource->searchRecords(post('query'), post('offset'), $count),
|
|
'count' => $this->dataSource->getCount()
|
|
];
|
|
}
|
|
|
|
public function onServerCreateRecord()
|
|
{
|
|
if ($this->isClientDataSource()) {
|
|
throw new SystemException('The Table widget is not configured to use the server data source.');
|
|
}
|
|
|
|
$this->dataSource->createRecord(
|
|
post('recordData'),
|
|
post('placement'),
|
|
post('relativeToKey')
|
|
);
|
|
|
|
return $this->onServerGetRecords();
|
|
}
|
|
|
|
public function onServerUpdateRecord()
|
|
{
|
|
if ($this->isClientDataSource()) {
|
|
throw new SystemException('The Table widget is not configured to use the server data source.');
|
|
}
|
|
|
|
$this->dataSource->updateRecord(post('key'), post('recordData'));
|
|
}
|
|
|
|
public function onServerDeleteRecord()
|
|
{
|
|
if ($this->isClientDataSource()) {
|
|
throw new SystemException('The Table widget is not configured to use the server data source.');
|
|
}
|
|
|
|
$this->dataSource->deleteRecord(post('key'));
|
|
|
|
return $this->onServerGetRecords();
|
|
}
|
|
|
|
public function onGetDropdownOptions()
|
|
{
|
|
$columnName = Input::get('column');
|
|
$rowData = Input::get('rowData');
|
|
|
|
$eventResults = $this->fireEvent('table.getDropdownOptions', [$columnName, $rowData]);
|
|
|
|
$options = [];
|
|
if (count($eventResults)) {
|
|
$options = $eventResults[0];
|
|
}
|
|
|
|
return [
|
|
'options' => $options
|
|
];
|
|
}
|
|
|
|
public function onGetAutocompleteOptions()
|
|
{
|
|
$columnName = Input::get('column');
|
|
$rowData = Input::get('rowData');
|
|
|
|
$eventResults = $this->fireEvent('table.getAutocompleteOptions', [$columnName, $rowData]);
|
|
|
|
$options = [];
|
|
if (count($eventResults)) {
|
|
$options = $eventResults[0];
|
|
}
|
|
|
|
return [
|
|
'options' => $options
|
|
];
|
|
}
|
|
}
|