2008-09-29 05:18:23 +02:00
|
|
|
<?php
|
|
|
|
/**
|
2010-09-03 07:06:13 +02:00
|
|
|
* A form field allowing a user to customize and create form fields.
|
|
|
|
* for saving into a {@link UserDefinedForm}
|
2009-04-15 06:23:43 +02:00
|
|
|
*
|
|
|
|
* @package userforms
|
2008-09-29 05:18:23 +02:00
|
|
|
*/
|
2009-12-07 03:04:20 +01:00
|
|
|
|
2008-09-29 05:18:23 +02:00
|
|
|
class FieldEditor extends FormField {
|
2012-05-07 07:09:08 +02:00
|
|
|
|
2013-04-03 03:34:43 +02:00
|
|
|
private static $allowed_actions = array(
|
2012-05-07 07:09:08 +02:00
|
|
|
'addfield',
|
|
|
|
'addoptionfield'
|
|
|
|
);
|
2008-09-29 05:18:23 +02:00
|
|
|
|
2010-09-03 07:06:13 +02:00
|
|
|
/**
|
|
|
|
* Field Editor Template
|
|
|
|
*
|
|
|
|
* @return String
|
|
|
|
*/
|
2012-05-04 03:59:10 +02:00
|
|
|
public function FieldHolder($properties = array()) {
|
2012-05-07 07:10:05 +02:00
|
|
|
$this->setAttribute('data-add-url', '\''.Controller::join_links($this->Link('addfield').'\''));
|
2008-09-29 05:18:23 +02:00
|
|
|
return $this->renderWith("FieldEditor");
|
|
|
|
}
|
|
|
|
|
2009-05-14 23:40:03 +02:00
|
|
|
/**
|
2010-09-03 07:06:13 +02:00
|
|
|
* Returns whether a user can edit the form
|
|
|
|
*
|
2009-06-08 06:42:17 +02:00
|
|
|
* @return boolean
|
2009-05-14 23:40:03 +02:00
|
|
|
*/
|
2012-05-04 03:59:10 +02:00
|
|
|
public function canEdit($member = null) {
|
2009-06-08 06:42:17 +02:00
|
|
|
if($this->readonly) return false;
|
2010-09-03 07:06:13 +02:00
|
|
|
|
2009-05-14 23:40:03 +02:00
|
|
|
return $this->form->getRecord()->canEdit();
|
|
|
|
}
|
2009-05-28 10:31:10 +02:00
|
|
|
|
2009-06-08 06:42:17 +02:00
|
|
|
/**
|
2010-09-03 07:06:13 +02:00
|
|
|
* Returns whether a user delete a field in the form. The {@link EditableFormField}s
|
|
|
|
* check if they can delete themselves but this counts as an {@link self::canEdit()}
|
|
|
|
* function rather than a delete
|
|
|
|
*
|
2009-06-08 06:42:17 +02:00
|
|
|
* @return boolean
|
|
|
|
*/
|
2012-05-04 03:59:10 +02:00
|
|
|
public function canDelete($member = null) {
|
2009-06-08 06:42:17 +02:00
|
|
|
if($this->readonly) return false;
|
2010-09-03 07:06:13 +02:00
|
|
|
|
|
|
|
return $this->form->getRecord()->canEdit();
|
2009-06-08 06:42:17 +02:00
|
|
|
}
|
|
|
|
|
2010-09-03 07:06:13 +02:00
|
|
|
/**
|
|
|
|
* Transform this form field to a readonly version.
|
|
|
|
*
|
|
|
|
* @return ViewableData_Customised
|
|
|
|
*/
|
2012-05-04 03:39:08 +02:00
|
|
|
public function performReadonlyTransformation() {
|
2009-06-08 08:17:15 +02:00
|
|
|
$clone = clone $this;
|
|
|
|
$clone->readonly = true;
|
|
|
|
$fields = $clone->Fields();
|
2009-06-08 06:42:17 +02:00
|
|
|
if($fields) foreach($fields as $field) {
|
|
|
|
$field->setReadonly();
|
|
|
|
}
|
|
|
|
|
2009-06-08 08:17:15 +02:00
|
|
|
return $clone->customise(array('Fields' => $fields));
|
2009-06-08 06:42:17 +02:00
|
|
|
}
|
|
|
|
|
2009-05-28 10:31:10 +02:00
|
|
|
/**
|
2010-09-03 07:06:13 +02:00
|
|
|
* Return the fields for the user forms
|
2009-05-28 10:31:10 +02:00
|
|
|
*
|
|
|
|
* @return DataObjectSet
|
|
|
|
*/
|
2012-05-04 03:39:08 +02:00
|
|
|
public function Fields() {
|
2009-06-08 08:17:15 +02:00
|
|
|
// Don't return any fields unless we actually have the dependent parameters set on the form field
|
|
|
|
if($this->form && $this->form->getRecord() && $this->name) {
|
|
|
|
$relationName = $this->name;
|
2010-03-22 05:18:09 +01:00
|
|
|
$fields = $this->form->getRecord()->getComponents($relationName);
|
2008-09-29 05:18:23 +02:00
|
|
|
|
2009-06-08 08:17:15 +02:00
|
|
|
if($fields) {
|
|
|
|
foreach($fields as $field) {
|
2012-05-07 07:10:27 +02:00
|
|
|
if(!$this->canEdit() && is_a($field, 'FormField')) {
|
|
|
|
$fields->remove($field);
|
|
|
|
$fields->push($field->performReadonlyTransformation());
|
2009-06-08 06:42:17 +02:00
|
|
|
}
|
2009-05-14 23:40:03 +02:00
|
|
|
}
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
2010-09-03 07:06:13 +02:00
|
|
|
|
2009-06-08 08:17:15 +02:00
|
|
|
return $fields;
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-25 03:06:28 +01:00
|
|
|
/**
|
|
|
|
* Return a DataObjectSet of all the addable fields to populate
|
|
|
|
* the add field menu
|
|
|
|
*
|
|
|
|
* @return DataObjectSet
|
|
|
|
*/
|
2012-05-04 03:39:08 +02:00
|
|
|
public function CreatableFields() {
|
2009-03-25 03:06:28 +01:00
|
|
|
$fields = ClassInfo::subclassesFor('EditableFormField');
|
|
|
|
|
|
|
|
if($fields) {
|
|
|
|
array_shift($fields); // get rid of subclass 0
|
2009-05-06 05:34:40 +02:00
|
|
|
asort($fields); // get in order
|
2012-04-22 21:17:42 +02:00
|
|
|
$output = new ArrayList();
|
2009-03-25 03:06:28 +01:00
|
|
|
foreach($fields as $field => $title) {
|
|
|
|
// get the nice title and strip out field
|
2013-04-03 03:34:43 +02:00
|
|
|
$niceTitle = Config::inst()->get($title, 'singular_name');
|
2009-04-27 02:20:20 +02:00
|
|
|
if($niceTitle) {
|
2009-04-21 07:10:39 +02:00
|
|
|
$output->push(new ArrayData(array(
|
2009-04-25 03:14:32 +02:00
|
|
|
'ClassName' => $field,
|
2009-04-21 07:10:39 +02:00
|
|
|
'Title' => "$niceTitle"
|
|
|
|
)));
|
|
|
|
}
|
2009-03-25 03:06:28 +01:00
|
|
|
}
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2009-04-27 02:20:20 +02:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
/**
|
|
|
|
* Handles saving the page. Needs to keep an eye on fields
|
|
|
|
* and options which have been removed / added
|
|
|
|
*
|
|
|
|
* @param DataObject Record to Save it In
|
|
|
|
*/
|
2012-05-04 03:59:10 +02:00
|
|
|
public function saveInto(DataObjectInterface $record) {
|
2008-09-29 05:18:23 +02:00
|
|
|
$name = $this->name;
|
2010-09-08 05:20:28 +02:00
|
|
|
$fieldSet = $record->$name();
|
2008-09-29 05:18:23 +02:00
|
|
|
|
|
|
|
// store the field IDs and delete the missing fields
|
|
|
|
// alternatively, we could delete all the fields and re add them
|
|
|
|
$missingFields = array();
|
2009-04-17 04:26:40 +02:00
|
|
|
|
2009-05-12 03:48:04 +02:00
|
|
|
foreach($fieldSet as $existingField) {
|
|
|
|
$missingFields[$existingField->ID] = $existingField;
|
|
|
|
}
|
2009-04-17 04:26:40 +02:00
|
|
|
|
2009-05-28 10:45:25 +02:00
|
|
|
if(isset($_REQUEST[$name]) && is_array($_REQUEST[$name])) {
|
2009-05-28 10:31:10 +02:00
|
|
|
foreach($_REQUEST[$name] as $newEditableID => $newEditableData) {
|
2009-05-12 03:48:04 +02:00
|
|
|
if(!is_numeric($newEditableID)) continue;
|
|
|
|
|
2009-05-28 10:31:10 +02:00
|
|
|
// get it from the db
|
2012-07-17 06:09:31 +02:00
|
|
|
$editable = DataObject::get_by_id('EditableFormField', $newEditableID);
|
2010-05-16 05:19:29 +02:00
|
|
|
|
2012-07-17 06:09:31 +02:00
|
|
|
// if it exists in the db update it
|
|
|
|
if($editable) {
|
2009-05-28 10:31:10 +02:00
|
|
|
|
|
|
|
// remove it from the removed fields list
|
|
|
|
if(isset($missingFields[$editable->ID]) && isset($newEditableData) && is_array($newEditableData)) {
|
2009-04-17 04:26:40 +02:00
|
|
|
unset($missingFields[$editable->ID]);
|
|
|
|
}
|
2009-05-28 10:31:10 +02:00
|
|
|
|
|
|
|
// set form id
|
2009-04-17 04:26:40 +02:00
|
|
|
if($editable->ParentID == 0) {
|
|
|
|
$editable->ParentID = $record->ID;
|
|
|
|
}
|
2009-05-28 10:31:10 +02:00
|
|
|
|
|
|
|
// save data
|
2009-04-17 04:26:40 +02:00
|
|
|
$editable->populateFromPostData($newEditableData);
|
|
|
|
}
|
2012-07-17 06:09:31 +02:00
|
|
|
}
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
2009-04-17 04:26:40 +02:00
|
|
|
|
2012-07-17 06:09:31 +02:00
|
|
|
// remove the fields not saved
|
2009-05-28 10:40:23 +02:00
|
|
|
if($this->canEdit()) {
|
2012-07-17 06:09:31 +02:00
|
|
|
foreach($missingFields as $removedField) {
|
|
|
|
if(is_numeric($removedField->ID)) {
|
|
|
|
// check we can edit this
|
|
|
|
$removedField->delete();
|
|
|
|
}
|
2009-05-28 10:31:10 +02:00
|
|
|
}
|
2012-07-17 06:09:31 +02:00
|
|
|
}
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
2009-04-17 04:26:40 +02:00
|
|
|
|
|
|
|
/**
|
2012-05-07 07:10:27 +02:00
|
|
|
* Add a field to the field editor. Called via a ajax get request from the userdefinedform javascript
|
2009-04-17 04:26:40 +02:00
|
|
|
*
|
|
|
|
* @return bool|html
|
|
|
|
*/
|
|
|
|
public function addfield() {
|
2013-09-04 13:34:21 +02:00
|
|
|
if(!SecurityToken::inst()->checkRequest($this->request)) {
|
|
|
|
return $this->httpError(400);
|
|
|
|
}
|
|
|
|
|
2008-09-29 05:18:23 +02:00
|
|
|
// get the last field in this form editor
|
|
|
|
$parentID = $this->form->getRecord()->ID;
|
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
if($parentID) {
|
2013-03-24 11:30:44 +01:00
|
|
|
$parentID = (int)$parentID;
|
2009-05-12 03:48:04 +02:00
|
|
|
|
2013-03-04 23:36:39 +01:00
|
|
|
$sqlQuery = new SQLQuery();
|
|
|
|
$sqlQuery = $sqlQuery
|
|
|
|
->setSelect('MAX(Sort)')
|
|
|
|
->setFrom("EditableFormField")
|
|
|
|
->setWhere("ParentID = $parentID");
|
|
|
|
|
|
|
|
$sort = $sqlQuery->execute()->value() + 1;
|
2009-04-17 04:26:40 +02:00
|
|
|
|
2009-04-25 03:14:32 +02:00
|
|
|
$className = (isset($_REQUEST['Type'])) ? $_REQUEST['Type'] : '';
|
2013-03-04 23:36:39 +01:00
|
|
|
|
|
|
|
if(!$className) {
|
|
|
|
user_error('Please select a field type to created', E_USER_WARNING);
|
|
|
|
}
|
2009-04-25 03:14:32 +02:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
if(is_subclass_of($className, "EditableFormField")) {
|
2009-04-20 01:52:44 +02:00
|
|
|
$field = new $className();
|
|
|
|
$field->ParentID = $this->form->getRecord()->ID;
|
2009-04-20 03:02:26 +02:00
|
|
|
$field->Name = $field->class . $field->ID;
|
2009-04-20 01:52:44 +02:00
|
|
|
$field->Sort = $sort;
|
|
|
|
$field->write();
|
2013-03-04 23:36:39 +01:00
|
|
|
|
2009-04-20 01:52:44 +02:00
|
|
|
return $field->EditSegment();
|
2009-04-17 04:26:40 +02:00
|
|
|
}
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
2013-03-04 23:36:39 +01:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
return false;
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
/**
|
|
|
|
* Return the html for a field option such as a
|
|
|
|
* dropdown field or a radio check box field
|
|
|
|
*
|
|
|
|
* @return bool|html
|
|
|
|
*/
|
|
|
|
public function addoptionfield() {
|
2013-09-04 13:34:21 +02:00
|
|
|
if(!SecurityToken::inst()->checkRequest($this->request)) {
|
|
|
|
return $this->httpError(400);
|
|
|
|
}
|
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
// passed via the ajax
|
|
|
|
$parent = (isset($_REQUEST['Parent'])) ? $_REQUEST['Parent'] : false;
|
2009-05-06 05:34:40 +02:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
// work out the sort by getting the sort of the last field in the form +1
|
|
|
|
if($parent) {
|
2013-03-24 11:30:44 +01:00
|
|
|
$sql_parent = (int)$parent;
|
2009-12-07 01:52:00 +01:00
|
|
|
|
2013-03-04 23:36:39 +01:00
|
|
|
$sqlQuery = new SQLQuery();
|
|
|
|
$sqlQuery = $sqlQuery
|
|
|
|
->setSelect('MAX(Sort)')
|
|
|
|
->setFrom("EditableOption")
|
|
|
|
->setWhere("ParentID = $sql_parent");
|
|
|
|
|
|
|
|
$sort = $sqlQuery->execute()->value() + 1;
|
2009-04-17 04:26:40 +02:00
|
|
|
|
|
|
|
if($parent) {
|
|
|
|
$object = new EditableOption();
|
|
|
|
$object->write();
|
|
|
|
$object->ParentID = $parent;
|
|
|
|
$object->Sort = $sort;
|
|
|
|
$object->Name = 'option' . $object->ID;
|
|
|
|
$object->write();
|
2013-03-04 23:36:39 +01:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
return $object->EditSegment();
|
|
|
|
}
|
|
|
|
}
|
2013-03-04 23:36:39 +01:00
|
|
|
|
2009-04-17 04:26:40 +02:00
|
|
|
return false;
|
2008-09-29 05:18:23 +02:00
|
|
|
}
|
2012-07-17 06:09:31 +02:00
|
|
|
}
|