mirror of
https://github.com/silverstripe/silverstripe-dms
synced 2024-10-22 14:05:56 +02:00
Merge pull request #119 from fspringveldt/feature/query-builder
Added Query Builder functionality to Document Sets
This commit is contained in:
commit
961306ecc0
16
_config/querybuilder.yml
Normal file
16
_config/querybuilder.yml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
DMSDocument:
|
||||||
|
searchable_fields:
|
||||||
|
Title:
|
||||||
|
title: "Document title matches ..."
|
||||||
|
Description:
|
||||||
|
title: "Document summary matches ..."
|
||||||
|
CreatedByID:
|
||||||
|
title: 'Document created by ...'
|
||||||
|
field: 'ListboxField'
|
||||||
|
filter: 'ExactMatchFilter'
|
||||||
|
LastEditedByID:
|
||||||
|
title: 'Document last changed by ...'
|
||||||
|
field: 'ListboxField'
|
||||||
|
filter: 'ExactMatchFilter'
|
||||||
|
Filename:
|
||||||
|
title: 'File name'
|
@ -120,8 +120,6 @@ class DMSDocumentAddController extends LeftAndMain
|
|||||||
_t('DMSDocumentAddController.MAINTAB', 'Main'),
|
_t('DMSDocumentAddController.MAINTAB', 'Main'),
|
||||||
new Tab(
|
new Tab(
|
||||||
_t('UploadField.FROMCOMPUTER', 'From your computer'),
|
_t('UploadField.FROMCOMPUTER', 'From your computer'),
|
||||||
new HiddenField('ID', false, $page->ID),
|
|
||||||
new HiddenField('DSID', false, $documentSet->ID),
|
|
||||||
$uploadField,
|
$uploadField,
|
||||||
new LiteralField(
|
new LiteralField(
|
||||||
'AllowedExtensions',
|
'AllowedExtensions',
|
||||||
@ -146,6 +144,8 @@ class DMSDocumentAddController extends LeftAndMain
|
|||||||
$form->Backlink = $backlink;
|
$form->Backlink = $backlink;
|
||||||
// Don't use AssetAdmin_EditForm, as it assumes a different panel structure
|
// Don't use AssetAdmin_EditForm, as it assumes a different panel structure
|
||||||
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
$form->setTemplate($this->getTemplatesWithSuffix('_EditForm'));
|
||||||
|
$form->Fields()->push(HiddenField::create('ID', false, $documentSet->ID));
|
||||||
|
$form->Fields()->push(HiddenField::create('DSID', false, $documentSet->ID));
|
||||||
|
|
||||||
return $form;
|
return $form;
|
||||||
}
|
}
|
||||||
|
@ -48,7 +48,7 @@ class DMSUploadField extends UploadField
|
|||||||
// Relate to the underlying document set being edited.
|
// Relate to the underlying document set being edited.
|
||||||
// Not applicable when editing the document itself and replacing it, or uploading from the ModelAdmin
|
// Not applicable when editing the document itself and replacing it, or uploading from the ModelAdmin
|
||||||
if ($record instanceof DMSDocumentSet) {
|
if ($record instanceof DMSDocumentSet) {
|
||||||
$record->Documents()->add($doc);
|
$record->Documents()->add($doc, array('BelongsToSet' => 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
return $doc;
|
return $doc;
|
||||||
|
118
code/forms/JsonField.php
Normal file
118
code/forms/JsonField.php
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class JsonField combines form inputs into a key-value pair
|
||||||
|
*/
|
||||||
|
class JsonField extends CompositeField
|
||||||
|
{
|
||||||
|
public function __construct($name, $children = null)
|
||||||
|
{
|
||||||
|
$this->setName($name);
|
||||||
|
|
||||||
|
if ($children instanceof FieldList || is_array($children)) {
|
||||||
|
foreach ($children as $child) {
|
||||||
|
$this->setChildName($child);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$children = is_array(func_get_args()) ? func_get_args() : array();
|
||||||
|
if (!empty($children)) {
|
||||||
|
array_shift($children);
|
||||||
|
}
|
||||||
|
foreach ($children as $child) {
|
||||||
|
$this->setChildName($child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
parent::__construct($children);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the name of the child object
|
||||||
|
*
|
||||||
|
* @param FormField $child
|
||||||
|
*/
|
||||||
|
private function setChildName($child)
|
||||||
|
{
|
||||||
|
$child->setName("{$this->getName()}[{$child->getName()}]");
|
||||||
|
}
|
||||||
|
|
||||||
|
public function hasData()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Override parent's behaviour as it's no longer required
|
||||||
|
*
|
||||||
|
* @param array $list
|
||||||
|
* @param bool $saveableOnly
|
||||||
|
*/
|
||||||
|
public function collateDataFields(&$list, $saveableOnly = false)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Recursively removed empty key-value pairs from $haystack
|
||||||
|
*
|
||||||
|
* @param $haystack
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function arrayFilterEmptyRecursive($haystack)
|
||||||
|
{
|
||||||
|
foreach ($haystack as $key => $value) {
|
||||||
|
if (is_array($value)) {
|
||||||
|
$haystack[$key] = $this->arrayFilterEmptyRecursive($haystack[$key]);
|
||||||
|
}
|
||||||
|
if (empty($haystack[$key])) {
|
||||||
|
unset($haystack[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $haystack;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Overrides parent behaviour to remove empty elements
|
||||||
|
*
|
||||||
|
* @return mixed|null|string
|
||||||
|
*/
|
||||||
|
public function dataValue()
|
||||||
|
{
|
||||||
|
$result = null;
|
||||||
|
if (is_array($this->value)) {
|
||||||
|
$this->value = $this->arrayFilterEmptyRecursive($this->value);
|
||||||
|
$result = (!empty($this->value)) ? Convert::array2json($this->value) : $result;
|
||||||
|
} else {
|
||||||
|
$result = parent::dataValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets
|
||||||
|
* @param mixed $value
|
||||||
|
*
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function setValue($value)
|
||||||
|
{
|
||||||
|
$this->value = $value;
|
||||||
|
if (is_string($value) && !empty($value)) {
|
||||||
|
$value = Convert::json2array($value);
|
||||||
|
} elseif (!is_array($value)) {
|
||||||
|
$value = array($value);
|
||||||
|
}
|
||||||
|
|
||||||
|
$pattern = "/^{$this->getName()}\[(.*)\]$/";
|
||||||
|
foreach ($this->children as $c) {
|
||||||
|
$title = $c->getName();
|
||||||
|
preg_match($pattern, $title, $matches);
|
||||||
|
if (!empty($matches[1]) && isset($value[$matches[1]])) {
|
||||||
|
$c->setValue($value[$matches[1]]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
}
|
@ -8,7 +8,6 @@
|
|||||||
* @property Varchar Title
|
* @property Varchar Title
|
||||||
* @property Text Description
|
* @property Text Description
|
||||||
* @property int ViewCount
|
* @property int ViewCount
|
||||||
* @property DateTime LastChanged
|
|
||||||
* @property Boolean EmbargoedIndefinitely
|
* @property Boolean EmbargoedIndefinitely
|
||||||
* @property Boolean EmbargoedUntilPublished
|
* @property Boolean EmbargoedUntilPublished
|
||||||
* @property DateTime EmbargoedUntilDate
|
* @property DateTime EmbargoedUntilDate
|
||||||
@ -22,6 +21,11 @@
|
|||||||
* @method ManyManyList ViewerGroups
|
* @method ManyManyList ViewerGroups
|
||||||
* @method ManyManyList EditorGroups
|
* @method ManyManyList EditorGroups
|
||||||
*
|
*
|
||||||
|
* @method Member CreatedBy
|
||||||
|
* @property Int CreatedByID
|
||||||
|
* @method Member LastEditedBy
|
||||||
|
* @property Int LastEditedByID
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
class DMSDocument extends DataObject implements DMSDocumentInterface
|
class DMSDocument extends DataObject implements DMSDocumentInterface
|
||||||
{
|
{
|
||||||
@ -31,9 +35,6 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
"Title" => 'Varchar(1024)', // eg. "Energy Saving Report for Year 2011, New Zealand LandCorp"
|
"Title" => 'Varchar(1024)', // eg. "Energy Saving Report for Year 2011, New Zealand LandCorp"
|
||||||
"Description" => 'Text',
|
"Description" => 'Text',
|
||||||
"ViewCount" => 'Int',
|
"ViewCount" => 'Int',
|
||||||
// When this document's file was created or last replaced (small changes like updating title don't count)
|
|
||||||
"LastChanged" => 'SS_DateTime',
|
|
||||||
|
|
||||||
"EmbargoedIndefinitely" => 'Boolean(false)',
|
"EmbargoedIndefinitely" => 'Boolean(false)',
|
||||||
"EmbargoedUntilPublished" => 'Boolean(false)',
|
"EmbargoedUntilPublished" => 'Boolean(false)',
|
||||||
"EmbargoedUntilDate" => 'SS_DateTime',
|
"EmbargoedUntilDate" => 'SS_DateTime',
|
||||||
@ -48,7 +49,9 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
);
|
);
|
||||||
|
|
||||||
private static $has_one = array(
|
private static $has_one = array(
|
||||||
'CoverImage' => 'Image'
|
'CoverImage' => 'Image',
|
||||||
|
'CreatedBy' => 'Member',
|
||||||
|
'LastEditedBy' => 'Member',
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $many_many = array(
|
private static $many_many = array(
|
||||||
@ -62,23 +65,13 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
'ID' => 'ID',
|
'ID' => 'ID',
|
||||||
'Title' => 'Title',
|
'Title' => 'Title',
|
||||||
'FilenameWithoutID' => 'Filename',
|
'FilenameWithoutID' => 'Filename',
|
||||||
'LastChanged' => 'LastChanged'
|
'LastEdited' => 'LastEdited'
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $singular_name = 'Document';
|
private static $singular_name = 'Document';
|
||||||
|
|
||||||
private static $plural_name = 'Documents';
|
private static $plural_name = 'Documents';
|
||||||
|
|
||||||
private static $searchable_fields = array(
|
|
||||||
'ID' => array(
|
|
||||||
'filter' => 'ExactMatchFilter',
|
|
||||||
'field' => 'NumericField'
|
|
||||||
),
|
|
||||||
'Title',
|
|
||||||
'Filename',
|
|
||||||
'LastChanged'
|
|
||||||
);
|
|
||||||
|
|
||||||
private static $summary_fields = array(
|
private static $summary_fields = array(
|
||||||
'Filename' => 'Filename',
|
'Filename' => 'Filename',
|
||||||
'Title' => 'Title',
|
'Title' => 'Title',
|
||||||
@ -126,7 +119,12 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($member && Permission::checkMember($member, array('ADMIN', 'SITETREE_EDIT_ALL', 'SITETREE_VIEW_ALL'))) {
|
if ($member && Permission::checkMember($member, array(
|
||||||
|
'ADMIN',
|
||||||
|
'SITETREE_EDIT_ALL',
|
||||||
|
'SITETREE_VIEW_ALL',
|
||||||
|
))
|
||||||
|
) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,7 +158,8 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Do early admin check
|
// Do early admin check
|
||||||
if ($member && Permission::checkMember($member,
|
if ($member && Permission::checkMember(
|
||||||
|
$member,
|
||||||
array(
|
array(
|
||||||
'ADMIN',
|
'ADMIN',
|
||||||
'SITETREE_EDIT_ALL',
|
'SITETREE_EDIT_ALL',
|
||||||
@ -817,7 +816,6 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
$this->Title = basename($filePath, '.'.$extension);
|
$this->Title = basename($filePath, '.'.$extension);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->LastChanged = SS_Datetime::now()->Rfc2822();
|
|
||||||
$this->write();
|
$this->write();
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
@ -994,7 +992,6 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
);
|
);
|
||||||
$versionsGridFieldConfig->getComponentByType('GridFieldDataColumns')
|
$versionsGridFieldConfig->getComponentByType('GridFieldDataColumns')
|
||||||
->setDisplayFields(Config::inst()->get('DMSDocument_versions', 'display_fields'))
|
->setDisplayFields(Config::inst()->get('DMSDocument_versions', 'display_fields'))
|
||||||
->setFieldCasting(array('LastChanged'=>"Datetime->Ago"))
|
|
||||||
->setFieldFormatting(
|
->setFieldFormatting(
|
||||||
array(
|
array(
|
||||||
'FilenameWithoutID' => '<a target=\'_blank\' class=\'file-url\' href=\'$Link\'>'
|
'FilenameWithoutID' => '<a target=\'_blank\' class=\'file-url\' href=\'$Link\'>'
|
||||||
@ -1143,7 +1140,7 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
//set the embargo options from the OptionSetField created in the getCMSFields method
|
//set the embargo options from the OptionSetField created in the getCMSFields method
|
||||||
//do not write after clearing the embargo (write happens automatically)
|
//do not write after clearing the embargo (write happens automatically)
|
||||||
$savedDate = $this->EmbargoedUntilDate;
|
$savedDate = $this->EmbargoedUntilDate;
|
||||||
$this->clearEmbargo(false); //clear all previous settings and re-apply them on save
|
$this->clearEmbargo(false); // Clear all previous settings and re-apply them on save
|
||||||
|
|
||||||
if ($this->Embargo == 'Published') {
|
if ($this->Embargo == 'Published') {
|
||||||
$this->embargoUntilPublished(false);
|
$this->embargoUntilPublished(false);
|
||||||
@ -1161,7 +1158,15 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
$this->expireAtDate($this->ExpireAtDate, false);
|
$this->expireAtDate($this->ExpireAtDate, false);
|
||||||
} else {
|
} else {
|
||||||
$this->clearExpiry(false);
|
$this->clearExpiry(false);
|
||||||
} //clear all previous settings
|
} // Clear all previous settings
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set user fields
|
||||||
|
if ($currentUserID = Member::currentUserID()) {
|
||||||
|
if (!$this->CreatedByID) {
|
||||||
|
$this->CreatedByID = $currentUserID;
|
||||||
|
}
|
||||||
|
$this->LastEditedByID = $currentUserID;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1289,11 +1294,6 @@ class DMSDocument extends DataObject implements DMSDocumentInterface
|
|||||||
_t('AssetTableField.LASTEDIT', 'Last changed') . ':',
|
_t('AssetTableField.LASTEDIT', 'Last changed') . ':',
|
||||||
$this->LastEdited
|
$this->LastEdited
|
||||||
),
|
),
|
||||||
new DateField_Disabled(
|
|
||||||
"LastChanged",
|
|
||||||
_t('AssetTableField.LASTCHANGED', 'Last replaced') . ':',
|
|
||||||
$this->LastChanged
|
|
||||||
),
|
|
||||||
new ReadonlyField("PublishedOn", "Published on". ':', $publishedOnValue),
|
new ReadonlyField("PublishedOn", "Published on". ':', $publishedOnValue),
|
||||||
new ReadonlyField("ReferencedOn", "Referenced on". ':', $relationListCountValue),
|
new ReadonlyField("ReferencedOn", "Referenced on". ':', $relationListCountValue),
|
||||||
new ReadonlyField("ViewCount", "View count". ':', $this->ViewCount)
|
new ReadonlyField("ViewCount", "View count". ':', $this->ViewCount)
|
||||||
|
@ -1,19 +1,34 @@
|
|||||||
<?php
|
<?php
|
||||||
/**
|
/**
|
||||||
* A document set is attached to Pages, and contains many DMSDocuments
|
* A document set is attached to Pages, and contains many DMSDocuments
|
||||||
|
*
|
||||||
|
* @property Varchar Title
|
||||||
|
* @property Text KeyValuePairs
|
||||||
|
* @property Enum SortBy
|
||||||
|
* @property Enum SortByDirection
|
||||||
*/
|
*/
|
||||||
class DMSDocumentSet extends DataObject
|
class DMSDocumentSet extends DataObject
|
||||||
{
|
{
|
||||||
|
|
||||||
private static $db = array(
|
private static $db = array(
|
||||||
'Title' => 'Varchar(255)'
|
'Title' => 'Varchar(255)',
|
||||||
|
'KeyValuePairs' => 'Text',
|
||||||
|
'SortBy' => "Enum('LastEdited,Created,Title')')",
|
||||||
|
'SortByDirection' => "Enum('DESC,ASC')')",
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $has_one = array(
|
private static $has_one = array(
|
||||||
'Page' => 'SiteTree'
|
'Page' => 'SiteTree',
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $many_many = array(
|
private static $many_many = array(
|
||||||
'Documents' => 'DMSDocument'
|
'Documents' => 'DMSDocument',
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $many_many_extraFields = array(
|
||||||
|
'Documents' => array(
|
||||||
|
'BelongsToSet' => 'Boolean(1)', // Flag indicating if a document was added directly to a set - in which case it is set - or added via the query-builder.
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -28,14 +43,12 @@ class DMSDocumentSet extends DataObject
|
|||||||
* }
|
* }
|
||||||
* </code>
|
* </code>
|
||||||
*
|
*
|
||||||
* @return DataList
|
* @return DataList|null
|
||||||
*/
|
*/
|
||||||
public function getDocuments()
|
public function getDocuments()
|
||||||
{
|
{
|
||||||
$documents = $this->Documents();
|
$documents = $this->Documents();
|
||||||
|
|
||||||
$this->extend('updateDocuments', $documents);
|
$this->extend('updateDocuments', $documents);
|
||||||
|
|
||||||
return $documents;
|
return $documents;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,8 +62,11 @@ class DMSDocumentSet extends DataObject
|
|||||||
{
|
{
|
||||||
// PHP 5.3 only
|
// PHP 5.3 only
|
||||||
$self = $this;
|
$self = $this;
|
||||||
|
|
||||||
$this->beforeUpdateCMSFields(function (FieldList $fields) use ($self) {
|
$this->beforeUpdateCMSFields(function (FieldList $fields) use ($self) {
|
||||||
|
$fields->removeFieldsFromTab(
|
||||||
|
'Root.Main',
|
||||||
|
array('KeyValuePairs', 'SortBy', 'SortByDirection')
|
||||||
|
);
|
||||||
// Don't put the GridField for documents in until the set has been created
|
// Don't put the GridField for documents in until the set has been created
|
||||||
if (!$self->isInDB()) {
|
if (!$self->isInDB()) {
|
||||||
$fields->addFieldToTab(
|
$fields->addFieldToTab(
|
||||||
@ -64,16 +80,13 @@ class DMSDocumentSet extends DataObject
|
|||||||
),
|
),
|
||||||
'Title'
|
'Title'
|
||||||
);
|
);
|
||||||
return;
|
} else {
|
||||||
}
|
|
||||||
|
|
||||||
// Document listing
|
// Document listing
|
||||||
$gridFieldConfig = GridFieldConfig::create()
|
$gridFieldConfig = GridFieldConfig::create()
|
||||||
->addComponents(
|
->addComponents(
|
||||||
new GridFieldToolbarHeader(),
|
new GridFieldToolbarHeader(),
|
||||||
new GridFieldFilterHeader(),
|
new GridFieldFilterHeader(),
|
||||||
new GridFieldSortableHeader(),
|
new GridFieldSortableHeader(),
|
||||||
// new GridFieldOrderableRows('DocumentSort'),
|
|
||||||
new GridFieldDataColumns(),
|
new GridFieldDataColumns(),
|
||||||
new GridFieldEditButton(),
|
new GridFieldEditButton(),
|
||||||
// Special delete dialog to handle custom behaviour of unlinking and deleting
|
// Special delete dialog to handle custom behaviour of unlinking and deleting
|
||||||
@ -90,34 +103,32 @@ class DMSDocumentSet extends DataObject
|
|||||||
|
|
||||||
if (class_exists('GridFieldSortableRows')) {
|
if (class_exists('GridFieldSortableRows')) {
|
||||||
$sortableComponent = new GridFieldSortableRows('DocumentSort');
|
$sortableComponent = new GridFieldSortableRows('DocumentSort');
|
||||||
// setUsePagenation method removed from newer version of SortableGridField.
|
// setUsePagination method removed from newer version of SortableGridField.
|
||||||
if (method_exists($sortableComponent, 'setUsePagination')) {
|
if (method_exists($sortableComponent, 'setUsePagination')) {
|
||||||
$sortableComponent->setUsePagination(false)->setForceRedraw(true);
|
$sortableComponent->setUsePagination(false)->setForceRedraw(true);
|
||||||
}
|
}
|
||||||
$gridFieldConfig->addComponent($sortableComponent);
|
$gridFieldConfig->addComponent($sortableComponent);
|
||||||
}
|
}
|
||||||
|
|
||||||
// HACK: Create a singleton of DMSDocument to ensure extensions are applied before we try to get display fields.
|
|
||||||
singleton('DMSDocument');
|
|
||||||
$gridFieldConfig->getComponentByType('GridFieldDataColumns')
|
$gridFieldConfig->getComponentByType('GridFieldDataColumns')
|
||||||
->setDisplayFields(Config::inst()->get('DMSDocument', 'display_fields'))
|
->setDisplayFields(DMSDocument::create()->config()->get('display_fields'))
|
||||||
->setFieldCasting(array('LastChanged' => 'Datetime->Ago'))
|
->setFieldCasting(array('LastEdited' => 'Datetime->Ago'))
|
||||||
->setFieldFormatting(
|
->setFieldFormatting(
|
||||||
array(
|
array(
|
||||||
'FilenameWithoutID' => '<a target=\'_blank\' class=\'file-url\' href=\'$Link\'>$FilenameWithoutID</a>'
|
'FilenameWithoutID' => '<a target=\'_blank\' class=\'file-url\' href=\'$Link\'>$FilenameWithoutID</a>',
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
// Override delete functionality with this class
|
// Override delete functionality with this class
|
||||||
$gridFieldConfig->getComponentByType('GridFieldDetailForm')
|
$gridFieldConfig->getComponentByType('GridFieldDetailForm')
|
||||||
->setItemRequestClass('DMSGridFieldDetailForm_ItemRequest');
|
->setItemRequestClass('DMSGridFieldDetailForm_ItemRequest');
|
||||||
|
|
||||||
$gridField = GridField::create(
|
$gridField = GridField::create(
|
||||||
'Documents',
|
'Documents',
|
||||||
false,
|
false,
|
||||||
$self->Documents(), //->Sort('DocumentSort'),
|
$self->Documents(),
|
||||||
$gridFieldConfig
|
$gridFieldConfig
|
||||||
);
|
);
|
||||||
|
$gridField->setModelClass('DMSDocument');
|
||||||
$gridField->addExtraClass('documents');
|
$gridField->addExtraClass('documents');
|
||||||
|
|
||||||
$gridFieldConfig->addComponent(
|
$gridFieldConfig->addComponent(
|
||||||
@ -128,6 +139,8 @@ class DMSDocumentSet extends DataObject
|
|||||||
|
|
||||||
$fields->removeByName('Documents');
|
$fields->removeByName('Documents');
|
||||||
$fields->addFieldToTab('Root.Main', $gridField);
|
$fields->addFieldToTab('Root.Main', $gridField);
|
||||||
|
$self->addQueryFields($fields);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
$this->addRequirements();
|
$this->addRequirements();
|
||||||
return parent::getCMSFields();
|
return parent::getCMSFields();
|
||||||
@ -150,4 +163,95 @@ class DMSDocumentSet extends DataObject
|
|||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds the query fields to build the document logic to the DMSDocumentSet.
|
||||||
|
*
|
||||||
|
* To extend use the following from within an Extension subclass:
|
||||||
|
*
|
||||||
|
* <code>
|
||||||
|
* public function updateQueryFields($result)
|
||||||
|
* {
|
||||||
|
* // Do something here
|
||||||
|
* }
|
||||||
|
* </code>
|
||||||
|
*
|
||||||
|
* @param FieldList $fields
|
||||||
|
*/
|
||||||
|
public function addQueryFields($fields)
|
||||||
|
{
|
||||||
|
/** @var DMSDocument $doc */
|
||||||
|
$doc = singleton('DMSDocument');
|
||||||
|
/** @var FormField $field */
|
||||||
|
$dmsDocFields = $doc->scaffoldSearchFields(array('fieldClasses' => true));
|
||||||
|
$membersMap = Member::get()->map('ID', 'Name')->toArray();
|
||||||
|
asort($membersMap);
|
||||||
|
foreach ($dmsDocFields as $field) {
|
||||||
|
// Apply field customisations where necessary
|
||||||
|
if (in_array($field->getName(), array('CreatedByID', 'LastEditedByID', 'LastEditedByID'))) {
|
||||||
|
/** @var ListboxField $field */
|
||||||
|
$field->setMultiple(true)->setSource($membersMap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$keyValPairs = JsonField::create('KeyValuePairs', $dmsDocFields->toArray());
|
||||||
|
|
||||||
|
// Now lastly add the sort fields
|
||||||
|
$sortedBy = FieldGroup::create('SortedBy', array(
|
||||||
|
DropdownField::create('SortBy', '', array(
|
||||||
|
'LastEdited' => 'Last changed',
|
||||||
|
'Created' => 'Created',
|
||||||
|
'Title' => 'Document title',
|
||||||
|
), 'LastEdited'),
|
||||||
|
DropdownField::create('SortByDirection', '', $this->dbObject('SortByDirection')->enumValues(), 'DESC'),
|
||||||
|
));
|
||||||
|
|
||||||
|
$sortedBy->setTitle(_t('DMSDocumentSet.SORTED_BY', 'Sort the document set by:'));
|
||||||
|
$fields->addFieldsToTab('Root.QueryBuilder', array($keyValPairs, $sortedBy));
|
||||||
|
$this->extend('updateQueryFields', $fields);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function onBeforeWrite()
|
||||||
|
{
|
||||||
|
parent::onBeforeWrite();
|
||||||
|
|
||||||
|
$this->saveLinkedDocuments();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve a list of the documents in this set. An extension hook is provided before the result is returned.
|
||||||
|
*
|
||||||
|
* @return ArrayList|null
|
||||||
|
*/
|
||||||
|
public function saveLinkedDocuments()
|
||||||
|
{
|
||||||
|
// Documents that belong to just this set.
|
||||||
|
/** @var ManyManyList $originals */
|
||||||
|
$originals = $this->Documents();
|
||||||
|
if (!(empty($this->KeyValuePairs)) && $this->isChanged('KeyValuePairs')) {
|
||||||
|
$keyValuesPair = Convert::json2array($this->KeyValuePairs);
|
||||||
|
/** @var DMSDocument $dmsDoc */
|
||||||
|
$dmsDoc = singleton('DMSDocument');
|
||||||
|
$context = $dmsDoc->getDefaultSearchContext();
|
||||||
|
|
||||||
|
$sortBy = $this->SortBy ? $this->SortBy : 'LastEdited';
|
||||||
|
$sortByDirection = $this->SortByDirection ? $this->SortByDirection : 'DESC';
|
||||||
|
$sortedBy = sprintf('%s %s', $sortBy, $sortByDirection);
|
||||||
|
/** @var DataList $documents */
|
||||||
|
$documents = $context->getResults($keyValuesPair, $sortedBy);
|
||||||
|
$now = SS_Datetime::now()->Rfc2822();
|
||||||
|
$documents = $documents->where(
|
||||||
|
"\"EmbargoedIndefinitely\" = 0 AND ".
|
||||||
|
" \"EmbargoedUntilPublished\" = 0 AND ".
|
||||||
|
"(\"EmbargoedUntilDate\" IS NULL OR " .
|
||||||
|
"(\"EmbargoedUntilDate\" IS NOT NULL AND '{$now}' >= \"EmbargoedUntilDate\")) AND " .
|
||||||
|
"\"ExpireAtDate\" IS NULL OR (\"ExpireAtDate\" IS NOT NULL AND '{$now}' < \"ExpireAtDate\")"
|
||||||
|
);
|
||||||
|
|
||||||
|
// Remove all BelongsToSet as the rules have changed
|
||||||
|
$originals->removeByFilter('"BelongsToSet" = 0');
|
||||||
|
foreach ($documents as $document) {
|
||||||
|
$originals->add($document, array('BelongsToSet' => 0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,7 +34,7 @@ class DMSDocument_versions extends DataObject
|
|||||||
private static $display_fields = array(
|
private static $display_fields = array(
|
||||||
'VersionCounter' => 'Version Counter',
|
'VersionCounter' => 'Version Counter',
|
||||||
'FilenameWithoutID' => 'Filename',
|
'FilenameWithoutID' => 'Filename',
|
||||||
'LastChanged' => 'Last Changed'
|
'LastEdited' => 'Last Changed'
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $summary_fields = array(
|
private static $summary_fields = array(
|
||||||
@ -47,7 +47,7 @@ class DMSDocument_versions extends DataObject
|
|||||||
);
|
);
|
||||||
|
|
||||||
private static $default_sort = array(
|
private static $default_sort = array(
|
||||||
'LastChanged' => 'DESC'
|
'LastEdited' => 'DESC'
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
|
@ -28,3 +28,31 @@ DMSDocumentAddController:
|
|||||||
- php
|
- php
|
||||||
- php5
|
- php5
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Adding fields to the Query Builder
|
||||||
|
Query builder fields are read from the DMSDocument::searchable_fields property set in [querybuilder.yml](../../_config/querybuilder.yml). Some default fields are provided and can be customised
|
||||||
|
by modifying the field and/or filter properties of a field or adding a new field entirely.
|
||||||
|
|
||||||
|
[See here for information](https://docs.silverstripe.org/en/developer_guides/model/searchfilters/) on how to modify search filters and [see here for more information](https://docs.silverstripe.org/en/developer_guides/forms/field_types/common_subclasses/)
|
||||||
|
on the field types available.
|
||||||
|
|
||||||
|
The default searchable filters available to query builder is as follows:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
DMSDocument:
|
||||||
|
searchable_fields:
|
||||||
|
Title:
|
||||||
|
title: "Document title matches ..."
|
||||||
|
Description:
|
||||||
|
title: "Document summary matches ..."
|
||||||
|
CreatedByID:
|
||||||
|
title: 'Document created by ...'
|
||||||
|
field: 'ListboxField'
|
||||||
|
filter: 'ExactMatchFilter'
|
||||||
|
LastEditedByID:
|
||||||
|
title: 'Document last changed by ...'
|
||||||
|
field: 'ListboxField'
|
||||||
|
filter: 'ExactMatchFilter'
|
||||||
|
Filename:
|
||||||
|
title: 'File name'
|
||||||
|
```
|
@ -1,4 +1,4 @@
|
|||||||
<strong>$FilenameWithoutID</strong>
|
<strong>$FilenameWithoutID</strong>
|
||||||
| $Extension
|
| $Extension
|
||||||
| $FileSizeFormatted
|
| $FileSizeFormatted
|
||||||
| <%t DMSDocument.LASTCHANGED "Last changed: {date}" date=$LastChanged.Nice %>
|
| <%t DMSDocument.LASTCHANGED "Last changed: {date}" date=$LastEdited.Nice %>
|
@ -83,4 +83,57 @@ class DMSDocumentSetTest extends SapphireTest
|
|||||||
$sortableAssertion = class_exists('GridFieldSortableRows') ? 'assertNotNull' : 'assertNull';
|
$sortableAssertion = class_exists('GridFieldSortableRows') ? 'assertNotNull' : 'assertNull';
|
||||||
$this->$sortableAssertion($config->getComponentByType('GridFieldSortableRows'));
|
$this->$sortableAssertion($config->getComponentByType('GridFieldSortableRows'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test that query fields can be added to the gridfield
|
||||||
|
*/
|
||||||
|
public function testAddQueryFields()
|
||||||
|
{
|
||||||
|
|
||||||
|
/** @var DMSDocumentSet $set */
|
||||||
|
$set = $this->objFromFixture('DMSDocumentSet', 'ds6');
|
||||||
|
/** @var FieldList $fields */
|
||||||
|
$fields = new FieldList(new TabSet('Root'));
|
||||||
|
/** @var FieldList $fields */
|
||||||
|
$set->addQueryFields($fields);
|
||||||
|
$keyValuePairs = $fields->dataFieldByName('KeyValuePairs');
|
||||||
|
$this->assertNotNull(
|
||||||
|
$keyValuePairs,
|
||||||
|
'addQueryFields() includes KeyValuePairs composite field'
|
||||||
|
);
|
||||||
|
$this->assertNotNull(
|
||||||
|
$keyValuePairs->fieldByName('KeyValuePairs[Title]'),
|
||||||
|
'addQueryFields() includes KeyValuePairs composite field'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testAddQueryFieldsIsExtensible()
|
||||||
|
{
|
||||||
|
|
||||||
|
DMSDocumentSet::add_extension('StubDocumentSetMockExtension');
|
||||||
|
|
||||||
|
$fields = new FieldList(new TabSet('Root'));
|
||||||
|
$set = new DMSDocumentSet;
|
||||||
|
$set->addQueryFields($fields);
|
||||||
|
|
||||||
|
$this->assertNotNull(
|
||||||
|
$fields->dataFieldByName('ExtendedField'),
|
||||||
|
'addQueryFields() is extendible as it included the field from the extension'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test that extra documents are added after write
|
||||||
|
*/
|
||||||
|
public function testSaveLinkedDocuments()
|
||||||
|
{
|
||||||
|
/** @var DMSDocumentSet $set */
|
||||||
|
$set = $this->objFromFixture('DMSDocumentSet', 'dsSaveLinkedDocuments');
|
||||||
|
// Assert initially docs
|
||||||
|
$this->assertEquals(1, $set->getDocuments()->count(), 'Set has 1 document');
|
||||||
|
// Now apply the query and see if 2 extras were added with CreatedByID filter
|
||||||
|
$set->KeyValuePairs = '{"Filename":"extradoc3"}';
|
||||||
|
$set->saveLinkedDocuments();
|
||||||
|
$this->assertEquals(2, $set->getDocuments()->count(), 'Set has 2 documents');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
47
tests/JsonFieldTest.php
Normal file
47
tests/JsonFieldTest.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
class JsonFieldTest extends SapphireTest
|
||||||
|
{
|
||||||
|
public function testJsonFieldConstructorMultiWays()
|
||||||
|
{
|
||||||
|
$jsonField = new JsonField('MyJsonField', new FieldList(
|
||||||
|
new TextField('FirstName', 'Given name'),
|
||||||
|
new TextField('Surname', 'Last name')
|
||||||
|
));
|
||||||
|
$this->assertEquals($jsonField->FieldList()->count(), 2);
|
||||||
|
$this->assertNotNull($jsonField->FieldList()->dataFieldByName('MyJsonField[FirstName]'));
|
||||||
|
|
||||||
|
$jsonField = new JsonField('MyJsonField', array(new TextField('FirstName', 'Given name'),
|
||||||
|
new TextField('Surname', 'Last name')));
|
||||||
|
$this->assertEquals($jsonField->FieldList()->count(), 2);
|
||||||
|
$this->assertNotNull($jsonField->FieldList()->dataFieldByName('MyJsonField[FirstName]'));
|
||||||
|
|
||||||
|
$jsonField = new JsonField(
|
||||||
|
'MyJsonField',
|
||||||
|
new TextField('FirstName', 'Given name'),
|
||||||
|
new TextField('Surname', 'Last name')
|
||||||
|
);
|
||||||
|
$this->assertEquals($jsonField->FieldList()->count(), 2);
|
||||||
|
$this->assertNotNull($jsonField->FieldList()->dataFieldByName('MyJsonField[FirstName]'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testJsonFieldDataValueCouldDealWithArray()
|
||||||
|
{
|
||||||
|
$jsonField = new JsonField('MyJsonField', new FieldList(
|
||||||
|
new TextField('FirstName', 'Given name'),
|
||||||
|
new TextField('Surname', 'Last name')
|
||||||
|
));
|
||||||
|
$jsonField->setValue($value = array(
|
||||||
|
'MyJsonField'=>array(
|
||||||
|
'FirstName' => 'Normann',
|
||||||
|
'Surname' => 'Lou',
|
||||||
|
),
|
||||||
|
));
|
||||||
|
|
||||||
|
$this->assertEquals($jsonField->dataValue(), Convert::array2json($value));
|
||||||
|
$jsonField->setValue($value = array(
|
||||||
|
'MyJsonField'=>array(),
|
||||||
|
));
|
||||||
|
$this->assertNull($jsonField->dataValue());
|
||||||
|
}
|
||||||
|
}
|
24
tests/Stub/StubDocumentSetMockExtension.php
Normal file
24
tests/Stub/StubDocumentSetMockExtension.php
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class StubDocumentSetMockExtension
|
||||||
|
*
|
||||||
|
* @package dms
|
||||||
|
*/
|
||||||
|
class StubDocumentSetMockExtension extends DataExtension implements TestOnly
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* For method {@link DMSDocumentSet::addQueryFields}
|
||||||
|
*
|
||||||
|
* @param FieldList $fields
|
||||||
|
*
|
||||||
|
* @return FieldList
|
||||||
|
*/
|
||||||
|
public function updateQueryFields($fields)
|
||||||
|
{
|
||||||
|
$fields->addFieldToTab('Root.QueryBuilder', new TextField('ExtendedField'));
|
||||||
|
|
||||||
|
return $fields;
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,10 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class StubRelatedDocumentExtension
|
||||||
|
*
|
||||||
|
* @package dms
|
||||||
|
*/
|
||||||
class StubRelatedDocumentExtension extends DataExtension implements TestOnly
|
class StubRelatedDocumentExtension extends DataExtension implements TestOnly
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
|
@ -20,6 +20,12 @@ SiteTree:
|
|||||||
s7:
|
s7:
|
||||||
Title: testPage7 has documents embargoed until publish
|
Title: testPage7 has documents embargoed until publish
|
||||||
URLSegment: s7
|
URLSegment: s7
|
||||||
|
s8:
|
||||||
|
Title: testPage8
|
||||||
|
URLSegment: s8
|
||||||
|
s9:
|
||||||
|
Title: testPage9
|
||||||
|
URLSegment: s9
|
||||||
DMSTag:
|
DMSTag:
|
||||||
t1:
|
t1:
|
||||||
Category: tag1
|
Category: tag1
|
||||||
@ -69,6 +75,12 @@ DMSDocumentSet:
|
|||||||
ds5:
|
ds5:
|
||||||
Title: Set containing embargoed until publish documents
|
Title: Set containing embargoed until publish documents
|
||||||
Page: =>SiteTree.s7
|
Page: =>SiteTree.s7
|
||||||
|
ds6:
|
||||||
|
Title: Test Set 6
|
||||||
|
Page: =>SiteTree.s8
|
||||||
|
dsSaveLinkedDocuments:
|
||||||
|
Title: Test Set 6
|
||||||
|
Page: =>SiteTree.s9
|
||||||
DMSDocument:
|
DMSDocument:
|
||||||
d1:
|
d1:
|
||||||
Filename: test-file-file-doesnt-exist-1
|
Filename: test-file-file-doesnt-exist-1
|
||||||
@ -122,3 +134,23 @@ DMSDocument:
|
|||||||
EmbargoUntilPublish: true
|
EmbargoUntilPublish: true
|
||||||
Folder: 5
|
Folder: 5
|
||||||
Sets: =>DMSDocumentSet.ds5
|
Sets: =>DMSDocumentSet.ds5
|
||||||
|
extraDoc1:
|
||||||
|
Filename: extradoc1
|
||||||
|
Folder: 5
|
||||||
|
Sets: =>DMSDocumentSet.ds6
|
||||||
|
BelongsToSet: 1
|
||||||
|
CreatedByID: 2
|
||||||
|
extraDoc2:
|
||||||
|
Filename: extradoc2
|
||||||
|
Folder: 5
|
||||||
|
CreatedByID: 2
|
||||||
|
BelongsToSet: 0
|
||||||
|
extraDoc3:
|
||||||
|
Filename: extradoc3
|
||||||
|
Folder: 5
|
||||||
|
BelongsToSet: 0
|
||||||
|
CreatedByID: 2
|
||||||
|
docSaveLinkedDocuments1:
|
||||||
|
Filename: saveLinkedDocument1
|
||||||
|
Folder: 5
|
||||||
|
Sets: =>DMSDocumentSet.dsSaveLinkedDocuments
|
Loading…
Reference in New Issue
Block a user