Automatic Fields detection + EditForm generation

Component now finds the Image relation and all 'editable fields' itself.
Also gets the edit Forms from the record's getCMSFields()
+ other little fixes
This commit is contained in:
colymba 2012-07-18 02:12:10 +03:00
parent f010fda4a6
commit 51b9518492
4 changed files with 186 additions and 110 deletions

View File

@ -7,7 +7,6 @@ class GridFieldBulkImageUpload implements GridField_HTMLProvider, GridField_URLH
* @var String * @var String
*/ */
protected $recordImageFieldName; protected $recordImageFieldName;
//protected $labelFieldName;
/** /**
* Target record editablez fields * Target record editablez fields
@ -20,29 +19,45 @@ class GridFieldBulkImageUpload implements GridField_HTMLProvider, GridField_URLH
* @param String $imageField * @param String $imageField
* @param String/Array $editableFields * @param String/Array $editableFields
*/ */
public function __construct($imageField, $editableFields) public function __construct($imageField = null, $editableFields = null)
{ {
$this->imageFieldName = $imageField; $this->setRecordImageField($imageField);
if ( !is_array($editableFields) ) $editableFields = array($editableFields); if ( !is_array($editableFields) && $editableFields != null ) $editableFields = array($editableFields);
$this->recordEditableFields = $editableFields; $this->setRecordEditableFields($editableFields);
} }
public function setRecordImageField($field) /**
*
* @param String $field
*/
function setRecordImageField($field)
{ {
$this->imageFieldName = $field; $this->recordImageFieldName = $field;
} }
public function setRecordEditableFields($fields) /**
*
* @param Array $fields
*/
function setRecordEditableFields($fields)
{ {
$this->recordEditableFields = $fields; $this->recordEditableFields = $fields;
} }
/**
*
* @return type
*/
public function getRecordImageField() public function getRecordImageField()
{ {
return $this->imageFieldName; return $this->recordImageFieldName;
} }
/**
*
* @return type
*/
public function getRecordEditableFields() public function getRecordEditableFields()
{ {
return $this->recordEditableFields; return $this->recordEditableFields;

View File

@ -22,13 +22,16 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
/** /**
* *
* @var Array * @var FieldList
*/ */
protected $createdRecords; protected $recordCMSFieldList;
/**
*
*/
static $url_handlers = array( static $url_handlers = array(
'$Action!' => '$Action' '$Action!' => '$Action'/*,
'' => 'uploadForm'*/
); );
/** /**
@ -41,12 +44,12 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
$this->gridField = $gridField; $this->gridField = $gridField;
$this->component = $component; $this->component = $component;
$this->controller = $controller; $this->controller = $controller;
$this->createdRecords = array();
parent::__construct(); parent::__construct();
} }
/** /**
* Returns the URL for this RequestHandler * Returns the URL for this RequestHandler
*
* @param String $action * @param String $action
* @return String * @return String
*/ */
@ -54,12 +57,97 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
return Controller::join_links($this->gridField->Link(), 'bulkimageupload', $action); return Controller::join_links($this->gridField->Link(), 'bulkimageupload', $action);
} }
/**
* Returns the name of the Image field name from the managed record
* Either as set in the component config or the default one
*
* @return string
*/
function getRecordImageField()
{
$fieldName = $this->component->getRecordImageField();
if ( $fieldName == null ) $fieldName = $this->getDefaultRecordImageField();
return $fieldName;
}
/**
* Returns the list of editable fields from the managed record
* Either as set in the component config or the default ones
*
* @return array
*/
function getRecordEditableFields()
{
$fields = $this->component->getRecordEditableFields();
if ( $fields == null ) $fields = $this->getDefaultRecordEditableFields();
return $fields;
}
/**
* Get the first has_one Image realtion from the GridField managed DataObject
*
* @return string
*/
function getDefaultRecordImageField()
{
$recordClass = $this->gridField->list->dataClass;
$recordHasOneFields = Config::inst()->get($recordClass, 'has_one', Config::UNINHERITED);
$imageField = null;
foreach( $recordHasOneFields as $field => $type )
{
if ( $type == 'Image' ) {
$imageField = $field . 'ID';
break;
}
}
return $imageField;
}
/**
* Return a list of the GridField managed DataObject (HTML)Text, (HTML)Varchar and Enum fields
*
* @return array
*/
function getDefaultRecordEditableFields()
{
$recordClass = $this->gridField->list->dataClass;
$recordDbFields = Config::inst()->get($recordClass, 'db', Config::UNINHERITED);
$editableFields = array();
foreach ( $recordDbFields as $field => $type )
{
if ( preg_match( '/(Text|Varchar|Enum)/i', $type ) > 0 ) {
array_push($editableFields, $field);
}
}
return $editableFields;
}
/**
* Return the CMS edit field for a given name. As set in the GridField managed DataObject getCMSFields method
* @param string $fieldName
* @return FormField
*/
function getFieldEditForm($fieldName)
{
if ( !$this->recordCMSFieldList ) {
$recordClass = $this->gridField->list->dataClass;
$this->recordCMSFieldList = singleton($recordClass)->getCMSFields();
}
return $this->recordCMSFieldList->fieldByName($fieldName);
}
/** /**
* Default and main action that returns the upload form etc... * Default and main action that returns the upload form etc...
* @param SS_HTTPRequest $request
* @return String Form HTML ??? * @return String Form HTML ???
*/ */
public function index(SS_HTTPRequest $request) public function index()
{ {
Requirements::javascript(FRAMEWORK_DIR . '/javascript/AssetUploadField.js'); Requirements::javascript(FRAMEWORK_DIR . '/javascript/AssetUploadField.js');
Requirements::css(FRAMEWORK_DIR . '/css/AssetUploadField.css'); Requirements::css(FRAMEWORK_DIR . '/css/AssetUploadField.css');
@ -68,13 +156,10 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
Requirements::javascript('GridFieldBulkImageUpload/javascript/GridFieldBulkImageUpload.js'); Requirements::javascript('GridFieldBulkImageUpload/javascript/GridFieldBulkImageUpload.js');
Requirements::css('GridFieldBulkImageUpload/css/GridFieldBulkImageUpload.css'); Requirements::css('GridFieldBulkImageUpload/css/GridFieldBulkImageUpload.css');
$crumbs = $this->Breadcrumbs();
if($crumbs && $crumbs->count()>=2) $one_level_up = $crumbs->offsetGet($crumbs->count()-2);
$actions = new FieldList(); $actions = new FieldList();
/*
$actions->push(FormAction::create('update', 'Finish')
->setUseButtonTag(true)->addExtraClass('ss-ui-action-constructive')->addExtraClass('bulkImageUploadUpdateBtn')->setAttribute('data-icon', 'accept'));
*/
$html = " $html = "
<a id=\"bulkImageUploadUpdateBtn\" class=\"cms-panel-link action ss-ui-action-constructive ss-ui-button ui-button ui-widget ui-state-default ui-corner-all ui-button-text-icon-primary\" data-icon=\"accept\" data-url=\"".$this->Link('update')."\" href=\"#\"> <a id=\"bulkImageUploadUpdateBtn\" class=\"cms-panel-link action ss-ui-action-constructive ss-ui-button ui-button ui-widget ui-state-default ui-corner-all ui-button-text-icon-primary\" data-icon=\"accept\" data-url=\"".$this->Link('update')."\" href=\"#\">
@ -82,10 +167,9 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
</a>"; </a>";
$actions->push(new LiteralField('savebutton', $html)); $actions->push(new LiteralField('savebutton', $html));
$crumbs = $this->Breadcrumbs();
if($crumbs && $crumbs->count()>=2){
$one_level_up = $crumbs->offsetGet($crumbs->count()-2);
if($crumbs && $crumbs->count()>=2)
{
$html = " $html = "
<a id=\"bulkImageUploadUpdateFinishBtn\" class=\"cms-panel-link action ss-ui-action-constructive ss-ui-button ui-button ui-widget ui-state-default ui-corner-all ui-button-text-icon-primary\" data-icon=\"accept\" data-url=\"".$this->Link('update')."\" href=\"".$one_level_up->Link."\"> <a id=\"bulkImageUploadUpdateFinishBtn\" class=\"cms-panel-link action ss-ui-action-constructive ss-ui-button ui-button ui-widget ui-state-default ui-corner-all ui-button-text-icon-primary\" data-icon=\"accept\" data-url=\"".$this->Link('update')."\" href=\"".$one_level_up->Link."\">
Save All &amp; Finish Save All &amp; Finish
@ -99,7 +183,6 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
$actions->push(new LiteralField('cancelbutton', $html)); $actions->push(new LiteralField('cancelbutton', $html));
} }
$uploadField = UploadField::create('BulkImageUploadField', ''); $uploadField = UploadField::create('BulkImageUploadField', '');
$uploadField->setConfig('previewMaxWidth', 40); $uploadField->setConfig('previewMaxWidth', 40);
$uploadField->setConfig('previewMaxHeight', 30); $uploadField->setConfig('previewMaxHeight', 30);
@ -126,22 +209,27 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
); );
$form->setTemplate('LeftAndMain_EditForm'); $form->setTemplate('LeftAndMain_EditForm');
$form->addExtraClass('center cms-edit-form'); $form->addExtraClass('center cms-edit-form cms-content');
$form->setAttribute('data-pjax-fragment', 'CurrentForm Content'); $form->setAttribute('data-pjax-fragment', 'CurrentForm Content');
if($crumbs && $crumbs->count()>=2){ if($crumbs && $crumbs->count()>=2){
$form->Backlink = $one_level_up->Link; $form->Backlink = $one_level_up->Link;
} }
return $form->forTemplate(); $response = new SS_HTTPResponse($form->forTemplate());
/* $response->addHeader('Content-Type', 'text/plain');
$response = new SS_HTTPResponse(Convert::raw2json(array('Content' => $form->forTemplate())));
$response->addHeader('Content-Type', 'text/json');
$response->addHeader('X-Title', 'SilverStripe - Bulk '.$this->gridField->list->dataClass.' Image Upload'); $response->addHeader('X-Title', 'SilverStripe - Bulk '.$this->gridField->list->dataClass.' Image Upload');
return $response; */ return $response;
} }
/**
* Process image upload and Object creation
* Create new DataObject and add image relation
* returns Image data and editable Fields forms
*
* @param SS_HTTPRequest $request
* @return \SS_HTTPResponse
*/
public function upload(SS_HTTPRequest $request) public function upload(SS_HTTPRequest $request)
{ {
$recordClass = $this->gridField->list->dataClass; $recordClass = $this->gridField->list->dataClass;
@ -165,10 +253,6 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
'type' => $tmpfile['type'], 'type' => $tmpfile['type'],
'error' => $tmpfile['error'] 'error' => $tmpfile['error']
); );
/*
$record->setField($this->component->getLabelFieldName(), $tmpfile['name']);
$record->write();
*/
} }
// Process the uploaded file // Process the uploaded file
@ -190,9 +274,17 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
$file = $upload->getFile(); $file = $upload->getFile();
// Attach the file to the related record. // Attach the file to the related record.
$record->setField($this->component->getRecordImageField(), $file->ID); $record->setField($this->getRecordImageField(), $file->ID);
$record->write(); $record->write();
// collect all editable fields forms
$recordEditableFieldsForms = array();
foreach ( $this->getRecordEditableFields() as $editableFieldName )
{
array_push($recordEditableFieldsForms, $this->parseFieldHTMLWithRecordID($this->getFieldEditForm($editableFieldName)->FieldHolder(), $record->ID) );
}
// Collect all output data. // Collect all output data.
$return = array_merge($return, array( $return = array_merge($return, array(
'id' => $file->ID, 'id' => $file->ID,
@ -204,14 +296,13 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
//'buttons' => $file->UploadFieldFileButtons, //'buttons' => $file->UploadFieldFileButtons,
'record' => array( 'record' => array(
'ID' => $record->ID, 'ID' => $record->ID,
'fields' => $this->component->getRecordEditableFields() 'fields' => $recordEditableFieldsForms
) )
)); ));
}
}
}
array_push($this->createdRecords, $record->ID); }
}
}
$response = new SS_HTTPResponse(Convert::raw2json(array($return))); $response = new SS_HTTPResponse(Convert::raw2json(array($return)));
$response->addHeader('Content-Type', 'text/plain'); $response->addHeader('Content-Type', 'text/plain');
@ -226,37 +317,24 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
*/ */
public function update(SS_HTTPRequest $request) public function update(SS_HTTPRequest $request)
{ {
$data = $request->requestVars(); $data = $this->getParsedPostData($request->requestVars());
$recordID = false;
$recordFields = array();
foreach( $data as $key => $val)
{
if ( stripos($key, 'record_') !== false )
{
if ( $key == 'record_ID' )
{
$recordID = $val;
}else{
$recordFields[str_ireplace('record_', '', $key)] = $val;
}
}
}
$recordClass = $this->gridField->list->dataClass; $recordClass = $this->gridField->list->dataClass;
$record = DataObject::get_by_id($recordClass, $recordID); $record = DataObject::get_by_id($recordClass, $data['ID']);
foreach($recordFields as $field => $value) foreach($data as $field => $value)
{ {
$record->setField($field, $value); $record->setField($field, $value);
} }
$record->write(); $record->write();
return '{done:1,recordID:'.$recordID.'}'; return '{done:1,recordID:'.$data['ID'].'}';
} }
/** /**
* Delete the Image Object and File as well as the DataObject
* according to the ID sent from the form
* *
* @param SS_HTTPRequest $request * @param SS_HTTPRequest $request
* @return String JSON * @return String JSON
@ -268,8 +346,8 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
$recordClass = $this->gridField->list->dataClass; $recordClass = $this->gridField->list->dataClass;
$record = DataObject::get_by_id($recordClass, $data['ID']); $record = DataObject::get_by_id($recordClass, $data['ID']);
$imageField = $this->component->getRecordImageField();
$imageField = $this->getRecordImageField();
$imageID = $record->$imageField; $imageID = $record->$imageField;
$image = DataObject::get_by_id('Image', $imageID); $image = DataObject::get_by_id('Image', $imageID);
@ -286,49 +364,39 @@ class GridFieldBulkImageUpload_Request extends RequestHandler {
} }
/** /**
* Simple function taht replace the 'record_XX_' off of the ID field name
* prefix needed since it was taken for a pageID if sent as is as well as fixing other things
* *
* @param array $data * @param array $data
* @return array * @return array
*/ */
public function getParsedPostData(array $data) function getParsedPostData(array $data)
{ {
$return = array(); $return = array();
$fields = array();
foreach( $data as $key => $val) foreach( $data as $key => $val)
{ {
if ( stripos($key, 'record_') !== false ) $return[ preg_replace( '/record_(\d+)_(\w+)/i', '$2', $key) ] = $val;
{
if ( $key == 'record_ID' )
{
$return['ID'] = $val;
}else{
$fields[str_ireplace('record_', '', $key)] = $val;
} }
}
}
$return['fields'] = $fields;
return $return; return $return;
} }
/** /**
* Traverse up nested requests until we reach the first that's not a GridFieldBulkImageUpload_Request. * Add a unique prefix to sensitive HTML attributes (ID, FOR, NAME)
* The opposite of {@link Controller::curr()}, required because * Fixes rendering issue (i.e. dropdown fields) and IDs being mistaken for page IDs
* Controller::$controller_stack is not directly accessible.
* *
* @return Controller * @param string $html
* @param string/int $id
* @return string
*/ */
/* function parseFieldHTMLWithRecordID($html, $id)
protected function getToplevelController() { {
$c = $this->controller; $prefix = 'record_'.$id.'_';
while($c && $c instanceof GridFieldBulkImageUpload_Request) { return str_ireplace ( array('id="', 'for="', 'name="'),
$c = $c->getController(); array('id="'.$prefix, 'for="'.$prefix, 'name="'.$prefix),
$html);
} }
return $c;
}
*/
/** /**
* CMS-specific functionality: Passes through navigation breadcrumbs * CMS-specific functionality: Passes through navigation breadcrumbs

View File

@ -137,7 +137,7 @@
$(this).parents('li.ss-uploadfield-item').empty().remove(); $(this).parents('li.ss-uploadfield-item').empty().remove();
if ( $('li.ss-uploadfield-item').lenght == 0 ) { if ( $('li.ss-uploadfield-item').length == 0 ) {
$('.ss-uploadfield-editandorganize').css('display', 'none'); $('.ss-uploadfield-editandorganize').css('display', 'none');
$('#Form_bulkImageUploadForm').removeClass('loading'); $('#Form_bulkImageUploadForm').removeClass('loading');
} }

View File

@ -23,18 +23,11 @@ window.tmpl.cache['GridFieldBulkImageUpload_downloadtemplate'] = tmpl(
'{% if (!file.error) { %}' + '{% if (!file.error) { %}' +
'<div class="ss-uploadfield-item-editform">'+ '<div class="ss-uploadfield-item-editform">'+
'<form action="update" method="post" class="bulkImageUploadUpdateForm" name="BIUUF_{%=file.record.ID%}">'+ '<form action="update" method="post" class="bulkImageUploadUpdateForm" name="BIUUF_{%=file.record.ID%}">'+
'<input type="hidden" name="record_ID" value="{%=file.record.ID%}"/>'+ '<input type="hidden" name="record_{%=file.record.ID%}_ID" value="{%=file.record.ID%}"/>'+
'<img class="imgPreview" src="{%=file.preview_url%}" />'+ '<img class="imgPreview" src="{%=file.preview_url%}" />'+
'{% for (var j=0; j<file.record.fields.length; j++) { %}' + '{% for (var j=0; j<file.record.fields.length; j++) { %}' +
'{%#file.record.fields[j]%}' +
'<div id="record_{%=file.record.ID%}_{%=file.record.fields[j]%}" class="field text">' +
'<label class="left" for="BIUUF_{%=file.record.ID%}_{%=file.record.fields[j]%}">{%=file.record.fields[j]%}</label>' +
'<div class="middleColumn">' +
'<input type="text" name="record_{%=file.record.fields[j]%}" value="" class="text" id="BIUUF_{%=file.record.ID%}_{%=file.record.fields[j]%}">' +
'</div>' +
'</div>' +
'{% } %}' + '{% } %}' +
'</form>'+ '</form>'+