group = $group; } else if(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"); 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') ); $csvFieldList = $fieldList; foreach( self::$addedCsvFields as $key => $value ) { $csvFieldList[$key] = $value; } 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"); // construct the filter and sort if(isset($_REQUEST['MemberOrderByField'])) { $this->sourceSort = '`' . Convert::raw2sql($_REQUEST['MemberOrderByField']) . '`' . Convert::raw2sql( $_REQUEST['MemberOrderByOrder'] ); } $SNG_member = singleton('Member'); // 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 ); } /** * Overridden functions */ 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) ); $orderByFields = new FieldGroup( new LabelField(_t('MemberTableField.ORDERBY', 'Order by')), new FieldSet( new DropdownField('MemberOrderByField','', array( 'FirstName' => _t('MemberTableField.FIRSTNAME', 'FirstName'), 'Surname' => _t('MemberTableField.SURNAME', 'Surname'), 'Email' => _t('MemberTableField.EMAIL', 'Email') )), new DropdownField('MemberOrderByOrder','',array( 'ASC' => _t('MemberTableField.ASC', 'Ascending'), 'DESC' => _t('MemberTableField.DESC', 'Descending') )) ) ); $groups = DataObject::get('Group'); $groupArray = array('' => _t('MemberTableField.ANYGROUP', 'Any group')); foreach( $groups as $group ) { $groupArray[$group->ID] = $group->Title; } $groupFields = new DropdownField('MemberGroup', _t('MemberTableField.FILTERBYGROUP', 'Filter by group'),$groupArray ); $actionFields = new LiteralField('MemberFilterButton',''); $fieldContainer = new FieldGroup( $searchFields, // $orderByFields, // $groupFields, $actionFields ); return $fieldContainer->FieldHolder(); } /** * Add existing member to group rather than creating a new member */ function addtogroup() { $data = $_REQUEST; unset($data['ID']); if(!is_numeric($data['ctf']['ID'])) { 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(); // To Avoid duplication in the Group_Members table if the ComponentSet.php is not modified just uncomment le line below //if( ! $record->inGroup( $data['ctf']['ID'] ) ) $record->Groups()->add( $data['ctf']['ID'] ); $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)); return new TabularStyle(new NestedForm(new Form($this, 'AddRecordForm', $fields, new FieldSet( new FormAction('addtogroup', _t('MemberTableField.ADD','Add')) ) ))); } /** * 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 ); $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'); } /** * Same behaviour as parent class, but adds the member to the passed GroupID. * * @return string */ function saveComplexTableField() { $id = (isset($_REQUEST['ctf']['childID'])) ? Convert::raw2sql($_REQUEST['ctf']['childID']) : false; if (is_numeric($id) && $id != 0) { $childObject = DataObject::get_by_id($this->sourceClass, $id); } else { $childObject = new $this->sourceClass(); $this->fields->removeByName('ID'); } $this->saveInto($childObject); $form = $this->controller->DetailForm($childObject->ID); $valid = $childObject->validate(); if($valid->valid()) { $childObject->write(); // custom behaviour for MemberTableField: add member to current group $childObject->Groups()->add($_REQUEST['GroupID']); $form->sessionMessage("Changes saved.", 'good'); } else { $form->sessionMessage(Convert::raw2xml("I couldn't save your changes:\n\n" . $valid->starredList()), 'bad'); } // if ajax-call in an iframe, update window if(Director::is_ajax()) { // Newly saved objects need their ID reflected in the reloaded form to avoid double saving $form->loadDataFrom($childObject); if($valid->valid()) { FormResponse::update_dom_id($form->FormName(), $form->formHtmlContent(), true, 'update'); } else { FormResponse::load_form($form->forTemplate(), $form->FormName()); } return FormResponse::respond(); } else { Director::redirectBack(); } } } 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); } } ?>