1527 lines
47 KiB
PHP
1527 lines
47 KiB
PHP
<?php
|
|
use LAM\TYPES\ConfiguredType;
|
|
|
|
/*
|
|
|
|
This code is part of LDAP Account Manager (http://www.ldap-account-manager.org/)
|
|
Copyright (C) 2003 - 2020 Roland Gruber
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
*/
|
|
|
|
/**
|
|
* This file includes functions to manage the list views.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*/
|
|
|
|
/** Used to get type information. */
|
|
include_once(__DIR__ . "/types.inc");
|
|
/** Used to get PDF information. */
|
|
include_once(__DIR__ . "/pdfstruct.inc");
|
|
/** Used to create PDF files. */
|
|
include_once(__DIR__ . "/pdf.inc");
|
|
|
|
|
|
/**
|
|
* Generates the list view.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*
|
|
*/
|
|
class lamList {
|
|
|
|
/** Account type */
|
|
protected $type;
|
|
|
|
/** current page number */
|
|
protected $page = 1;
|
|
|
|
/** list of LDAP attributes */
|
|
protected $attrArray = array();
|
|
|
|
/** list of attribute descriptions */
|
|
protected $descArray = array();
|
|
|
|
/** maximum count of entries per page */
|
|
protected $maxPageEntries = 30;
|
|
|
|
/** sort column name */
|
|
protected $sortColumn;
|
|
|
|
/** sort direction: 1 for ascending, -1 for descending */
|
|
protected $sortDirection = 1;
|
|
|
|
/** LDAP suffix */
|
|
protected $suffix;
|
|
|
|
/** refresh page switch */
|
|
protected $refresh = true;
|
|
|
|
/** entries to show */
|
|
protected $entries;
|
|
|
|
/** entries from LDAP */
|
|
protected $ldapEntries;
|
|
|
|
/** sort mapping for entries array(original index => sorted index) */
|
|
protected $sortMapping;
|
|
|
|
/** list of filters (attribute name => filter input) */
|
|
protected $filters = array();
|
|
|
|
/** list of possible LDAP suffixes(organizational units) */
|
|
protected $possibleSuffixes;
|
|
|
|
/** list of account specific labels */
|
|
protected $labels;
|
|
|
|
/** configuration options */
|
|
private $configOptions;
|
|
|
|
/** tabindex for GUI elements */
|
|
protected $tabindex = 1;
|
|
|
|
/** defines if the server side filter changed */
|
|
protected $serverSideFilterChanged;
|
|
|
|
/** ID for list size config option */
|
|
const LIST_SIZE_OPTION_NAME = "L_SIZE";
|
|
|
|
/** prefix for virtual (non-LDAP) attributes */
|
|
const VIRTUAL_ATTRIBUTE_PREFIX = 'lam_virtual_';
|
|
|
|
/**
|
|
* List of attributes to filter on server side.
|
|
*
|
|
* @var string[]
|
|
*/
|
|
protected $serverSideFilterAttributes = array(
|
|
'cn', 'commonname', 'uid', 'memberuid', 'description',
|
|
'sn', 'surname', 'gn', 'givenname', 'company', 'mail'
|
|
);
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param LAM\TYPES\ConfiguredType $type account type
|
|
* @return lamList list object
|
|
*/
|
|
public function __construct($type) {
|
|
$this->type = $type;
|
|
$this->labels = array(
|
|
'nav' => _("Object count: %s"),
|
|
'error_noneFound' => _("No objects found!"),
|
|
'newEntry' => _("New object"),
|
|
'deleteEntry' => _("Delete selected objects"));
|
|
$this->configOptions = $this->listGetAllConfigOptions();
|
|
$this->listReadOptionsFromCookie();
|
|
}
|
|
|
|
/**
|
|
* Reads the list options from the cookie value.
|
|
*/
|
|
private function listReadOptionsFromCookie() {
|
|
if ((sizeof($this->configOptions) > 0)
|
|
&& isset($_COOKIE["ListOptions_" . $this->type->getId()])) {
|
|
$cookieValue = $_COOKIE["ListOptions_" . $this->type->getId()];
|
|
$valueParts = explode(";", $cookieValue);
|
|
$values = array();
|
|
for ($i = 0; $i < sizeof($valueParts); $i++) {
|
|
$key_value = explode('=', $valueParts[$i]);
|
|
if (sizeof($key_value) == 2) {
|
|
$values[$key_value[0]] = $key_value[1];
|
|
}
|
|
}
|
|
for ($i = 0; $i < sizeof($this->configOptions); $i++) {
|
|
if (isset($values[$this->configOptions[$i]->getID()])) {
|
|
$this->configOptions[$i]->setValue($values[$this->configOptions[$i]->getID()]);
|
|
}
|
|
}
|
|
// notify subclasses
|
|
$this->listConfigurationChanged();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Prints the HTML code to display the list view.
|
|
*/
|
|
public function showPage() {
|
|
$this->tabindex = 1;
|
|
// do POST actions
|
|
$postFragment = $this->listDoPost();
|
|
// update filter
|
|
$this->listBuildFilter();
|
|
// get some parameters
|
|
$this->listGetParams();
|
|
// print HTML head
|
|
$this->printHeader();
|
|
// print messages when redirected from other pages
|
|
$this->listPrintRedirectMessages();
|
|
// refresh data if needed
|
|
if ($this->refresh) {
|
|
$this->listRefreshData();
|
|
}
|
|
// local filtering
|
|
$this->applyLocalFilters();
|
|
// sort rows by sort column
|
|
if (isset($this->entries)) {
|
|
$this->listCreateSortMapping($this->entries);
|
|
}
|
|
// insert HTML fragment from listDoPost
|
|
echo $postFragment;
|
|
// config dialog
|
|
$this->listPrintConfigurationPage();
|
|
// show form
|
|
echo "<div class=\"ui-tabs-panel ui-widget-content ui-corner-bottom\">";
|
|
echo "<div id=\"listTabContentArea\">\n";
|
|
echo "<form action=\"list.php?type=" . $this->type->getId() . "&norefresh=true\" method=\"post\">\n";
|
|
// draw account list if accounts were found
|
|
if (sizeof($this->entries) > 0) {
|
|
// buttons
|
|
$this->listPrintButtons(false);
|
|
// navigation bar
|
|
$this->listDrawNavigationBar(sizeof($this->entries));
|
|
$this->printAccountTable($this->entries);
|
|
}
|
|
else {
|
|
// buttons
|
|
$this->listPrintButtons(true);
|
|
// navigation bar
|
|
$this->listDrawNavigationBar(sizeof($this->entries));
|
|
$accounts = array();
|
|
$this->printAccountTable($accounts);
|
|
echo "</table><br>\n";
|
|
}
|
|
$this->printFooter();
|
|
}
|
|
|
|
/**
|
|
* Builds the regular expressions from the filter values.
|
|
*/
|
|
protected function listBuildFilter() {
|
|
if (isset($_GET['accountEditBack'])) {
|
|
return;
|
|
}
|
|
$oldFilter = $this->filters;
|
|
$this->serverSideFilterChanged = false;
|
|
$this->filters = array();
|
|
if (!isset($_POST['clear_filter'])) {
|
|
// build filter array
|
|
for ($i = 0; $i < sizeof($this->attrArray); $i++) {
|
|
$foundFilter = null;
|
|
if (isset($_GET["filter" . strtolower($this->attrArray[$i])])) {
|
|
$foundFilter = $_GET["filter" . strtolower($this->attrArray[$i])];
|
|
}
|
|
if (isset($_POST["filter" . strtolower($this->attrArray[$i])])) {
|
|
$foundFilter = $_POST["filter" . strtolower($this->attrArray[$i])];
|
|
}
|
|
if (isset($foundFilter) && ($foundFilter != '')) {
|
|
if (preg_match('/^[\\^]?([\p{L}\p{N} _\\*\\$\\.:@-])+[\\$]?$/iu', $foundFilter)) { // \p{L} matches any Unicode letter
|
|
$this->filters[strtolower($this->attrArray[$i])] = $foundFilter;
|
|
}
|
|
else {
|
|
StatusMessage('ERROR', _('Please enter a valid filter. Only letters, numbers and " _*$.@-" are allowed.'), htmlspecialchars($foundFilter));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
$filterAttrs = array_merge(array_keys($oldFilter), array_keys($this->filters));
|
|
$filterAttrs = array_unique($filterAttrs);
|
|
foreach ($filterAttrs as $attrName) {
|
|
if (!$this->isAttributeFilteredByServer($attrName)) {
|
|
continue;
|
|
}
|
|
if (!isset($oldFilter[$attrName]) || !isset($this->filters[$attrName]) || ($oldFilter[$attrName] != $this->filters[$attrName])) {
|
|
$this->serverSideFilterChanged = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Determines the sort mapping and stores it in $this->sortMapping.
|
|
* The sort mapping is used to display the right rows when the account table is created.
|
|
*
|
|
* @param array $info the account list
|
|
*/
|
|
protected function listCreateSortMapping(&$info) {
|
|
if (!is_array($this->attrArray) || !is_string($this->sortColumn)) {
|
|
return;
|
|
}
|
|
$toSort = array();
|
|
$col = $this->sortColumn;
|
|
$size = sizeof($info);
|
|
if ($this->sortColumn != 'dn') {
|
|
for ($i = 0; $i < $size; $i++) {
|
|
// sort by first attribute with name $sort
|
|
$toSort[] = &$info[$i][$col][0];
|
|
}
|
|
}
|
|
else {
|
|
for ($i = 0; $i < $size; $i++) {
|
|
$toSort[] = &$info[$i][$col];
|
|
}
|
|
}
|
|
natcasesort($toSort);
|
|
$sortResult = array();
|
|
if ($this->sortDirection == 1) {
|
|
foreach ($toSort as $orig => $val) {
|
|
$sortResult[] = $orig;
|
|
}
|
|
}
|
|
else {
|
|
$counter = sizeof($toSort);
|
|
foreach ($toSort as $orig => $val) {
|
|
$counter--;
|
|
$sortResult[$counter] = $orig;
|
|
}
|
|
}
|
|
$this->sortMapping = &$sortResult;
|
|
}
|
|
|
|
/**
|
|
* Draws a navigation bar to switch between pages
|
|
*
|
|
* @param integer $count number of account entries
|
|
*/
|
|
protected function listDrawNavigationBar($count) {
|
|
$filter = $this->getFilterAsTextForURL();
|
|
$row = new htmlResponsiveRow();
|
|
$row->setCSSClasses(array('maxrow'));
|
|
$countLabel = new htmlOutputText(sprintf($this->labels['nav'], $count));
|
|
$row->add($countLabel, 12, 6, 6);
|
|
$navGroup = new htmlGroup();
|
|
if ($count > $this->maxPageEntries) {
|
|
if ($this->page != 1) {
|
|
$linkHref = "list.php?type=" . $this->type->getId() . "&norefresh=true&page=1" .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$link = new htmlLink(null, $linkHref, '../../graphics/go-first.png');
|
|
$link->setTitle(_('Jump to first page'));
|
|
$link->setCSSClasses(array('margin5'));
|
|
$navGroup->addElement($link);
|
|
}
|
|
if ($this->page > 11) {
|
|
$linkHref = "list.php?type=" . $this->type->getId() . "&norefresh=true&page=" . ($this->page - 10) .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$link = new htmlLink(null, $linkHref, '../../graphics/go-previous.png');
|
|
$link->setTitle(_('Jump 10 pages backward'));
|
|
$link->setCSSClasses(array('margin5'));
|
|
$navGroup->addElement($link);
|
|
}
|
|
$pageCount = ceil($count / $this->maxPageEntries);
|
|
for ($i = $this->page - 6; $i < ($this->page + 5); $i++) {
|
|
if ($i >= $pageCount) {
|
|
break;
|
|
}
|
|
elseif ($i < 0) {
|
|
continue;
|
|
}
|
|
if ($i == $this->page - 1) {
|
|
$url = "list.php?type=" . $this->type->getId() . "&norefresh=true" .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$navInput = new htmlInputField('listNavPage', ($i + 1));
|
|
$navInput->setMinimumAndMaximumNumber(1, $pageCount);
|
|
$navInput->setCSSClasses(array('listPageInput'));
|
|
$navInput->setOnKeyPress('listPageNumberKeyPress(\'' . $url . '\', event);');
|
|
$navGroup->addElement($navInput);
|
|
}
|
|
else {
|
|
$linkHref = "list.php?type=" . $this->type->getId() . "&norefresh=true&page=" . ($i + 1) .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$link = new htmlLink(($i + 1), $linkHref);
|
|
$link->setCSSClasses(array('margin5'));
|
|
$navGroup->addElement($link);
|
|
}
|
|
}
|
|
if ($this->page < ($pageCount - 10)) {
|
|
$linkHref = "list.php?type=" . $this->type->getId() . "&norefresh=true&page=" . ($this->page + 10) .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$link = new htmlLink(null, $linkHref, '../../graphics/go-next.png');
|
|
$link->setTitle(_('Jump 10 pages forward'));
|
|
$link->setCSSClasses(array('margin5'));
|
|
$navGroup->addElement($link);
|
|
}
|
|
if ($this->page < $pageCount) {
|
|
$linkHref = "list.php?type=" . $this->type->getId() . "&norefresh=true&page=" . $pageCount .
|
|
"&sort=" . $this->sortColumn . "&sortdirection=" . $this->sortDirection . $filter;
|
|
$link = new htmlLink(null, $linkHref, '../../graphics/go-last.png');
|
|
$link->setTitle(_('Jump to last page'));
|
|
$link->setCSSClasses(array('margin5'));
|
|
$navGroup->addElement($link);
|
|
}
|
|
}
|
|
$row->add($navGroup, 12, 6, 6, 'responsiveLabel');
|
|
parseHtml(null, $row, array(), false, $this->tabindex, $this->type->getScope());
|
|
}
|
|
|
|
/**
|
|
* Returns the filter as text to be used as URL parameter.
|
|
*
|
|
* @return String filter text
|
|
*/
|
|
protected function getFilterAsTextForURL() {
|
|
$text = '';
|
|
foreach ($this->filters as $attr => $filter) {
|
|
$text .= "&filter" . strtolower($attr) . '=' . $filter;
|
|
}
|
|
return $text;
|
|
}
|
|
|
|
/**
|
|
* Prints the entry list
|
|
*
|
|
* @param array $info entries
|
|
*/
|
|
private function printAccountTable(&$info) {
|
|
$scope = $this->type->getScope();
|
|
$titles = $this->descArray;
|
|
array_unshift($titles, _('Actions'));
|
|
$data = array();
|
|
$data[] = $this->getSortingElements();
|
|
$data[] = $this->getFilterElements();
|
|
$onClickEvents = array();
|
|
$onDoubleClickEvents = array();
|
|
$this->addDataElements($data, $info, $onClickEvents, $onDoubleClickEvents);
|
|
|
|
$table = new htmlResponsiveTable($titles, $data);
|
|
$table->setRowClasses($scope . '-dark', $scope . '-bright');
|
|
$table->setCSSClasses(array($scope . '-border accountlist'));
|
|
$table->setOnClickEvents($onClickEvents);
|
|
$table->setOnDoubleClickEvents($onDoubleClickEvents);
|
|
$columnCount = sizeof($titles);
|
|
|
|
parseHtml(null, $table, array(), false, $this->tabindex, $scope);
|
|
}
|
|
|
|
/**
|
|
* Returns the elements to show in sorting row.
|
|
*
|
|
* @return htmlElement[] elements
|
|
*/
|
|
private function getSortingElements() {
|
|
$filter = $this->getFilterAsTextForURL();
|
|
$sortElements = array(new htmlSpan(new htmlOutputText(_('Sort sequence')), array('nowrap')));
|
|
foreach ($this->attrArray as $attributeName) {
|
|
$link = "list.php?type=" . $this->type->getId() . "&".
|
|
"sort=" . strtolower($attributeName) . $filter . "&norefresh=y";
|
|
$buttons = new htmlGroup();
|
|
if (strtolower($attributeName) == $this->sortColumn) {
|
|
if ($this->sortDirection < 0) {
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=1', '../../graphics/downarrows-black.png'));
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=-1', '../../graphics/uparrows.png'));
|
|
}
|
|
else {
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=1', '../../graphics/downarrows.png'));
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=-1', '../../graphics/uparrows-black.png'));
|
|
}
|
|
}
|
|
else {
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=1', '../../graphics/downarrows-black.png'));
|
|
$buttons->addElement(new htmlLink(null, $link . '&sortdirection=-1', '../../graphics/uparrows-black.png'));
|
|
}
|
|
$sortElements[] = $buttons;
|
|
}
|
|
return $sortElements;
|
|
}
|
|
|
|
/**
|
|
* Returns the elements to show in filter row.
|
|
*
|
|
* @return htmlElement[] elements
|
|
*/
|
|
private function getFilterElements() {
|
|
$actionElement = new htmlGroup();
|
|
$selectAll = new htmlInputCheckbox('tableSelectAll', false);
|
|
$selectAll->setCSSClasses(array('align-middle'));
|
|
$selectAll->setOnClick('list_switchAccountSelection();');
|
|
$actionElement->addElement($selectAll);
|
|
$actionElement->addElement(new htmlSpacer('1rem', null));
|
|
$actionElement->addElement(new htmlOutputText(_('Filter')));
|
|
$filterButton = new htmlButton('apply_filter', 'filter.png', true);
|
|
$filterButton->setTitle(_("Here you can input simple filter expressions (e.g. 'value' or 'v*'). The filter is case-insensitive."));
|
|
$actionElement->addElement($filterButton);
|
|
if (sizeof($this->filters) > 0) {
|
|
$clearFilterButton = new htmlButton('clear_filter', 'clearFilter.png', true);
|
|
$clearFilterButton->setTitle(_('Clear filter'));
|
|
$actionElement->addElement($clearFilterButton);
|
|
}
|
|
|
|
$filterElements = array(new htmlDiv(null, $actionElement, array('lam-listtools')));
|
|
$clearFilter = isset($_POST['clear_filter']);
|
|
foreach ($this->attrArray as $attributeName) {
|
|
$attributeName = strtolower($attributeName);
|
|
if ($this->canBeFiltered($attributeName)) {
|
|
$filterElements[] = $this->getFilterArea($attributeName, $clearFilter);
|
|
}
|
|
else {
|
|
$filterElements[] = new htmlOutputText('');
|
|
}
|
|
}
|
|
return $filterElements;
|
|
}
|
|
|
|
/**
|
|
* Prints the content of a single attribute filter area.
|
|
*
|
|
* @param String $attrName attribute name
|
|
* @param boolean $clearFilter true if filter value should be cleared
|
|
* @return htmlElement element to show
|
|
*/
|
|
protected function getFilterArea($attrName, $clearFilter) {
|
|
$value = "";
|
|
if (!$clearFilter && isset($this->filters[$attrName])) {
|
|
$value = $this->filters[$attrName];
|
|
}
|
|
$filterInput = new htmlInputField('filter' . $attrName, $value, null);
|
|
$filterInput->setCSSClasses(array($this->type->getScope() . '-bright'));
|
|
$filterInput->setOnKeyPress("SubmitForm('apply_filter', event);");
|
|
$filterInput->setFieldSize(null);
|
|
return $filterInput;
|
|
}
|
|
|
|
/**
|
|
* Adds the LDAP data elements to the given array.
|
|
*
|
|
* @param array $data data for responsible table
|
|
* @param array $info entries
|
|
* @param array $onClickEvents row number => code
|
|
* @param array $onDoubleClickEvents row number => code
|
|
*/
|
|
private function addDataElements(&$data, &$info, &$onClickEvents, &$onDoubleClickEvents) {
|
|
// calculate which rows to show
|
|
$table_begin = ($this->page - 1) * $this->maxPageEntries;
|
|
if (($this->page * $this->maxPageEntries) > sizeof($info)) {
|
|
$table_end = sizeof($info);
|
|
}
|
|
else {
|
|
$table_end = ($this->page * $this->maxPageEntries);
|
|
}
|
|
// get sort mapping
|
|
if (empty($this->sortMapping)) {
|
|
$this->sortMapping = array();
|
|
$infoSize = sizeof($info);
|
|
for ($i = 0; $i < $infoSize; $i++) {
|
|
$this->sortMapping[$i] = $i;
|
|
}
|
|
}
|
|
// print account list
|
|
for ($i = $table_begin; $i < $table_end; $i++) {
|
|
$row = array();
|
|
$index = $this->sortMapping[$i];
|
|
$rowID = base64_encode($info[$index]['dn']);
|
|
$actionElement = new htmlGroup();
|
|
$checkbox = new htmlInputCheckbox($rowID, false);
|
|
$checkbox->setOnClick("list_click('" . $rowID . "');");
|
|
$checkbox->setCSSClasses(array('accountBoxUnchecked align-middle'));
|
|
$actionElement->addElement($checkbox);
|
|
$actionElement->addElement(new htmlSpacer('0.5rem', null));
|
|
$rowNumber = $i - $table_begin + 2;
|
|
$onClickEvents[$rowNumber] = "list_click('" . $rowID . "');";
|
|
$onDoubleClickEvents[$rowNumber] = "top.location.href='../account/edit.php?type=" . $this->type->getId() . "&DN=" . rawurlencode($info[$index]['dn']) . "';";
|
|
$this->addToolLinks($info[$index], $rowID, $actionElement);
|
|
$row[] = new htmlDiv(null, $actionElement, array('lam-listtools'));
|
|
foreach ($this->attrArray as $attributeName) {
|
|
$attributeName = strtolower($attributeName);
|
|
$row[] = $this->getTableCellContent($info[$index], $attributeName);
|
|
}
|
|
$data[] = $row;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the tool image links (e.g. edit and delete) for each account.
|
|
*
|
|
* @param array $account LDAP attributes
|
|
* @param String $id account ID
|
|
* @param htmlGroup $element location where to add tools
|
|
*/
|
|
private function addToolLinks($account, $id, &$element) {
|
|
$toolCount = 0;
|
|
// edit link
|
|
$editLink = new htmlLink('', "../account/edit.php?type=" . $this->type->getId() . "&DN='" . rawurlencode($account['dn']) . "'", '../../graphics/edit.png');
|
|
$editLink->setTitle(_("Edit"));
|
|
$element->addElement($editLink);
|
|
$toolCount++;
|
|
// delete link
|
|
if (checkIfWriteAccessIsAllowed($this->type->getId()) && checkIfDeleteEntriesIsAllowed($this->type->getId())) {
|
|
$deleteLink = new htmlLink('', "deletelink.php?type=" . $this->type->getId() . "&DN='" . rawurlencode($account['dn']) . "'", '../../graphics/delete.png');
|
|
$deleteLink->setTitle(_("Delete"));
|
|
$element->addElement($deleteLink);
|
|
$toolCount++;
|
|
}
|
|
// PDF button
|
|
$pdfButton = new htmlButton("createPDF_" . $id, 'pdf.png', true);
|
|
$pdfButton->setTitle(_('Create PDF file'));
|
|
$element->addElement($pdfButton);
|
|
$toolCount++;
|
|
// additional tools
|
|
$tools = $this->getAdditionalTools();
|
|
for ($i = 0; $i < sizeof($tools); $i++) {
|
|
$toolLink = new htmlLink('', $tools[$i]->getLinkTarget() . "?type=" . $this->type->getId() . "&DN='" . rawurlencode($account['dn']) . "'", '../../graphics/' . $tools[$i]->getImage());
|
|
$toolLink->setTitle($tools[$i]->getName());
|
|
$element->addElement($toolLink);
|
|
$toolCount++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns if the given attribute can be filtered.
|
|
* If filtering is not possible then no filter box will be displayed.
|
|
* By default all attributes can be filtered.
|
|
*
|
|
* @param String $attr attribute name
|
|
* @return boolean filtering possible
|
|
*/
|
|
protected function canBeFiltered($attr) {
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Returns the content of a cell in the account list for a given LDAP entry and attribute.
|
|
*
|
|
* @param array $entry LDAP attributes
|
|
* @param string $attribute attribute name
|
|
* @return htmlElement content
|
|
*/
|
|
protected function getTableCellContent(&$entry, &$attribute) {
|
|
// print all attribute entries seperated by "; "
|
|
if (isset($entry[$attribute]) && sizeof($entry[$attribute]) > 0) {
|
|
if (is_array($entry[$attribute])) {
|
|
if (($attribute == 'entryexpiretimestamp') && !empty($entry[$attribute][0])) {
|
|
return new htmlOutputText(formatLDAPTimestamp($entry[$attribute][0]));
|
|
}
|
|
else {
|
|
// sort array
|
|
sort($entry[$attribute]);
|
|
return new htmlOutputText(implode("; ", $entry[$attribute]));
|
|
}
|
|
}
|
|
else {
|
|
return new htmlOutputText($entry[$attribute]);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Manages all POST actions (e.g. button pressed) for the account lists.
|
|
*
|
|
* @return String HTML fragment to insert into beginning of account list
|
|
*/
|
|
protected function listDoPost() {
|
|
if (!empty($_POST)) {
|
|
validateSecurityToken();
|
|
}
|
|
// check if button was pressed and if we have to add/delete an account or call file upload
|
|
if (isset($_POST['new']) || isset($_POST['del']) || isset($_POST['fileUpload'])){
|
|
if (!checkIfWriteAccessIsAllowed($this->type->getId())) {
|
|
die();
|
|
}
|
|
// add new account
|
|
if (isset($_POST['new']) && checkIfNewEntriesAreAllowed($this->type->getId())){
|
|
metaRefresh("../account/edit.php?type=" . $this->type->getId() . "&suffix=" . $this->suffix);
|
|
exit;
|
|
}
|
|
// delete account(s)
|
|
elseif (isset($_POST['del']) && checkIfDeleteEntriesIsAllowed($this->type->getId())){
|
|
// search for checkboxes
|
|
$accounts = array_keys($_POST, "on");
|
|
// build DN list
|
|
$_SESSION['delete_dn'] = array();
|
|
for ($i = 0; $i < sizeof($accounts); $i++) {
|
|
if ($accounts[$i] == 'tableSelectAll') {
|
|
continue;
|
|
}
|
|
$_SESSION['delete_dn'][] = base64_decode($accounts[$i]);
|
|
}
|
|
if (sizeof($accounts) > 0) {
|
|
metaRefresh("../delete.php?type=" . $this->type->getId());
|
|
exit;
|
|
}
|
|
}
|
|
// file upload
|
|
elseif (isset($_POST['fileUpload']) && checkIfNewEntriesAreAllowed($this->type->getId())){
|
|
metaRefresh("../upload/masscreate.php?type=" . $this->type->getId());
|
|
exit;
|
|
}
|
|
}
|
|
// PDF button
|
|
foreach ($_POST as $key => $value) {
|
|
if (strpos($key, 'createPDF_') > -1) {
|
|
$parts = explode("_", $key);
|
|
if (sizeof($parts) == 2) {
|
|
$this->showPDFPage($parts[1]);
|
|
exit;
|
|
}
|
|
}
|
|
}
|
|
// PDF creation Ok
|
|
if (isset($_POST['createPDFok'])) {
|
|
$pdfStruct = $_POST['pdf_structure'];
|
|
$pdfFont = $_POST['pdf_font'];
|
|
$option = $_POST['createFor'];
|
|
$filename = '';
|
|
// create for clicked account
|
|
if ($option == 'DN') {
|
|
$_SESSION["accountPDF"] = new accountContainer($this->type, "accountPDF");
|
|
$_SESSION["accountPDF"]->load_account(base64_decode($_POST['clickedAccount']));
|
|
$filename = \LAM\PDF\createModulePDF(array($_SESSION["accountPDF"]), $pdfStruct, $pdfFont);
|
|
unset($_SESSION["accountPDF"]);
|
|
}
|
|
// create for all selected accounts
|
|
elseif ($option == 'SELECTED') {
|
|
// search for checkboxes
|
|
$accounts = array_keys($_POST, "on");
|
|
$list = array();
|
|
// load accounts from LDAP
|
|
for ($i = 0; $i < sizeof($accounts); $i++) {
|
|
$_SESSION["accountPDF-$i"] = new accountContainer($this->type, "accountPDF-$i");
|
|
$_SESSION["accountPDF-$i"]->load_account(base64_decode($accounts[$i]));
|
|
$list[$i] = $_SESSION["accountPDF-$i"];
|
|
}
|
|
if (sizeof($list) > 0) {
|
|
$filename = \LAM\PDF\createModulePDF($list, $pdfStruct, $pdfFont);
|
|
for ($i = 0; $i < sizeof($accounts); $i++) {
|
|
unset($_SESSION["accountPDF-$i"]);
|
|
}
|
|
}
|
|
}
|
|
// create for all accounts
|
|
elseif ($option == 'ALL') {
|
|
$list = array();
|
|
$entriesCount = sizeof($this->entries);
|
|
for ($i = 0; $i < $entriesCount; $i++) {
|
|
$_SESSION["accountPDF-$i"] = new accountContainer($this->type, "accountPDF-$i");
|
|
$_SESSION["accountPDF-$i"]->load_account($this->entries[$i]['dn']);
|
|
$list[$i] = $_SESSION["accountPDF-$i"];
|
|
}
|
|
if (sizeof($list) > 0) {
|
|
$filename = \LAM\PDF\createModulePDF($list, $pdfStruct, $pdfFont);
|
|
for ($i = 0; $i < $entriesCount; $i++) {
|
|
// clean session
|
|
unset($_SESSION["accountPDF-$i"]);
|
|
}
|
|
}
|
|
}
|
|
elseif ($option == 'SESSION') {
|
|
$filename = \LAM\PDF\createModulePDF(array($_SESSION[$_POST['PDFSessionID']]), $pdfStruct, $pdfFont);
|
|
unset($_SESSION[$_POST['PDFSessionID']]);
|
|
}
|
|
if ($filename != '') {
|
|
return "<script type=\"text/javascript\">window.open('" . $filename . "', '_blank');</script>";
|
|
}
|
|
}
|
|
// check if back from configuration page
|
|
if ((sizeof($this->configOptions) > 0) && isset($_POST['saveConfigOptions'])) {
|
|
$cookieValue = '';
|
|
for ($i = 0; $i < sizeof($this->configOptions); $i++) {
|
|
$this->configOptions[$i]->fillFromPostData();
|
|
$cookieValue .= $this->configOptions[$i]->getID() . "=" . $this->configOptions[$i]->getValue() . ';';
|
|
}
|
|
// save options as cookie for one year
|
|
setcookie("ListOptions_" . $this->type->getId(), $cookieValue, time()+60*60*24*365, "/", null, null, true);
|
|
// notify subclasses
|
|
$this->listConfigurationChanged();
|
|
}
|
|
return '';
|
|
}
|
|
|
|
/**
|
|
* Shows the page where the user may select the PDF options.
|
|
*
|
|
* @param String $id account ID
|
|
*/
|
|
private function showPDFPage($id) {
|
|
$sessionObject = null;
|
|
$PDFSessionID = null;
|
|
if (($id == null) && isset($_GET['PDFSessionID'])) {
|
|
$PDFSessionID = $_GET['PDFSessionID'];
|
|
$sessionObject = $_SESSION[$PDFSessionID];
|
|
}
|
|
// search for checkboxes
|
|
$selAccounts = array_keys($_POST, "on");
|
|
if (!in_array($id, $selAccounts)) {
|
|
$selAccounts[] = $id;
|
|
}
|
|
// get possible PDF structures
|
|
$pdf_structures = \LAM\PDF\getPDFStructures($this->type->getId());
|
|
|
|
$this->printHeader();
|
|
|
|
echo "<div class=\"ui-tabs-nav " . $this->type->getScope() . "-bright\">";
|
|
echo "<div class=\"smallPaddingContent\">\n";
|
|
$refreshParam = '&norefresh=true';
|
|
if (isset($_GET['refresh']) && ($_GET['refresh'] == 'true')) {
|
|
$refreshParam = '&refresh=true';
|
|
}
|
|
echo "<form action=\"list.php?type=" . $this->type->getId() . $refreshParam . "\" method=\"post\">\n";
|
|
|
|
$container = new htmlResponsiveRow();
|
|
$container->add(new htmlSubTitle(_('Create PDF file')), 12);
|
|
|
|
$container->add(new htmlResponsiveSelect('pdf_structure', $pdf_structures, array('default'), _('PDF structure'), '405'), 12);
|
|
$fonts = \LAM\PDF\getPdfFonts();
|
|
$fontSelection = new htmlResponsiveSelect('pdf_font', $fonts, array(), _('Font'), '411');
|
|
$fontSelection->setCSSClasses(array('lam-save-selection'));
|
|
$fontSelection->setHasDescriptiveElements(true);
|
|
$fontSelection->setSortElements(false);
|
|
$container->add($fontSelection, 12);
|
|
|
|
$container->addVerticalSpacer('1rem');
|
|
$container->addLabel(new htmlOutputText(_('Create for')));
|
|
// check if account object is already in session
|
|
if ($sessionObject != null) {
|
|
$container->addField(new htmlOutputText($sessionObject->finalDN));
|
|
$container->add(new htmlHiddenInput('createFor', 'SESSION'), 0);
|
|
$container->add(new htmlHiddenInput('PDFSessionID', $PDFSessionID), 0);
|
|
}
|
|
else {
|
|
$radioOptions = array(
|
|
getAbstractDN(base64_decode($id)) => 'DN',
|
|
sprintf(_('All selected accounts (%s)'), sizeof($selAccounts)) => 'SELECTED',
|
|
sprintf(_('All accounts (%s)'), sizeof($this->entries)) => 'ALL'
|
|
);
|
|
$container->addField(new htmlRadio('createFor', $radioOptions, 'DN'));
|
|
}
|
|
|
|
$container->addVerticalSpacer('2rem');
|
|
$container->addLabel(new htmlOutputText(' ', false));
|
|
$buttonContainer = new htmlGroup();
|
|
$buttonContainer->addElement(new htmlButton('createPDFok', _('Ok')));
|
|
$buttonContainer->addElement(new htmlSpacer('0.5rem', null));
|
|
$buttonContainer->addElement(new htmlButton('createPDFCancel', _('Cancel')));
|
|
$container->addField($buttonContainer);
|
|
// hidden inputs for selected accounts
|
|
for ($i = 0; $i < sizeof($selAccounts); $i++) {
|
|
$container->add(new htmlHiddenInput($selAccounts[$i], 'on'), 0);
|
|
}
|
|
$container->add(new htmlHiddenInput('clickedAccount', $id), 0);
|
|
addSecurityTokenToMetaHTML($container);
|
|
|
|
parseHtml(null, $container, array(), false, $this->tabindex, $this->type->getScope());
|
|
|
|
$this->printFooter();
|
|
}
|
|
|
|
/**
|
|
* Prints a combobox with possible sub-DNs.
|
|
*
|
|
* @return htmlGroup OU selection (may be empty)
|
|
*/
|
|
protected function listShowOUSelection() {
|
|
$group = new htmlGroup();
|
|
if (sizeof($this->possibleSuffixes) > 1) {
|
|
$suffixList = array();
|
|
for ($i = 0; $i < sizeof($this->possibleSuffixes); $i++) {
|
|
$suffixList[getAbstractDN($this->possibleSuffixes[$i])] = $this->possibleSuffixes[$i];
|
|
}
|
|
$suffixSelect = new htmlSelect('suffix', $suffixList, array($this->suffix));
|
|
$suffixSelect->setOnchangeEvent("listOUchanged('" . $this->type->getId() . "', this)");
|
|
$suffixSelect->setRightToLeftTextDirection(true);
|
|
$suffixSelect->setSortElements(false);
|
|
$suffixSelect->setHasDescriptiveElements(true);
|
|
$suffixSelect->setCSSClasses(array('margin5', 'lam-listouselection-height'));
|
|
$group->addElement($suffixSelect);
|
|
$group->addElement(new htmlSpacer('5px', null));
|
|
}
|
|
return $group;
|
|
}
|
|
|
|
/**
|
|
* Prints the create and delete buttons.
|
|
*
|
|
* @param boolean $createOnly true if only the create button should be displayed
|
|
* @param int $tabindex HTML tabindex counter
|
|
*/
|
|
protected function listPrintButtons($createOnly) {
|
|
$row = new htmlResponsiveRow();
|
|
$row->setCSSClasses(array('maxrow lam-list-buttons'));
|
|
$left = new htmlGroup();
|
|
// button part
|
|
if (checkIfWriteAccessIsAllowed($this->type->getId())) {
|
|
// add button
|
|
if (checkIfNewEntriesAreAllowed($this->type->getId())) {
|
|
$newButton = new htmlButton('new', $this->labels['newEntry']);
|
|
$newButton->setIconClass('createButton');
|
|
$newButton->setCSSClasses(array('fullwidth-mobile-only'));
|
|
$left->addElement($newButton);
|
|
}
|
|
// delete button
|
|
if (!$createOnly && checkIfDeleteEntriesIsAllowed($this->type->getId())) {
|
|
$delButton = new htmlButton('del', $this->labels['deleteEntry']);
|
|
$delButton->setIconClass('deleteButton');
|
|
$delButton->setCSSClasses(array('fullwidth-mobile-only'));
|
|
$left->addElement($delButton);
|
|
}
|
|
$toolSettings = $_SESSION['config']->getToolSettings();
|
|
if ($this->type->getBaseType()->supportsFileUpload() && checkIfNewEntriesAreAllowed($this->type->getId())
|
|
&& !(isset($toolSettings['tool_hide_toolFileUpload']) && ($toolSettings['tool_hide_toolFileUpload'] == 'true'))) {
|
|
$uploadButton = new htmlButton('fileUpload', _('File upload'));
|
|
$uploadButton->setIconClass('upButton');
|
|
$uploadButton->setCSSClasses(array('fullwidth-mobile-only'));
|
|
$left->addElement($uploadButton);
|
|
}
|
|
}
|
|
|
|
// OU selection and settings
|
|
$right = new htmlResponsiveRow();
|
|
$right->add($this->listShowOUSelection(), 12, 12, 10);
|
|
$rightButtonGroup = new htmlGroup();
|
|
$refreshButton = new htmlButton('refresh', 'refresh.png', true);
|
|
$refreshButton->setCSSClasses(array('margin5 lam-listouselection-height'));
|
|
$refreshButton->setTitle(_("Refresh"));
|
|
$rightButtonGroup->addElement($refreshButton);
|
|
$settingsLink = new htmlLink('', '#', '../../graphics/tools.png');
|
|
$settingsLink->setOnClick('listShowSettingsDialog(\'' . _('Change list settings') . '\', \'' . _('Ok') . '\', \'' . _('Cancel') . '\');');
|
|
$settingsLink->setTitle(_('Change settings'));
|
|
$settingsLink->setCSSClasses(array('margin5 lam-listouselection-height'));
|
|
$rightButtonGroup->addElement($settingsLink);
|
|
$right->add($rightButtonGroup, 12, 12, 2);
|
|
|
|
$this->addExtraInputElementsToTopArea($left, $right);
|
|
$row->add($left, 12, 6, 6, 'text-left');
|
|
$row->add($right, 12, 6, 6, 'text-right');
|
|
parseHtml(null, $row, array(), false, $this->tabindex, $this->type->getScope());
|
|
}
|
|
|
|
/**
|
|
* Can be used by subclasses to add e.g. additional buttons to the top area.
|
|
*
|
|
* @param htmlGroup $left left part
|
|
* @param htmlGroup $right right part
|
|
*/
|
|
protected function addExtraInputElementsToTopArea(&$left, &$right) {
|
|
// only used by subclasses
|
|
}
|
|
|
|
/**
|
|
* Prints the header part of the page.
|
|
*/
|
|
private function printHeader() {
|
|
include 'adminHeader.inc';
|
|
$this->printHeaderContent();
|
|
}
|
|
|
|
/**
|
|
* Prints any extra HTML for the header part.
|
|
*/
|
|
protected function printHeaderContent() {
|
|
// implemented by child classes if needed
|
|
}
|
|
|
|
/**
|
|
* Prints the footer area of the page.
|
|
*/
|
|
private function printFooter() {
|
|
$this->printFooterContent();
|
|
include 'adminFooter.inc';
|
|
}
|
|
|
|
/**
|
|
* Prints any extra HTML for the footer part.
|
|
*/
|
|
protected function printFooterContent() {
|
|
?>
|
|
<input type="hidden" name="<?php echo getSecurityTokenName(); ?>" value="<?php echo getSecurityTokenValue(); ?>">
|
|
</form></div></div>
|
|
<script type="text/javascript">
|
|
jQuery(document).ready(function() {
|
|
jQuery('#tab_<?php echo $this->type->getId(); ?>').addClass('ui-tabs-active');
|
|
jQuery('#tab_<?php echo $this->type->getId(); ?>').addClass('ui-state-active');
|
|
});
|
|
</script>
|
|
<?php
|
|
}
|
|
|
|
/**
|
|
* Returns an hash array containing with all attributes to be shown and their descriptions.
|
|
* Format: array(attribute => description)
|
|
*
|
|
* @return array attribute list
|
|
*/
|
|
protected function listGetAttributeDescriptionList() {
|
|
$attrs = $this->type->getAttributes();
|
|
$ret = array();
|
|
foreach ($attrs as $attr) {
|
|
$ret[$attr->getAttributeName()] = $attr->getAlias();
|
|
}
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* Sets some internal parameters.
|
|
*/
|
|
protected function listGetParams() {
|
|
if (isset($_GET['accountEditBack'])) {
|
|
$this->refresh = true;
|
|
return;
|
|
}
|
|
// check if only PDF should be shown
|
|
if (isset($_GET['printPDF'])) {
|
|
$this->showPDFPage(null);
|
|
exit();
|
|
}
|
|
// get current page
|
|
if (!empty($_GET["page"])) {
|
|
$this->page = $_GET["page"];
|
|
}
|
|
else {
|
|
$this->page = 1;
|
|
}
|
|
// generate attribute-description table
|
|
$temp_array = $this->listGetAttributeDescriptionList();
|
|
$this->attrArray = array_keys($temp_array); // list of LDAP attributes to show
|
|
$this->descArray = array_values($temp_array); // list of descriptions for the attributes
|
|
// get sorting column
|
|
if (isset($_GET["sort"])) {
|
|
if ($_GET["sort"] == $this->sortColumn) {
|
|
$this->sortDirection = -$this->sortDirection;
|
|
}
|
|
else {
|
|
$this->sortColumn = $_GET["sort"];
|
|
$this->sortDirection = 1;
|
|
}
|
|
}
|
|
else {
|
|
$this->sortColumn = strtolower($this->attrArray[0]);
|
|
$this->sortDirection = 1;
|
|
}
|
|
// get sort order
|
|
if (isset($_GET['sortdirection'])) {
|
|
$this->sortDirection = htmlspecialchars($_GET['sortdirection']);
|
|
}
|
|
// check search suffix
|
|
if (isset($_POST['suffix'])) {
|
|
// new suffix selected via combobox
|
|
$this->suffix = $_POST['suffix'];
|
|
}
|
|
elseif (isset($_GET['suffix'])) {
|
|
// new suffix selected via combobox
|
|
$this->suffix = $_GET['suffix'];
|
|
}
|
|
elseif (!$this->suffix) {
|
|
// default suffix
|
|
$this->suffix = $this->type->getSuffix();
|
|
}
|
|
// check if LDAP data should be refreshed
|
|
$this->refresh = true;
|
|
if (isset($_GET['norefresh'])) {
|
|
$this->refresh = false;
|
|
}
|
|
if (isset($_POST['refresh'])) {
|
|
$this->refresh = true;
|
|
}
|
|
if ((isset($_POST['apply_filter']) || isset($_POST['clear_filter']))
|
|
&& $this->serverSideFilterChanged) {
|
|
$this->refresh = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Rereads the entries from LDAP.
|
|
*/
|
|
protected function listRefreshData() {
|
|
// check suffix
|
|
if (!$this->suffix) {
|
|
$this->suffix = $this->type->getSuffix(); // default suffix
|
|
}
|
|
// configure search filter
|
|
$module_filter = get_ldap_filter($this->type->getId()); // basic filter is provided by modules
|
|
$filter = "(&" . $module_filter . $this->buildLDAPAttributeFilter() . ")";
|
|
$attrs = $this->attrArray;
|
|
// remove virtual attributes from list
|
|
for ($i = 0; $i < sizeof($attrs); $i++) {
|
|
if (strpos($attrs[$i], self::VIRTUAL_ATTRIBUTE_PREFIX) === 0) {
|
|
unset($attrs[$i]);
|
|
}
|
|
}
|
|
$attrs = array_values($attrs);
|
|
// include additional attributes
|
|
$additionalAttrs = $this->getAdditionalLDAPAttributesToRead();
|
|
for ($i = 0; $i < sizeof($additionalAttrs); $i++) {
|
|
if (!in_array_ignore_case($additionalAttrs[$i], $attrs)) {
|
|
$attrs[] = $additionalAttrs[$i];
|
|
}
|
|
}
|
|
$this->ldapEntries = searchLDAP($this->suffix, $filter, $attrs);
|
|
$this->entries = array();
|
|
foreach ($this->ldapEntries as $index => &$attrs) {
|
|
$this->entries[$index] = &$attrs;
|
|
}
|
|
$lastError = getLastLDAPError();
|
|
if ($lastError != null) {
|
|
call_user_func_array('StatusMessage', $lastError);
|
|
}
|
|
// generate list of possible suffixes
|
|
$this->possibleSuffixes = $this->type->getSuffixList();
|
|
}
|
|
|
|
/**
|
|
* Builds the LDAP filter based on the filter entries in the GUI.
|
|
*
|
|
* @return String LDAP filter
|
|
*/
|
|
protected function buildLDAPAttributeFilter() {
|
|
$text = '';
|
|
foreach ($this->filters as $attr => $filter) {
|
|
if (!$this->isAttributeFilteredByServer($attr)) {
|
|
continue;
|
|
}
|
|
$filterExpression = $filter;
|
|
if (strpos($filter, '^') === 0) {
|
|
$filterExpression = substr($filterExpression, 1);
|
|
}
|
|
elseif (strpos($filter, '*') !== 0) {
|
|
$filterExpression = '*' . $filterExpression;
|
|
}
|
|
if (strrpos($filter, '$') === (strlen($filter) - 1)) {
|
|
$filterExpression = substr($filterExpression, 0, -1);
|
|
}
|
|
elseif (strrpos($filter, '*') !== (strlen($filter) - 1)) {
|
|
$filterExpression = $filterExpression . '*';
|
|
}
|
|
$text .= '(' . $attr . '=' . $filterExpression . ')';
|
|
}
|
|
return $text;
|
|
}
|
|
|
|
/**
|
|
* Specifies if the given attribute name is used for server side filtering (LDAP filter string).
|
|
*
|
|
* @param string $attrName attribute name
|
|
* @return bool filter server side
|
|
*/
|
|
protected function isAttributeFilteredByServer($attrName) {
|
|
return in_array(strtolower($attrName), $this->serverSideFilterAttributes);
|
|
}
|
|
|
|
/**
|
|
* Applies any local filters for attributes that cannot be filtered server side.
|
|
*/
|
|
protected function applyLocalFilters() {
|
|
$this->entries = array();
|
|
foreach ($this->ldapEntries as $index => &$data) {
|
|
$this->entries[$index] = &$data;
|
|
}
|
|
$toFilter = array();
|
|
foreach ($this->filters as $filterAttribute => $filterValue) {
|
|
if ($this->isAttributeFilteredByServer($filterAttribute) || ($filterValue === '')) {
|
|
continue;
|
|
}
|
|
foreach ($this->entries as $index => &$data) {
|
|
if (in_array($index, $toFilter)) {
|
|
continue;
|
|
}
|
|
$regex = str_replace(array('*'), array('.*'), $filterValue);
|
|
$regex = '/' . $regex . '/i';
|
|
if (!$this->isFilterMatching($data, $filterAttribute, $regex)) {
|
|
$toFilter[] = $index;
|
|
}
|
|
}
|
|
}
|
|
foreach ($toFilter as $index) {
|
|
unset($this->entries[$index]);
|
|
}
|
|
$this->entries = array_values($this->entries);
|
|
}
|
|
|
|
/**
|
|
* Checks if the given LDAP data matches the filter.
|
|
*
|
|
* @param array $data LDAP attributes
|
|
* @param string $filterAttribute filter attribute name
|
|
* @param string $regex filter attribute regex
|
|
*/
|
|
protected function isFilterMatching(&$data, $filterAttribute, $regex) {
|
|
if (!isset($data[$filterAttribute])) {
|
|
return false;
|
|
}
|
|
foreach ($data[$filterAttribute] as $value) {
|
|
if (preg_match($regex, $value)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Forces a refresh of the LDAP data.
|
|
* Function must be called before $this->refresh option is checked to load new LDAP data (e.g. in listGetParams).
|
|
*/
|
|
protected function forceRefresh() {
|
|
$this->refresh = true;
|
|
if (isset($_GET['norefresh'])) {
|
|
unset($_GET['norefresh']);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns a list of additional LDAP attributes that should be read.
|
|
* This can be used to show additional data even if the user selected other attributes to show in the list.
|
|
*
|
|
* @return array additional attribute names
|
|
*/
|
|
protected function getAdditionalLDAPAttributesToRead() {
|
|
return array();
|
|
}
|
|
|
|
/**
|
|
* Returns a list of lamListTool objects to display next to the edit/delete buttons.
|
|
*
|
|
* @return lamListTool[] tools
|
|
*/
|
|
protected function getAdditionalTools() {
|
|
return array();
|
|
}
|
|
|
|
/**
|
|
* Returns a list of possible configuration options.
|
|
*
|
|
* @return array list of lamListOption objects
|
|
*/
|
|
protected function listGetAllConfigOptions() {
|
|
$listSizeOption = new lamSelectListOption(_("Maximum list entries"), array(10, 20, 30, 50, 75, 100, 500, 1000), self::LIST_SIZE_OPTION_NAME);
|
|
$listSizeOption->setHelpID('208');
|
|
$listSizeOption->setValue($this->maxPageEntries);
|
|
return array($listSizeOption);
|
|
}
|
|
|
|
/**
|
|
* Prints the list configuration page.
|
|
*/
|
|
protected function listPrintConfigurationPage() {
|
|
echo "<div id=\"settingsDialog\" class=\"hidden dialog-content\">\n";
|
|
echo "<form id=\"settingsDialogForm\" action=\"list.php?type=" . $this->type->getId() . "&norefresh=true\" method=\"post\">\n";
|
|
|
|
$configContainer = new htmlResponsiveRow();
|
|
for ($i = 0; $i < sizeof($this->configOptions); $i++) {
|
|
$configContainer->add($this->configOptions[$i]->getMetaHTML(), 12);
|
|
}
|
|
$configContainer->add(new htmlHiddenInput('saveConfigOptions', 'ok'), 12);
|
|
addSecurityTokenToMetaHTML($configContainer);
|
|
|
|
parseHtml('', $configContainer, array(), false, $this->tabindex, $this->type->getScope());
|
|
|
|
echo '</form>';
|
|
echo "</div>\n";
|
|
}
|
|
|
|
/**
|
|
* Returns the configuration option with the given ID.
|
|
*
|
|
* @param String $ID ID
|
|
*/
|
|
protected function listGetConfigOptionByID($ID) {
|
|
for ($i = 0; $i < sizeof($this->configOptions); $i++) {
|
|
if ($this->configOptions[$i]->getID() === $ID) {
|
|
return $this->configOptions[$i];
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Called when the configuration options changed.
|
|
*/
|
|
protected function listConfigurationChanged() {
|
|
$sizeOption = $this->listGetConfigOptionByID(self::LIST_SIZE_OPTION_NAME);
|
|
if ($sizeOption->getValue() != null) {
|
|
$this->maxPageEntries = $sizeOption->getValue();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Prints messages when another page (e.g. delete/upload) redirects to the list view.
|
|
*/
|
|
protected function listPrintRedirectMessages() {
|
|
if (isset($_GET['deleteAllOk'])) {
|
|
StatusMessage('INFO', _('Deletion was successful.'));
|
|
}
|
|
elseif (isset($_GET['uploadAllOk'])) {
|
|
StatusMessage('INFO', _("Upload has finished"));
|
|
if (isset($_SESSION['mass_pdf']['file'])) {
|
|
StatusMessage('INFO', sprintf(_('You can download your PDF files {link=%s}{color=#d2131a}here{endcolor}{endlink}.'), $_SESSION['mass_pdf']['file']));
|
|
}
|
|
}
|
|
elseif (isset($_GET['accountEditInvalidID'])) {
|
|
StatusMessage('WARN', _('Please do not edit multiple accounts in parallel in multiple browser tabs.'));
|
|
}
|
|
if (isset($_SESSION['listRedirectMessages'])) {
|
|
for ($i = 0; $i < sizeof($_SESSION['listRedirectMessages']); $i++) {
|
|
call_user_func_array('StatusMessage', $_SESSION['listRedirectMessages'][$i]);
|
|
}
|
|
unset($_SESSION['listRedirectMessages']);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Represents a tool which can be included in the account lists.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*/
|
|
class lamListTool {
|
|
|
|
/** tool name */
|
|
private $name;
|
|
/** tool image */
|
|
private $image;
|
|
/** link target */
|
|
private $target;
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param String $name tool name
|
|
* @param String $image image file
|
|
* @param String $target target page
|
|
* @return lamListTool tool object
|
|
*/
|
|
public function __construct($name, $image, $target) {
|
|
$this->name = $name;
|
|
$this->image = $image;
|
|
$this->target = $target;
|
|
}
|
|
|
|
/**
|
|
* Returns the name of the tool image.
|
|
* The image is returned without path (e.g. mytool.png). All images must reside in the graphics folder.
|
|
*
|
|
* @return String image name
|
|
*/
|
|
public function getImage() {
|
|
return $this->image;
|
|
}
|
|
|
|
/**
|
|
* Returns the tool name.
|
|
* This is used for the tool tip.
|
|
*
|
|
* @return String name
|
|
*/
|
|
public function getName() {
|
|
return $this->name;
|
|
}
|
|
|
|
/**
|
|
* Returns the PHP file (relative to 'templates/lists') which will be the target for this tool.
|
|
* The target page will be opened with two GET parameters: DN and type (e.g. user)
|
|
*
|
|
* @return String page file (e.g. 'mytool.php')
|
|
*/
|
|
public function getLinkTarget() {
|
|
return $this->target;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Represents a list configuration option.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*/
|
|
abstract class lamListOption {
|
|
|
|
/** unique ID */
|
|
private $ID;
|
|
/** option value */
|
|
private $value;
|
|
|
|
/**
|
|
* Creates a new config option.
|
|
*
|
|
* @param String $ID unique ID
|
|
* @return lamConfigOption config option
|
|
*/
|
|
public function __construct($ID) {
|
|
$this->ID = $ID;
|
|
}
|
|
|
|
/**
|
|
* Returns the option ID.
|
|
*
|
|
* @return String ID
|
|
*/
|
|
public function getID() {
|
|
return $this->ID;
|
|
}
|
|
|
|
/**
|
|
* Fills the config option from POST data.
|
|
*
|
|
* @return array list of StatusMessages (array(<type>, <head line>, <body>))
|
|
*/
|
|
public abstract function fillFromPostData();
|
|
|
|
/**
|
|
* Returns the option value. The value must not contain "=" and ";".
|
|
*
|
|
* @return String value
|
|
*/
|
|
public function getValue() {
|
|
return $this->value;
|
|
}
|
|
|
|
/**
|
|
* Sets the config option value. The value must not contain "=" and ";".
|
|
*
|
|
* @param String $value
|
|
*/
|
|
public function setValue($value) {
|
|
if ((strpos($value, '=') > -1) || (strpos($value, ';') > -1)) {
|
|
user_error("Invalid value for list option: " . $value, E_ERROR);
|
|
}
|
|
$this->value = $value;
|
|
}
|
|
|
|
/**
|
|
* Returns the meta HTML data to display this option.
|
|
*
|
|
* @return htmlResponsiveRow meta HTML
|
|
*/
|
|
public abstract function getMetaHTML();
|
|
|
|
}
|
|
|
|
/**
|
|
* Boolean option for list configuration.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*/
|
|
class lamBooleanListOption extends lamListOption {
|
|
|
|
/** option name */
|
|
private $name;
|
|
|
|
/**
|
|
* Creates a new boolean option.
|
|
*
|
|
* @param String $name name to show on config page
|
|
* @param String $ID unique ID
|
|
* @return lamBooleanListOption config option
|
|
*/
|
|
public function __construct($name, $ID) {
|
|
parent::__construct($ID);
|
|
$this->name = $name;
|
|
}
|
|
|
|
/**
|
|
* Returns if this option is selected.
|
|
*
|
|
* @return boolean true, if selected
|
|
*/
|
|
public function isSelected() {
|
|
return ($this->getValue() === "1");
|
|
}
|
|
|
|
/**
|
|
* Fills the config option from POST data.
|
|
*
|
|
* @return array list of StatusMessages (array(<type>, <head line>, <body>))
|
|
*/
|
|
public function fillFromPostData() {
|
|
if (isset($_POST[$this->getID()])) {
|
|
$this->setValue("1");
|
|
}
|
|
else {
|
|
$this->setValue("0");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
* @see lamListOption::getMetaHTML()
|
|
*/
|
|
public function getMetaHTML() {
|
|
$return = new htmlResponsiveRow();
|
|
$return->add(new htmlResponsiveInputCheckbox($this->getID(), $this->isSelected(), $this->name), 12);
|
|
return $return;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Boolean option for list configuration.
|
|
*
|
|
* @package lists
|
|
* @author Roland Gruber
|
|
*/
|
|
class lamSelectListOption extends lamListOption {
|
|
|
|
/** option name */
|
|
private $name;
|
|
/** possible select options */
|
|
private $options;
|
|
/** help ID */
|
|
private $helpID;
|
|
|
|
/**
|
|
* Creates a new selection list option.
|
|
*
|
|
* @param String $name name to show on config page
|
|
* @param array $options list of possible values
|
|
* @param String $ID unique ID
|
|
* @return lamBooleanListOption config option
|
|
*/
|
|
public function __construct($name, $options, $ID) {
|
|
parent::__construct($ID);
|
|
$this->name = $name;
|
|
$this->options = $options;
|
|
}
|
|
|
|
/**
|
|
* Sets the help ID.
|
|
*
|
|
* @param Strign $id help ID
|
|
*/
|
|
public function setHelpID($id) {
|
|
$this->helpID = $id;
|
|
}
|
|
|
|
/**
|
|
* Fills the config option from POST data.
|
|
*
|
|
* @return array list of StatusMessages (array(<type>, <head line>, <body>))
|
|
*/
|
|
public function fillFromPostData() {
|
|
if (isset($_POST[$this->getID()])) {
|
|
$this->setValue($_POST[$this->getID()]);
|
|
}
|
|
else {
|
|
$this->setValue(null);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
* @see lamListOption::getMetaHTML()
|
|
*/
|
|
public function getMetaHTML() {
|
|
$return = new htmlResponsiveRow();
|
|
$return->add(new htmlResponsiveSelect($this->getID(), $this->options, array($this->getValue()), $this->name, $this->helpID), 12);
|
|
return $return;
|
|
}
|
|
|
|
}
|
|
|
|
?>
|