| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439 |
- <?php
- /*
- * This file is part of MODX Revolution.
- *
- * Copyright (c) MODX, LLC. All Rights Reserved.
- *
- * For complete copyright and license information, see the COPYRIGHT and LICENSE
- * files found in the top-level directory of this distribution.
- */
- /**
- * Interface for implementation on derivative Resource types. Please define the following methods in your derivative
- * class to properly implement a Custom Resource Type in MODX.
- *
- * @see modResource
- * @interface
- * @package modx
- */
- interface modResourceInterface {
- /**
- * Determine the controller path for this Resource class. Return an absolute path.
- *
- * @static
- * @param xPDO $modx A reference to the modX object
- * @return string The absolute path to the controller for this Resource class
- */
- public static function getControllerPath(xPDO &$modx);
- /**
- * Use this in your extended Resource class to display the text for the context menu item, if showInContextMenu is
- * set to true. Return in the following format:
- *
- * array(
- * 'text_create' => 'ResourceTypeName',
- * 'text_create_here' => 'Create ResourceTypeName Here',
- * );
- *
- * @return array
- */
- public function getContextMenuText();
- /**
- * Use this in your extended Resource class to return a translatable name for the Resource Type.
- * @return string
- */
- public function getResourceTypeName();
- /**
- * Allows you to manipulate the tree node for a Resource before it is sent
- * @abstract
- * @param array $node
- */
- public function prepareTreeNode(array $node = array());
- }
- /**
- * Represents a web resource managed by the MODX framework.
- *
- * @property int $id The ID of the Resource
- * @property string $type The type of the resource; document/reference
- * @property string $contentType The content type string of the Resource, such as text/html
- * @property string $pagetitle The page title of the Resource
- * @property string $longtitle The long title of the Resource
- * @property string $description The description of the Resource
- * @property string $alias The FURL alias of the resource
- * @property boolean $aliasVisible Whether or not we should exclude the resource alias for children
- * @property string $link_attributes Any link attributes for the URL generated for the Resource
- * @property boolean $published Whether or not this Resource is published, or viewable by users without the 'view_unpublished' permission
- * @property int $pub_date The UNIX time that this Resource will be automatically marked as published
- * @property int $unpub_date The UNIX time that this Resource will be automatically marked as unpublished
- * @property int $parent The parent ID of the Resource
- * @property boolean $isfolder Whether or not this Resource is a container
- * @property string $introtext The intro text of this Resource, often used as an excerpt
- * @property string $content The actual content of this Resource
- * @property boolean $richtext Whether or not this Resource is edited with a Rich Text Editor, if installed
- * @property int $template The Template this Resource is tied to, or 0 to use an empty Template
- * @property int $menuindex The menuindex, or rank, that this Resource shows in.
- * @property boolean $searchable Whether or not this Resource should be searchable
- * @property boolean $cacheable Whether or not this Resource should be cacheable
- * @property int $createdby The ID of the User that created this Resource
- * @property int $createdon The UNIX time of when this Resource was created
- * @property int $editedby The ID of the User, if any, that last edited this Resource
- * @property int $editedon The UNIX time, if set, of when this Resource was last edited
- * @property boolean $deleted Whether or not this Resource is marked as deleted
- * @property int $deletedon The UNIX time of when this Resource was deleted
- * @property int $deletedby The User that deleted this Resource
- * @property int $publishedon The UNIX time that this Resource was marked as published
- * @property int $publishedby The User that published this Resource
- * @property string $menutitle The title to show when this Resource is displayed in a menu
- * @property boolean $donthit Deprecated.
- * @property boolean $privateweb Deprecated.
- * @property boolean $privatemgr Deprecated.
- * @property int $content_dispo The type of Content Disposition that is used when displaying this Resource
- * @property boolean $hidemenu Whether or not this Resource should show in menus
- * @property string $class_key The Class Key of this Resource. Useful for derivative Resource types
- * @property string $context_key The Context that this Resource resides in
- * @property int $content_type The Content Type ID of this Resource
- * @property string $uri The generated URI of this Resource
- * @property boolean $uri_override Whether or not this URI is "frozen": where the URI will stay as specified and will not be regenerated
- * @property boolean $hide_children_in_tree Whether or not this Resource should show in the mgr tree any of its children
- * @property boolean $show_in_tree Whether or not this Resource should show in the mgr tree
- * @see modTemplate
- * @see modContentType
- * @package modx
- */
- class modResource extends modAccessibleSimpleObject implements modResourceInterface {
- /**
- * Represents the cacheable content for a resource.
- *
- * Note that this is not the raw source content, but the content that is the
- * result of processing cacheable tags within the raw source content.
- * @var string
- */
- public $_content= '';
- /**
- * Represents the output the resource produces.
- * @var string
- */
- public $_output= '';
- /**
- * The context the resource is requested from.
- *
- * Note that this is different than the context_key field that describes a
- * primary context for the resource.
- * @var string
- */
- public $_contextKey= null;
- /**
- * Indicates if the resource has already been processed.
- * @var boolean
- */
- protected $_processed= false;
- /**
- * The cache filename for the resource in the context.
- * @var string
- */
- protected $_cacheKey= null;
- /**
- * Indicates if the site cache should be refreshed when saving changes.
- * @var boolean
- */
- protected $_refreshCache= true;
- /**
- * Indicates if this Resource was generated from a forward.
- * @var boolean
- */
- public $_isForward= false;
- /**
- * An array of Javascript/CSS to be appended to the footer of this Resource
- * @var array $_jscripts
- */
- public $_jscripts = array();
- /**
- * An array of Javascript/CSS to be appended to the HEAD of this Resource
- * @var array $_sjscripts
- */
- public $_sjscripts = array();
- /**
- * All loaded Javascript/CSS that has been calculated to be loaded
- * @var array
- */
- public $_loadedjscripts = array();
- /**
- * Use if extending modResource to state whether or not to show the extended class in the tree context menu
- * @var boolean
- */
- public $showInContextMenu = false;
- /**
- * Use if extending modResource to state whether or not to allow drop on extended class in the resource tree
- * Set 1 for allow drop, 0 for disable drop or -1 for default behavior
- * @var int
- */
- public $allowDrop = -1;
- /**
- * Use if extending modResource to state whether or not the derivative class can be listed in the class_key
- * dropdown users can change when editing a resource.
- * @var boolean
- */
- public $allowListingInClassKeyDropdown = true;
- /**
- * Whether or not to allow creation of children resources in tree. Can be overridden in a derivative Resource class.
- * @var boolean
- */
- public $allowChildrenResources = true;
- /** @var modX $xpdo */
- public $xpdo;
- /**
- * Filter a string for use as a URL path segment.
- *
- * @param modX|xPDO &$xpdo A reference to a modX or xPDO instance.
- * @param string $segment The string to filter into a path segment.
- * @param array $options Local options to override global filter settings.
- *
- * @return string The filtered string ready to use as a path segment.
- */
- public static function filterPathSegment(&$xpdo, $segment, array $options = array()) {
- /* setup the various options */
- $iconv = function_exists('iconv');
- $mbext = function_exists('mb_strlen') && (boolean) $xpdo->getOption('use_multibyte', false);
- $charset = strtoupper((string) $xpdo->getOption('modx_charset', $options, 'UTF-8'));
- $delimiter = $xpdo->getOption('friendly_alias_word_delimiter', $options, '-');
- $delimiters = $xpdo->getOption('friendly_alias_word_delimiters', $options, '-_');
- $maxlength = (integer) $xpdo->getOption('friendly_alias_max_length', $options, 0);
- $stripElementTags = (boolean) $xpdo->getOption('friendly_alias_strip_element_tags', $options, true);
- $trimchars = $xpdo->getOption('friendly_alias_trim_chars', $options, '/.' . $delimiters);
- $restrictchars = $xpdo->getOption('friendly_alias_restrict_chars', $options, 'pattern');
- $restrictcharspattern = $xpdo->getOption('friendly_alias_restrict_chars_pattern', $options, '/[\0\x0B\t\n\r\f\a&=+%#<>"~`@\?\[\]\{\}\|\^\'\\\\]/');
- $lowercase = (boolean) $xpdo->getOption('friendly_alias_lowercase_only', $options, true);
- $translit = $xpdo->getOption('friendly_alias_translit', $options, $iconv ? 'iconv' : 'none');
- $translitClass = $xpdo->getOption('friendly_alias_translit_class', $options, 'translit.modTransliterate');
- /* strip html and optionally MODX element tags (stripped by default) */
- if ($xpdo instanceof modX) {
- $segment = $xpdo->stripTags($segment, '', $stripElementTags ? array() : null);
- }
- /* replace with the specified word delimiter */
- $segment = str_replace(' ', $delimiter, $segment);
- /* decode named entities to the appropriate character for the character set */
- $segment = html_entity_decode($segment, ENT_QUOTES, $charset);
- /* prepare '&' replacement */
- if ($xpdo instanceof modX && $xpdo->getService('lexicon','modLexicon') && $xpdo->lexicon('and')) {
- $ampersand = ' ' . $xpdo->lexicon('and') . ' ';
- } else {
- $ampersand = ' and ';
- }
- /* apply transliteration as configured */
- switch ($translit) {
- case '':
- case 'none':
- /* no transliteration */
- break;
- case 'iconv':
- /* if iconv is available, use the built-in transliteration it provides */
- $segment = iconv($mbext ? mb_detect_encoding($segment) : $charset, $charset . '//TRANSLIT//IGNORE', $segment);
- $ampersand = iconv($mbext ? mb_detect_encoding($segment) : $charset, $charset . '//TRANSLIT//IGNORE', $ampersand);
- break;
- case 'iconv_ascii':
- /* if iconv is available, use the built-in transliteration to ASCII it provides */
- $segment = iconv(($mbext) ? mb_detect_encoding($segment) : $charset, 'ASCII//TRANSLIT//IGNORE', $segment);
- break;
- default:
- /* otherwise look for a transliteration service class that will accept named transliteration tables */
- if ($xpdo instanceof modX) {
- $translitClassPath = $xpdo->getOption('friendly_alias_translit_class_path', $options, $xpdo->getOption('core_path', $options, MODX_CORE_PATH) . 'components/');
- if ($xpdo->getService('translit', $translitClass, $translitClassPath, $options)) {
- $segment = $xpdo->translit->translate($segment, $translit);
- $ampersand = $xpdo->translit->translate($ampersand, $translit);
- }
- }
- break;
- }
- /* replace any remaining '&' with a translit ampersand */
- $segment = str_replace('&', $ampersand, $segment);
- /* restrict characters as configured */
- switch ($restrictchars) {
- case 'alphanumeric':
- /* restrict segment to alphanumeric characters only */
- $segment = preg_replace('/[^\.%A-Za-z0-9 _-]/', '', $segment);
- break;
- case 'alpha':
- /* restrict segment to alpha characters only */
- $segment = preg_replace('/[^\.%A-Za-z _-]/', '', $segment);
- break;
- case 'legal':
- /* restrict segment to legal URL characters only */
- $segment = preg_replace('/[\0\x0B\t\n\r\f\a&=+%#<>"~`@\?\[\]\{\}\|\^\'\\\\]/', '', $segment);
- break;
- case 'pattern':
- default:
- /* restrict segment using regular expression pattern configured (same as legal by default) */
- if (!empty($restrictcharspattern)) {
- $segment = preg_replace($restrictcharspattern, '', $segment);
- }
- }
- /* replace one or more space characters with word delimiter */
- $segment = preg_replace('/\s+/u', $delimiter, $segment);
- /* replace one or more instances of word delimiters with word delimiter */
- $delimiterTokens = array();
- for ($d = 0; $d < strlen($delimiters); $d++) {
- $delimiterTokens[] = preg_quote($delimiters{$d}, '/');
- }
- if (!empty($delimiterTokens)) {
- $delimiterPattern = '/[' . implode('|', $delimiterTokens) . ']+/';
- $segment = preg_replace($delimiterPattern, $delimiter, $segment);
- }
- /* unless lowercase_only preference is explicitly off, change case to lowercase */
- if ($lowercase) {
- if ($mbext) {
- /* if the mb extension is available use it to protect multi-byte chars */
- $segment = mb_convert_case($segment, MB_CASE_LOWER, $charset);
- } else {
- /* otherwise, just use strtolower */
- $segment = strtolower($segment);
- }
- }
- /* trim specified chars from both ends of the segment */
- $segment = trim($segment, $trimchars);
- /* get the strlen of the segment (use mb extension if available) */
- $length = $mbext ? mb_strlen($segment, $charset) : strlen($segment);
- /* if maxlength is specified and exceeded, return substr with additional trim applied */
- if ($maxlength > 0 && $length > $maxlength) {
- $segment = substr($segment, 0, $maxlength);
- $segment = trim($segment, $trimchars);
- }
- return $segment;
- }
- /**
- * Get a sortable, limitable collection (and total count) of Resource Groups for a given Resource.
- *
- * @static
- * @param modResource &$resource A reference to the modResource to get the groups from.
- * @param array $sort An array of sort columns in column => direction format.
- * @param int $limit A limit of records to retrieve in the collection.
- * @param int $offset A record offset for a limited collection.
- * @return array An array containing the collection and total.
- */
- public static function listGroups(modResource &$resource, array $sort = array('id' => 'ASC'), $limit = 0, $offset = 0) {
- $result = array('collection' => array(), 'total' => 0);
- $c = $resource->xpdo->newQuery('modResourceGroup');
- $c->leftJoin('modResourceGroupResource', 'ResourceGroupResource', array(
- "ResourceGroupResource.document_group = modResourceGroup.id",
- 'ResourceGroupResource.document' => $resource->get('id')
- ));
- $result['total'] = $resource->xpdo->getCount('modResourceGroup',$c);
- $c->select($resource->xpdo->getSelectColumns('modResourceGroup', 'modResourceGroup'));
- $c->select(array("IF(ISNULL(ResourceGroupResource.document),0,1) AS access"));
- foreach ($sort as $sortKey => $sortDir) {
- $c->sortby($resource->xpdo->escape('modResourceGroup') . '.' . $resource->xpdo->escape($sortKey), $sortDir);
- }
- if ($limit > 0) $c->limit($limit, $offset);
- $result['collection'] = $resource->xpdo->getCollection('modResourceGroup', $c);
- return $result;
- }
- /**
- * Retrieve a collection of Template Variables for a Resource.
- *
- * @static
- * @param modResource &$resource A reference to the modResource to retrieve TemplateVars for.
- * @return A collection of modTemplateVar instances for the modResource.
- */
- public static function getTemplateVarCollection(modResource &$resource) {
- $c = $resource->xpdo->newQuery('modTemplateVar');
- $c->query['distinct'] = 'DISTINCT';
- $c->select($resource->xpdo->getSelectColumns('modTemplateVar', 'modTemplateVar'));
- $c->select($resource->xpdo->getSelectColumns('modTemplateVarTemplate', 'tvtpl', '', array('rank')));
- if ($resource->isNew()) {
- $c->select(array(
- 'modTemplateVar.default_text AS value',
- '0 AS resourceId'
- ));
- } else {
- $c->select(array(
- 'IF(ISNULL(tvc.value),modTemplateVar.default_text,tvc.value) AS value',
- $resource->get('id').' AS resourceId'
- ));
- }
- $c->innerJoin('modTemplateVarTemplate','tvtpl',array(
- 'tvtpl.tmplvarid = modTemplateVar.id',
- 'tvtpl.templateid' => $resource->get('template'),
- ));
- if (!$resource->isNew()) {
- $c->leftJoin('modTemplateVarResource','tvc',array(
- 'tvc.tmplvarid = modTemplateVar.id',
- 'tvc.contentid' => $resource->get('id'),
- ));
- }
- $c->sortby('tvtpl.rank,modTemplateVar.rank');
- return $resource->xpdo->getCollection('modTemplateVar', $c);
- }
- /**
- * Refresh Resource URI fields for children of the specified parent.
- *
- * @static
- * @param modX &$modx A reference to a valid modX instance.
- * @param int $parent The id of a Resource parent to start from (default is 0, the root)
- * @param array $options An array of various options for the method:
- * - resetOverrides: if true, Resources with uri_override set to true will be included
- * - contexts: an optional array of context keys to limit the refresh scope
- * @return void
- */
- public static function refreshURIs(modX &$modx, $parent = 0, array $options = array()) {
- $resetOverrides = array_key_exists('resetOverrides', $options) ? (boolean) $options['resetOverrides'] : false;
- $contexts = array_key_exists('contexts', $options) ? explode(',', $options['contexts']) : null;
- $criteria = $modx->newQuery('modResource', array('parent' => $parent));
- if (!$resetOverrides) {
- $criteria->where(array('uri_override' => false));
- }
- if (!empty($contexts)) {
- $criteria->where(array('context_key:IN' => $contexts));
- }
- $criteria->sortby('menuindex', 'ASC');
- /** @var modResource $resource */
- foreach ($modx->getIterator('modResource', $criteria) as $resource) {
- $resource->set('refreshURIs', true);
- if ($resetOverrides) {
- $resource->set('uri_override', false);
- }
- if (!$resource->get('uri_override')) {
- $resource->set('uri', '');
- }
- $resource->save();
- }
- }
- /**
- * Updates the Context of all Children recursively to that of the parent.
- *
- * @static
- * @param modX &$modx A reference to an initialized modX instance.
- * @param modResource $parent The parent modResource instance.
- * @param array $options An array of options.
- * @return int The number of children updated.
- */
- public static function updateContextOfChildren(modX &$modx, $parent, array $options = array()) {
- $count = 0;
- /** @var modResource $child */
- foreach ($parent->getIterator('Children') as $child) {
- $child->set('context_key', $parent->get('context_key'));
- if ($child->save()) {
- $count++;
- } else {
- $modx->log(modX::LOG_LEVEL_ERROR, "Could not change Context of child resource {$child->get('id')}", '', __METHOD__, __FILE__, __LINE__);
- }
- }
- return $count;
- }
- /**
- * @param xPDO $xpdo A reference to the xPDO|modX instance
- */
- function __construct(xPDO & $xpdo) {
- parent :: __construct($xpdo);
- $this->_contextKey= isset ($this->xpdo->context) ? $this->xpdo->context->get('key') : 'web';
- $this->_cacheKey= "[contextKey]/resources/[id]";
- }
- /**
- * Prepare the resource for output.
- */
- public function prepare()
- {
- # 1. Parse cacheable elements if exist.
- $this->process();
- # 2. Copy registered scripts added by the cacheable elements.
- $this->syncScripts();
- # 3. Parse uncacheable elements.
- $this->parseContent();
- }
-
- /**
- * Process a resource, transforming source content to output.
- *
- * @return string The processed cacheable content of a resource.
- */
- public function process() {
- if (!$this->get('cacheable') || !$this->_processed || !$this->_content) {
- $this->_content= '';
- $this->_output= '';
- $this->xpdo->getParser();
- /** @var modTemplate $baseElement */
- if ($baseElement= $this->getOne('Template')) {
- if ($baseElement->process()) {
- $this->_content= $baseElement->_output;
- $this->_processed= true;
- }
- } else {
- $this->_content= $this->getContent();
- $maxIterations= intval($this->xpdo->getOption('parser_max_iterations',10));
- $this->xpdo->parser->processElementTags('', $this->_content, false, false, '[[', ']]', array(), $maxIterations);
- $this->_processed= true;
- }
- }
- return $this->_content;
- }
- /**
- * @param array $data Data for placeholders
- * @return string
- */
- public function parseContent($data = array())
- {
- $this->xpdo->getParser();
- $maxIterations = intval($this->xpdo->getOption('parser_max_iterations', null, 10));
- $oldResource = $this->xpdo->resource;
- $this->xpdo->resource = $this;
- if (!empty($data)) {
- $scope = $this->xpdo->toPlaceholders($data, '', '.', true);
- }
- if (!$this->_processed) {
- $this->_content = $this->getContent();
- $this->xpdo->parser->processElementTags('', $this->_content, false, false, '[[', ']]', array(), $maxIterations);
- $this->_processed = true;
- }
- $this->_output = $this->_content;
- $this->xpdo->parser->processElementTags('', $this->_output, true, false, '[[', ']]', array(), $maxIterations);
- $this->xpdo->parser->processElementTags('', $this->_output, true, true, '[[', ']]', array(), $maxIterations);
- $this->xpdo->resource = $oldResource;
- if (isset($scope['keys'])) $this->xpdo->unsetPlaceholders($scope['keys']);
- if (isset($scope['restore'])) $this->xpdo->toPlaceholders($scope['restore']);
- return $this->_output;
- }
- /**
- * Store scripts registered by cached elements.
- */
- public function syncScripts()
- {
- $this->_jscripts = $this->xpdo->jscripts;
- $this->_sjscripts = $this->xpdo->sjscripts;
- $this->_loadedjscripts = $this->xpdo->loadedjscripts;
- }
- /**
- * Gets the raw, unprocessed source content for a resource.
- *
- * @param array $options An array of options implementations can use to
- * accept language, revision identifiers, or other information to alter the
- * behavior of the method.
- * @return string The raw source content for the resource.
- */
- public function getContent(array $options = array()) {
- $content = '';
- if (isset($options['content'])) {
- $content = $options['content'];
- } else {
- $content = $this->get('content');
- }
- return $content;
- }
- /**
- * Set the raw source content for this element.
- *
- * @param mixed $content The source content; implementations can decide if
- * it can only be a string, or some other source from which to retrieve it.
- * @param array $options An array of options implementations can use to
- * accept language, revision identifiers, or other information to alter the
- * behavior of the method.
- * @return boolean True indicates the content was set.
- */
- public function setContent($content, array $options = array()) {
- return $this->set('content', $content);
- }
- /**
- * Returns the cache key for this instance in the specified or current context.
- *
- * @param string $context A specific Context to get the cache key from.
- *
- * @return string The cache key.
- */
- public function getCacheKey($context = '') {
- $id = $this->get('id') ? (string) $this->get('id') : '0';
- if (!is_string($context) || $context === '') {
- $context = !empty($this->_contextKey)
- ? $this->_contextKey
- : $this->get('context_key');
- }
- $cacheKey = $this->_cacheKey;
- if (strpos($cacheKey, '[') !== false) {
- $cacheKey= str_replace('[contextKey]', $context, $cacheKey);
- $cacheKey= str_replace('[id]', $id, $cacheKey);
- }
- return $cacheKey;
- }
- /**
- * Gets a collection of objects related by aggregate or composite relations.
- *
- * {@inheritdoc}
- *
- * Includes special handling for related objects with alias {@link
- * modTemplateVar}, respecting framework security unless specific criteria
- * are provided.
- *
- * @todo Refactor to use the new ABAC security model.
- */
- public function & getMany($alias, $criteria= null, $cacheFlag= false) {
- $collection= array ();
- if ($alias === 'TemplateVars' || $alias === 'modTemplateVar' && ($criteria === null || strtolower($criteria) === 'all')) {
- $collection= $this->getTemplateVars();
- } else {
- $collection= parent :: getMany($alias, $criteria, $cacheFlag);
- }
- return $collection;
- }
- /**
- * Get a collection of the Template Variable values for the Resource.
- *
- * @return array A collection of TemplateVar values for this Resource.
- */
- public function getTemplateVars() {
- return $this->xpdo->call('modResource', 'getTemplateVarCollection', array(&$this));
- }
- /**
- * Set a field value by the field key or name.
- *
- * {@inheritdoc}
- *
- * Additional logic added for the following fields:
- * -alias: Applies {@link modResource::cleanAlias()}
- * -contentType: Calls {@link modResource::addOne()} to sync contentType
- * -content_type: Sets the contentType field appropriately
- */
- public function set($k, $v= null, $vType= '') {
- $rt= false;
- switch ($k) {
- case 'alias' :
- $v= $this->cleanAlias($v);
- break;
- case 'contentType' :
- if ($v !== $this->get('contentType')) {
- if ($contentType= $this->xpdo->getObject('modContentType', array ('mime_type' => $v))) {
- if ($contentType->get('mime_type') != $this->get('contentType')) {
- $this->addOne($contentType, 'ContentType');
- }
- }
- }
- break;
- case 'content_type' :
- if ($v !== $this->get('content_type')) {
- /** @var modContentType $contentType */
- if ($contentType= $this->xpdo->getObject('modContentType', $v)) {
- if ($contentType->get('mime_type') != $this->get('contentType')) {
- $this->_fields['contentType']= $contentType->get('mime_type');
- $this->_dirty['contentType']= 'contentType';
- }
- }
- }
- break;
- }
- return parent :: set($k, $v, $vType);
- }
- /**
- * Adds an object related to this modResource by a foreign key relationship.
- *
- * {@inheritdoc}
- *
- * Adds legacy support for keeping the existing contentType field in sync
- * when a modContentType is set using this function.
- *
- * @param xPDOObject $obj
- * @param string $alias
- * @return boolean
- */
- public function addOne(& $obj, $alias= '') {
- $added= parent :: addOne($obj, $alias);
- if ($obj instanceof modContentType && $alias= 'ContentType') {
- $this->_fields['contentType']= $obj->get('mime_type');
- $this->_dirty['contentType']= 'contentType';
- }
- return $added;
- }
- /**
- * Transforms a string to form a valid URL representation.
- *
- * @param string $alias A string to transform into a valid URL representation.
- * @param array $options Options to append to or override configuration settings.
- * @return string The transformed string.
- */
- public function cleanAlias($alias, array $options = array()) {
- if ($this->xpdo instanceof modX && $ctx = $this->xpdo->getContext($this->get('context_key'))) {
- $options = array_merge($ctx->config, $options);
- }
- return $this->xpdo->call($this->_class, 'filterPathSegment', array(&$this->xpdo, $alias, $options));
- }
- /**
- * Persist new or changed modResource instances to the database container.
- *
- * If the modResource is new, the createdon and createdby fields will be set
- * using the current time and user authenticated in the context.
- *
- * If uri is empty or uri_overridden is not set and something has been changed which
- * might affect the Resource's uri, it is (re-)calculated using getAliasPath(). This
- * can be forced recursively by setting refreshURIs to true before calling save().
- *
- * @param boolean $cacheFlag
- * @return boolean
- */
- public function save($cacheFlag= null) {
- if ($this->isNew()) {
- if (!$this->get('createdon')) $this->set('createdon', time());
- if (!$this->get('createdby') && $this->xpdo instanceof modX) $this->set('createdby', $this->xpdo->getLoginUserID());
- }
- $refreshChildURIs = false;
- if ($this->xpdo instanceof modX && $this->xpdo->getOption('friendly_urls')) {
- $refreshChildURIs = ($this->get('refreshURIs') || $this->isDirty('uri') || $this->isDirty('alias') || $this->isDirty('alias_visible') || $this->isDirty('parent') || $this->isDirty('context_key'));
- if ($this->get('uri') == '' || (!$this->get('uri_override') && ($this->isDirty('uri_override') || $this->isDirty('content_type') || $this->isDirty('isfolder') || $refreshChildURIs))) {
- $this->set('uri', $this->getAliasPath($this->get('alias')));
- }
- }
- $changeContext = false;
- if ($this->xpdo instanceof modX) {
- $changeContext = $this->isDirty('context_key');
- }
- $rt= parent :: save($cacheFlag);
- if ($rt && $refreshChildURIs) {
- $this->xpdo->call('modResource', 'refreshURIs', array(
- &$this->xpdo,
- $this->get('id'),
- ));
- }
- if ($rt && $changeContext) {
- $this->xpdo->call($this->_class, 'updateContextOfChildren', array(&$this->xpdo, $this));
- }
- return $rt;
- }
- /**
- * Return whether or not the resource has been processed.
- *
- * @access public
- * @return boolean
- */
- public function getProcessed() {
- return $this->_processed;
- }
- /**
- * Set the field indicating the resource has been processed.
- *
- * @param boolean $processed Pass true to indicate the Resource has been processed.
- */
- public function setProcessed($processed) {
- $this->_processed= (boolean) $processed;
- }
- /**
- * Adds a lock on the Resource
- *
- * @access public
- * @param integer $user
- * @param array $options An array of options for the lock.
- * @return boolean True if the lock was successful.
- */
- public function addLock($user = 0, array $options = array()) {
- $locked = false;
- if ($this->xpdo instanceof modX) {
- if (!$user) {
- $user = $this->xpdo->user->get('id');
- }
- $lockedBy = $this->getLock();
- if (empty($lockedBy) || ($lockedBy == $user)) {
- $this->xpdo->registry->locks->subscribe('/resource/');
- $this->xpdo->registry->locks->send('/resource/', array(md5($this->get('id')) => $user), array('ttl' => $this->xpdo->getOption('lock_ttl', $options, 360)));
- $locked = true;
- } elseif ($lockedBy != $user) {
- $locked = $lockedBy;
- }
- }
- return $locked;
- }
- /**
- * Gets the lock on the Resource.
- *
- * @access public
- * @return int
- */
- public function getLock() {
- $lock = 0;
- if ($this->xpdo instanceof modX) {
- if ($this->xpdo->getService('registry', 'registry.modRegistry')) {
- $this->xpdo->registry->addRegister('locks', 'registry.modDbRegister', array('directory' => 'locks'));
- $this->xpdo->registry->locks->connect();
- $this->xpdo->registry->locks->subscribe('/resource/' . md5($this->get('id')));
- if ($msgs = $this->xpdo->registry->locks->read(array('remove_read' => false, 'poll_limit' => 1))) {
- $msg = reset($msgs);
- $lock = intval($msg);
- }
- }
- }
- return $lock;
- }
- /**
- * Removes all locks on a Resource.
- *
- * @access public
- * @param int $user
- * @return boolean True if locks were removed.
- */
- public function removeLock($user = 0) {
- $removed = false;
- if ($this->xpdo instanceof modX) {
- if (!$user) {
- $user = $this->xpdo->user->get('id');
- }
- $lockedBy = $this->getLock();
- if (empty($lockedBy) || $lockedBy == $user) {
- if ($this->xpdo->getService('registry', 'registry.modRegistry')) {
- $this->xpdo->registry->addRegister('locks', 'registry.modDbRegister', array('directory' => 'locks'));
- $this->xpdo->registry->locks->connect();
- $this->xpdo->registry->locks->subscribe('/resource/' . md5($this->get('id')));
- $this->xpdo->registry->locks->read(array('remove_read' => true, 'poll_limit' => 1));
- $removed = true;
- }
- }
- }
- return $removed;
- }
- /**
- * Loads the access control policies applicable to this resource.
- *
- * {@inheritdoc}
- */
- public function findPolicy($context = '') {
- $policy = array();
- $enabled = true;
- $context = !empty($context) ? $context : $this->xpdo->context->get('key');
- if ($context === $this->xpdo->context->get('key')) {
- $enabled = (boolean) $this->xpdo->getOption('access_resource_group_enabled', null, true);
- } elseif ($this->xpdo->getContext($context)) {
- $enabled = (boolean) $this->xpdo->contexts[$context]->getOption('access_resource_group_enabled', true);
- }
- if ($enabled) {
- if (empty($this->_policies) || !isset($this->_policies[$context])) {
- $accessTable = $this->xpdo->getTableName('modAccessResourceGroup');
- $policyTable = $this->xpdo->getTableName('modAccessPolicy');
- $resourceGroupTable = $this->xpdo->getTableName('modResourceGroupResource');
- $sql = "SELECT Acl.target, Acl.principal, Acl.authority, Acl.policy, Policy.data FROM {$accessTable} Acl " .
- "LEFT JOIN {$policyTable} Policy ON Policy.id = Acl.policy " .
- "JOIN {$resourceGroupTable} ResourceGroup ON Acl.principal_class = 'modUserGroup' " .
- "AND (Acl.context_key = :context OR Acl.context_key IS NULL OR Acl.context_key = '') " .
- "AND ResourceGroup.document = :resource " .
- "AND ResourceGroup.document_group = Acl.target " .
- "GROUP BY Acl.target, Acl.principal, Acl.authority, Acl.policy";
- $bindings = array(
- ':resource' => $this->get('id'),
- ':context' => $context
- );
- $query = new xPDOCriteria($this->xpdo, $sql, $bindings);
- if ($query->stmt && $query->stmt->execute()) {
- while ($row = $query->stmt->fetch(PDO::FETCH_ASSOC)) {
- $policy['modAccessResourceGroup'][$row['target']][] = array(
- 'principal' => $row['principal'],
- 'authority' => $row['authority'],
- 'policy' => $row['data'] ? $this->xpdo->fromJSON($row['data'], true) : array(),
- );
- }
- }
- $this->_policies[$context] = $policy;
- } else {
- $policy = $this->_policies[$context];
- }
- }
- return $policy;
- }
- /**
- * Checks to see if the Resource has children or not. Returns the number of
- * children.
- *
- * @access public
- * @return integer The number of children of the Resource
- */
- public function hasChildren() {
- $c = $this->xpdo->newQuery('modResource');
- $c->where(array(
- 'parent' => $this->get('id'),
- ));
- return $this->xpdo->getCount('modResource',$c);
- }
- /**
- * Gets the value of a TV for the Resource.
- *
- * @access public
- * @param mixed $pk Either the ID of the TV, or the name of the TV.
- * @return null/mixed The value of the TV for the Resource, or null if the
- * TV is not found.
- */
- public function getTVValue($pk) {
- $byName = !is_numeric($pk);
- /** @var modTemplateVar $tv */
- if ($byName && $this->xpdo instanceof modX) {
- $tv = $this->xpdo->getParser()->getElement('modTemplateVar', $pk);
- } else {
- $tv = $this->xpdo->getObject('modTemplateVar', $byName ? array('name' => $pk) : $pk);
- }
- return $tv == null ? null : $tv->renderOutput($this->get('id'));
- }
- /**
- * Sets a value for a TV for this Resource
- *
- * @param mixed $pk The TV name or ID to set
- * @param string $value The value to set for the TV
- * @return bool Whether or not the TV saved successfully
- */
- public function setTVValue($pk,$value) {
- $success = false;
- if (is_numeric($pk)) {
- $pk = intval($pk);
- } elseif (is_string($pk)) {
- $pk = array('name' => $pk);
- }
- /** @var modTemplateVar $tv */
- $tv = $this->xpdo->getObject('modTemplateVar',$pk);
- if ($tv) {
- $tv->setValue($this->get('id'),$value);
- $success = $tv->save();
- }
- return $success;
- }
- /**
- * Get the Resource's full alias path.
- *
- * @param string $alias Optional. The alias to check. If not set, will
- * then build it from the pagetitle if automatic_alias is set to true.
- * @param array $fields Optional. An array of field values to use instead of
- * using the current modResource fields.
- * @return string
- */
- public function getAliasPath($alias = '',array $fields = array()) {
- if (empty($fields)) $fields = $this->toArray();
- $workingContext = $this->xpdo->getContext($fields['context_key']);
- if (empty($fields['uri_override']) || empty($fields['uri'])) {
- /* auto assign alias if using automatic_alias */
- if (empty($alias) && $workingContext->getOption('automatic_alias', false)) {
- $alias = $this->cleanAlias($fields['pagetitle']);
- } elseif (empty($alias) && isset($fields['id']) && !empty($fields['id'])) {
- $alias = $this->cleanAlias($fields['id']);
- } else {
- $alias = $this->cleanAlias($alias);
- }
- $fullAlias= $alias;
- $isHtml= true;
- $extension= '';
- $containerSuffix= $workingContext->getOption('container_suffix', '');
- /* @var modContentType $contentType process content type */
- if (!empty($fields['content_type']) && $contentType= $this->xpdo->getObject('modContentType', $fields['content_type'])) {
- $extension= $contentType->getExtension();
- $isHtml= (strpos($contentType->get('mime_type'), 'html') !== false);
- }
- /* set extension to container suffix if Resource is a folder, HTML content type, and the container suffix is set */
- if (!empty($fields['isfolder']) && $isHtml && !empty ($containerSuffix)) {
- $extension= $containerSuffix;
- }
- $aliasPath= '';
- /* if using full alias paths, calculate here */
- if ($workingContext->getOption('use_alias_path', false)) {
- $useFrozenPathUris = $workingContext->getOption('use_frozen_parent_uris', false);
- $pathParentId= $fields['parent'];
- $parentResources= array ();
- $query = $this->xpdo->newQuery('modResource');
- $query->select($this->xpdo->getSelectColumns('modResource', '', '', array('parent', 'alias', 'alias_visible', 'uri', 'uri_override')));
- $query->where("{$this->xpdo->escape('id')} = ?");
- $query->prepare();
- $query->stmt->execute(array($pathParentId));
- $currResource= $query->stmt->fetch(PDO::FETCH_ASSOC);
- while ($currResource) {
- // If the use_frozen_parent_uris setting is enabled, we will look at the parent frozen uri instead
- // of building the full uri from all parents. This makes sure children will have an uri relative
- // from the parent at all times.
- if ($useFrozenPathUris && $currResource['uri_override'] && !empty($currResource['uri'])) {
- $parentResources[] = rtrim($currResource['uri'], '/');
- break;
- }
- $parentAlias= $currResource['alias'];
- if (empty ($parentAlias)) {
- $parentAlias= "{$pathParentId}";
- }
- // If we are ignoring the alias for this parent, simply skip adding it to the array for the alias
- // path.
- if ($currResource['alias_visible'] == 1) {
- $parentResources[]= "{$parentAlias}";
- }
- $pathParentId= $currResource['parent'];
- $query->stmt->execute(array($pathParentId));
- $currResource= $query->stmt->fetch(PDO::FETCH_ASSOC);
- }
- $aliasPath= !empty ($parentResources) ? implode('/', array_reverse($parentResources)) : '';
- if (strlen($aliasPath) > 0 && $aliasPath[strlen($aliasPath) - 1] !== '/') $aliasPath .= '/';
- }
- $fullAlias= $aliasPath . $fullAlias . $extension;
- } else {
- $fullAlias= $fields['uri'];
- }
- return $fullAlias;
- }
- /**
- * Tests to see if an alias is a duplicate.
- *
- * @param string $aliasPath The current full alias path. If none is passed,
- * will build it from the Resource's currently set alias.
- * @param string $contextKey The context to search for a duplicate alias in.
- * @return mixed The ID of the Resource using the alias, if a duplicate, otherwise false.
- */
- public function isDuplicateAlias($aliasPath = '', $contextKey = '') {
- if (empty($aliasPath)) $aliasPath = $this->getAliasPath($this->get('alias'));
- $criteria = $this->xpdo->newQuery('modResource');
- $where = array(
- 'id:!=' => $this->get('id'),
- 'uri' => $aliasPath,
- 'deleted' => false,
- );
- if (!empty($contextKey)) {
- $where['context_key'] = $contextKey;
- }
- $criteria->select('id');
- $criteria->where($where);
- $criteria->prepare();
- $duplicate = $this->xpdo->getValue($criteria->stmt);
- return $duplicate > 0 ? (integer) $duplicate : false;
- }
- /**
- * Duplicate the Resource.
- *
- * @param array $options An array of options.
- * @return mixed Returns either an error message, or the newly created modResource object.
- */
- public function duplicate(array $options = array()) {
- if (!($this->xpdo instanceof modX)) return false;
- /* duplicate resource */
- $prefixDuplicate = !empty($options['prefixDuplicate']) ? true : false;
- $newName = !empty($options['newName']) ? $options['newName'] : ($prefixDuplicate ? $this->xpdo->lexicon('duplicate_of', array(
- 'name' => $this->get('pagetitle'),
- )) : $this->get('pagetitle'));
- /** @var modResource $newResource */
- $newResource = $this->xpdo->newObject($this->get('class_key'));
- $newResource->fromArray($this->toArray('', true), '', false, true);
- $newResource->set('pagetitle', $newName);
- /* do published status preserving */
- $publishedMode = $this->getOption('publishedMode',$options,'preserve');
- switch ($publishedMode) {
- case 'unpublish':
- $newResource->set('published',false);
- $newResource->set('publishedon',0);
- $newResource->set('publishedby',0);
- break;
- case 'publish':
- $newResource->set('published',true);
- $newResource->set('publishedon',time());
- $newResource->set('publishedby',$this->xpdo->user->get('id'));
- break;
- case 'preserve':
- default:
- $newResource->set('published',$this->get('published'));
- $newResource->set('publishedon',$this->get('publishedon'));
- $newResource->set('publishedby',$this->get('publishedby'));
- break;
- }
- /* allow overrides for every item */
- if (!empty($options['overrides']) && is_array($options['overrides'])) {
- $newResource->fromArray($options['overrides']);
- }
- $newResource->set('id',0);
- /* make sure children get assigned to new parent */
- $newResource->set('parent',isset($options['parent']) ? $options['parent'] : $this->get('parent'));
- $newResource->set('createdby',$this->xpdo->user->get('id'));
- $newResource->set('createdon',time());
- $newResource->set('editedby',0);
- $newResource->set('editedon',0);
- /* get new alias */
- $preserve_alias = $this->xpdo->getOption('preserve_alias', $options, false);
- $alias = $newResource->cleanAlias($newName);
- if ($this->xpdo->getOption('friendly_urls', $options, false)) {
- if(!($preserve_alias)){
- /* auto assign alias */
- $aliasPath = $newResource->getAliasPath($newName);
- $dupeContext = $this->xpdo->getOption('global_duplicate_uri_check', $options, false) ? '' : $newResource->get('context_key');
- if ($newResource->isDuplicateAlias($aliasPath, $dupeContext)) {
- $alias = '';
- if ($newResource->get('uri_override')) {
- $newResource->set('uri_override', false);
- }
- }
- $newResource->set('alias',$alias);
- }
- }
- $preserve_menuindex = $this->xpdo->getOption('preserve_menuindex', $options, false);
- /* set new menuindex */
- if(!$preserve_menuindex){
- $menuindex = $this->xpdo->getCount('modResource',array('parent' => $this->get('parent')));
- $newResource->set('menuindex',$menuindex);
- }
- /* save resource */
- if (!$newResource->save()) {
- return $this->xpdo->lexicon('resource_err_duplicate');
- }
- $tvds = $this->getMany('TemplateVarResources');
- /** @var modTemplateVarResource $oldTemplateVarResource */
- foreach ($tvds as $oldTemplateVarResource) {
- /** @var modTemplateVarResource $newTemplateVarResource */
- $newTemplateVarResource = $this->xpdo->newObject('modTemplateVarResource');
- $newTemplateVarResource->set('contentid',$newResource->get('id'));
- $newTemplateVarResource->set('tmplvarid',$oldTemplateVarResource->get('tmplvarid'));
- $newTemplateVarResource->set('value',$oldTemplateVarResource->get('value'));
- $newTemplateVarResource->save();
- }
- $groups = $this->getMany('ResourceGroupResources');
- /** @var modResourceGroupResource $oldResourceGroupResource */
- foreach ($groups as $oldResourceGroupResource) {
- /** @var modResourceGroupResource $newResourceGroupResource */
- $newResourceGroupResource = $this->xpdo->newObject('modResourceGroupResource');
- $newResourceGroupResource->set('document_group',$oldResourceGroupResource->get('document_group'));
- $newResourceGroupResource->set('document',$newResource->get('id'));
- $newResourceGroupResource->save();
- }
- /* duplicate resource, recursively */
- $duplicateChildren = isset($options['duplicateChildren']) ? $options['duplicateChildren'] : true;
- if ($duplicateChildren) {
- if (!$this->checkPolicy('add_children')) return $newResource;
- $criteria = array(
- 'context_key' => $this->get('context_key'),
- 'parent' => $this->get('id')
- );
- $count = $this->xpdo->getCount('modResource',$criteria);
- if ($count > 0) {
- $children = $this->xpdo->getIterator('modResource',$criteria);
- /** @var modResource $child */
- foreach ($children as $child) {
- $child->duplicate(array(
- 'duplicateChildren' => true,
- 'parent' => $newResource->get('id'),
- 'prefixDuplicate' => $prefixDuplicate,
- 'overrides' => !empty($options['overrides']) ? $options['overrides'] : false,
- 'publishedMode' => $publishedMode,
- 'preserve_alias' => $preserve_alias,
- 'preserve_menuindex' => $preserve_menuindex
- ));
- }
- }
- }
- return $newResource;
- }
- /**
- * Joins a Resource to a Resource Group
- *
- * @access public
- * @param mixed $resourceGroupPk Either the ID, name or object of the Resource Group
- * @param boolean $byName Force the criteria to check by name for Numeric usergroup's name
- * @return boolean True if successful.
- */
- public function joinGroup($resourceGroupPk, $byName = false) {
- if (!is_object($resourceGroupPk) && !($resourceGroupPk instanceof modResourceGroup)) {
- if ($byName) {
- $c = array(
- 'name' => $resourceGroupPk,
- );
- } else {
- $c = array(
- is_int($resourceGroupPk) ? 'id' : 'name' => $resourceGroupPk,
- );
- }
- /** @var modResourceGroup $resourceGroup */
- $resourceGroup = $this->xpdo->getObject('modResourceGroup',$c);
- if (empty($resourceGroup) || !is_object($resourceGroup) || !($resourceGroup instanceof modResourceGroup)) {
- $this->xpdo->log(modX::LOG_LEVEL_ERROR, __METHOD__ . ' - No resource group: ' . $resourceGroupPk);
- return false;
- }
- } else {
- $resourceGroup =& $resourceGroupPk;
- }
- if ($this->isMember($resourceGroup->get('name'))) {
- $this->xpdo->log(modX::LOG_LEVEL_ERROR, __METHOD__ . ' - Resource '.$this->get('id') . ' already in resource group: ' . $resourceGroupPk);
- return false;
- }
- /** @var modResourceGroupResource $resourceGroupResource */
- $resourceGroupResource = $this->xpdo->newObject('modResourceGroupResource');
- $resourceGroupResource->set('document',$this->get('id'));
- $resourceGroupResource->set('document_group',$resourceGroup->get('id'));
- return $resourceGroupResource->save();
- }
- /**
- * Removes a Resource from a Resource Group
- *
- * @access public
- * @param int|string|modResourceGroup $resourceGroupPk Either the ID, name or object of the Resource Group
- * @return boolean True if successful.
- */
- public function leaveGroup($resourceGroupPk) {
- if (!is_object($resourceGroupPk) && !($resourceGroupPk instanceof modResourceGroup)) {
- $c = array(
- is_int($resourceGroupPk) ? 'id' : 'name' => $resourceGroupPk,
- );
- /** @var modResourceGroup $resourceGroup */
- $resourceGroup = $this->xpdo->getObject('modResourceGroup',$c);
- if (empty($resourceGroup) || !is_object($resourceGroup) || !($resourceGroup instanceof modResourceGroup)) {
- $this->xpdo->log(modX::LOG_LEVEL_ERROR, __METHOD__ . ' - No resource group: ' . (is_object($resourceGroupPk) ? $resourceGroupPk->get('name') : $resourceGroupPk));
- return false;
- }
- } else {
- $resourceGroup =& $resourceGroupPk;
- }
- if (!$this->isMember($resourceGroup->get('name'))) {
- $this->xpdo->log(modX::LOG_LEVEL_ERROR, __METHOD__ . ' - Resource ' . $this->get('id') . ' is not in resource group: ' . (is_object($resourceGroupPk) ? $resourceGroupPk->get('name') : $resourceGroupPk));
- return false;
- }
- /** @var modResourceGroupResource $resourceGroupResource */
- $resourceGroupResource = $this->xpdo->getObject('modResourceGroupResource',array(
- 'document' => $this->get('id'),
- 'document_group' => $resourceGroup->get('id'),
- ));
- return $resourceGroupResource->remove();
- }
- /**
- * Gets a sortable, limitable collection (and total count) of Resource Groups for the Resource.
- *
- * @param array $sort An array of sort columns in column => direction format.
- * @param int $limit A limit of records to retrieve in the collection.
- * @param int $offset A record offset for a limited collection.
- * @return array An array containing the collection and total.
- */
- public function getGroupsList(array $sort = array('id' => 'ASC'), $limit = 0, $offset = 0) {
- return $this->xpdo->call('modResource', 'listGroups', array(&$this, $sort, $limit, $offset));
- }
- /**
- * Gets all the Resource Group names of the resource groups this resource is assigned to.
- *
- * @access public
- * @return array An array of Resource Group names.
- */
- public function getResourceGroupNames() {
- $resourceGroupNames= array();
- $resourceGroups = $this->xpdo->getCollectionGraph('modResourceGroup', '{"ResourceGroupResources":{}}', array('ResourceGroupResources.document' => $this->get('id')));
- if ($resourceGroups) {
- /** @var modResourceGroup $resourceGroup */
- foreach ($resourceGroups as $resourceGroup) {
- $resourceGroupNames[] = $resourceGroup->get('name');
- }
- }
- return $resourceGroupNames;
- }
- /**
- * States whether a resource is a member of a resource group or groups. You may specify
- * either a string name of the resource group, or an array of names.
- *
- * @access public
- * @param string|array $groups Either a string of a resource group name or an array
- * of names.
- * @param boolean $matchAll If true, requires the resource to be a member of all
- * the resource groups specified. If false, the resource can be a member of only one to
- * pass. Defaults to false.
- * @return boolean True if the resource is a member of any of the resource groups
- * specified.
- */
- public function isMember($groups, $matchAll = false) {
- $isMember = false;
- $resourceGroupNames = $this->getResourceGroupNames();
- if ($resourceGroupNames) {
- if (is_array($groups)) {
- if ($matchAll) {
- $matches = array_diff($groups, $resourceGroupNames);
- $isMember = empty($matches);
- } else {
- $matches = array_intersect($groups, $resourceGroupNames);
- $isMember = !empty($matches);
- }
- } else {
- $isMember = (array_search($groups, $resourceGroupNames) !== false);
- }
- }
- return $isMember;
- }
- /**
- * Determine the controller path for this Resource class
- * @static
- * @param xPDO $modx A reference to the modX object
- * @return string The absolute path to the controller for this Resource class
- */
- public static function getControllerPath(xPDO &$modx) {
- $theme = $modx->getOption('manager_theme',null,'default');
- $controllersPath = $modx->getOption('manager_path',null,MODX_MANAGER_PATH).'controllers/'.$theme.'/';
- return $controllersPath.'resource/';
- }
- /**
- * Use this in your extended Resource class to display the text for the context menu item, if showInContextMenu is
- * set to true.
- * @return array
- */
- public function getContextMenuText() {
- return array(
- 'text_create' => $this->xpdo->lexicon('resource'),
- 'text_create_here' => $this->xpdo->lexicon('resource_create_here'),
- );
- }
- /**
- * Use this in your extended Resource class to return a translatable name for the Resource Type.
- * @return string
- */
- public function getResourceTypeName() {
- $className = $this->_class;
- if ($className == 'modDocument') $className = 'document';
- return $this->xpdo->lexicon($className);
- }
- /**
- * Use this in your extended Resource class to modify the tree node contents
- * @param array $node
- * @return array
- */
- public function prepareTreeNode(array $node = array()) {
- return $node;
- }
- /**
- * Get a namespaced property for the Resource
- * @param string $key
- * @param string $namespace
- * @param null $default
- * @return null
- */
- public function getProperty($key,$namespace = 'core',$default = null) {
- $properties = $this->get('properties');
- $properties = !empty($properties) ? $properties : array();
- return array_key_exists($namespace,$properties) && array_key_exists($key,$properties[$namespace]) ? $properties[$namespace][$key] : $default;
- }
- /**
- * Get the properties for the specific namespace for the Resource
- * @param string $namespace
- * @return array
- */
- public function getProperties($namespace = 'core') {
- $properties = $this->get('properties');
- $properties = !empty($properties) ? $properties : array();
- return array_key_exists($namespace,$properties) ? $properties[$namespace] : array();
- }
- /**
- * Set a namespaced property for the Resource
- * @param string $key
- * @param mixed $value
- * @param string $namespace
- * @return bool
- */
- public function setProperty($key,$value,$namespace = 'core') {
- $properties = $this->get('properties');
- $properties = !empty($properties) ? $properties : array();
- if (!array_key_exists($namespace,$properties)) $properties[$namespace] = array();
- $properties[$namespace][$key] = $value;
- return $this->set('properties',$properties);
- }
- /**
- * Set properties for a namespace on the Resource, optionally merging them with existing ones.
- * @param array $newProperties
- * @param string $namespace
- * @param bool $merge
- * @return boolean
- */
- public function setProperties(array $newProperties,$namespace = 'core',$merge = true) {
- $properties = $this->get('properties');
- $properties = !empty($properties) ? $properties : array();
- if (!array_key_exists($namespace,$properties)) $properties[$namespace] = array();
- $properties[$namespace] = $merge ? array_merge($properties[$namespace],$newProperties) : $newProperties;
- return $this->set('properties',$properties);
- }
- /**
- * Clear the cache of this resource in the current or specified Context.
- *
- * @param string $context Key of context for clearing
- *
- * @return void
- */
- public function clearCache($context = '') {
- /** @var xPDOFileCache $cache */
- $cache = $this->xpdo->cacheManager->getCacheProvider(
- $this->xpdo->getOption('cache_resource_key', null, 'resource'),
- array(
- xPDO::OPT_CACHE_HANDLER => $this->xpdo->getOption('cache_resource_handler', null, $this->xpdo->getOption(xPDO::OPT_CACHE_HANDLER, null, 'xPDOFileCache')),
- xPDO::OPT_CACHE_EXPIRES => (integer)$this->xpdo->getOption('cache_resource_expires', null, $this->xpdo->getOption(xPDO::OPT_CACHE_EXPIRES, null, 0)),
- xPDO::OPT_CACHE_FORMAT => (integer)$this->xpdo->getOption('cache_resource_format', null, $this->xpdo->getOption(xPDO::OPT_CACHE_FORMAT, null, xPDOCacheManager::CACHE_PHP)),
- xPDO::OPT_CACHE_ATTEMPTS => (integer)$this->xpdo->getOption('cache_resource_attempts', null, $this->xpdo->getOption(xPDO::OPT_CACHE_ATTEMPTS, null, 10)),
- xPDO::OPT_CACHE_ATTEMPT_DELAY => (integer)$this->xpdo->getOption('cache_resource_attempt_delay', null, $this->xpdo->getOption(xPDO::OPT_CACHE_ATTEMPT_DELAY, null, 1000)),
- )
- );
- $key = $this->getCacheKey($context);
- $cache->delete($key, array('deleteTop' => true));
- $cache->delete($key);
- if ($this->xpdo instanceof modX) {
- $this->xpdo->invokeEvent('OnResourceCacheUpdate', array('id' => $this->get('id')));
- }
- }
- }
|