array('CMSMain.PAGETYPE', 'Page Type'), 'Status' => array('CMSMain.STATUS', 'Status'), 'MetaDescription' => array('CMSMain.METADESC', 'Description'), 'MetaKeywords' => array('CMSMain.METAKEYWORDS', 'Keywords') ); static function T_SiteTreeFilterOptions(){ return array( 'ClassName' => _t('CMSMain.PAGETYPEOPT', 'Page Type', 0, "The dropdown title in CMSMain left SiteTreeFilterOptions"), 'Status' => _t('CMSMain.STATUSOPT', 'Status', 0, "The dropdown title in CMSMain left SiteTreeFilterOptions"), 'MetaDescription' => _t('CMSMain.METADESCOPT', 'Description', 0, "The dropdown title in CMSMain left SiteTreeFilterOptions"), 'MetaKeywords' => _t('CMSMain.METAKEYWORDSOPT', 'Keywords', 0, "The dropdown title in CMSMain left SiteTreeFilterOptions"), ); } public function init() { parent::init(); // "Lang" attribute is either explicitly added by LeftAndMain Javascript logic, // or implied on a translated record (see {@link Translatable->updateCMSFields()}). if(Translatable::is_enabled()) { // $Lang serves as a "context" which can be inspected by Translatable - hence it // has the same name as the database property on Translatable. if($this->getRequest()->requestVar("Lang")) { $this->Lang = $this->getRequest()->requestVar("Lang"); } elseif($this->getRequest()->requestVar("lang")) { $this->Lang = $this->getRequest()->requestVar("lang"); } else { $this->Lang = Translatable::default_lang(); } Translatable::set_reading_lang($this->Lang); } // collect languages for TinyMCE spellchecker plugin if(Translatable::is_enabled()) { $spellcheckLangs = Translatable::get_existing_content_languages(); } else { $defaultLang = Translatable::default_lang(); $spellcheckLangs = array($defaultLang => i18n::get_language_name($defaultLang)); } $spellcheckSpec = array(); foreach($spellcheckLangs as $lang => $title) $spellcheckSpec[] = "{$title}={$lang}"; // We don't want this showing up in every ajax-response, it should always be present in a CMS-environment if(!Director::is_ajax()) { Requirements::javascript(MCE_ROOT . "tiny_mce_src.js"); Requirements::javascriptTemplate(CMS_DIR . "/javascript/tinymce.template.js", array( "ContentCSS" => (SSViewer::current_theme() ? THEMES_DIR . "/" . SSViewer::current_theme() : project()) . "/css/editor.css", "BaseURL" => Director::absoluteBaseURL(), "Lang" => i18n::get_tinymce_lang(), 'SpellcheckLangs' => '+' . implode(',', $spellcheckSpec) )); } // Always block the HtmlEditorField.js otherwise it will be sent with an ajax request Requirements::block(SAPPHIRE_DIR . '/javascript/HtmlEditorField.js'); Requirements::javascript(CMS_DIR . '/javascript/CMSMain.js'); Requirements::javascript(CMS_DIR . '/javascript/CMSMain_left.js'); Requirements::javascript(CMS_DIR . '/javascript/CMSMain_right.js'); } /** * If this is set to true, the "switchView" context in the * template is shown, with links to the staging and publish site. * * @return boolean */ function ShowSwitchView() { return true; } //------------------------------------------------------------------------------------------// // Main controllers //------------------------------------------------------------------------------------------// // Main UI components /** * Override {@link LeftAndMain} Link to allow blank URL segment for CMSMain. * * @return string */ public function Link($action = null) { return Controller::join_links( $this->stat('url_base', true), $this->stat('url_segment', true), // in case we want to change the segment '/', // trailing slash needed if $action is null! "$action" ); } /** * Return the entire site tree as a nested set of ULs */ public function SiteTreeAsUL() { $this->generateDataTreeHints(); $this->generateTreeStylingJS(); // Pre-cache sitetree version numbers for querying efficiency Versioned::prepopulate_versionnumber_cache("SiteTree", "Stage"); Versioned::prepopulate_versionnumber_cache("SiteTree", "Live"); return $this->getSiteTreeFor("SiteTree"); } /** * Returns the SiteTree columns that can be filtered using the the Site Tree Search button as a DataObjectSet */ public function SiteTreeFilterOptions() { $filter_options = new DataObjectSet(); foreach(self::T_SiteTreeFilterOptions() as $key => $value) { $record = array( 'Column' => $key, 'Title' => $value, ); $filter_options->push(new ArrayData($record)); } return $filter_options; } public function SiteTreeFilterDateField() { $dateField = new CalendarDateField('SiteTreeFilterDate'); return $dateField->Field(); } /** * Returns a filtered Site Tree */ public function filtersitetree() { // Pre-cache sitetree version numbers for querying efficiency Versioned::prepopulate_versionnumber_cache("SiteTree", "Stage"); Versioned::prepopulate_versionnumber_cache("SiteTree", "Live"); $className = 'SiteTree'; $rootID = null; $obj = $rootID ? $this->getRecord($rootID) : singleton($className); $obj->setMarkingFilterFunction('cmsMainMarkingFilterFunction'); $obj->markPartialTree(); if($p = $this->currentPage()) $obj->markToExpose($p); // getChildrenAsUL is a flexible and complex way of traversing the tree $siteTree = $obj->getChildrenAsUL("", ' "
  • ID\" class=\"" . $child->CMSTreeClasses($extraArg) . "\">" . "Link(),0,-1), "show", $child->ID) . "\" " . (($child->canEdit() || $child->canAddChildren()) ? "" : "class=\"disabled\"") . " title=\"' . _t('LeftAndMain.PAGETYPE') . '".$child->class."\" >" . ($child->TreeTitle()) . "" ' ,$this, true); // Wrap the root if needs be. if(!$rootID) { $rootLink = $this->Link() . '0'; $siteTree = ""; } return $siteTree; } public function generateDataTreeHints() { $classes = ClassInfo::subclassesFor( $this->stat('tree_class') ); $def['Root'] = array(); foreach($classes as $class) { $obj = singleton($class); if($obj instanceof HiddenClass) continue; $allowedChildren = $obj->allowedChildren(); if($allowedChildren != "none") $def[$class]['allowedChildren'] = $allowedChildren; $def[$class]['defaultChild'] = $obj->defaultChild(); $def[$class]['defaultParent'] = isset(SiteTree::get_by_url($obj->defaultParent())->ID) ? SiteTree::get_by_url($obj->defaultParent())->ID : null; if(is_array($allowedChildren)) foreach($allowedChildren as $allowedChild) { $def[$allowedChild]['allowedParents'][] = $class; } if($obj->stat('can_be_root')) { $def['Root']['allowedChildren'][] = $class; } } // Put data hints into a script tag at the top Requirements::customScript("siteTreeHints = " . $this->jsDeclaration($def) . ";"); } public function generateTreeStylingJS() { $classes = ClassInfo::subclassesFor('SiteTree'); foreach($classes as $class) { $obj = singleton($class); if($obj instanceof HiddenClass) continue; if($icon = $obj->stat('icon')) $iconInfo[$class] = $icon; } $iconInfo['BrokenLink'] = 'cms/images/treeicons/brokenlink'; $js = "var _TREE_ICONS = [];\n"; foreach($iconInfo as $class => $icon) { // SiteTree::$icon can be set to array($icon, $option) // $option can be "file" or "folder" to force the icon to always be the file or the folder form $option = null; if(is_array($icon)) list($icon, $option) = $icon; $fileImage = ($option == "folder") ? $icon . '-openfolder.gif' : $icon . '-file.gif'; $openFolderImage = $icon . '-openfolder.gif'; if(!Director::fileExists($openFolderImage) || $option = "file") $openFolderImage = $fileImage; $closedFolderImage = $icon . '-closedfolder.gif'; if(!Director::fileExists($closedFolderImage) || $option = "file") $closedFolderImage = $fileImage; $js .= << $v) { $parts[] = "$k : " . $this->jsDeclaration($v); } return " {\n " . implode(", \n", $parts) . " }\n"; } else { foreach($array as $part) $parts[] = $this->jsDeclaration($part); return " [ " . implode(", ", $parts) . " ]\n"; } } else { return "'" . addslashes($array) . "'"; } } /** * Populates an array of classes in the CMS * which allows the user to change the page type. * * @return DataObjectSet */ public function PageTypes() { $classes = ClassInfo::getValidSubClasses(); array_shift($classes); $result = new DataObjectSet(); $kill_ancestors = array(); // figure out if there are any classes we don't want to appear foreach($classes as $class) { $instance = singleton($class); // do any of the progeny want to hide an ancestor? if($ancestor_to_hide = $instance->stat('hide_ancestor')) { // note for killing later $kill_ancestors[] = $ancestor_to_hide; } } // If any of the descendents don't want any of the elders to show up, cruelly render the elders surplus to requirements. if($kill_ancestors) { foreach ($kill_ancestors as $mark) { // unset from $classes $idx = array_search($mark, $classes); unset($classes[$idx]); } } foreach($classes as $class) { $instance = singleton($class); if($instance instanceof HiddenClass) continue; if(!$instance->canCreate()) continue; // skip this type if it is restricted if($instance->stat('need_permission') && !$this->can(singleton($class)->stat('need_permission'))) continue; $addAction = $instance->i18n_singular_name(); // if we're in translation mode, the link between the translated pagetype // title and the actual classname might not be obvious, so we add it in parantheses // Example: class "RedirectorPage" has the title "Weiterleitung" in German, // so it shows up as "Weiterleitung (RedirectorPage)" if(i18n::get_locale() != 'en_US') { $addAction .= " ({$class})"; } $result->push(new ArrayData(array( 'ClassName' => $class, 'AddAction' => $addAction, ))); } $result->sort('AddAction'); return $result; } /** * Get a databsae record to be managed by the CMS */ public function getRecord($id) { $treeClass = $this->stat('tree_class'); if($id && is_numeric($id)) { $record = DataObject::get_one( $treeClass, "\"$treeClass\".\"ID\" = $id"); if(!$record) { // $record = Versioned::get_one_by_stage($treeClass, "Live", "\"$treeClass\".\"ID\" = $id"); Versioned::reading_stage('Live'); singleton($treeClass)->flushCache(); $record = DataObject::get_one( $treeClass, "\"$treeClass\".\"ID\" = $id"); if($record) Versioned::reading_stage(null); } return $record; } else if(substr($id,0,3) == 'new') { return $this->getNewItem($id); } } public function getEditForm($id) { $record = $this->getRecord($id); if($record) { if($record->IsDeletedFromStage) $record->Status = _t('CMSMain.REMOVEDFD',"Removed from the draft site"); $fields = $record->getCMSFields($this); if ($fields == null) { user_error("getCMSFields returned null on a 'Page' object - it should return a FieldSet object. Perhaps you forgot to put a return statement at the end of your method?", E_USER_ERROR); } $fields->push($idField = new HiddenField("ID")); $fields->push($liveURLField = new HiddenField("LiveURLSegment")); $fields->push($stageURLField = new HiddenField("StageURLSegment")); /*if( substr($record->ID, 0, 3 ) == 'new' )*/ $fields->push(new HiddenField('Sort','', $record->Sort )); $idField->setValue($id); if($record->ID && is_numeric( $record->ID ) ) { $liveRecord = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree\".\"ID\" = $record->ID"); if($liveRecord) $liveURLField->setValue($liveRecord->AbsoluteLink()); } if(!$record->IsDeletedFromStage) { $stageURLField->setValue($record->AbsoluteLink()); } // getAllCMSActions can be used to completely redefine the action list if($record->hasMethod('getAllCMSActions')) { $actions = $record->getAllCMSActions(); } else { $actions = $record->getCMSActions(); // add default actions if none are defined if(!$actions || !$actions->Count()) { if($record->canEdit()) { $actions->push(new FormAction('save',_t('CMSMain.SAVE','Save'))); $actions->push($deleteAction = new FormAction('delete',_t('CMSMain.DELETE','Delete from the draft site'))); $deleteAction->addExtraClass('delete'); } } } $form = new Form($this, "EditForm", $fields, $actions); $form->loadDataFrom($record); $form->disableDefaultAction(); if(!$record->canEdit() || $record->IsDeletedFromStage) { $readonlyFields = $form->Fields()->makeReadonly(); $form->setFields($readonlyFields); } return $form; } else if($id) { return new Form($this, "EditForm", new FieldSet( new LabelField('PageDoesntExistLabel',_t('CMSMain.PAGENOTEXISTS',"This page doesn't exist"))), new FieldSet()); } } //------------------------------------------------------------------------------------------// // 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; if(!$parent && isset($_REQUEST['Parent'])) { $page = SiteTree::get_by_url($_REQUEST['Parent']); if($page) $parent = $page->ID; } if(is_numeric($parent)) $parentObj = DataObject::get_by_id("SiteTree", $parent); if(!$parentObj || !$parentObj->ID) $parent = 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->write(); return $this->returnItemToUser($p); } /** * @uses LeftAndMainDecorator->augmentNewSiteTreeItem() */ public function getNewItem($id, $setID = true) { list($dummy, $className, $parentID, $suffix) = array_pad(explode('-',$id),4,null); if(Translatable::is_enabled()) { if (Translatable::default_lang() != $this->Lang) { Translatable::set_reading_lang(Translatable::default_lang()); $originalItem = DataObject::get_by_id($className,$suffix); Translatable::set_reading_lang($this->Lang); $translation = $originalItem->getTranslation($this->Lang); if(!$translation) $translation = $originalItem->createTranslation($this->Lang); if($setID) $translation->ID = $id; return $translation; } } $newItem = new $className(); if( !$suffix ) { $sessionTag = "NewItems." . $parentID . "." . $className; if(Session::get($sessionTag)) { $suffix = '-' . Session::get($sessionTag); Session::set($sessionTag, Session::get($sessionTag) + 1); } else Session::set($sessionTag, 1); $id = $id . $suffix; } $newItem->Title = _t('CMSMain.NEW',"New ",PR_MEDIUM,'"New " followed by a className').$className; $newItem->URLSegment = "new-" . strtolower($className); $newItem->ClassName = $className; $newItem->ParentID = $parentID; // DataObject::fieldExists only checks the current class, not the hierarchy // This allows the CMS to set the correct sort value if($newItem->castingHelperPair('Sort')) { $newItem->Sort = DB::query("SELECT MAX(\"Sort\") FROM \"SiteTree\" WHERE \"ParentID\" = '" . Convert::raw2sql($parentID) . "'")->value() + 1; } if( Member::currentUser() ) $newItem->OwnerID = Member::currentUser()->ID; if($setID) $newItem->ID = $id; # Some modules like subsites add extra fields that need to be set when the new item is created $this->extend('augmentNewSiteTreeItem', $newItem); return $newItem; } /** * Delete the page from live. This means a page in draft mode might still exist. * * @see delete() */ public function deletefromlive($urlParams, $form) { $id = $_REQUEST['ID']; Versioned::reading_stage('Live'); $record = DataObject::get_by_id("SiteTree", $id); if($record && !$record->canDelete()) return Security::permissionFailure($this); $descRemoved = ''; $descendantsRemoved = 0; // before deleting the records, get the descendants of this tree if($record) { $descendantIDs = $record->getDescendantIDList('SiteTree'); // then delete them from the live site too $descendantsRemoved = 0; foreach( $descendantIDs as $descID ) if( $descendant = DataObject::get_by_id('SiteTree', $descID) ) { $descendant->delete(); $descendantsRemoved++; } // delete the record $record->delete(); } Versioned::reading_stage('Stage'); if(isset($descendantsRemoved)) { $descRemoved = " and $descendantsRemoved descendants"; $descRemoved = sprintf(' '._t('CMSMain.DESCREMOVED', 'and %s descendants'), $descendantsRemoved); } else { $descRemoved = ''; } FormResponse::add($this->deleteTreeNodeJS($record)); FormResponse::status_message(sprintf(_t('CMSMain.REMOVED', 'Deleted \'%s\'%s from live site'), $record->Title, $descRemoved), 'good'); return FormResponse::respond(); } /** * Actually perform the publication step */ public function performPublish($record) { if($record && !$record->canPublish()) return Security::permissionFailure($this); $record->doPublish(); } /** * Reverts a page by publishing it to live. * Use {@link restorepage()} if you want to restore a page * which was deleted from draft without publishing. * * @uses SiteTree->doRevertToLive() */ public function revert($urlParams, $form) { $id = (int)$_REQUEST['ID']; $record = Versioned::get_one_by_stage('SiteTree', 'Live', "\"SiteTree_Live\".\"ID\" = {$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); $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'); return FormResponse::respond(); } /** * Delete the current page from draft stage. * @see deletefromlive() */ public function delete($urlParams, $form) { $id = $_REQUEST['ID']; $record = DataObject::get_one("SiteTree", "\"SiteTree\".\"ID\" = $id"); if($record && !$record->canDelete()) return Security::permissionFailure(); // save ID and delete record $recordID = $record->ID; $record->delete(); 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(); } } else { Director::redirectBack(); } } /* * Return a dropdown for selecting reports */ function ReportSelector() { $reports = ClassInfo::subclassesFor("SideReport"); $options[""] = _t('CMSMain.CHOOSEREPORT',"(Choose a report)"); foreach($reports as $report) { if($report != 'SideReport') $options[$report] = singleton($report)->title(); } return new DropdownField("ReportSelector", _t('CMSMain.REPORT', 'Report'),$options); } /** * Get the content for a side report */ function sidereport() { $reportClass = $this->urlParams['ID']; $report = ClassInfo::exists($reportClass) ? new $reportClass() : false; return $report ? $report->getHTML() : false; } /** * Get the versions of the current page */ function versions() { $pageID = $this->urlParams['ID']; $page = $this->getRecord($pageID); if($page) { $versions = $page->allVersions($_REQUEST['unpublished'] ? "" : "\"SiteTree\".WasPublished = 1"); return array( 'Versions' => $versions, ); } else { return sprintf(_t('CMSMain.VERSIONSNOPAGE',"Can't find page #%d",PR_LOW),$pageID); } } /** * 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); } else { $record = $this->performRollback($_REQUEST['ID'], "Live"); echo sprintf(_t('CMSMain.ROLLEDBACKPUB',"Rolled back to published version. New version number is #%d"),$record->Version); } } function unpublish() { $SQL_id = Convert::raw2sql($_REQUEST['ID']); $page = DataObject::get_by_id("SiteTree", $SQL_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)); } /** * 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()); $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);"); } 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(); } function performRollback($id, $version) { $record = DataObject::get_by_id($this->stat('tree_class'), $id); if($record && !$record->canPublish()) return Security::permissionFailure($this); $record->doRollbackTo($version); return $record; } function getversion() { $id = $this->urlParams['ID']; $version = str_replace('&ajax=1','',$this->urlParams['OtherID']); $record = Versioned::get_version("SiteTree", $id, $version); $versionAuthor = DataObject::get_by_id('Member', $record->AuthorID); if($record) { if($record && !$record->canView()) return Security::permissionFailure($this); $fields = $record->getCMSFields($this); $fields->removeByName("Status"); $fields->push(new HiddenField("ID")); $fields->push(new HiddenField("Version")); $fields->insertBefore( new LiteralField( 'YouAreViewingHeader', '

    ' . sprintf( _t( 'CMSMain.VIEWING', "You are viewing version #%s, created %s by %s", PR_MEDIUM, 'Version number is a linked string, created is a relative time (e.g. 2 days ago), by a specific author' ), "ID/$version\" title=\"" . $versionAuthor->Title . "\">$version", $record->obj('LastEdited')->Ago(), $versionAuthor->Title ) . '

    ' ), 'Root' ); $actions = new FieldSet( new FormAction("email", _t('CMSMain.EMAIL',"Email")), new FormAction("print", _t('CMSMain.PRINT',"Print")), new FormAction("rollback", _t('CMSMain.ROLLBACK',"Roll back to this version")) ); // encode the message to appear in the body of the email $archiveURL = Director::absoluteBaseURL() . $record->URLSegment . '?archiveDate=' . $record->obj('LastEdited')->URLDatetime(); $archiveEmailMessage = urlencode( $this->customise( array( 'ArchiveDate' => $record->obj('LastEdited'), 'ArchiveURL' => $archiveURL ) )->renderWith( 'ViewArchivedEmail' ) ); $archiveEmailMessage = preg_replace( '/\+/', '%20', $archiveEmailMessage ); $fields->push( new HiddenField( 'ArchiveEmailMessage', '', $archiveEmailMessage ) ); $fields->push( new HiddenField( 'ArchiveEmailSubject', '', preg_replace( '/\+/', '%20', urlencode( 'Archived version of ' . $record->Title ) ) ) ); $fields->push( new HiddenField( 'ArchiveURL', '', $archiveURL ) ); $form = new Form($this, "EditForm", $fields, $actions); $form->loadDataFrom($record); $form->loadDataFrom(array( "ID" => $id, "Version" => $version, )); // historical version shouldn't be editable $readonlyFields = $form->Fields()->makeReadonly(); $form->setFields($readonlyFields); $templateData = $this->customise(array( "EditForm" => $form )); SSViewer::setOption('rewriteHashlinks', false); if(Director::is_ajax()) { $result = $templateData->renderWith($this->class . '_right'); $parts = split(']*>', $result); return $parts[sizeof($parts)-2]; } else { return $templateData->renderWith('LeftAndMain'); } } } function compareversions() { $id = (int)$this->urlParams['ID']; $version1 = (int)$_REQUEST['From']; $version2 = (int)$_REQUEST['To']; if( $version1 > $version2 ) { $toVersion = $version1; $fromVersion = $version2; } else { $toVersion = $version2; $fromVersion = $version1; } $page = DataObject::get_by_id("SiteTree", $id); if($page && !$page->canView()) return Security::permissionFailure($this); $record = $page->compareVersions($fromVersion, $toVersion); $fromVersionRecord = Versioned::get_version('SiteTree', $id, $fromVersion); $toVersionRecord = Versioned::get_version('SiteTree', $id, $toVersion); if($record) { $fromDateNice = $fromVersionRecord->obj('LastEdited')->Ago(); $toDateNice = $toVersionRecord->obj('LastEdited')->Ago(); $fromAuthor = DataObject::get_by_id('Member', $fromVersionRecord->AuthorID); $toAuthor = DataObject::get_by_id('Member', $toVersionRecord->AuthorID); $fields = $record->getCMSFields($this); $fields->push(new HiddenField("ID")); $fields->push(new HiddenField("Version")); $fields->insertBefore( new LiteralField( 'YouAreComparingHeader', '

    ' . sprintf( _t('CMSMain.COMPARINGV',"Comparing versions %s and %s"), "Version\" title=\"$fromAuthor->Title\">$fromVersionRecord->Version ($fromDateNice)", "Version\" title=\"$toAuthor->Title\">$toVersionRecord->Version ($toDateNice)" ) . '

    ' ), "Root" ); $actions = new FieldSet(); $form = new Form($this, "EditForm", $fields, $actions); $form->loadDataFrom($record); $form->loadDataFrom(array( "ID" => $id, "Version" => $fromVersion, )); // comparison views shouldn't be editable $readonlyFields = $form->Fields()->makeReadonly(); $form->setFields($readonlyFields); foreach($form->Fields()->dataFields() as $field) { $field->dontEscape = true; } return $this->sendFormToBrowser(array( "EditForm" => $form )); } } function sendFormToBrowser($templateData) { if(Director::is_ajax()) { SSViewer::setOption('rewriteHashlinks', false); $result = $this->customise($templateData)->renderWith($this->class . '_right'); $parts = split(']*>', $result); return $parts[sizeof($parts)-2]; } else { return array( "Right" => $this->customise($templateData)->renderWith($this->class . '_right'), ); } } function dialog() { Requirements::clear(); $buttons = new DataObjectSet; if($_REQUEST['Buttons']) foreach($_REQUEST['Buttons'] as $button) { list($name, $title) = explode(',',$button,2); $buttons->push(new ArrayData(array( "Name" => $name, "Title" => $title, ))); } return array( "Message" => htmlentities($_REQUEST['Message']), "Buttons" => $buttons, "Modal" => $_REQUEST['Modal'] ? true : false, ); } function canceldraftchangesdialog() { Requirements::clear(); Requirements::css(CMS_DIR . 'css/dialog.css'); Requirements::javascript(THIRDPARTY_DIR . '/prototype.js'); Requirements::javascript(THIRDPARTY_DIR . '/behaviour.js'); Requirements::javascript(THIRDPARTY_DIR . '/prototype_improvements.js'); Requirements::javascript(CMS_DIR . '/javascript/dialog.js'); $message = _t('CMSMain.COPYPUBTOSTAGE',"Do you really want to copy the published content to the stage site?"); $buttons = ""; return $this->customise( array( 'Message' => $message, 'Buttons' => $buttons, 'DialogType' => 'alert' ))->renderWith('Dialog'); } /** * Publishes a number of items. * Called by AJAX */ public function publishitems() { // This method can't be called without ajax. if(!Director::is_ajax()) { Director::redirectBack(); return; } $ids = split(' *, *', $this->requestParams['csvIDs']); $notifications = array(); $idList = array(); // make sure all the ids are numeric. // Add all the children to the list of IDs if they are missing foreach($ids as $id) { $brokenPageList = ''; if(is_numeric($id)) { $record = DataObject::get_by_id($this->stat('tree_class'), $id); if($record) { if($record && !$record->canPublish()) return Security::permissionFailure($this); // Publish this page $record->doPublish(); // Now make sure the 'changed' icon is removed $publishedRecord = DataObject::get_by_id($this->stat('tree_class'), $id); $JS_title = Convert::raw2js($publishedRecord->TreeTitle()); FormResponse::add("\$('sitetree').setNodeTitle($id, '$JS_title');"); FormResponse::add("$('Form_EditForm').reloadIfSetTo($record->ID);"); $record->destroy(); unset($record); } } } if (sizeof($ids) > 1) $message = sprintf(_t('CMSMain.PAGESPUB', "%d pages published "), sizeof($ids)); else $message = sprintf(_t('CMSMain.PAGEPUB', "%d page published "), sizeof($ids)); FormResponse::add('statusMessage("'.$message.'","good");'); return FormResponse::respond(); } /** * @return Form */ public function PublishItemsForm() { $form = new Form( $this, 'PublishItemsForm', new FieldSet( new HiddenField('csvIDs'), new CheckboxField('ShowDrafts', _t('CMSMain_left.ss.SHOWONLYCHANGED','Show only changed pages')) ), new FieldSet( new FormAction('publishitems', _t('CMSMain_left.ss.PUBLISHCONFIRM','Publish the selected pages')) ) ); $form->addExtraClass('actionparams'); return $form; } /** * Delete a number of items. * This code supports notification */ public function deleteitems() { // This method can't be called without ajax. if(!Director::is_ajax()) { Director::redirectBack(); return; } $ids = split(' *, *', $_REQUEST['csvIDs']); $notifications = array(); $idList = array(); // make sure all the ids are numeric. // Add all the children to the list of IDs if they are missing foreach($ids as $id) { $brokenPageList = ''; if(is_numeric($id)) { $record = DataObject::get_by_id($this->stat('tree_class'), $id); if($record) { if($record && !$record->canDelete()) return Security::permissionFailure($this); // add all the children for this record if they are not already in the list // this check is a little slower but will prevent circular dependencies // (should they exist, which they probably shouldn't) from causing // the function to not terminate $children = $record->AllChildren(); if( $children ) foreach( $children as $child ) if( array_search( $child->ID, $ids ) !== FALSE ) $ids[] = $child->ID; if($record->hasMethod('BackLinkTracking')) { $brokenPages = $record->BackLinkTracking(); foreach($brokenPages as $brokenPage) { $brokenPageList .= "
  • " . $brokenPage->Breadcrumbs(3, true) . "
  • "; $brokenPage->HasBrokenLink = true; $notifications[$brokenPage->OwnerID][] = $brokenPage; $brokenPage->writeWithoutVersion(); } } $record->delete(); $record->destroy(); // DataObject::delete_by_id($this->stat('tree_class'), $id); // check to see if the record exists on the live site, if it doesn't remove the tree node $liveRecord = Versioned::get_one_by_stage( $this->stat('tree_class'), 'Live', "\"{$this->stat('tree_class')}\".\"ID\"={$id}"); if($liveRecord) { $title = Convert::raw2js($record->TreeTitle()); FormResponse::add("$('sitetree').setNodeTitle($record->OldID, '$title');"); FormResponse::add("$('Form_EditForm').reloadIfSetTo($record->OldID);"); } else { FormResponse::add("var node = $('sitetree').getTreeNodeByIdx('$id');"); FormResponse::add("if(node && node.parentTreeNode) node.parentTreeNode.removeTreeNode(node);"); FormResponse::add("$('Form_EditForm').reloadIfSetTo($record->OldID);"); } } } } if($notifications) foreach($notifications as $memberID => $pages) { if(class_exists('Page_BrokenLinkEmail')) { $email = new Page_BrokenLinkEmail(); $email->populateTemplate(new ArrayData(array( "Recipient" => DataObject::get_by_id("Member", $memberID), "BrokenPages" => new DataObjectSet($pages), ))); $email->debug(); $email->send(); } } if (sizeof($ids) > 1) $message = sprintf(_t('CMSMain.PAGESDEL', "%d pages deleted "), sizeof($ids)); else $message = sprintf(_t('CMSMain.PAGEDEL', "%d page deleted "), sizeof($ids)); if(isset($brokenPageList) && $brokenPageList != '') { $message .= _t('CMSMain.NOWBROKEN'," The following pages now have broken links:")."" . _t('CMSMain.NOWBROKEN2',"Their owners have been emailed and they will fix up those pages."); } FormResponse::add('statusMessage("'.$message.'","good");'); return FormResponse::respond(); } /** * @return Form */ function DeleteItemsForm() { $form = new Form( $this, 'DeleteItemsForm', new FieldSet( new LiteralField('SelectedPagesNote', sprintf('

    %s

    ', _t('CMSMain_left.ss.SELECTPAGESACTIONS','Select the pages that you want to change & then click an action:')) ), new HiddenField('csvIDs') ), new FieldSet( new FormAction('deleteitems', _t('CMSMain_left.ss.DELETECONFIRM','Delete the selected pages')) ) ); $form->addExtraClass('actionparams'); return $form; } function buildbrokenlinks() { if($this->urlParams['ID']) { $newPageSet[] = DataObject::get_by_id("Page", $this->urlParams['ID']); } else { $pages = DataObject::get("Page"); foreach($pages as $page) $newPageSet[] = $page; $pages = null; } $content = new HtmlEditorField('Content'); $download = new HtmlEditorField('Download'); foreach($newPageSet as $i => $page) { $page->HasBrokenLink = 0; $page->HasBrokenFile = 0; $lastUsage = (memory_get_usage() - $lastPoint); $lastPoint = memory_get_usage(); $content->setValue($page->Content); $content->saveInto($page); $download->setValue($page->Download); $download->saveInto($page); echo "
  • $page->Title (link:$page->HasBrokenLink, file:$page->HasBrokenFile)"; $page->writeWithoutVersion(); $page->destroy(); $newPageSet[$i] = null; } } function AddPageOptionsForm() { $pageTypes = array(); foreach( $this->PageTypes() as $arrayData ) { $pageTypes[$arrayData->getField('ClassName')] = $arrayData->getField('AddAction'); } return new Form($this, "AddPageOptionsForm", new FieldSet( new HiddenField("ParentID"), new DropdownField("PageType", "", $pageTypes) // "Page to copy" => new TreeDropdownField("DuplicateSection", "", "SiteTree"), ), new FieldSet( new FormAction("addpage", _t('CMSMain.GO',"Go")) )); } /** * Helper function to get page count */ function getpagecount() { ini_set('max_execution_time', 0); $excludePages = split(" *, *", $_GET['exclude']); $pages = DataObject::get("SiteTree", "\"ParentID\" = 0"); foreach($pages as $page) $pageArr[] = $page; while(list($i,$page) = each($pageArr)) { if(!in_array($page->URLSegment, $excludePages)) { if($children = $page->AllChildren()) { foreach($children as $child) $pageArr[] = $child; } if(!$_GET['onlywithcontent'] || strlen(Convert::xml2raw($page->Content)) > 100) { echo "
  • " . $page->Breadcrumbs(null, true) . "
  • "; $count++; } else { echo "
  • " . $page->Breadcrumbs(null, true) . " - " . _t('CMSMain.NOCONTENT',"no content") . "
  • "; } } } echo '

    ' . _t('CMSMain.TOTALPAGES',"Total pages: ") . "$count

    "; } function publishall() { ini_set("memory_limit", -1); ini_set('max_execution_time', 0); $response = ""; if(isset($this->requestParams['confirm'])) { $start = 0; $pages = DataObject::get("SiteTree", "", "", "", "$start,30"); $count = 0; while(true) { foreach($pages as $page) { if($page && !$page->canPublish()) return Security::permissionFailure($this); $page->doPublish(); $page->destroy(); unset($page); $count++; $response .= "
  • $count
  • "; } if($pages->Count() > 29) { $start += 30; $pages = DataObject::get("SiteTree", "", "", "", "$start,30"); } else { break; } } $response .= sprintf(_t('CMSMain.PUBPAGES',"Done: Published %d pages"), $count); } else { $response .= '

    ' . _t('CMSMain.PUBALLFUN','"Publish All" functionality') . '

    ' . _t('CMSMain.PUBALLFUN2', 'Pressing this button will do the equivalent of going to every page and pressing "publish". It\'s intended to be used after there have been massive edits of the content, such as when the site was first built.') . '

    '; } return $response; } /** * Restore a previously deleted page. * Internal action which shouldn't be executed through URL-handlers. */ function restorepage() { if($id = $this->urlParams['ID']) { $restoredPage = Versioned::get_latest_version("SiteTree", $id); $restoredPage->ID = $restoredPage->RecordID; // if no record can be found on draft stage (meaning it has been "deleted from draft" before), // create an empty record if(!DB::query("SELECT \"ID\" FROM \"SiteTree\" WHERE \"ID\" = $restoredPage->ID")->value()) { DB::query("INSERT INTO \"SiteTree\" (\"ID\") VALUES ($restoredPage->ID)"); } $restoredPage->forceChange(); $restoredPage->writeWithoutVersion(); } else { echo _t('CMSMain.VISITRESTORE',"visit restorepage/(ID)",PR_LOW,'restorepage/(ID) should not be translated (is an URL)'); } } function duplicate() { if(($id = $this->urlParams['ID']) && is_numeric($id)) { $page = DataObject::get_by_id("SiteTree", $id); if($page && !$page->canEdit()) return Security::permissionFailure($this); $newPage = $page->duplicate(); // ParentID can be hard-set in the URL. This is useful for pages with multiple parents if($_GET['parentID'] && is_numeric($_GET['parentID'])) { $newPage->ParentID = $_GET['parentID']; $newPage->write(); } return $this->returnItemToUser($newPage); } else { user_error("CMSMain::duplicate() Bad ID: '$id'", E_USER_WARNING); } } function duplicatewithchildren() { if(($id = $this->urlParams['ID']) && is_numeric($id)) { $page = DataObject::get_by_id("SiteTree", $id); if($page && !$page->canEdit()) return Security::permissionFailure($this); $newPage = $page->duplicateWithChildren(); return $this->returnItemToUser($newPage); } else { user_error("CMSMain::duplicate() Bad ID: '$id'", E_USER_WARNING); } } /** * Switch the cms language and reload the site tree * */ function switchlanguage($lang, $donotcreate = null) { //is it's a clean switch (to an existing language deselect the current page) if (is_string($lang)) $dontunloadPage = true; $lang = (is_string($lang) ? $lang : urldecode($this->urlParams['ID'])); if ($lang != Translatable::default_lang()) { Translatable::set_reading_lang(Translatable::default_lang()); $tree_class = $this->stat('tree_class'); $obj = new $tree_class; $allIDs = $obj->getDescendantIDList(); $allChildren = $obj->AllChildren(); $classesMap = $allChildren->map('ID','ClassName'); $titlesMap = $allChildren->map(); Translatable::set_reading_lang($lang); $obj = new $tree_class; $languageIDs = $obj->getDescendantIDList(); $notcreatedlist = array_diff($allIDs,$languageIDs); FormResponse::add("$('addpage').getElementsByTagName('button')[0].disabled=true;"); //FormResponse::add("$('Form_AddPageOptionsForm').getElementsByTagName('div')[1].getElementsByTagName('input')[0].disabled=true;"); FormResponse::add("$('Translating_Message').innerHTML = 'Translating mode - ".i18n::get_language_name($lang)."';"); FormResponse::add("Element.removeClassName('Translating_Message','nonTranslating');"); } else { Translatable::set_reading_lang($lang); FormResponse::add("$('addpage').getElementsByTagName('button')[0].disabled=false;"); //FormResponse::add("$('Form_AddPageOptionsForm').getElementsByTagName('div')[1].getElementsByTagName('input')[0].disabled=false;"); FormResponse::add("Element.addClassName('Translating_Message','nonTranslating');"); } $obj = singleton($this->stat('tree_class')); $obj->markPartialTree(); $siteTree = $obj->getChildrenAsUL("", ' "
  • ID\" class=\"" . $child->CMSTreeClasses($extraArg) . "\">" . "Link(),0,-1), "show", $child->ID) . "\" " . (($child->canEdit() || $child->canAddChildren()) ? "" : "class=\"disabled\"") . " title=\"' . _t('LeftAndMain.PAGETYPE') . '".$child->class."\" >" . (Convert::raw2js($child->TreeTitle())) . "" ' ,$this, true); $rootLink = $this->Link() . '0'; $siteTree = "
  • " . _t('LeftAndMain.SITECONTENT') . "" . $siteTree . "
  • "; FormResponse::add("$('sitetree').innerHTML ='". ereg_replace("[\n]","\\\n",$siteTree) ."';"); FormResponse::add("SiteTree.applyTo('#sitetree');"); if (isset($notcreatedlist)) { foreach ($notcreatedlist as $notcreated) { if ($notcreated == $donotcreate) continue; $id = "new-{$classesMap[$notcreated]}-0-$notcreated"; Session::set($id . '_originalLangID',$notcreated); $treeTitle = Convert::raw2js($titlesMap[$notcreated]); $response = <<getRecord($originalLangID); $this->Lang = $langCode; Translatable::set_reading_lang($langCode); // creating a record in-memory, which means setting the $Lang property // will have no effect as the record is loaded through another javascript // call and CMSMain->getitem(). The CMS submits the currently selected language // through javascript, which will cause $Lang to be written to the database. // @todo Explicitly set $Lang property for in-memory object so we don't need javascript modifying GET-calls to save the new object $translatedRecord = $record->createTranslation($langCode); $url = sprintf( "%s/%d/?lang=%s", $this->Link('show'), $translatedRecord->ID, $langCode ); FormResponse::add(sprintf('window.location.href = "%s";', $url)); return FormResponse::respond(); } /** * Provide the permission codes used by LeftAndMain. * Can't put it on LeftAndMain since that's an abstract base class. */ function providePermissions() { $classes = ClassInfo::subclassesFor('LeftAndMain'); foreach($classes as $class) { $title = _t("{$class}.MENUTITLE", LeftAndMain::menu_title_for_class($class)); $perms["CMS_ACCESS_" . $class] = sprintf( _t( 'CMSMain.ACCESS', "Access to '%s' (%s)", PR_MEDIUM, "Item in permission selection identifying the admin section, with title and classname. Example: Access to 'Files & Images' (AssetAdmin)" ), $title, $class ); } $perms["CMS_ACCESS_LeftAndMain"] = _t( 'CMSMain.ACCESSALLINTERFACES', 'Access to all CMS interfaces' ); return $perms; } /** * Return a dropdown with existing languages */ function LangSelector() { $langs = Translatable::get_existing_content_languages('SiteTree'); return new DropdownField("LangSelector","Language",$langs,Translatable::current_lang()); } /** * Determine if there are more than one languages in our site tree */ function MultipleLanguages() { $langs = Translatable::get_existing_content_languages('SiteTree'); return (count($langs) > 1); } /** * Get the name of the language that we are translating in */ function EditingLang() { if(!Translatable::is_default_lang()) { return i18n::get_language_name(Translatable::current_lang()); } else { return false; } } } $filterCache = array(); // TODO: Find way to put this in a class function cmsMainMarkingFilterFunction($node) { // Expand all nodes // $node->markingFinished(); $failed_filter = false; // First check for the generic search term in the URLSegment, Title, MenuTitle, & Content if (!empty($_REQUEST['SiteTreeSearchTerm'])) { // For childless nodes, show only those matching the filter $filter = strtolower($_REQUEST['SiteTreeSearchTerm']); if ( strpos( strtolower($node->URLSegment) , $filter) === false && strpos( strtolower($node->Title) , $filter) === false && strpos( strtolower($node->MenuTitle) , $filter) === false && strpos( strtolower($node->Content) , $filter) === false) { $failed_filter = true; } } // Check the 'Edited Since' date if (!empty($_REQUEST['SiteTreeFilterDate'])) { $edited_since = mktime(0, 0, 0, substr($_REQUEST['SiteTreeFilterDate'], 3, 2), substr($_REQUEST['SiteTreeFilterDate'], 0, 2), substr($_REQUEST['SiteTreeFilterDate'], 6, 4)); if ( strtotime($node->LastEdited) < $edited_since ) { $failed_filter = true; } } // Now check if a specified Criteria attribute matches foreach (CMSMain::T_SiteTreeFilterOptions() as $key => $value) { if (!empty($_REQUEST[$key])) { $parameterName = $key; $filter = strtolower($_REQUEST[$key]); // Show node only if the filter string exists anywere in the filter paramater (ignoring case) if (strpos( strtolower($node->$parameterName) , $filter) === false) { $failed_filter = true; } } } // Each filter must match or it fails if (true == $failed_filter) { // Don't ever hide nodes with children, because otherwise if one of their children matches the search, it wouldn't be shown. if($node->AllChildrenIncludingDeleted()->count() > 0) { // Open all nodes with children so it is easy to see any children that match the search. foreach($node->AllChildrenIncludingDeleted() as $childNode) { if(cmsMainMarkingFilterFunction($childNode)) { $node->markOpened(); $filterCache[$node->ID] = true; return true; } } } $filterCache[$node->ID] = false; return false; } else { if($node->AllChildrenIncludingDeleted()->count() > 0) { $node->markOpened(); } $filterCache[$node->ID] = true; return true; } } ?>