2011-03-23 10:51:00 +01:00
|
|
|
<?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.
|
|
|
|
*
|
|
|
|
* @todo write a better description about what this field does.
|
|
|
|
*
|
|
|
|
* 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 $detailFormValidator;
|
|
|
|
|
|
|
|
protected $group;
|
|
|
|
|
|
|
|
protected $template = 'MemberTableField';
|
|
|
|
|
|
|
|
public $popupClass = 'MemberTableField_Popup';
|
|
|
|
|
|
|
|
public $itemClass = 'MemberTableField_Item';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the page size for this table.
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
public static $page_size = 20;
|
|
|
|
|
|
|
|
protected $permissions = array(
|
|
|
|
"add",
|
|
|
|
"edit",
|
|
|
|
"show",
|
|
|
|
"delete",
|
|
|
|
'inlineadd'
|
|
|
|
//"export",
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor method for MemberTableField.
|
|
|
|
*
|
|
|
|
* @param Controller $controller Controller class which created this field
|
|
|
|
* @param string $name Name of the field (e.g. "Members")
|
|
|
|
* @param mixed $group Can be the ID of a Group instance, or a Group instance itself
|
|
|
|
* @param DataObjectSet $members Optional set of Members to set as the source items for this field
|
|
|
|
* @param boolean $hidePassword Hide the password field or not in the summary?
|
|
|
|
*/
|
|
|
|
function __construct($controller, $name, $group = null, $members = null, $hidePassword = true) {
|
2011-03-30 03:28:50 +02:00
|
|
|
|
|
|
|
if(!$members) {
|
|
|
|
if($group) {
|
|
|
|
if(is_numeric($group)) $group = DataObject::get_by_id('Group', $group);
|
|
|
|
$this->group = $group;
|
|
|
|
$members = $group->Members();
|
|
|
|
|
|
|
|
} elseif(isset($_REQUEST['ctf'][$this->Name()]["ID"]) && is_numeric($_REQUEST['ctf'][$this->Name()]["ID"])) {
|
|
|
|
throw new Exception("Is this still being used? It's a hack and we should remove it.");
|
|
|
|
$group = DataObject::get_by_id('Group', $_REQUEST['ctf'][$this->Name()]["ID"]);
|
|
|
|
$this->group = $group;
|
|
|
|
$members = $group->Members();
|
|
|
|
} else {
|
|
|
|
$members = DataObject::get("Member");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$SNG_member = singleton('Member');
|
2011-03-23 10:51:00 +01:00
|
|
|
$fieldList = $SNG_member->summaryFields();
|
|
|
|
$memberDbFields = $SNG_member->db();
|
|
|
|
$csvFieldList = array();
|
|
|
|
|
|
|
|
foreach($memberDbFields as $field => $dbFieldType) {
|
|
|
|
$csvFieldList[$field] = $field;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!$hidePassword) {
|
|
|
|
$fieldList["SetPassword"] = "Password";
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->hidePassword = $hidePassword;
|
|
|
|
|
2011-03-30 03:28:50 +02:00
|
|
|
// Add a search filter
|
2011-03-23 10:51:00 +01:00
|
|
|
$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}%'";
|
|
|
|
}
|
2011-03-30 03:28:50 +02:00
|
|
|
$members = $members->filter('(' . implode(' OR ', $searchFilters) . ')');
|
2011-03-23 10:51:00 +01:00
|
|
|
}
|
|
|
|
|
2011-03-30 03:28:50 +02:00
|
|
|
parent::__construct($controller, $name, $members, $fieldList);
|
|
|
|
|
2011-03-23 10:51:00 +01:00
|
|
|
$this->setFieldListCsv($csvFieldList);
|
|
|
|
$this->setPageSize($this->stat('page_size'));
|
|
|
|
}
|
|
|
|
|
|
|
|
function FieldHolder() {
|
|
|
|
$ret = parent::FieldHolder();
|
|
|
|
|
|
|
|
Requirements::javascript(SAPPHIRE_DIR . "/thirdparty/scriptaculous/controls.js");
|
2011-03-23 11:07:31 +01:00
|
|
|
Requirements::javascript(SAPPHIRE_ADMIN_DIR . '/javascript/MemberTableField.js');
|
|
|
|
Requirements::javascript(SAPPHIRE_ADMIN_DIR . "/javascript/MemberTableField_popup.js");
|
2011-03-23 10:51:00 +01:00
|
|
|
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function SearchForm() {
|
|
|
|
$groupID = (isset($this->group)) ? $this->group->ID : 0;
|
|
|
|
$query = isset($_GET['MemberSearch']) ? $_GET['MemberSearch'] : null;
|
|
|
|
|
|
|
|
$searchFields = new FieldGroup(
|
|
|
|
new TextField('MemberSearch', _t('MemberTableField.SEARCH', 'Search'), $query),
|
|
|
|
new HiddenField("ctf[ID]", '', $groupID),
|
|
|
|
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() {
|
|
|
|
// Protect against CSRF on destructive action
|
|
|
|
$token = $this->getForm()->getSecurityToken();
|
|
|
|
if(!$token->checkRequest($this->controller->getRequest())) return $this->httpError(400);
|
|
|
|
|
|
|
|
$data = $_REQUEST;
|
2011-03-30 03:28:50 +02:00
|
|
|
|
2011-03-23 10:51:00 +01:00
|
|
|
$groupID = (isset($data['ctf']['ID'])) ? $data['ctf']['ID'] : null;
|
|
|
|
|
|
|
|
if(!is_numeric($groupID)) {
|
|
|
|
FormResponse::status_messsage(_t('MemberTableField.ADDINGFIELD', 'Adding failed'), 'bad');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get existing record either by ID or unique identifier.
|
|
|
|
$identifierField = Member::get_unique_identifier_field();
|
2011-03-30 03:28:50 +02:00
|
|
|
$className = 'Member';
|
2011-03-23 10:51:00 +01:00
|
|
|
$record = null;
|
|
|
|
if(isset($data[$identifierField])) {
|
|
|
|
$record = DataObject::get_one(
|
|
|
|
$className,
|
|
|
|
sprintf('"%s" = \'%s\'', $identifierField, $data[$identifierField])
|
|
|
|
);
|
|
|
|
|
|
|
|
if($record && !$record->canEdit()) return $this->httpError('401');
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall back to creating a new record
|
|
|
|
if(!$record) $record = new $className();
|
|
|
|
|
|
|
|
// Update an existing record, or populate a new one.
|
|
|
|
// If values on an existing (autocompleted) record have been changed,
|
|
|
|
// they will overwrite current data. We need to unset 'ID'
|
|
|
|
// record as it points to the group rather than the member record, and would
|
|
|
|
// cause the member to be written to a potentially existing record.
|
|
|
|
unset($data['ID']);
|
|
|
|
$record->update($data);
|
|
|
|
|
|
|
|
// Validate record, mainly password restrictions.
|
|
|
|
// Note: Doesn't use Member_Validator
|
|
|
|
$valid = $record->validate();
|
|
|
|
if($valid->valid()) {
|
|
|
|
$record->write();
|
2011-03-30 03:28:50 +02:00
|
|
|
$this->getDataList()->add($record);
|
2011-03-23 10:51:00 +01:00
|
|
|
|
|
|
|
$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 {
|
|
|
|
$message = sprintf(
|
|
|
|
_t(
|
|
|
|
'MemberTableField.ERRORADDINGUSER',
|
|
|
|
'There was an error adding the user to the group: %s'
|
|
|
|
),
|
|
|
|
Convert::raw2xml($valid->starredList())
|
|
|
|
);
|
|
|
|
|
|
|
|
FormResponse::status_message($message, 'bad');
|
|
|
|
}
|
|
|
|
|
|
|
|
return FormResponse::respond();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* #################################
|
|
|
|
* Custom Functions
|
|
|
|
* #################################
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Group
|
|
|
|
*/
|
|
|
|
function getGroup() {
|
|
|
|
return $this->group;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add existing member to group by name (with JS-autocompletion)
|
|
|
|
*/
|
|
|
|
function AddRecordForm() {
|
|
|
|
$fields = new FieldSet();
|
|
|
|
foreach($this->FieldList() as $fieldName => $fieldTitle) {
|
|
|
|
// If we're adding the set password field, we want to hide the text from any peeping eyes
|
|
|
|
if($fieldName == 'SetPassword') {
|
|
|
|
$fields->push(new PasswordField($fieldName));
|
|
|
|
} else {
|
|
|
|
$fields->push(new TextField($fieldName));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if($this->group) {
|
|
|
|
$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
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function AddForm() {
|
|
|
|
$form = parent::AddForm();
|
|
|
|
|
|
|
|
// Set default groups - also implemented in MemberTableField_Popup::__construct()
|
|
|
|
if($this->group) {
|
|
|
|
$groupsField = $form->Fields()->dataFieldByName('Groups');
|
|
|
|
// TODO Needs to be a string value (not int) because of TreeMultiselectField->getItems(),
|
|
|
|
// see http://open.silverstripe.org/ticket/5836
|
|
|
|
if($groupsField) $groupsField->setValue((string)$this->group->ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $form;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Same behaviour as parent class, but adds the
|
|
|
|
* member to the passed GroupID.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2011-03-30 03:28:50 +02:00
|
|
|
function saveComplexTableField($data, $form, $params) {
|
2011-03-23 10:51:00 +01:00
|
|
|
$className = $this->sourceClass();
|
|
|
|
$childData = new $className();
|
|
|
|
|
|
|
|
// Needs to write before saveInto() to ensure the 'Groups' TreeMultiselectField saves
|
|
|
|
$childData->write();
|
|
|
|
|
|
|
|
try {
|
|
|
|
$form->saveInto($childData);
|
|
|
|
$childData->write();
|
|
|
|
} catch(ValidationException $e) {
|
|
|
|
$form->sessionMessage($e->getResult()->message(), 'bad');
|
|
|
|
return Director::redirectBack();
|
|
|
|
}
|
|
|
|
|
|
|
|
$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() . '">' . htmlspecialchars($childData->Title, ENT_QUOTES, 'UTF-8') . '</a>',
|
|
|
|
$closeLink
|
|
|
|
);
|
|
|
|
$form->sessionMessage($message, 'good');
|
|
|
|
|
|
|
|
$this->controller->redirectBack();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Popup window for {@link MemberTableField}.
|
|
|
|
* @package cms
|
|
|
|
* @subpackage security
|
|
|
|
*/
|
|
|
|
class MemberTableField_Popup extends ComplexTableField_Popup {
|
|
|
|
|
|
|
|
function __construct($controller, $name, $fields, $validator, $readonly, $dataObject) {
|
2011-03-30 03:28:50 +02:00
|
|
|
$group = ($controller instanceof MemberTableField) ? $controller->getGroup() : $controller->getParentController()->getGroup();
|
2011-03-23 10:51:00 +01:00
|
|
|
// Set default groups - also implemented in AddForm()
|
|
|
|
if($group) {
|
|
|
|
$groupsField = $fields->dataFieldByName('Groups');
|
|
|
|
if($groupsField) $groupsField->setValue($group->ID);
|
|
|
|
}
|
|
|
|
|
|
|
|
parent::__construct($controller, $name, $fields, $validator, $readonly, $dataObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
function forTemplate() {
|
|
|
|
$ret = parent::forTemplate();
|
|
|
|
|
2011-03-23 11:07:31 +01:00
|
|
|
Requirements::css(SAPPHIRE_ADMIN_DIR . '/css/SecurityAdmin.css');
|
|
|
|
Requirements::javascript(SAPPHIRE_ADMIN_DIR . '/javascript/MemberTableField.js');
|
|
|
|
Requirements::javascript(SAPPHIRE_ADMIN_DIR . '/javascript/MemberTableField_popup.js');
|
2011-03-23 10:51:00 +01:00
|
|
|
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @package cms
|
|
|
|
* @subpackage security
|
|
|
|
*/
|
|
|
|
class MemberTableField_Item extends ComplexTableField_Item {
|
|
|
|
|
|
|
|
function Actions() {
|
|
|
|
$actions = parent::Actions();
|
|
|
|
|
|
|
|
foreach($actions as $action) {
|
|
|
|
if($action->Name == 'delete') {
|
|
|
|
if($this->parent->getGroup()) {
|
|
|
|
$action->TitleText = _t('MemberTableField.DeleteTitleText',
|
|
|
|
'Delete from this group',
|
|
|
|
PR_MEDIUM,
|
|
|
|
'Delete button hover text'
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$action->TitleText = _t('MemberTableField.DeleteTitleTextDatabase',
|
|
|
|
'Delete from database and all groups',
|
|
|
|
PR_MEDIUM,
|
|
|
|
'Delete button hover text'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $actions;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
?>
|