"SiteTree", ); static $db = array( "VersionID" => "Int", ); /** * Generates the array of fields required for the page type. */ function getVirtualFields() { $nonVirtualFields = array( "SecurityTypeID", "OwnerID", "URLSegment", "Sort", "Status", 'ShowInMenus', // 'Locale' 'ShowInSearch', 'Version', ); $allFields = $this->db(); if($hasOne = $this->has_one()) foreach($hasOne as $link) $allFields[$link . 'ID'] = "Int"; foreach($allFields as $field => $type) { if(!in_array($field, $nonVirtualFields)) $virtualFields[] = $field; } return $virtualFields; } function ContentSource() { return $this->CopyContentFrom(); } function allowedChildren() { if($this->CopyContentFrom()) { return $this->CopyContentFrom()->allowedChildren(); } } public function syncLinkTracking() { if($this->CopyContentFromID) { $this->HasBrokenLink = !(bool) DataObject::get_by_id('SiteTree', $this->CopyContentFromID); } } /** * We can only publish the page if there is a published source page */ public function canPublish($member = null) { return $this->isPublishable() && parent::canPublish($member); } /** * Return true if we can delete this page from the live site, which is different from can * we publish it. */ public function canDeleteFromLive($member = null) { return parent::canPublish($member); } /** * Returns true if is page is publishable by anyone at all * Return false if the source page isn't published yet. * * Note that isPublishable doesn't affect ete from live, only publish. */ public function isPublishable() { // No source if(!$this->CopyContentFrom() || !$this->CopyContentFrom()->ID) { return false; } // Unpublished source if(!Versioned::get_versionnumber_by_stage('SiteTree', 'Live', $this->CopyContentFromID)) { return false; } // Default - publishable return true; } /** * Generate the CMS fields from the fields from the original page. */ function getCMSFields($cms = null) { $fields = parent::getCMSFields($cms); // Setup the linking to the original page. $copyContentFromField = new TreeDropdownField( "CopyContentFromID", _t('VirtualPage.CHOOSE', "Choose a page to link to"), "SiteTree" ); $copyContentFromField->setFilterFunction(create_function('$item', 'return $item->ClassName != "VirtualPage";')); // Setup virtual fields if($virtualFields = $this->getVirtualFields()) { $roTransformation = new ReadonlyTransformation(); foreach($virtualFields as $virtualField) { if($fields->dataFieldByName($virtualField)) $fields->replaceField($virtualField, $fields->dataFieldByName($virtualField)->transform($roTransformation)); } } // Add fields to the tab $fields->addFieldToTab("Root.Content.Main", new HeaderField('VirtualPageHeader',_t('VirtualPage.HEADER', "This is a virtual page")), "Title" ); $fields->addFieldToTab("Root.Content.Main", $copyContentFromField, "Title"); // Create links back to the original object in the CMS if($this->CopyContentFromID) { $linkToContent = "CopyContentFromID\">" . _t('VirtualPage.EDITCONTENT', 'click here to edit the content') . ""; $fields->addFieldToTab("Root.Content.Main", $linkToContentLabelField = new LabelField('VirtualPageContentLinkLabel', $linkToContent), "Title" ); $linkToContentLabelField->setAllowHTML(true); } return $fields; } /** * We have to change it to copy all the content from the original page first. */ function onBeforeWrite() { // On regular write, this will copy from published source. This happens on every publish if($this->extension_instances['Versioned']->migratingVersion && Versioned::current_stage() == 'Live') { $performCopyFrom = true; $stageSourceVersion = DB::query("SELECT \"Version\" FROM \"SiteTree\" WHERE \"ID\" = $this->CopyContentFromID")->value(); $liveSourceVersion = DB::query("SELECT \"Version\" FROM \"SiteTree_Live\" WHERE \"ID\" = $this->CopyContentFromID")->value(); // We're going to create a new VP record in SiteTree_versions because the published // version might not exist, unless we're publishing the latest version if($stageSourceVersion != $liveSourceVersion) { $this->extension_instances['Versioned']->migratingVersion = null; } // On regular write, this will copy from draft source. This is only executed when the source // page changeds } else { $performCopyFrom = $this->isChanged('CopyContentFromID') && $this->CopyContentFromID != 0; } // On publish, this will copy from published source if($performCopyFrom && $this instanceof VirtualPage) { // This flush is needed because the get_one cache doesn't respect site version :-( singleton('SiteTree')->flushCache(); $source = DataObject::get_one("SiteTree",sprintf('"SiteTree"."ID" = %d', $this->CopyContentFromID)); // Leave the updating of image tracking until after write, in case its a new record $this->copyFrom($source, false); $this->URLSegment = $source->URLSegment . '-' . $this->ID; } parent::onBeforeWrite(); } function onAfterWrite() { parent::onAfterWrite(); // Don't do this stuff when we're publishing if(!$this->extension_instances['Versioned']->migratingVersion) { if( $this->isChanged('CopyContentFromID') && $this->CopyContentFromID != 0 && $this instanceof VirtualPage ) { $this->updateImageTracking(); } } } /** * Ensure we have an up-to-date version of everything. */ function copyFrom($source, $updateImageTracking = true) { if($source) { foreach($this->getVirtualFields() as $virtualField) { $this->$virtualField = $source->$virtualField; } // We also want to copy ShowInMenus, but only if we're copying the // source page for the first time. if($this->isChanged('CopyContentFromID')) { $this->ShowInMenus = $source->ShowInMenus; } if($updateImageTracking) $this->updateImageTracking(); } } function updateImageTracking() { // Doesn't work on unsaved records if(!$this->ID) return; // Remove CopyContentFrom() from the cache unset($this->components['CopyContentFrom']); // Update ImageTracking $this->ImageTracking()->setByIdList($this->CopyContentFrom()->ImageTracking()->column('ID')); } /** * Allow attributes on the master page to pass * through to the virtual page * * @param string $field * @return mixed */ function __get($field) { $return = parent::__get($field); if ($return === null) { if($this->copyContentFrom()->hasMethod($funcName = "get$field")) { $return = $this->copyContentFrom()->$funcName(); } else if($this->copyContentFrom()->hasField($field)) { $return = $this->copyContentFrom()->getField($field); } else if($field == 'Content') { return '
' . _t('VirtualPage.NOTFOUND', 'We could not find the content for this virtual page.') . '
'; } } return $return; } /** * Pass unrecognized method calls on to the original data object * * @param string $method * @param string $args */ function __call($method, $args) { try { return parent::__call($method, $args); } catch (Exception $e) { // Hack... detect exception type. We really should use exception subclasses. // if the exception isn't a 'no method' error, rethrow it if ($e->getCode() !== 2175) throw $e; $original = $this->copyContentFrom(); return call_user_func_array(array($original, $method), $args); } } /** * Overwrite to also check for method on the original data object * * @param string $method * @return bool */ function hasMethod($method) { $haveIt = parent::hasMethod($method); if (!$haveIt) $haveIt = $this->copyContentFrom()->hasMethod($method); return $haveIt; } } /** * Controller for the virtual page. * @package cms */ class VirtualPage_Controller extends Page_Controller { static $allowed_actions = array( 'loadcontentall' => 'ADMIN', ); /** * Reloads the content if the version is different ;-) */ function reloadContent() { $this->failover->copyFrom($this->failover->CopyContentFrom()); $this->failover->write(); return; } function getViewer($action) { $originalClass = get_class($this->CopyContentFrom()); if ($originalClass == 'SiteTree') $name = 'Page_Controller'; else $name = $originalClass."_Controller"; $controller = new $name(); return $controller->getViewer($action); } /** * When the virtualpage is loaded, check to see if the versions are the same * if not, reload the content. * NOTE: Virtual page must have a container object of subclass of sitetree. * We can't load the content without an ID or record to copy it from. */ function init(){ if(isset($this->record) && $this->record->ID){ if($this->record->VersionID != $this->failover->CopyContentFrom()->Version){ $this->reloadContent(); $this->VersionID = $this->failover->CopyContentFrom()->VersionID; } } parent::init(); } function loadcontentall() { $pages = DataObject::get("VirtualPage"); foreach($pages as $page) { $page->copyFrom($page->CopyContentFrom()); $page->write(); $page->publish("Stage", "Live"); echo "