mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
d04e908358
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/cms/branches/2.3@70956 467b73ca-7a2a-4603-9d3b-597d59a354a9
382 lines
11 KiB
PHP
Executable File
382 lines
11 KiB
PHP
Executable File
<?php
|
|
/**
|
|
* Enhances {ComplexTableField} with the ability to list groups and given members.
|
|
* It is based around groups, so it deletes Members from a Group rather than from the entire system.
|
|
*
|
|
* In contrast to the original implementation, the URL-parameters "ParentClass" and "ParentID" are used
|
|
* to specify "Group" (hardcoded) and the GroupID-relation.
|
|
*
|
|
* Returns either:
|
|
* - provided members
|
|
* - members of a provided group
|
|
* - all members
|
|
* - members based on a search-query
|
|
* @package cms
|
|
* @subpackage security
|
|
*/
|
|
class MemberTableField extends ComplexTableField {
|
|
|
|
protected $members;
|
|
|
|
protected $hidePassword;
|
|
|
|
protected $pageSize;
|
|
|
|
protected $detailFormValidator;
|
|
|
|
protected $group;
|
|
|
|
protected $template = "MemberTableField";
|
|
|
|
public $popupClass = 'MemberTableField_Popup';
|
|
|
|
static $data_class = "Member";
|
|
|
|
protected $permissions = array(
|
|
'add',
|
|
'edit',
|
|
'delete'
|
|
);
|
|
|
|
private static $addedPermissions = array();
|
|
|
|
private static $addedFields = array();
|
|
|
|
private static $addedCsvFields = array();
|
|
|
|
public static function addPermissions($addingPermissionList) {
|
|
self::$addedPermissions = $addingPermissionList;
|
|
}
|
|
|
|
public static function addMembershipFields($addingFieldList, $addingCsvFieldList = null) {
|
|
self::$addedFields = $addingFieldList;
|
|
$addingCsvFieldList == null ? self::$addedCsvFields = $addingFieldList : self::$addedCsvFields = $addingCsvFieldList;
|
|
}
|
|
|
|
function __construct($controller, $name, $group, $members = null, $hidePassword = true, $pageLimit = 10) {
|
|
if($group) {
|
|
if(is_object($group)) {
|
|
$this->group = $group;
|
|
} elseif(is_numeric($group)) {
|
|
$this->group = DataObject::get_by_id('Group', $group);
|
|
}
|
|
} else if(is_numeric($_REQUEST['ctf'][$this->Name()]["ID"])) {
|
|
$this->group = DataObject::get_by_id('Group', $_REQUEST['ctf'][$this->Name()]["ID"]);
|
|
}
|
|
|
|
$sourceClass = $this->stat('data_class');
|
|
$SNG_member = singleton($this->stat('data_class'));
|
|
|
|
foreach(self::$addedPermissions as $permission) {
|
|
array_push($this->permissions, $permission);
|
|
}
|
|
|
|
$fieldList = array(
|
|
"FirstName" => _t('MemberTableField.FIRSTNAME', 'Firstname'),
|
|
"Surname" => _t('MemberTableField.SURNAME', 'Surname'),
|
|
"Email" => _t('MemberTableField.EMAIL', 'Email')
|
|
);
|
|
|
|
$memberDbFields = $SNG_member->db();
|
|
$csvFieldList = array();
|
|
|
|
foreach($memberDbFields as $field => $dbFieldType) {
|
|
$csvFieldList[$field] = $field;
|
|
}
|
|
|
|
foreach(self::$addedFields as $key => $value) {
|
|
$fieldList[$key] = $value;
|
|
}
|
|
|
|
if(!$hidePassword) {
|
|
$fieldList["SetPassword"] = "Password";
|
|
}
|
|
|
|
// Legacy: Use setCustomSourceItems() instead.
|
|
if($members) {
|
|
$this->customSourceItems = $this->memberListWithGroupID($members, $group);
|
|
}
|
|
|
|
$this->hidePassword = $hidePassword;
|
|
|
|
parent::__construct($controller, $name, $sourceClass, $fieldList);
|
|
|
|
Requirements::javascript(CMS_DIR . '/javascript/MemberTableField.js');
|
|
Requirements::javascript(CMS_DIR . "/javascript/MemberTableField_popup.js");
|
|
|
|
// search
|
|
$SQL_search = isset($_REQUEST['MemberSearch']) ? Convert::raw2sql($_REQUEST['MemberSearch']) : null;
|
|
if(!empty($_REQUEST['MemberSearch'])) {
|
|
$searchFilters = array();
|
|
foreach($SNG_member->searchableFields() as $fieldName => $fieldSpec) {
|
|
if(strpos($fieldName,'.') === false) $searchFilters[] = "`$fieldName` LIKE '%{$SQL_search}%'";
|
|
}
|
|
$this->sourceFilter[] = '(' . implode(' OR ', $searchFilters) . ')';
|
|
}
|
|
|
|
// filter by groups
|
|
// TODO Not implemented yet
|
|
if(isset($_REQUEST['ctf'][$this->Name()]['GroupID']) && is_numeric($_REQUEST['ctf'][$this->Name()]['GroupID'])) {
|
|
$this->sourceFilter[] = "`GroupID`='{$_REQUEST['ctf'][$this->Name()]['GroupID']}'";
|
|
}
|
|
|
|
$this->sourceJoin = " INNER JOIN `Group_Members` ON `MemberID` = `Member`.`ID`";
|
|
$this->setFieldListCsv($csvFieldList);
|
|
}
|
|
|
|
function sourceID() {
|
|
return $this->group->ID;
|
|
}
|
|
|
|
function AddLink() {
|
|
return $this->Link() . '/add';
|
|
}
|
|
|
|
function SearchForm() {
|
|
$searchFields = new FieldGroup(
|
|
new TextField('MemberSearch', _t('MemberTableField.SEARCH', 'Search')),
|
|
new HiddenField("ctf[ID]", '', $this->group->ID),
|
|
new HiddenField('MemberFieldName', '', $this->name),
|
|
new HiddenField('MemberDontShowPassword', '', $this->hidePassword)
|
|
);
|
|
|
|
$actionFields = new LiteralField('MemberFilterButton','<input type="submit" class="action" name="MemberFilterButton" value="'._t('MemberTableField.FILTER', 'Filter').'" id="MemberFilterButton"/>');
|
|
|
|
$fieldContainer = new FieldGroup(
|
|
$searchFields,
|
|
$actionFields
|
|
);
|
|
|
|
return $fieldContainer->FieldHolder();
|
|
}
|
|
|
|
/**
|
|
* Add existing member to group rather than creating a new member
|
|
*/
|
|
function addtogroup() {
|
|
$data = $_REQUEST;
|
|
unset($data['ID']);
|
|
$ctfID = isset($data['ctf']) ? $data['ctf']['ID'] : null;
|
|
|
|
if(!is_numeric($ctfID)) {
|
|
FormResponse::status_messsage(_t('MemberTableField.ADDINGFIELD', 'Adding failed'), 'bad');
|
|
}
|
|
|
|
$className = Object::getCustomClass($this->stat('data_class'));
|
|
$record = new $className();
|
|
|
|
$record->update($data);
|
|
|
|
$valid = $record->validate();
|
|
|
|
if($valid->valid()) {
|
|
$record->write();
|
|
$record->Groups()->add($ctfID);
|
|
|
|
$this->sourceItems();
|
|
|
|
// TODO add javascript to highlight added row (problem: might not show up due to sorting/filtering)
|
|
FormResponse::update_dom_id($this->id(), $this->renderWith($this->template), true);
|
|
FormResponse::status_message(_t('MemberTableField.ADDEDTOGROUP','Added member to group'), 'good');
|
|
|
|
} else {
|
|
FormResponse::status_message(Convert::raw2xml("I couldn't add that user to this group:\n\n" . $valid->starredlist()), 'bad');
|
|
}
|
|
|
|
return FormResponse::respond();
|
|
}
|
|
|
|
/**
|
|
* Custom delete implementation:
|
|
* Remove member from group rather than from the database
|
|
*/
|
|
function delete() {
|
|
$groupID = Convert::raw2sql($_REQUEST['ctf']['ID']);
|
|
$memberID = Convert::raw2sql($_REQUEST['ctf']['childID']);
|
|
if(is_numeric($groupID) && is_numeric($memberID)) {
|
|
$member = DataObject::get_by_id('Member', $memberID);
|
|
$member->Groups()->remove($groupID);
|
|
} else {
|
|
user_error("MemberTableField::delete: Bad parameters: Group=$groupID, Member=$memberID", E_USER_ERROR);
|
|
}
|
|
|
|
return FormResponse::respond();
|
|
|
|
}
|
|
|
|
/**
|
|
* #################################
|
|
* Utility Functions
|
|
* #################################
|
|
*/
|
|
function getParentClass() {
|
|
return 'Group';
|
|
}
|
|
|
|
function getParentIdName($childClass,$parentClass){
|
|
return 'GroupID';
|
|
}
|
|
|
|
/**
|
|
* #################################
|
|
* Custom Functions
|
|
* #################################
|
|
*/
|
|
function memberListWithGroupID($members, $group) {
|
|
$newMembers = new DataObjectSet();
|
|
foreach($members as $member) {
|
|
$newMembers->push($member->customise(array('GroupID' => $group->ID)));
|
|
}
|
|
return $newMembers;
|
|
}
|
|
|
|
function setGroup($group) {
|
|
$this->group = $group;
|
|
}
|
|
|
|
function setController($controller) {
|
|
$this->controller = $controller;
|
|
}
|
|
|
|
function GetControllerName() {
|
|
return $this->controller->class;
|
|
}
|
|
|
|
/**
|
|
* Add existing member to group by name (with JS-autocompletion)
|
|
*/
|
|
function AddRecordForm() {
|
|
$fields = new FieldSet();
|
|
foreach($this->FieldList() as $fieldName => $fieldTitle) {
|
|
$fields->push(new TextField($fieldName));
|
|
}
|
|
$fields->push(new HiddenField('ctf[ID]', null, $this->group->ID));
|
|
|
|
$actions = new FieldSet(
|
|
new FormAction('addtogroup', _t('MemberTableField.ADD','Add'))
|
|
);
|
|
|
|
return new TabularStyle(
|
|
new NestedForm(
|
|
new Form(
|
|
$this,
|
|
'AddRecordForm',
|
|
$fields,
|
|
$actions
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Same behaviour as parent class, but adds the
|
|
* member to the passed GroupID.
|
|
*
|
|
* @return string
|
|
*/
|
|
function saveComplexTableField($data, $form, $params) {
|
|
$className = $this->sourceClass();
|
|
$childData = new $className();
|
|
$form->saveInto($childData);
|
|
$childData->write();
|
|
|
|
$childData->Groups()->add($data['GroupID']);
|
|
|
|
$closeLink = sprintf(
|
|
'<small><a href="' . $_SERVER['HTTP_REFERER'] . '" onclick="javascript:window.top.GB_hide(); return false;">(%s)</a></small>',
|
|
_t('ComplexTableField.CLOSEPOPUP', 'Close Popup')
|
|
);
|
|
$message = sprintf(
|
|
_t('ComplexTableField.SUCCESSADD', 'Added %s %s %s'),
|
|
$childData->singular_name(),
|
|
'<a href="' . $this->Link() . '">' . $childData->Title . '</a>',
|
|
$closeLink
|
|
);
|
|
$form->sessionMessage($message, 'good');
|
|
|
|
Director::redirectBack();
|
|
}
|
|
|
|
/**
|
|
* Cached version for getting the appropraite members for this particular group.
|
|
*
|
|
* This includes getting inherited groups, such as groups under groups.
|
|
*/
|
|
function sourceItems() {
|
|
// Caching.
|
|
if($this->sourceItems) {
|
|
return $this->sourceItems;
|
|
}
|
|
|
|
// Setup limits
|
|
$limitClause = '';
|
|
if(isset($_REQUEST['ctf'][$this->Name()]['start']) && is_numeric($_REQUEST['ctf'][$this->Name()]['start'])) {
|
|
$limitClause = ($_REQUEST['ctf'][$this->Name()]['start']) . ", {$this->pageSize}";
|
|
} else {
|
|
$limitClause = "0, {$this->pageSize}";
|
|
}
|
|
|
|
// We use the group to get the members, as they already have the bulk of the look up functions
|
|
$start = isset($_REQUEST['ctf'][$this->Name()]['start']) ? $_REQUEST['ctf'][$this->Name()]['start'] : 0;
|
|
|
|
$this->sourceItems = $this->group->Members(
|
|
$this->pageSize, // limit
|
|
$start, // offset
|
|
$this->sourceFilter,
|
|
$this->sourceSort
|
|
);
|
|
|
|
// Because we are not used $this->upagedSourceItems any more, and the DataObjectSet is usually the source
|
|
// that a large member set runs out of memory. we disable it here.
|
|
//$this->unpagedSourceItems = $this->group->Members('', '', $this->sourceFilter, $this->sourceSort);
|
|
$this->totalCount = ($this->sourceItems) ? $this->sourceItems->TotalItems() : 0;
|
|
|
|
return $this->sourceItems;
|
|
}
|
|
|
|
function TotalCount() {
|
|
$this->sourceItems(); // Called for its side-effect of setting total count
|
|
return $this->totalCount;
|
|
}
|
|
|
|
/**
|
|
* Handles item requests
|
|
* MemberTableField needs its own item request class so that it can overload the delete method
|
|
*/
|
|
function handleItem($request) {
|
|
return new MemberTableField_ItemRequest($this, $request->param('ID'));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Popup window for {@link MemberTableField}.
|
|
* @package cms
|
|
* @subpackage security
|
|
*/
|
|
class MemberTableField_Popup extends ComplexTableField_Popup {
|
|
|
|
function __construct($controller, $name, $fields, $sourceClass, $readonly=false, $validator = null) {
|
|
parent::__construct($controller, $name, $fields, $sourceClass, $readonly, $validator);
|
|
|
|
Requirements::javascript(CMS_DIR . '/javascript/MemberTableField.js');
|
|
Requirements::javascript(CMS_DIR . '/javascript/MemberTableField_popup.js');
|
|
}
|
|
|
|
}
|
|
|
|
class MemberTableField_ItemRequest extends ComplexTableField_ItemRequest {
|
|
/**
|
|
* Deleting an item from a member table field should just remove that member from the group
|
|
*/
|
|
function delete() {
|
|
if($this->ctf->Can('delete') !== true) {
|
|
return false;
|
|
}
|
|
|
|
$groupID = $this->ctf->sourceID();
|
|
$this->dataObj()->Groups()->remove($groupID);
|
|
}
|
|
|
|
}
|
|
|
|
?>
|