mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
FEATURE Changed CMSMain and LeftAndMain form submissions to return raw HTML instead of using FormResponse logic and evaluated javascript. This allows a more customizeable UI layer that is decoupled from the serverside logic. Any state changes should be propagated through the form itself.
ENHANCEMENT Using new 'X-STATUS' HTTP response header for CMS form responses, as it is more robust for submitting variable length strings than the original 'Status' header. The status is evaluated in LeftAndMain.EditForm.js API CHANGE Removed CMSMain->tellBrowserAboutPublicationChange(), LeftAndMain->returnItemToUser(), LeftAndMain->getActionUpdateJS(), LeftAndMain->addTreeNodeJS(), LeftAndMain->deleteTreeNodeJS(). Use javascript to respond to state changes API CHANGE Removed CMSForm and CMSRightForm javascript classes, superseded by LeftAndMain.EditForm.js ENHANCEMENT Removed custom change detection in LeftAndMain->save(), this should be handled by DataObject->write() ENHANCEMENT Removed switch in LeftAndMain->save() which doesnt process saving if the record hasn't been altered, to simplify the saving logic ENHANCEMENT Removed custom add/remove tree node logic in LeftAndMain->save() which was retrieving state from DataObjectLog. This was never actively used, and should be handled by custom clientside logic. git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/cms/trunk@92674 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
a397804db6
commit
c2d24f9022
@ -578,7 +578,6 @@ JS;
|
||||
if(is_numeric($id)) {
|
||||
$record = DataObject::get_by_id($this->stat('tree_class'), $id);
|
||||
if($record) {
|
||||
$script .= $this->deleteTreeNodeJS($record);
|
||||
$record->delete();
|
||||
$record->destroy();
|
||||
}
|
||||
|
228
code/CMSMain.php
228
code/CMSMain.php
@ -369,6 +369,10 @@ JS;
|
||||
}
|
||||
}
|
||||
|
||||
$form = new Form($this, "EditForm", $fields, $actions);
|
||||
$form->loadDataFrom($record);
|
||||
$form->disableDefaultAction();
|
||||
|
||||
// Add a default or custom validator.
|
||||
// @todo Currently the default Validator.js implementation
|
||||
// adds javascript to the document body, meaning it won't
|
||||
@ -388,15 +392,6 @@ JS;
|
||||
$form->unsetValidator();
|
||||
}
|
||||
|
||||
// The clientside (mainly LeftAndMain*.js) rely on ajax responses
|
||||
// which can be evaluated as javascript, hence we need
|
||||
// to override any global changes to the validation handler.
|
||||
$validator->setJavascriptValidationHandler('prototype');
|
||||
|
||||
$form = new Form($this, "EditForm", $fields, $actions, $validator);
|
||||
$form->loadDataFrom($record);
|
||||
$form->disableDefaultAction();
|
||||
|
||||
if(!$record->canEdit() || $record->IsDeletedFromStage) {
|
||||
$readonlyFields = $form->Fields()->makeReadonly();
|
||||
$form->setFields($readonlyFields);
|
||||
@ -421,27 +416,29 @@ JS;
|
||||
// Data saving handlers
|
||||
|
||||
|
||||
public function addpage() {
|
||||
$className = isset($_REQUEST['PageType']) ? $_REQUEST['PageType'] : "Page";
|
||||
$parent = isset($_REQUEST['ParentID']) ? $_REQUEST['ParentID'] : 0;
|
||||
$suffix = isset($_REQUEST['Suffix']) ? "-" . $_REQUEST['Suffix'] : null;
|
||||
public function addpage($data, $form) {
|
||||
$className = isset($data['PageType']) ? $data['PageType'] : "Page";
|
||||
$parentID = isset($data['ParentID']) ? (int)$data['ParentID'] : 0;
|
||||
$suffix = isset($data['Suffix']) ? "-" . $data['Suffix'] : null;
|
||||
|
||||
if(!$parent && isset($_REQUEST['Parent'])) {
|
||||
$page = SiteTree::get_by_link($_REQUEST['Parent']);
|
||||
if($page) $parent = $page->ID;
|
||||
if(!$parentID && isset($data['Parent'])) {
|
||||
$page = SiteTree:: get_by_link(Convert::raw2sql($data['Parent']));
|
||||
if($page) $parentID = $page->ID;
|
||||
}
|
||||
|
||||
if(is_numeric($parent)) $parentObj = DataObject::get_by_id("SiteTree", $parent);
|
||||
if(!$parentObj || !$parentObj->ID) $parent = 0;
|
||||
if(is_numeric($parentID)) $parentObj = DataObject::get_by_id("SiteTree", $parentID);
|
||||
if(!$parentObj || !$parentObj->ID) $parentID = 0;
|
||||
|
||||
if($parentObj && !$parentObj->canAddChildren()) return Security::permissionFailure($this);
|
||||
if(!singleton($className)->canCreate()) return Security::permissionFailure($this);
|
||||
|
||||
$p = $this->getNewItem("new-$className-$parent".$suffix, false);
|
||||
$p->Locale = $_REQUEST['Locale'];
|
||||
$p->Locale = $data['Locale'];
|
||||
$p->write();
|
||||
|
||||
return $this->returnItemToUser($p);
|
||||
$form = $this->getEditForm($p->ID);
|
||||
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
/**
|
||||
@ -491,10 +488,9 @@ JS;
|
||||
*
|
||||
* @see delete()
|
||||
*/
|
||||
public function deletefromlive($urlParams, $form) {
|
||||
$id = $_REQUEST['ID'];
|
||||
public function deletefromlive($data, $form) {
|
||||
Versioned::reading_stage('Live');
|
||||
$record = DataObject::get_by_id("SiteTree", $id);
|
||||
$record = DataObject::get_by_id("SiteTree", $data['ID']);
|
||||
if($record && !$record->canDelete()) return Security::permissionFailure($this);
|
||||
|
||||
$descRemoved = '';
|
||||
@ -525,10 +521,17 @@ JS;
|
||||
$descRemoved = '';
|
||||
}
|
||||
|
||||
FormResponse::add($this->deleteTreeNodeJS($record));
|
||||
FormResponse::status_message(sprintf(_t('CMSMain.REMOVED', 'Deleted \'%s\'%s from live site'), $record->Title, $descRemoved), 'good');
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(
|
||||
_t('CMSMain.REMOVED', 'Deleted \'%s\'%s from live site'),
|
||||
$record->Title,
|
||||
$descRemoved
|
||||
)
|
||||
);
|
||||
|
||||
return FormResponse::respond();
|
||||
// nothing to return
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
@ -547,22 +550,37 @@ JS;
|
||||
*
|
||||
* @uses SiteTree->doRevertToLive()
|
||||
*/
|
||||
public function revert($urlParams, $form) {
|
||||
$id = (int)$_REQUEST['ID'];
|
||||
$record = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree_Live\".\"ID\" = '{$id}'");
|
||||
public function revert($data, $form) {
|
||||
if(!isset($data['ID'])) return new HTTPResponse("Please pass an ID in the form content", 400);
|
||||
|
||||
$restoredPage = Versioned::get_latest_version("SiteTree", $data['ID']);
|
||||
if(!$restoredPage) return new HTTPResponse("SiteTree #$id not found", 400);
|
||||
|
||||
$record = Versioned::get_one_by_stage(
|
||||
'SiteTree',
|
||||
'Live',
|
||||
sprintf("\"SiteTree_Live\".\"ID\" = '%d'", (int)$data['ID'])
|
||||
);
|
||||
|
||||
// a user can restore a page without publication rights, as it just adds a new draft state
|
||||
// (this action should just be available when page has been "deleted from draft")
|
||||
if(isset($record) && $record && !$record->canEdit()) return Security::permissionFailure($this);
|
||||
if(isset($record) && $record && !$record->canEdit()) {
|
||||
return Security::permissionFailure($this);
|
||||
}
|
||||
|
||||
$record->doRevertToLive();
|
||||
|
||||
$title = Convert::raw2js($record->Title);
|
||||
FormResponse::get_page($id);
|
||||
FormResponse::add("$('sitetree').setNodeTitle($id, '$title');");
|
||||
FormResponse::status_message(sprintf(_t('CMSMain.RESTORED',"Restored '%s' successfully",PR_MEDIUM,'Param %s is a title'),$title),'good');
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(
|
||||
_t('CMSMain.RESTORED',"Restored '%s' successfully",PR_MEDIUM,'Param %s is a title'),
|
||||
$record->Title
|
||||
)
|
||||
);
|
||||
|
||||
return FormResponse::respond();
|
||||
$form = $this->getEditForm($record->ID);
|
||||
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
/**
|
||||
@ -580,20 +598,23 @@ JS;
|
||||
$recordID = $record->ID;
|
||||
$record->delete();
|
||||
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(
|
||||
_t('CMSMain.REMOVEDPAGEFROMDRAFT',"Removed '%s' from the draft site"),
|
||||
$record->Title
|
||||
)
|
||||
);
|
||||
|
||||
if(Director::is_ajax()) {
|
||||
// need a valid ID value even if the record doesn't have one in the database
|
||||
// (its still present in the live tables)
|
||||
$liveRecord = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree_Live\".\"ID\" = $recordID");
|
||||
// if the page has never been published to live, we need to act the same way as in deletefromlive()
|
||||
if($liveRecord) {
|
||||
// the form is readonly now, so we need to refresh the representation
|
||||
FormResponse::get_page($recordID);
|
||||
return $this->tellBrowserAboutPublicationChange($liveRecord, sprintf(_t('CMSMain.REMOVEDPAGEFROMDRAFT',"Removed '%s' from the draft site"),$record->Title));
|
||||
} else {
|
||||
FormResponse::add($this->deleteTreeNodeJS($record));
|
||||
FormResponse::status_message(sprintf(_t('CMSMain.REMOVEDPAGEFROMDRAFT',"Removed '%s' from the draft site"),$record->Title), 'good');
|
||||
return FormResponse::respond();
|
||||
}
|
||||
$liveRecord = Versioned::get_one_by_stage(
|
||||
'SiteTree',
|
||||
'Live',
|
||||
"\"SiteTree_Live\".\"ID\" = $recordID"
|
||||
);
|
||||
return ($liveRecord) ? $form->formHtmlContent() : "";
|
||||
} else {
|
||||
Director::redirectBack();
|
||||
}
|
||||
@ -674,65 +695,49 @@ JS;
|
||||
/**
|
||||
* Roll a page back to a previous version
|
||||
*/
|
||||
function rollback() {
|
||||
if(isset($_REQUEST['Version']) && (bool)$_REQUEST['Version']) {
|
||||
$record = $this->performRollback($_REQUEST['ID'], $_REQUEST['Version']);
|
||||
echo sprintf(_t('CMSMain.ROLLEDBACKVERSION',"Rolled back to version #%d. New version number is #%d"),$_REQUEST['Version'],$record->Version);
|
||||
function rollback($data, $form) {
|
||||
if(isset($data['Version']) && (bool)$data['Version']) {
|
||||
$record = $this->performRollback($data['ID'], $data['Version']);
|
||||
$message = sprintf(
|
||||
_t('CMSMain.ROLLEDBACKVERSION',"Rolled back to version #%d. New version number is #%d"),
|
||||
$data['Version'],
|
||||
$record->Version
|
||||
);
|
||||
} else {
|
||||
$record = $this->performRollback($_REQUEST['ID'], "Live");
|
||||
echo sprintf(_t('CMSMain.ROLLEDBACKPUB',"Rolled back to published version. New version number is #%d"),$record->Version);
|
||||
}
|
||||
$record = $this->performRollback($data['ID'], "Live");
|
||||
$message = sprintf(
|
||||
_t('CMSMain.ROLLEDBACKPUB',"Rolled back to published version. New version number is #%d"),
|
||||
$record->Version
|
||||
);
|
||||
}
|
||||
|
||||
function publish($urlParams, $form) {
|
||||
$urlParams['publish'] = '1';
|
||||
$this->response->addHeader('X-Status', $message);
|
||||
|
||||
return $this->save($urlParams, $form);
|
||||
$form = $this->getEditForm($record->ID);
|
||||
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
function unpublish() {
|
||||
$SQL_id = Convert::raw2sql($_REQUEST['ID']);
|
||||
function publish($data, $form) {
|
||||
$data['publish'] = '1';
|
||||
|
||||
$page = DataObject::get_by_id("SiteTree", $SQL_id);
|
||||
return $this->save($data, $form);
|
||||
}
|
||||
|
||||
function unpublish($data, $form) {
|
||||
$page = DataObject::get_by_id("SiteTree", $data['ID']);
|
||||
if($page && !$page->canPublish()) return Security::permissionFailure($this);
|
||||
|
||||
$page->doUnpublish();
|
||||
|
||||
return $this->tellBrowserAboutPublicationChange($page, sprintf(_t('CMSMain.REMOVEDPAGE',"Removed '%s' from the published site"),$page->Title));
|
||||
}
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(_t('CMSMain.REMOVEDPAGE',"Removed '%s' from the published site"),$page->Title)
|
||||
);
|
||||
|
||||
/**
|
||||
* Return a few pieces of information about a change to a page
|
||||
* - Send the new status message
|
||||
* - Update the action buttons
|
||||
* - Update the treenote
|
||||
* - Send a status message
|
||||
*/
|
||||
function tellBrowserAboutPublicationChange($page, $statusMessage) {
|
||||
$JS_title = Convert::raw2js($page->TreeTitle());
|
||||
$form->loadDataFrom($page);
|
||||
|
||||
$JS_stageURL = $page->IsDeletedFromStage ? '' : Convert::raw2js($page->AbsoluteLink());
|
||||
$liveRecord = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree\".\"ID\" = $page->ID");
|
||||
|
||||
$JS_liveURL = $liveRecord ? Convert::raw2js($liveRecord->AbsoluteLink()) : '';
|
||||
|
||||
FormResponse::add($this->getActionUpdateJS($page));
|
||||
FormResponse::update_status($page->Status);
|
||||
|
||||
if($JS_stageURL || $JS_liveURL) {
|
||||
FormResponse::add("\$('sitetree').setNodeTitle($page->ID, '$JS_title');");
|
||||
} else {
|
||||
FormResponse::add("var node = $('sitetree').getTreeNodeByIdx('$page->ID');");
|
||||
FormResponse::add("if(node && node.parentTreeNode) node.parentTreeNode.removeTreeNode(node);");
|
||||
FormResponse::add("$('Form_EditForm').reloadIfSetTo($page->ID);");
|
||||
}
|
||||
|
||||
if($statusMessage) FormResponse::status_message($statusMessage, 'good');
|
||||
FormResponse::add("$('Form_EditForm').elements.StageURLSegment.value = '$JS_stageURL';");
|
||||
FormResponse::add("$('Form_EditForm').elements.LiveURLSegment.value = '$JS_liveURL';");
|
||||
FormResponse::add("$('Form_EditForm').notify('PagePublished', $('Form_EditForm').elements.ID.value);");
|
||||
|
||||
return FormResponse::respond();
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
function performRollback($id, $version) {
|
||||
@ -1085,6 +1090,7 @@ JS;
|
||||
)
|
||||
)
|
||||
);
|
||||
$form->unsetValidator();
|
||||
|
||||
return $form;
|
||||
}
|
||||
@ -1167,24 +1173,26 @@ JS;
|
||||
/**
|
||||
* Restore a completely deleted page from the SiteTree_versions table.
|
||||
*/
|
||||
function restore() {
|
||||
if(($id = $_REQUEST['ID']) && is_numeric($id)) {
|
||||
function restore($data, $form) {
|
||||
if(!isset($data['ID']) || !is_numeric($data['ID'])) {
|
||||
return new HTTPResponse("Please pass an ID in the form content", 400);
|
||||
}
|
||||
|
||||
$restoredPage = Versioned::get_latest_version("SiteTree", $id);
|
||||
if($restoredPage) {
|
||||
if(!$restoredPage) return new HTTPResponse("SiteTree #$id not found", 400);
|
||||
|
||||
$restoredPage = $restoredPage->doRestoreToStage();
|
||||
|
||||
FormResponse::get_page($id);
|
||||
$title = Convert::raw2js($restoredPage->TreeTitle());
|
||||
FormResponse::add("$('sitetree').setNodeTitle($id, '$title');");
|
||||
FormResponse::status_message(sprintf(_t('CMSMain.RESTORED',"Restored '%s' successfully",PR_MEDIUM,'Param %s is a title'),$title),'good');
|
||||
return FormResponse::respond();
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(
|
||||
_t('CMSMain.RESTORED',"Restored '%s' successfully",PR_MEDIUM,'Param %s is a title'),
|
||||
$restoredPage->TreeTitle
|
||||
)
|
||||
);
|
||||
|
||||
} else {
|
||||
return new SS_HTTPResponse("SiteTree #$id not found", 400);
|
||||
}
|
||||
} else {
|
||||
return new SS_HTTPResponse("Please pass an ID in the form content", 400);
|
||||
}
|
||||
$form = $this->getEditForm($id);
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
function duplicate() {
|
||||
@ -1202,7 +1210,8 @@ JS;
|
||||
$newPage->write();
|
||||
}
|
||||
|
||||
return $this->returnItemToUser($newPage);
|
||||
$form = $this->getEditForm($newPage->ID);
|
||||
return $form->formHtmlContent();
|
||||
} else {
|
||||
user_error("CMSMain::duplicate() Bad ID: '$id'", E_USER_WARNING);
|
||||
}
|
||||
@ -1217,14 +1226,13 @@ JS;
|
||||
|
||||
$newPage = $page->duplicateWithChildren();
|
||||
|
||||
return $this->returnItemToUser($newPage);
|
||||
$form = $this->getEditForm($newPage->ID);
|
||||
return $form->formHtmlContent();
|
||||
} else {
|
||||
user_error("CMSMain::duplicate() Bad ID: '$id'", E_USER_WARNING);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Create a new translation from an existing item, switch to this language and reload the tree.
|
||||
*/
|
||||
|
@ -344,6 +344,7 @@ class LeftAndMain extends Controller {
|
||||
SSViewer::setOption('rewriteHashlinks', false);
|
||||
return $this->EditForm()->formHtmlContent();
|
||||
} else {
|
||||
// Rendering is handled by template, which will call EditForm() eventually
|
||||
return array();
|
||||
}
|
||||
}
|
||||
@ -545,163 +546,44 @@ class LeftAndMain extends Controller {
|
||||
return $tree;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allows you to returns a new data object to the tree (subclass of sitetree)
|
||||
* and updates the tree via javascript.
|
||||
*/
|
||||
public function returnItemToUser($p) {
|
||||
if(Director::is_ajax()) {
|
||||
// Prepare the object for insertion.
|
||||
$parentID = (int) $p->ParentID;
|
||||
$id = $p->ID ? $p->ID : "new-$p->class-$p->ParentID";
|
||||
$treeTitle = Convert::raw2js($p->TreeTitle());
|
||||
$hasChildren = (is_numeric($id) && $p->AllChildren() && $p->AllChildren()->Count()) ? ' unexpanded' : '';
|
||||
$singleInstanceCSSClass = $p->stat('single_instance_only') ? $p->stat('single_instance_only_css_class') : "";
|
||||
|
||||
// Ensure there is definitly a node avaliable. if not, append to the home tree.
|
||||
$response = <<<JS
|
||||
var tree = $('sitetree');
|
||||
var newNode = tree.createTreeNode("$id", "$treeTitle", "{$p->class}{$hasChildren} {$singleInstanceCSSClass}");
|
||||
node = tree.getTreeNodeByIdx($parentID);
|
||||
if(!node) {
|
||||
node = tree.getTreeNodeByIdx(0);
|
||||
}
|
||||
node.open();
|
||||
node.appendTreeNode(newNode);
|
||||
newNode.selectTreeNode();
|
||||
JS;
|
||||
FormResponse::add($response);
|
||||
FormResponse::add($this->hideSingleInstanceOnlyFromCreateFieldJS($p));
|
||||
|
||||
return FormResponse::respond();
|
||||
} else {
|
||||
Director::redirect('admin/' . self::$url_segment . '/show/' . $p->ID);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Save and Publish page handler
|
||||
*/
|
||||
public function save($urlParams, $form) {
|
||||
public function save($data, $form) {
|
||||
$className = $this->stat('tree_class');
|
||||
$result = '';
|
||||
|
||||
$SQL_id = Convert::raw2sql($_REQUEST['ID']);
|
||||
// Existing or new record?
|
||||
$SQL_id = Convert::raw2sql($data['ID']);
|
||||
if(substr($SQL_id,0,3) != 'new') {
|
||||
$record = DataObject::get_one($className, "\"$className\".\"ID\" = {$SQL_id}");
|
||||
$record = DataObject::get_by_id($className, $SQL_id);
|
||||
if($record && !$record->canEdit()) return Security::permissionFailure($this);
|
||||
} else {
|
||||
if(!singleton($this->stat('tree_class'))->canCreate()) return Security::permissionFailure($this);
|
||||
$record = $this->getNewItem($SQL_id, false);
|
||||
}
|
||||
|
||||
// We don't want to save a new version if there are no changes
|
||||
$dataFields_new = $form->Fields()->dataFields();
|
||||
$dataFields_old = $record->getAllFields();
|
||||
$changed = false;
|
||||
$hasNonRecordFields = false;
|
||||
foreach($dataFields_new as $datafield) {
|
||||
// if the form has fields not belonging to the record
|
||||
if(!isset($dataFields_old[$datafield->Name()])) {
|
||||
$hasNonRecordFields = true;
|
||||
}
|
||||
// if field-values have changed
|
||||
if(!isset($dataFields_old[$datafield->Name()]) || $dataFields_old[$datafield->Name()] != $datafield->dataValue()) {
|
||||
$changed = true;
|
||||
}
|
||||
}
|
||||
|
||||
if(!$changed && !$hasNonRecordFields) {
|
||||
// Tell the user we have saved even though we haven't, as not to confuse them
|
||||
if(is_a($record, "Page")) {
|
||||
$record->Status = "Saved (update)";
|
||||
}
|
||||
FormResponse::status_message(_t('LeftAndMain.SAVEDUP',"Saved"), "good");
|
||||
FormResponse::update_status($record->Status);
|
||||
return FormResponse::respond();
|
||||
}
|
||||
|
||||
$form->dataFieldByName('ID')->Value = 0;
|
||||
|
||||
if(isset($urlParams['Sort']) && is_numeric($urlParams['Sort'])) {
|
||||
$record->Sort = $urlParams['Sort'];
|
||||
}
|
||||
|
||||
// HACK: This should be turned into something more general
|
||||
$originalClass = $record->ClassName;
|
||||
$originalStatus = $record->Status;
|
||||
$originalParentID = $record->ParentID;
|
||||
|
||||
$record->HasBrokenLink = 0;
|
||||
$record->HasBrokenFile = 0;
|
||||
|
||||
$record->writeWithoutVersion();
|
||||
|
||||
// HACK: This should be turned into something more general
|
||||
$originalURLSegment = $record->URLSegment;
|
||||
|
||||
$form->saveInto($record, true);
|
||||
|
||||
if(is_a($record, "Page")) {
|
||||
$record->Status = ($record->Status == "New page" || $record->Status == "Saved (new)") ? "Saved (new)" : "Saved (update)";
|
||||
}
|
||||
|
||||
if(Director::is_ajax()) {
|
||||
if($SQL_id != $record->ID) {
|
||||
FormResponse::add("$('sitetree').setNodeIdx(\"{$SQL_id}\", \"$record->ID\");");
|
||||
FormResponse::add("$('Form_EditForm').elements.ID.value = \"$record->ID\";");
|
||||
}
|
||||
|
||||
if($added = DataObjectLog::getAdded('SiteTree')) {
|
||||
foreach($added as $page) {
|
||||
if($page->ID != $record->ID) $result .= $this->addTreeNodeJS($page);
|
||||
}
|
||||
}
|
||||
if($deleted = DataObjectLog::getDeleted('SiteTree')) {
|
||||
foreach($deleted as $page) {
|
||||
if($page->ID != $record->ID) $result .= $this->deleteTreeNodeJS($page);
|
||||
}
|
||||
}
|
||||
if($changed = DataObjectLog::getChanged('SiteTree')) {
|
||||
foreach($changed as $page) {
|
||||
if($page->ID != $record->ID) {
|
||||
$title = Convert::raw2js($page->TreeTitle());
|
||||
FormResponse::add("$('sitetree').setNodeTitle($page->ID, \"$title\");");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$message = _t('LeftAndMain.SAVEDUP');
|
||||
|
||||
// Update the class instance if necessary
|
||||
if($originalClass != $record->ClassName) {
|
||||
if($data['ClassName'] != $record->ClassName) {
|
||||
$newClassName = $record->ClassName;
|
||||
// The records originally saved attribute was overwritten by $form->saveInto($record) before.
|
||||
// This is necessary for newClassInstance() to work as expected, and trigger change detection
|
||||
// on the ClassName attribute
|
||||
$record->setClassName($originalClass);
|
||||
$record->setClassName($data['ClassName']);
|
||||
// Replace $record with a new instance
|
||||
$record = $record->newClassInstance($newClassName);
|
||||
|
||||
// update the tree icon
|
||||
FormResponse::add("if(\$('sitetree').setNodeIcon) \$('sitetree').setNodeIcon($record->ID, '$originalClass', '$record->ClassName');");
|
||||
}
|
||||
|
||||
// HACK: This should be turned into somethign more general
|
||||
if( ($record->class == 'VirtualPage' && $originalURLSegment != $record->URLSegment) ||
|
||||
($originalClass != $record->ClassName) || self::$ForceReload == true) {
|
||||
FormResponse::add("$('Form_EditForm').getPageFromServer($record->ID);");
|
||||
}
|
||||
|
||||
// After reloading action
|
||||
if($originalStatus != $record->Status) {
|
||||
$message .= sprintf(_t('LeftAndMain.STATUSTO'," Status changed to '%s'"),$record->Status);
|
||||
}
|
||||
|
||||
if($originalParentID != $record->ParentID) {
|
||||
FormResponse::add("if(\$('sitetree').setNodeParentID) \$('sitetree').setNodeParentID($record->ID, $record->ParentID);");
|
||||
}
|
||||
|
||||
// save form data into record
|
||||
$form->saveInto($record, true);
|
||||
$record->write();
|
||||
|
||||
// if changed to a single_instance_only page type
|
||||
@ -713,29 +595,22 @@ JS;
|
||||
FormResponse::add("jQuery('#sitetree li.{$record->ClassName}').removeClass('{$record->stat('single_instance_only_css_class')}');");
|
||||
}
|
||||
// if chnaged from a single_instance_only page type
|
||||
$sampleOriginalClassObject = new $originalClass();
|
||||
$sampleOriginalClassObject = new $data['ClassName']();
|
||||
if($sampleOriginalClassObject->stat('single_instance_only')) {
|
||||
FormResponse::add($this->showSingleInstanceOnlyInCreateFieldJS($sampleOriginalClassObject));
|
||||
}
|
||||
|
||||
if( ($record->class != 'VirtualPage') && $originalURLSegment != $record->URLSegment) {
|
||||
$message .= sprintf(_t('LeftAndMain.CHANGEDURL'," Changed URL to '%s'"),$record->URLSegment);
|
||||
FormResponse::add("\$('Form_EditForm').elements.URLSegment.value = \"$record->URLSegment\";");
|
||||
FormResponse::add("\$('Form_EditForm_StageURLSegment').value = \"{$record->URLSegment}\";");
|
||||
}
|
||||
|
||||
// If the 'Save & Publish' button was clicked, also publish the page
|
||||
if (isset($urlParams['publish']) && $urlParams['publish'] == 1) {
|
||||
if (isset($data['publish']) && $data['publish'] == 1) {
|
||||
$record->doPublish();
|
||||
$this->extend('onAfterSave', $record);
|
||||
|
||||
$record->doPublish();
|
||||
|
||||
// Update classname with original and get new instance (see above for explanation)
|
||||
$record->setClassName($originalClass);
|
||||
$record->setClassName($data['ClassName']);
|
||||
$publishedRecord = $record->newClassInstance($record->ClassName);
|
||||
|
||||
return $this->tellBrowserAboutPublicationChange(
|
||||
$publishedRecord,
|
||||
$this->response->addHeader(
|
||||
'X-Status',
|
||||
sprintf(
|
||||
_t(
|
||||
'LeftAndMain.STATUSPUBLISHEDSUCCESS',
|
||||
@ -743,22 +618,20 @@ JS;
|
||||
PR_MEDIUM,
|
||||
'Status message after publishing a page, showing the page title'
|
||||
),
|
||||
$record->Title
|
||||
$publishedRecord->Title
|
||||
)
|
||||
);
|
||||
|
||||
$form->loadDataFrom($publishedRecord);
|
||||
} else {
|
||||
// BUGFIX: Changed icon only shows after Save button is clicked twice http://support.silverstripe.com/gsoc/ticket/76
|
||||
$title = Convert::raw2js($record->TreeTitle());
|
||||
FormResponse::add("$('sitetree').setNodeTitle(\"$record->ID\", \"$title\");");
|
||||
$result .= $this->getActionUpdateJS($record);
|
||||
FormResponse::status_message($message, "good");
|
||||
FormResponse::update_status($record->Status);
|
||||
|
||||
$this->extend('onAfterSave', $record);
|
||||
$this->response->addHeader('X-Status', _t('LeftAndMain.SAVEDUP'));
|
||||
|
||||
return FormResponse::respond();
|
||||
}
|
||||
// write process might've changed the record, so we reload before returning
|
||||
$form->loadDataFrom($record);
|
||||
}
|
||||
|
||||
return $form->formHtmlContent();
|
||||
}
|
||||
|
||||
/**
|
||||
@ -809,54 +682,6 @@ if (singleSingleOnlyOfThisPageType.length == 0) {
|
||||
JS;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a piece of javascript that will update the actions of the main form
|
||||
*/
|
||||
public function getActionUpdateJS($record) {
|
||||
// Get the new action buttons
|
||||
|
||||
$tempForm = $this->getEditForm($record->ID);
|
||||
$actionList = '';
|
||||
foreach($tempForm->Actions() as $action) {
|
||||
$actionList .= $action->Field() . ' ';
|
||||
}
|
||||
|
||||
FormResponse::add("$('Form_EditForm').loadActionsFromString('" . Convert::raw2js($actionList) . "');");
|
||||
|
||||
return FormResponse::respond();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return JavaScript code to generate a tree node for the given page, if visible
|
||||
*/
|
||||
public function addTreeNodeJS($page, $select = false) {
|
||||
$parentID = (int)$page->ParentID;
|
||||
$title = Convert::raw2js($page->TreeTitle());
|
||||
$response = <<<JS
|
||||
var newNode = $('sitetree').createTreeNode($page->ID, "$title", "$page->class");
|
||||
var parentNode = $('sitetree').getTreeNodeByIdx($parentID);
|
||||
if(parentNode) parentNode.appendTreeNode(newNode);
|
||||
JS;
|
||||
$response .= ($select ? "newNode.selectTreeNode();\n" : "") ;
|
||||
FormResponse::add($response);
|
||||
return FormResponse::respond();
|
||||
}
|
||||
/**
|
||||
* Return JavaScript code to remove a tree node for the given page, if it exists.
|
||||
*/
|
||||
public function deleteTreeNodeJS($page) {
|
||||
$id = $page->ID ? $page->ID : $page->OldID;
|
||||
$response = <<<JS
|
||||
var node = $('sitetree').getTreeNodeByIdx($id);
|
||||
if(node && node.parentTreeNode) node.parentTreeNode.removeTreeNode(node);
|
||||
$('Form_EditForm').closeIfSetTo($id);
|
||||
JS;
|
||||
FormResponse::add($response);
|
||||
|
||||
if ($this instanceof LeftAndMain) FormResponse::add($this->showSingleInstanceOnlyInCreateFieldJS($page));
|
||||
return FormResponse::respond();
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets a static variable on this class which means the panel will be reloaded.
|
||||
*/
|
||||
|
@ -329,6 +329,8 @@ SiteTree.prototype = {
|
||||
*/
|
||||
|
||||
this.observeMethod('SelectionChanged', this.interruptLoading.bind(this) );
|
||||
|
||||
jQuery('#Form_EditForm').bind('loadnewpage', this.onLoadNewPage.bind(this));
|
||||
},
|
||||
destroy: function () {
|
||||
if(this.Tree) this.Tree.destroy();
|
||||
@ -344,6 +346,42 @@ SiteTree.prototype = {
|
||||
interruptLoading: function( newLoadingNode ) {
|
||||
if( this.loadingNode ) this.loadingNode.removeNodeClass('loading');
|
||||
this.loadingNode = newLoadingNode;
|
||||
},
|
||||
|
||||
onLoadNewPage: function(e, eventData) {
|
||||
// finds a certain value in an array generated by jQuery.serializeArray()
|
||||
var findInSerializedArray = function(arr, name) {
|
||||
for(var i=0; i<arr.length; i++) {
|
||||
if(arr[i].name == name) return arr[i].value;
|
||||
};
|
||||
return false;
|
||||
};
|
||||
|
||||
var id = jQuery(e.target.ID).val();
|
||||
|
||||
// check if a form with a valid ID exists
|
||||
if(id) {
|
||||
// set current tree element
|
||||
this.setCurrentByIdx(id);
|
||||
|
||||
// set correct parent (only if it has changed)
|
||||
var parentID = jQuery(e.target.ParentID).val();
|
||||
if(parentID) this.setNodeParentID(id, jQuery(e.target.ParentID).val());
|
||||
|
||||
// set title (either from TreeTitle or from Title fields)
|
||||
// Treetitle has special HTML formatting to denote the status changes.
|
||||
var title = jQuery((e.target.TreeTitle) ? e.target.TreeTitle : e.target.Title).val();
|
||||
if(title) this.setNodeTitle(id, title);
|
||||
|
||||
// update icon (only if it has changed)
|
||||
var className = jQuery(e.target.ClassName).val();
|
||||
if(className) this.setNodeIcon(id, className);
|
||||
|
||||
} else {
|
||||
var node = this.getTreeNodeByIdx(origData.id);
|
||||
if(node && node.parentTreeNode) node.parentTreeNode.removeTreeNode(node);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,322 +1,3 @@
|
||||
|
||||
CMSForm = Class.extend('ChangeTracker').extend('Observable');
|
||||
CMSForm.prototype = {
|
||||
initialize : function(fn) {
|
||||
this.ChangeTracker.initialize();
|
||||
this.formName = fn;
|
||||
this.prepareForm();
|
||||
},
|
||||
|
||||
/**
|
||||
* Trigger normal save event, helpful e.g. when enter key is pressed in
|
||||
* single line input fields.
|
||||
*/
|
||||
onsubmit: function(e) {
|
||||
this.save();
|
||||
|
||||
Event.stop(e);
|
||||
return false;
|
||||
},
|
||||
|
||||
/**
|
||||
* Processing called whenever a page is loaded in the right - including the initial one
|
||||
*/
|
||||
prepareForm : function() {
|
||||
},
|
||||
|
||||
/**
|
||||
* Load actions from a string containing the HTML content
|
||||
*/
|
||||
loadActionsFromString : function(actionHTML) {
|
||||
var actionHolder = $('form_actions_' + this.formName);
|
||||
actionHolder.innerHTML = actionHTML;
|
||||
},
|
||||
|
||||
/**
|
||||
* Close the form down without any corrective action, after its been deleted.
|
||||
*/
|
||||
closeIfSetTo: function(id) {
|
||||
if(this.elements.ID && this.elements.ID.value == id) {
|
||||
// Note: TinyMCE coupling
|
||||
jQuery('#Form_EditForm').concrete('ss').cleanup();
|
||||
this.innerHTML = "<p>" + ss.i18n._t('LeftAndMain.PAGEWASDELETED') + "</p>";
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Reload lose the form if the current page is open.
|
||||
*/
|
||||
reloadIfSetTo: function(id) {
|
||||
if(this.elements.ID && this.elements.ID.value == id) {
|
||||
this.getPageFromServer(id);
|
||||
}
|
||||
},
|
||||
|
||||
close: function() {
|
||||
this.innerHTML = "<p> </p>";
|
||||
var actions;
|
||||
if(actions = $('form_actions_' + this.formName)) {
|
||||
actions.parentNode.removeChild(actions);
|
||||
}
|
||||
},
|
||||
|
||||
updateStatus: function( newStatus ) {
|
||||
|
||||
if( $('Form_EditForm_Status') )
|
||||
$('Form_EditForm_Status').innerHTML = "STATUS: " + newStatus;
|
||||
},
|
||||
|
||||
/**
|
||||
* Load a new page into the right-hand form.
|
||||
*
|
||||
* @param formContent string
|
||||
* @param response object (optional)
|
||||
* @param evalResponse boolean (optional)
|
||||
*/
|
||||
loadNewPage : function(formContent, response, evalResponse) {
|
||||
//alert('here: ' + formContent);
|
||||
var rightHTML = formContent;
|
||||
|
||||
// Rewrite # links
|
||||
rightHTML = rightHTML.replace(/(<a[^>]+href *= *")#/g, '$1' + window.location.href.replace(/#.*$/,'') + '#');
|
||||
|
||||
// Rewrite iframe links (for IE)
|
||||
rightHTML = rightHTML.replace(/(<iframe[^>]*src=")([^"]+)("[^>]*>)/g, '$1' + jQuery('base').attr('href') + '$2$3');
|
||||
|
||||
// Note: TinyMCE coupling
|
||||
jQuery('#Form_EditForm').concrete('ss').cleanup();
|
||||
|
||||
// Prepare iframes for removal, otherwise we get loading bugs
|
||||
var i, allIframes = this.getElementsByTagName('iframe');
|
||||
if(allIframes) for(i=0;i<allIframes.length;i++) {
|
||||
allIframes[i].contentWindow.location.href = 'about:blank';
|
||||
allIframes[i].parentNode.removeChild(allIframes[i]);
|
||||
}
|
||||
|
||||
if(response && evalResponse) {
|
||||
Ajax.Evaluator(response);
|
||||
} else {
|
||||
this.innerHTML = rightHTML;
|
||||
}
|
||||
|
||||
// Get the form attributes from embedded fields
|
||||
var attr;
|
||||
|
||||
for(attr in {'action':true ,'method':true,'enctype':true,'name':true}) {
|
||||
if(this.elements['_form_' + attr]) {
|
||||
this[attr] = this.elements['_form_' + attr].value;
|
||||
this.elements['_form_' + attr].parentNode.removeChild(this.elements['_form_' + attr]);
|
||||
}
|
||||
}
|
||||
|
||||
allIframes = this.getElementsByTagName('iframe');
|
||||
if(allIframes) for(i=0;i<allIframes.length;i++) {
|
||||
try {
|
||||
allIframes[i].contentWindow.location.href = allIframes[i].src;
|
||||
} catch(er) { alert('Error in LeftAndMain_right.js CMSForm.loadNewPage: ' + er.message); }
|
||||
}
|
||||
|
||||
_TAB_DIVS_ON_PAGE = [];
|
||||
|
||||
//initTabstripe() become livequery in tabstrip.js, so we don't need to call it for each tab strip here.
|
||||
|
||||
|
||||
// We assume that an evaluated response is generated by FormResponse
|
||||
// which takes care of calling these method it
|
||||
if (!evalResponse) {
|
||||
if (this.prepareForm) this.prepareForm();
|
||||
Behaviour.apply(this);
|
||||
}
|
||||
|
||||
if(this.resetElements) this.resetElements();
|
||||
|
||||
window.ontabschanged();
|
||||
|
||||
// If there's a title field and it's got a "new XX" value, focus/select that first
|
||||
// This is really a little too CMS-specific (as opposed to LeftAndMain), but the cleanup can happen after jQuery refactoring
|
||||
if($('input#Form_EditForm_Title') && $('input#Form_EditForm_Title').value.match(/^new/i)) {
|
||||
$('input#Form_EditForm_Title').select();
|
||||
}
|
||||
},
|
||||
/**
|
||||
* Save the contens of the form, by submitting it and resetting is changed checker
|
||||
* on success.
|
||||
*
|
||||
* @param publish boolean (optional) whether to publish in addition to saving
|
||||
*/
|
||||
save: function(ifChanged, callAfter, action, publish) {
|
||||
_AJAX_LOADING = true;
|
||||
// Note: TinyMCE coupling
|
||||
if(typeof tinyMCE != 'undefined') tinyMCE.triggerSave();
|
||||
if(!action) action = "save";
|
||||
|
||||
var __callAfter = callAfter;
|
||||
var __form = this;
|
||||
|
||||
if(__form.notify && __form.elements.ID != undefined) __form.notify('BeforeSave', __form.elements.ID.value);
|
||||
|
||||
// validate if required
|
||||
if(this.validate && !this.validate()) {
|
||||
// TODO Automatically switch to the tab/position of the first error
|
||||
statusMessage("Validation failed.", "bad");
|
||||
|
||||
if($('Form_EditForm_action_save') && $('Form_EditForm_action_save').stopLoading) $('Form_EditForm_action_save').stopLoading();
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
var success = function(response) {
|
||||
Ajax.Evaluator(response);
|
||||
|
||||
__form.resetElements();
|
||||
if(__callAfter) __callAfter();
|
||||
if(__form.notify && __form.elements.ID != undefined) __form.notify('PageSaved', __form.elements.ID.value);
|
||||
if($('Form_EditForm_action_save') && $('Form_EditForm_action_save').stopLoading) $('Form_EditForm_action_save').stopLoading();
|
||||
_AJAX_LOADING = false;
|
||||
}
|
||||
|
||||
if(ifChanged) {
|
||||
var data = this.serializeChangedFields('ID') + '&ajax=1&action_' + action + '=1';
|
||||
} else {
|
||||
var data = this.serializeAllFields() + '&ajax=1&action_' + action + '=1';
|
||||
}
|
||||
if(publish)
|
||||
{
|
||||
data += '&publish=1';
|
||||
}
|
||||
|
||||
statusMessage(ss.i18n._t('CMSMAIN.SAVING'), null, true);
|
||||
|
||||
new Ajax.Request(this.action, {
|
||||
method : this.method,
|
||||
postBody: data,
|
||||
onSuccess : success,
|
||||
onFailure : function(response) {
|
||||
errorMessage('Error saving content', response);
|
||||
if($('Form_EditForm_action_save') && $('Form_EditForm_action_save').stopLoading) $('Form_EditForm_action_save').stopLoading();
|
||||
_AJAX_LOADING = false;
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
loadPage_url : 'admin/getpage'
|
||||
}
|
||||
|
||||
CMSRightForm = Class.extend('CMSForm');
|
||||
CMSRightForm.prototype = {
|
||||
intialize: function() {
|
||||
this.CMSForm.initialize('right');
|
||||
},
|
||||
|
||||
|
||||
/**
|
||||
* Load the given URL (with &ajax=1) into this form
|
||||
*/
|
||||
loadURLFromServer : function(url) {
|
||||
var urlParts = url.match( /ID=(\d+)/ );
|
||||
var id = urlParts ? urlParts[1] : null;
|
||||
|
||||
if( !url.match( /^https?:\/\/.*/ ) )
|
||||
url = document.getElementsByTagName('base')[0].href + url;
|
||||
|
||||
new Ajax.Request( url + '&ajax=1', {
|
||||
asynchronous : true,
|
||||
onSuccess : function( response ) {
|
||||
$('Form_EditForm').successfullyReceivedPage(response,id);
|
||||
},
|
||||
onFailure : function(response) {
|
||||
alert(response.responseText);
|
||||
errorMessage('error loading page',response);
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
successfullyReceivedPage : function(response,pageID) {
|
||||
var loadingNode = $('sitetree').loadingNode;
|
||||
|
||||
if( loadingNode && pageID && parseInt( $('sitetree').getIdxOf( loadingNode ) ) != pageID ) {
|
||||
return;
|
||||
}
|
||||
|
||||
// must wait until the javascript has finished
|
||||
document.body.style.cursor = 'wait';
|
||||
|
||||
this.loadNewPage(response.responseText);
|
||||
|
||||
var subform;
|
||||
if(subform = $('Form_MemberForm')) subform.close();
|
||||
if(subform = $('Form_SubForm')) subform.close();
|
||||
|
||||
if(this.elements.ID) {
|
||||
this.notify('PageLoaded', this.elements.ID.value);
|
||||
}
|
||||
|
||||
if(this.receivingID) {
|
||||
// Treenode might not exist if that part of the tree is closed
|
||||
var treeNode = loadingNode ? loadingNode : $('sitetree').getTreeNodeByIdx(this.receivingID);
|
||||
if(treeNode) {
|
||||
$('sitetree').setCurrentByIdx(treeNode.getIdx());
|
||||
treeNode.removeNodeClass('loading');
|
||||
}
|
||||
statusMessage('');
|
||||
}
|
||||
|
||||
// must wait until the javascript has finished
|
||||
document.body.style.cursor = 'default';
|
||||
|
||||
},
|
||||
didntReceivePage : function(response) {
|
||||
errorMessage('error loading page', response);
|
||||
$('sitetree').getTreeNodeByIdx(this.elements.ID.value).removeNodeClass('loading');
|
||||
},
|
||||
|
||||
|
||||
/**
|
||||
* Request a page from the server via Ajax
|
||||
*/
|
||||
getPageFromServer : function(id, treeNode) {
|
||||
// if(id && id.match(/^[A-Za-z0-9_]+$/)) {
|
||||
if(id && (id == 'root' || parseInt(id) == id || (id.substr && id.substr(0,3) == 'new') )) {
|
||||
this.receivingID = id;
|
||||
|
||||
// Treenode might not exist if that part of the tree is closed
|
||||
if(!treeNode) treeNode = $('sitetree').getTreeNodeByIdx(id);
|
||||
|
||||
if(treeNode) {
|
||||
$('sitetree').loadingNode = treeNode;
|
||||
treeNode.addNodeClass('loading');
|
||||
url = treeNode.aTag.href + (treeNode.aTag.href.indexOf('?')==-1?'?':'&') + 'ajax=1';
|
||||
}
|
||||
if(SiteTreeHandlers.loadPage_url) {
|
||||
var sep = (SiteTreeHandlers.loadPage_url.indexOf('?') == -1) ? '?' : '&';
|
||||
url = SiteTreeHandlers.loadPage_url + sep + 'ID=' + id;
|
||||
}
|
||||
|
||||
// used to set language in CMSMain->init()
|
||||
var lang = $('LangSelector') ? $F('LangSelector') : null;
|
||||
if(lang) {
|
||||
url += '&locale='+lang;
|
||||
}
|
||||
|
||||
statusMessage("loading...");
|
||||
this.loadURLFromServer(url);
|
||||
} else {
|
||||
throw("getPageFromServer: Bad page ID: " + id);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the status field
|
||||
*/
|
||||
/*setStatus: function(newStatus) {
|
||||
var statusLabel = document.getElementsBySelector('label.pageStatusMessage')[0];
|
||||
if(statusLabel) statusLabel.innerHTML = "STATUS: " + newStatus;
|
||||
}*/
|
||||
}
|
||||
|
||||
CMSForm.applyTo('#Form_SubForm');
|
||||
CMSRightForm.applyTo('#Form_EditForm', 'right');
|
||||
|
||||
/**
|
||||
* Handle auto-saving. Detects if changes have been made, and if so save everything on the page.
|
||||
* If confirmation is true it will ask for confirmation.
|
||||
|
Loading…
x
Reference in New Issue
Block a user