1
0
mirror of https://github.com/e107inc/e107.git synced 2025-01-18 05:09:05 +01:00
php-e107/e107_handlers/userclass_class.php

1172 lines
37 KiB
PHP
Raw Normal View History

2006-12-02 04:36:16 +00:00
<?php
/*
+ ----------------------------------------------------------------------------+
| e107 website system
|
| <EFBFBD>Steve Dunstan 2001-2002
| http://e107.org
| jalist@e107.org
|
| Released under the terms and conditions of the
| GNU General Public License (http://gnu.org).
|
| $Source: /cvs_backup/e107_0.8/e107_handlers/userclass_class.php,v $
| $Revision: 1.18 $
| $Date: 2008-11-09 20:31:10 $
2008-09-15 16:20:27 +00:00
| $Author: secretr $
2006-12-02 04:36:16 +00:00
+----------------------------------------------------------------------------+
*/
2007-12-22 12:39:27 +00:00
/*
This class handles everything a user needs. Admin functions inherit from it.
*/
2006-12-02 04:36:16 +00:00
if (!defined('e107_INIT')) { exit; }
2007-12-22 12:39:27 +00:00
include_lan(e_LANGUAGEDIR.e_LANGUAGE."/lan_userclass.php");
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
/*
For info:
define("e_UC_PUBLIC", 0);
define("e_UC_MAINADMIN", 250);
define("e_UC_READONLY", 251);
define("e_UC_GUEST", 252);
define("e_UC_MEMBER", 253);
define("e_UC_ADMIN", 254);
define("e_UC_NOBODY", 255);
2006-12-02 04:36:16 +00:00
*/
2007-12-22 12:39:27 +00:00
// Move these definitions to class2.php later if they're adopted
define("e_UC_ADMINMOD",249);
define("e_UC_MODS",248);
//define("e_UC_USERS",247);
define('UC_CLASS_ICON_DIR','userclasses/'); // Directory for userclass icons
define('UC_ICON_DIR',e_IMAGE.'generic/'); // Directory for the icons used in the admin tree displays
2008-04-05 16:30:18 +00:00
define('e_UC_BLANK','-1');
2007-12-22 12:39:27 +00:00
class user_class
{
2008-02-16 12:03:27 +00:00
var $class_tree; // Simple array, filled with current tree. Additional field class_children is an array of child user classes (by ID)
var $class_parents; // Array of class IDs of 'parent' (i.e. top level) classes
2007-12-22 12:39:27 +00:00
2008-02-16 12:03:27 +00:00
var $fixed_classes = array(); // The 'predefined' classes of 0.7
var $text_class_link = array(); // List of 'core' user classes and the related constants
2007-12-22 12:39:27 +00:00
var $sql_r; // We'll use our own DB to avoid interactions
// Constructor
function user_class()
{
global $imode;
$this->sql_r = new db;
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
$this->fixed_classes = array(e_UC_PUBLIC => UC_LAN_0,
e_UC_GUEST => UC_LAN_1,
e_UC_NOBODY => UC_LAN_2,
e_UC_MEMBER => UC_LAN_3,
e_UC_ADMIN => UC_LAN_5,
e_UC_MAINADMIN => UC_LAN_6,
e_UC_READONLY => UC_LAN_4
);
2008-02-16 12:03:27 +00:00
$this->text_class_link = array('public' => e_UC_PUBLIC, 'guest' => e_UC_GUEST, 'nobody' => e_UC_NOBODY, 'member' => e_UC_MEMBER,
2007-12-22 12:39:27 +00:00
'admin' => e_UC_ADMIN, 'main' => e_UC_MAINADMIN, 'readonly' => e_UC_READONLY);
$this->read_tree(TRUE); // Initialise the classes on entry
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
/*
Ensure the tree of userclass data is stored in our object.
Only read if its either not present, or the $force flag is set
*/
function read_tree($force = FALSE)
{
2008-03-23 10:11:17 +00:00
if (isset($this->class_tree) && !$force) return $this->class_tree;
2007-12-22 12:39:27 +00:00
$this->class_tree = array();
$this->class_parents = array();
$this->sql_r->db_Select("userclass_classes", '*', "ORDER BY userclass_parent", 'nowhere'); // The order statement should give a consistent return
while ($row = $this->sql_r->db_Fetch())
{
$this->class_tree[$row['userclass_id']] = $row;
$this->class_tree[$row['userclass_id']]['class_children'] = array(); // Create the child array in case needed
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
2008-03-23 10:11:17 +00:00
// Add in any fixed classes that aren't already defined
foreach ($this->fixed_classes as $c => $d)
{
if (!isset($this->class_tree[$c]) && ($c != e_UC_PUBLIC))
{
// $this->class_tree[$c]['userclass_parent'] = (($c == e_UC_MEMBER) || ($c == e_UC_NOBODY)) ? e_UC_PUBLIC : e_UC_MEMBER;
$this->class_tree[$c]['userclass_parent'] = (($c == e_UC_ADMIN) || ($c == e_UC_MAINADMIN)) ? e_UC_MEMBER : e_UC_PUBLIC ;
$this->class_tree[$c]['userclass_id'] = $c;
$this->class_tree[$c]['userclass_name'] = $d;
$this->class_tree[$c]['userclass_description'] = 'Fixed class';
$this->class_tree[$c]['userclass_visibility'] = e_UC_PUBLIC;
$this->class_tree[$c]['userclass_editclass'] = e_UC_MAINADMIN;
2008-03-23 10:11:17 +00:00
$this->class_tree[$c]['userclass_accum'] = $c;
// $this->class_parents[] = $c;
}
}
2007-12-22 12:39:27 +00:00
// Now build the tree
foreach ($this->class_tree as $uc)
{
2008-03-23 10:11:17 +00:00
if ($uc['userclass_parent'] == e_UC_PUBLIC)
// if (($uc['userclass_parent'] == e_UC_PUBLIC) || ($uc['userclass_parent'] == e_UC_NOBODY) || ($uc['userclass_parent'] == e_UC_MEMBER))
2007-12-22 12:39:27 +00:00
{ // Note parent (top level) classes
$this->class_parents[$uc['userclass_id']] = $uc['userclass_id'];
}
else
{
if (!array_key_exists($uc['userclass_parent'],$this->class_tree))
{
echo "Orphaned class record: ID=".$uc['userclass_id']." Name=".$uc['userclass_name']." Parent=".$uc['userclass_parent']."<br />";
}
else
{ // Add to array
$this->class_tree[$uc['userclass_parent']]['class_children'][] = $uc['userclass_id'];
}
}
}
}
// Given the list of 'base' classes a user belongs to, returns a comma separated list including ancestors. Duplicates stripped
function get_all_user_classes($start_list)
{
$is = array();
$start_array = explode(',', $start_list);
foreach ($start_array as $sa)
{ // Merge in latest values - should eliminate duplicates as it goes
if (isset($this->class_tree[$sa]))
{
$is = array_merge($is,explode(',',$this->class_tree[$sa]['userclass_accum']));
}
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
return implode(',',array_unique($is));
}
2008-01-07 22:30:29 +00:00
// Returns a list of user classes which can be edited by the specified classlist (defaults to current user's classes)
function get_editable_classes($class_list = USERCLASS_LIST)
{
$ret = array();
2008-01-09 22:44:19 +00:00
$blockers = array(e_UC_PUBLIC => 1, e_UC_READONLY => 1, e_UC_MEMBER => 1, e_UC_NOBODY => 1, e_UC_GUEST => 1);
2008-01-07 22:30:29 +00:00
$possibles = array_flip(explode(',',$class_list));
2008-01-09 22:44:19 +00:00
// unset($possibles[e_UC_PUBLIC]);
2008-01-08 22:24:22 +00:00
unset($possibles[e_UC_READONLY]);
foreach ($this->class_tree as $uc => $uv)
2008-01-07 22:30:29 +00:00
{
2008-01-08 22:24:22 +00:00
if (!isset($blockers[$uc]))
2008-01-07 22:30:29 +00:00
{
2008-01-08 22:24:22 +00:00
$ec = $this->class_tree[$uc]['userclass_editclass'];
// echo "Check class: {$uc} editclass {$ec} in array: ".(isset($possibles[$ec]) ? 'yes' : 'no').'<br />';
if (isset($possibles[$ec]))
{
$ret[] = $uc;
}
2008-01-07 22:30:29 +00:00
}
}
return implode(',',$ret);
}
2007-12-22 12:39:27 +00:00
// Given a comma separated list, returns the minimum number of class memberships required to achieve this (i.e. strips classes 'above' another in the tree)
// Requires the class tree to have been initialised
function normalise_classes($class_list)
{
$drop_classes = array();
$old_classes = explode(',',$class_list);
foreach ($old_classes as $c)
{ // Look at our parents (which are in 'userclass_accum') - if any of them are contained in old_classes, we can drop them.
$tc = array_flip(explode(',',$this->class_tree[$c]['userclass_accum']));
unset($tc[$c]); // Current class should be in $tc anyway
foreach ($tc as $tc_c => $v)
{
if (in_array($tc_c,$old_classes))
{
$drop_classes[] = $tc_c;
}
}
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
$new_classes = array_diff($old_classes,$drop_classes);
return implode(',',$new_classes);
}
/* Generate a dropdown list of user classes from which to select - virtually as r_userclass()
$optlist allows selection of the classes to be shown in the dropdown. All or none can be included, separated by comma. Valid options are:
public
guest
nobody
member
readonly
admin
main - main admin
classes - shows all classes
matchclass - if 'classes' is set, this option will only show the classes that the user is a member of
language - list of languages.
2008-04-05 16:30:18 +00:00
blank - puts an empty option at the top of select dropdowns
2007-12-22 12:39:27 +00:00
filter - only show those classes where member is in a class permitted to view them - i.e. as the new 'visible to' field - added for 0.8
force - show all classes (subject to the other options, including matchclass) - added for 0.8
$extra_js - can add JS handlers (e.g. 'onclick', 'onchange') if required
[ $mode parameter of r_userclass() removed - $optlist is more flexible) ]
*/
function uc_dropdown($fieldname, $curval = 0, $optlist = "", $extra_js = '')
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
global $pref;
$show_classes = $this->uc_required_class_list($optlist);
2008-09-15 16:20:27 +00:00
$text = '';
2007-12-22 12:39:27 +00:00
foreach ($show_classes as $k => $v)
{
2008-04-05 16:30:18 +00:00
if ($k == e_UC_BLANK)
{
$text .= "<option value=''>&nbsp;</option>\n";
}
else
{
2008-09-15 16:20:27 +00:00
$s = ($curval == $k && $curval !== '') ? "selected='selected'" : "";
2008-04-05 16:30:18 +00:00
$text .= "<option value='".$k."' ".$s.">".$v."</option>\n";
}
2007-12-22 12:39:27 +00:00
}
2008-09-15 16:20:27 +00:00
if (strpos($optlist, "language") !== FALSE && $pref['multilanguage'])
2007-12-22 12:39:27 +00:00
{
$text .= "<optgroup label=' ------ ' />\n";
$tmpl = explode(",",e_LANLIST);
foreach($tmpl as $lang)
{
$s = ($curval == $lang) ? " selected='selected'" : "";
$text .= "<option value='$lang' ".$s.">".$lang."</option>\n";
}
}
// Only return the select box if we've ended up with some options
if ($text) $text = "<select class='tbox' name='{$fieldname}' {$extra_js}>\n".$text."</select>\n";
return $text;
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
/*
Generate an ordered array classid=>classname - used for dropdown and check box lists
If $just_ids is TRUE, array value is just '1'
*/
function uc_required_class_list($optlist = '', $just_ids = FALSE)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$ret = array();
if (!$optlist) $optlist = 'public,guest,nobody,member,classes'; // Set defaults to simplify ongoing processing
2008-01-08 22:24:22 +00:00
if ($optlist == 'editable')
{
$temp = array_flip(explode(',',$this->get_editable_classes()));
if ($just_ids) return $temp;
foreach ($temp as $c => $t)
{
$temp[$c] = $this->class_tree[$c]['userclass_name'];
}
return $temp;
}
2007-12-22 12:39:27 +00:00
$opt_arr = explode(',',$optlist);
foreach ($opt_arr as $k => $v)
{
$opt_arr[$k] = trim($v);
}
$opt_arr = array_flip($opt_arr); // This also eliminates duplicates which could arise from applying the other options, although shouldn't matter
if (isset($opt_arr['force'])) unset($opt_arr['filter']);
2008-04-05 16:30:18 +00:00
if (isset($opt_arr['blank']))
{
$ret[e_UC_BLANK] = 1;
}
2007-12-22 12:39:27 +00:00
// Do the 'fixed' classes next
foreach ($this->text_class_link as $k => $v)
{
2008-02-16 12:03:27 +00:00
// if (isset($opt_arr[$k]) || isset($opt_arr['force']))
if (isset($opt_arr[$k]))
2007-12-22 12:39:27 +00:00
{
$ret[$v] = $just_ids ? '1' : $this->fixed_classes[$v];
}
}
// Now do the user-defined classes
2007-12-29 22:32:58 +00:00
if (isset($opt_arr['classes']) || isset($opt_arr['force']))
2007-12-22 12:39:27 +00:00
{ // Display those classes the user is allowed to:
// Main admin always sees the lot
// a) Mask the 'fixed' user classes which have already been processed
// b) Apply the visibility option field ('userclass_visibility')
// c) Apply the matchclass option if appropriate
foreach($this->class_tree as $uc_id => $row)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
if (!array_key_exists($uc_id,$this->fixed_classes)
&& ( getperms("0")
|| (
(!isset($optlist["matchclass"]) || check_class($uc_id))
&& (!isset($optlist["filter"]) || check_class($row['userclass_visibility']))
)
)
)
2006-12-02 04:36:16 +00:00
{
2008-01-08 22:24:22 +00:00
$ret[$uc_id] = $just_ids ? '1' : $this->class_tree[$uc_id]['userclass_name'];
2006-12-02 04:36:16 +00:00
}
}
2007-12-22 12:39:27 +00:00
}
/* Above loop slightly changes the display order of earlier code versions.
If readonly must be last (after language), delete it from the $text_class_link array, and uncomment the following code
if (isset($opt_arr['readonly']))
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$ret[e_UC_READONLY] = $this->class_tree[e_UC_READONLY]['userclass_description'];
}
*/
return $ret;
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
/*
Very similar to r_userclass, but returns a list of check boxes. Doesn't encapsulate it.
$fieldname is the name for the array of checkboxes
$curval is a comma separated list of class IDs for boxes which are checked.
$optlist as for uc_dropdown
if $showdescription is TRUE, appends the class description in brackets
*/
function uc_checkboxes($fieldname, $curval='', $optlist = '', $showdescription = FALSE)
{
global $pref;
$show_classes = $this->uc_required_class_list($optlist);
$curArray = explode(",", $curval); // Array of current values
$ret = "";
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
foreach ($show_classes as $k => $v)
{
2008-04-05 16:30:18 +00:00
if ($k != e_UC_BLANK)
{
2007-12-22 12:39:27 +00:00
$c = (in_array($k,$curArray)) ? " checked='checked'" : "";
if ($showdescription) $v .= " (".$this->uc_get_classdescription($k).")";
$ret .= "<label><input type='checkbox' name='{$fieldname}[{$k}]' value='{$k}' {$c} /> ".$v."</label><br />\n";
2008-04-05 16:30:18 +00:00
}
2007-12-22 12:39:27 +00:00
}
if (strpos($optlist, "language") !== FALSE && $pref['multilanguage'])
{
$ret .= "<hr />\n";
2006-12-02 04:36:16 +00:00
$tmpl = explode(",",e_LANLIST);
2007-12-22 12:39:27 +00:00
foreach($tmpl as $lang)
{
$c = (in_array($lang, $curArray)) ? " checked='checked' " : "";
$ret .= "<label><input type='checkbox' name='{$fieldname}[{$lang}]' value='1' {$c} /> {$lang}</label><br />";
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
}
return $ret;
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
/*
Next two routines create an indented tree - for example within a select box or a list of check boxes.
For each displayed element, the callback routine is called
$treename is the name given to the elements where required
$callback is a routine used to generate each element; there are two implemented within this class:
select (the default) - generates the option list. Text requires to be encapsulated in a <select......./select> tag set
- can also be used with multi-select boxes
checkbox - generates a set of checkboxes
Alternative callbacks can be used to achieve different layouts/styles
$current_value is a single class number for single-select dropdown; comma separated array of class numbers for checkbox list or multi-select
$optlist works the same as for other class displays
*/
function vetted_sub_tree($treename, $callback,$listnum,$nest_level,$current_value, $perms)
{
$ret = '';
$nest_level++;
foreach ($this->class_tree[$listnum]['class_children'] as $p)
{
2007-12-22 12:39:27 +00:00
// Looks like we don't need to differentiate between function and class calls
if (isset($perms[$p]))
{
$ret .= call_user_func($callback,$treename, $p,$current_value,$nest_level);
}
$ret .= $this->vetted_sub_tree($treename, $callback,$p,$nest_level,$current_value, $perms);
}
2007-12-22 12:39:27 +00:00
return $ret;
}
function vetted_tree($treename, $callback='', $current_value='', $optlist = '')
{
$ret = '';
if (!$callback) $callback=array($this,'select');
$current_value = str_replace(' ','',$current_value); // Simplifies parameter passing for the tidy-minded
$perms = $this->uc_required_class_list($optlist,TRUE); // List of classes which we can display
2008-04-05 16:30:18 +00:00
if (isset($perms[e_UC_BLANK]))
{
$ret .= call_user_func($callback,$treename, e_UC_BLANK, $current_value,0);
}
2007-12-22 12:39:27 +00:00
foreach ($this->class_parents as $p)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
if (isset($perms[$p]))
{
$ret .= call_user_func($callback,$treename, $p,$current_value,0);
}
$ret .= $this->vetted_sub_tree($treename, $callback,$p,0, $current_value, $perms);
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
return $ret;
}
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
2008-03-23 10:11:17 +00:00
// Callback for vetted_tree - Creates the option list for a selection box
2007-12-22 12:39:27 +00:00
function select($treename, $classnum, $current_value, $nest_level)
{
2008-04-05 16:30:18 +00:00
if ($classnum == e_UC_BLANK) return "<option value=''>&nbsp;</option>\n";
2007-12-22 12:39:27 +00:00
$tmp = explode(',',$current_value);
$sel = in_array($classnum,$tmp) ? " selected='selected'" : '';
if ($nest_level == 0)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$prefix = '';
$style = " style='font-weight:bold; font_style: italic;'";
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
elseif ($nest_level == 1)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$prefix = '&nbsp;&nbsp;';
$style = " style='font-weight:bold'";
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
else
{
2008-02-16 12:03:27 +00:00
$prefix = '&nbsp;&nbsp;'.str_repeat('--',$nest_level-1).'>';
2007-12-22 12:39:27 +00:00
$style = '';
}
return "<option value='{$classnum}'{$sel}{$style}>".$prefix.$this->class_tree[$classnum]['userclass_name']."</option>\n";
}
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
// Callback for vetted_tree - displays indented checkboxes with class name only
function checkbox($treename, $classnum, $current_value, $nest_level)
{
2008-04-05 16:30:18 +00:00
if ($classnum == e_UC_BLANK) return '';
2007-12-22 12:39:27 +00:00
$tmp = explode(',',$current_value);
$chk = in_array($classnum, $tmp) ? " checked='checked'" : '';
if ($nest_level == 0)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$style = " style='font-weight:bold'";
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
else
{
// $style = " style='text-indent:".(12*$nest_level)."px'";
$style = " style='text-indent:".(1.2*$nest_level)."em'";
}
return "<div {$style}><input type='checkbox' name='{$treename}[]' value='{$classnum}'{$chk} />".$this->class_tree[$classnum]['userclass_name']."</div>\n";
}
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
// Callback for vetted_tree - displays indented checkboxes with class name, and description in brackets
function checkbox_desc($treename, $classnum, $current_value, $nest_level)
{
2008-04-05 16:30:18 +00:00
if ($classnum == e_UC_BLANK) return '';
2007-12-22 12:39:27 +00:00
$tmp = explode(',',$current_value);
$chk = in_array($classnum, $tmp) ? " checked='checked'" : '';
if ($nest_level == 0)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$style = " style='font-weight:bold'";
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
else
{
$style = " style='text-indent:".(1.2*$nest_level)."em'";
}
return "<div {$style}><input type='checkbox' name='{$treename}[]' value='{$classnum}'{$chk} />".$this->class_tree[$classnum]['userclass_name'].' ('.$this->class_tree[$classnum]['userclass_description'].")</div>\n";
}
2006-12-02 04:36:16 +00:00
2007-12-22 12:39:27 +00:00
/*
Return array of all classes, limited according to membership of the userclass_visibility field if $filter is set.
Index field - userclass_id
Data fields - userclass_name, userclass_description, userclass_editclass
*/
function uc_get_classlist($filter = FALSE)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$ret = array();
$this->read_tree(FALSE); // Make sure we have data
foreach ($this->class_tree as $k => $v)
{
if (!$filter || check_class($filter))
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$ret[$k] = array('userclass_name' => $v, 'userclass_description' => $v['userclass_description'], 'userclass_editclass' => $v['userclass_editclass']);
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
}
return $ret;
}
function uc_get_classname($id)
{
if (isset($this->class_tree[$id]))
{
return $this->class_tree[$id]['userclass_name'];
}
if (isset($this->fixed_classes[$id]))
{
return $this->fixed_classes[$id];
}
return '';
}
function uc_get_classdescription($id)
{
if (isset($this->class_tree[$id]))
{
return $this->class_tree[$id]['userclass_description'];
}
if (isset($this->fixed_classes[$id]))
{
return $this->fixed_classes[$id]; // Name and description the same for fixed classes
}
return '';
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
function uc_get_classicon($id)
{
if (isset($this->class_tree[$id]))
{
return $this->class_tree[$id]['userclass_icon'];
}
return '';
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
/*
Return all users in a particular class or set of classes.
$classlist is a comma separated list of classes - if the 'predefined' classes are required, they must be included. No spaces allowed
$field_list is used to select the returned fields ($user_id is the primary index)
****** Can be verrrrryyyy slow - has to scan the whole user database at present ******
********* NOT TESTED **********
***** NOT SURE WHETHER THIS IS REALLY A USER OR A USER CLASS FUNCTION *****
*/
function get_users_in_class($classlist, $field_list = 'user_name, user_loginname', $include_ancestors = FALSE, $order_by = 'user_id')
{
2007-12-22 12:39:27 +00:00
$ret = array();
if ($include_ancestors) $classlist = $this->get_all_user_classes($classlist);
$class_regex = "(^|,)(".str_replace(' ','',str_replace(",", "|", $classlist)).")(,|$)";
$qry = "SELECT 'user_id,{$field_list}' FROM `user` WHERE user_class REGEXP '{$class_regex}' ORDER BY '{$order_by}'";
if ($this->sql_r->db_Select_gen($qry))
{
while ($row = $this->sql_r->db_Fetch(MYSQL_ASSOC))
{
$ret[$row['user_id']] = $row;
}
}
return $ret;
}
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
//========================================================================
// Functions from previous userclass_class handler
//========================================================================
// Implemented for backwards compatibility/convenience
/*
With $optlist you can now specify which classes are shown in the dropdown.
All or none can be included, separated by comma (or whatever).
Valid options are:
public
guest
nobody
member
readonly
admin
main - main admin
classes - shows all classes
matchclass - if 'classes' is set, this option will only show the classes that the user is a member of
language - list of languages.
filter - only show those classes where member is in a class permitted to view them - i.e. as the new 'visible to' field
force - show all classes (subject to the other options, including matchclass)
$mode = 'off' turns off listing of admin/main admin classes unless enabled in $optlist (can probably be deprecated - barely used)
*/
function r_userclass($fieldname, $curval = 0, $mode = "off", $optlist = "")
{
// echo "Call r_userclass{$fieldname}, CV: {$curval} opts: {$optlist}<br />";
global $e_userclass;
if ($mode != 'off')
{ // Handle legacy code
if ($optlist) $optlist .= ',';
$optlist .= 'admin,main';
if ($mode != 'admin') $optlist .= ',readonly';
}
if (!is_object($e_userclass)) $e_userclass = new user_class;
return $e_userclass->uc_dropdown($fieldname,$curval,$optlist);
}
// Very similar to r_userclass, but returns a list of check boxes. (currently only used in newspost.php)
// $curval is a comma separated list of class IDs for boxes which are checked.
function r_userclass_check($fieldname, $curval = '', $optlist = "")
{
// echo "Call r_userclass_check: {$curval}<br />";
global $e_userclass;
if (!is_object($e_userclass)) $e_userclass = new user_class;
$ret = $e_userclass->uc_checkboxes($fieldname,$curval,$optlist);
if ($ret) $ret = "<div class='tbox' style='margin-left:0px;margin-right:auto;width:60%;height:58px;overflow:auto'>".$ret."</div>";
return $ret;
}
2006-12-02 04:36:16 +00:00
function get_userclass_list()
{
2007-12-22 12:39:27 +00:00
// echo "Call get_userclass_list<br />";
global $e_userclass;
if (!is_object($e_userclass)) $e_userclass = new user_class;
return $e_userclass->uc_get_classlist();
}
function r_userclass_name($id)
{
// echo "Call r_userclass_name<br />";
global $e_userclass;
if (!is_object($e_userclass)) $e_userclass = new user_class;
return $e_userclass->uc_get_classname($id);
}
// Deprecated functions to hopefully be removed
function r_userclass_radio($fieldname, $curval = '')
{
echo "Deprecated function r_userclass_radio not used in core - mutter if you'd like it implemented<br />";
}
//========================================================================
// Admin Class handler - could go into separate file later
//========================================================================
class user_class_admin extends user_class
{
var $field_list = array('userclass_name' => "varchar(100) NOT NULL default ''",
'userclass_description' => "varchar(250) NOT NULL default ''",
'userclass_editclass' => "tinyint(3) unsigned NOT NULL default '0'",
'userclass_parent' => "tinyint(3) unsigned NOT NULL default '0'",
'userclass_accum' => "varchar(250) NOT NULL default ''",
'userclass_visibility' => "tinyint(3) unsigned NOT NULL default '0'",
'userclass_icon' => "varchar(250) NOT NULL default ''"
); // Note - 'userclass_id' intentionally not in this list
// Icons to use for graphical tree display
// First index - no children, children
// Second index - not last item, last item
// Third index - closed tree, open tree
var $tree_icons = array( );
var $graph_debug = FALSE; // Shows extra info on graphical tree when TRUE
function user_class_admin()
{
$this->user_class(); // Call constructor from ancestor class
// Have to initialise the images this way - PHP4 won't take a nested array assignment in the variable list
$this->tree_icons = array(
FALSE => array( // No children
FALSE => array( // Not last item
FALSE => '', // Closed tree - don't display
TRUE => 'branch.gif'
)
,
TRUE => array( // Last item
FALSE => '', // Closed tree - don't display
TRUE => 'branchbottom.gif'
)
),
TRUE => array( // children
FALSE => array( // Not last item
FALSE => 'plus.gif', // Closed tree - option to expand
TRUE => 'minus.gif'
)
,
TRUE => array( // Last item
FALSE => 'plusbottom.gif', // Closed tree - option to expand
TRUE => 'minusbottom.gif'
))
);
}
/*
Next three routines are used to update the database after adding/deleting a class
*/
function calc_tree()
{
// echo "Calc Tree<br />";
$this->read_tree(TRUE); // Make sure we have accurate data
foreach ($this->class_parents as $cp)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$rights = array();
2008-03-23 10:11:17 +00:00
$this->rebuild_tree($cp,$rights); // increasing rights going down the tree
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
}
2008-03-23 10:11:17 +00:00
// Internal function, called recursively to rebuild the permissions tree where rights increase going down the tree
// $parent is the class number being processed.
// $rights is the array of rights accumulated so far in the walk down the tree
2007-12-22 12:39:27 +00:00
function rebuild_tree($parent, $rights)
{
2008-03-23 10:11:17 +00:00
if ($this->class_tree[$parent]['userclass_parent'] == e_UC_NOBODY)
{
$this->topdown_tree($parent);
return;
}
2007-12-22 12:39:27 +00:00
$rights[] = $parent;
$imp_rights = implode(',',$rights);
if ($this->class_tree[$parent]['userclass_accum'] != $imp_rights)
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
$this->class_tree[$parent]['userclass_accum'] = $imp_rights;
if (!isset($this->class_tree[$cp]['change_flag'])) $this->class_tree[$parent]['change_flag'] = 'UPDATE';
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
foreach ($this->class_tree[$parent]['class_children'] as $cc)
{
$this->rebuild_tree($cc,$rights); // Recursive call
}
}
2006-12-02 04:36:16 +00:00
2008-03-23 10:11:17 +00:00
// Internal function, called recursively to rebuild the permissions tree where rights increase going up the tree
// Returns an array
function topdown_tree($our_class)
{
$rights = array($our_class); // Accumulator always has rights to its own class
foreach ($this->class_tree[$our_class]['class_children'] as $cc)
{
$rights = array_merge($rights,$this->topdown_tree($cc)); // Recursive call
}
$rights = array_unique($rights);
$imp_rights = implode(',',$rights);
if ($this->class_tree[$our_class]['userclass_accum'] != $imp_rights)
{
$this->class_tree[$our_class]['userclass_accum'] = $imp_rights;
$this->class_tree[$our_class]['change_flag'] = 'UPDATE';
}
return $rights;
}
2007-12-22 12:39:27 +00:00
function save_tree()
{
// echo "Save Tree<br />";
foreach ($this->class_tree as $tree)
{
if (isset($tree['change_flag']))
{
switch ($tree['change_flag'])
2006-12-02 04:36:16 +00:00
{
2007-12-22 12:39:27 +00:00
case 'INSERT' :
$this->add_new_class($tree);
break;
case 'UPDATE' :
$this->save_edited_class($tree);
break;
default :
continue;
}
}
}
}
/*
Next two routines show a text-based tree with markers to indicate the hierarchy.
*/
function show_sub_tree($listnum,$marker, $add_class = FALSE)
{
$ret = '';
$marker = '--'.$marker;
foreach ($this->class_tree[$listnum]['class_children'] as $p)
{
$ret .= $marker.$this->class_tree[$p]['userclass_id'].':'.$this->class_tree[$p]['userclass_name'];
if ($add_class) $ret .= " (".$this->class_tree[$p]['userclass_accum'].")";
$ret .= " Children: ".count($this->class_tree[$p]['class_children']);
$ret .= "<br />";
$ret .= $this->show_sub_tree($p,$marker, $add_class);
}
return $ret;
}
function show_tree($add_class = FALSE)
{
$ret = '';
foreach ($this->class_parents as $p)
{
$ret .= $this->class_tree[$p]['userclass_id'].':'.$this->class_tree[$p]['userclass_name'];
if ($add_class) $ret .= " (".$this->class_tree[$p]['userclass_accum'].")";
$ret .= " Children: ".count($this->class_tree[$p]['class_children']);
$ret .= "<br />";
$ret .= $this->show_sub_tree($p,'>', $add_class);
}
return $ret;
}
/*
Next two routines generate a graphical tree, including option to open/close branches
*/
function show_graphical_subtree($listnum, $indent_images, $is_last = FALSE)
{
$num_children = count($this->class_tree[$listnum]['class_children']);
$is_open = TRUE;
$tag_name = 'uclass_tree_'.$listnum;
$ret = "<div class='uclass_tree' style='height: 20px'>\n";
2007-12-22 12:39:27 +00:00
foreach ($indent_images as $im)
{
$ret .= "<img src='".UC_ICON_DIR.$im."' alt='class icon' />";
}
// If this link has children, wrap the next image in a link and an expand/hide option
if ($num_children)
{
$ret .= "<span onclick=\"javascript: expandit('{$tag_name}'); expandit('{$tag_name}_p'); expandit('{$tag_name}_m')\"><img src='".UC_ICON_DIR.$this->tree_icons[TRUE][$is_last][TRUE]."' alt='class icon' id='{$tag_name}_m' />";
$ret .= "<img src='".UC_ICON_DIR.$this->tree_icons[TRUE][$is_last][FALSE]."' style='display:none' id='{$tag_name}_p' alt='class icon' /></span>\n";
2007-12-22 12:39:27 +00:00
}
else
{
$ret .= "<img src='".UC_ICON_DIR.$this->tree_icons[FALSE][$is_last][$is_open]."' alt='class icon' />\n";
2007-12-22 12:39:27 +00:00
}
$name_line = '';
if ($this->graph_debug) $name_line = $this->class_tree[$listnum]['userclass_id'].":";
$name_line .= $this->class_tree[$listnum]['userclass_name'];
2008-01-07 22:30:29 +00:00
if ($this->graph_debug) $name_line .= "[vis:".$this->class_tree[$listnum]['userclass_visibility'].", edit:".$this->class_tree[$listnum]['userclass_editclass']."] = ".$this->class_tree[$listnum]['userclass_accum'];
2007-12-22 12:39:27 +00:00
// Next (commented out) line gives a 'conventional' link
$ret .= "<img src='".UC_ICON_DIR."topicon.png' alt='class icon' /><a style='text-decoration: none' class='userclass_edit' href='".e_ADMIN_ABS."userclass2.php?config.edit.{$this->class_tree[$listnum]['userclass_id']}'>".$this->class_tree[$listnum]['userclass_name']."</a>
</div>";
//$ret .= "<img src='".UC_ICON_DIR."topicon.png' alt='class icon' />
//<span style='cursor:pointer; vertical-align: bottom' onclick=\"javascript: document.location.href='".e_ADMIN."userclass2.php?config.edit.{$this->class_tree[$listnum]['userclass_id']}'\">".$name_line."</span></div>";
// vertical-align: middle doesn't work! Nor does text-top
2007-12-22 12:39:27 +00:00
if ($num_children)
{
$ret .= "<div class='uclass_tree' id='{$tag_name}'>\n";
2007-12-22 12:39:27 +00:00
$image_level = count($indent_images);
if ($is_last)
{
$indent_images[] = 'linebottom.gif';
}
else
{
$indent_images[] = 'line.gif';
}
foreach ($this->class_tree[$listnum]['class_children'] as $p)
{
$num_children--;
if ($num_children)
{ // Use icon indicating more below
$ret .= $this->show_graphical_subtree($p, $indent_images, FALSE);
}
else
{ // else last entry on this tree
$ret .= $this->show_graphical_subtree($p, $indent_images, TRUE);
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
}
$ret .= "</div>";
}
return $ret;
}
2008-03-23 10:11:17 +00:00
2007-12-22 12:39:27 +00:00
function show_graphical_tree($show_debug=FALSE)
{
$this->graph_debug = $show_debug;
$indent_images = array();
$ret = "<div class='uclass_tree' style='height:16px'>
<img src='".UC_ICON_DIR."topicon.png' alt='class icon' style='vertical-align: bottom' />
<span style='top:3px'>".UC_LAN_0."</span>
</div>"; // 'Everyone' link
2007-12-22 12:39:27 +00:00
$num_parents = count($this->class_parents);
foreach ($this->class_parents as $p)
{
$num_parents--;
$ret .= $this->show_graphical_subtree($p, $indent_images, ($num_parents == 0));
}
return $ret;
}
// Creates an array which contains only DB fields (i.e. strips the added status)
function copy_rec($classrec, $inc_id = FALSE)
{
$ret = array();
if ($inc_id && isset($classrec['userclass_id'])) $ret['userclass_id'] = $classrec['userclass_id'];
foreach ($this->field_list as $fl => $val)
{
if (isset($classrec[$fl])) $ret[$fl] = $classrec[$fl];
}
return $ret;
}
function add_new_class($classrec)
{
// echo "Add new class<br />";
$this->sql_r->db_Insert('userclass_classes',$this->copy_rec($classrec, TRUE));
}
function save_edited_class($classrec)
{
// echo "Save edited class: ".implode(',', $classrec)."<br />";
if (!$classrec['userclass_id'])
{
echo "Programming bungle on save<br />";
return;
}
$qry = '';
$spacer = '';
foreach ($this->field_list as $fl => $val)
{
if (isset($classrec[$fl]))
{
$qry .= $spacer."`".$fl."` = '".$classrec[$fl]."'";
$spacer = ", ";
}
}
$this->sql_r->db_Update('userclass_classes', $qry." WHERE `userclass_id`='{$classrec['userclass_id']}'");
}
function delete_class($class_id)
{
2008-02-16 12:03:27 +00:00
if (isset($this->fixed_classes[$class_id])) return FALSE; // Some classes can't be deleted
if (!isset($this->class_tree[$class_id])) return FALSE;
if (count($this->class_tree[$class_id]['class_children'])) return FALSE; // Can't delete class with descendants
foreach ($this->class_tree as $c)
{
if ($c['userclass_editclass'] == $class_id) return FALSE;
if ($c['userclass_visibility'] == $class_id) return FALSE;
}
if (!$this->sql_r->db_Delete('userclass_classes', "`userclass_id`='{$class_id}'")) return FALSE;
$this->read_tree(TRUE); // Re-read the class tree
return TRUE;
2007-12-22 12:39:27 +00:00
}
// Update the userclass table with the extra fields for 0.8
// Return TRUE if all fields present. Return FALSE if update needed
function update_db($check_only = FALSE)
{ // Just run through all the fields that should be there, and add if not
$fn = 1; // Count fields
$prev_field = 'userclass_id';
foreach ($this->field_list as $fl => $parms)
{
$field_name = $this->sql_r->db_Field("userclass_classes",$fn);
// echo "Compare: {$field_name} : {$fl}<br />";
if ($field_name != $fl)
{
if ($check_only) return FALSE;
$this->sql_r->db_Select_gen("ALTER TABLE #userclass_classes ADD `{$fl}` {$parms} AFTER `{$prev_field}`;");
}
$fn++;
$prev_field = $fl;
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
return TRUE;
}
// Set default tree structure
function set_default_structure()
{
// If they don't exist, we need to create class records for the 'standard' user classes
$init_list = array(
2008-03-23 10:11:17 +00:00
array('userclass_id' => e_UC_MEMBER, 'userclass_name' => UC_LAN_3,
'userclass_description' => UCSLAN_75,
2007-12-22 12:39:27 +00:00
'userclass_editclass' => e_UC_MAINADMIN,
'userclass_parent' => e_UC_PUBLIC,
'userclass_visibility' => e_UC_MEMBER
),
2008-03-23 10:11:17 +00:00
array('userclass_id' => e_UC_ADMINMOD, 'userclass_name' => UC_LAN_8,
'userclass_description' => UCSLAN_74,
2007-12-22 12:39:27 +00:00
'userclass_editclass' => e_UC_MAINADMIN,
2008-03-23 10:11:17 +00:00
'userclass_parent' => e_UC_MAINADMIN,
2007-12-22 12:39:27 +00:00
'userclass_visibility' => e_UC_MEMBER
),
2008-03-23 10:11:17 +00:00
array('userclass_id' => e_UC_ADMIN, 'userclass_name' => UC_LAN_5,
'userclass_description' => UCSLAN_76,
2007-12-22 12:39:27 +00:00
'userclass_editclass' => e_UC_MAINADMIN,
'userclass_parent' => e_UC_ADMINMOD,
'userclass_visibility' => e_UC_MEMBER
),
2008-03-23 10:11:17 +00:00
array('userclass_id' => e_UC_MAINADMIN, 'userclass_name' => UC_LAN_6,
'userclass_description' => UCSLAN_77,
2007-12-22 12:39:27 +00:00
'userclass_editclass' => e_UC_MAINADMIN,
2008-03-23 10:11:17 +00:00
'userclass_parent' => e_UC_NOBODY,
2007-12-22 12:39:27 +00:00
'userclass_visibility' => e_UC_MEMBER
),
2008-03-23 10:11:17 +00:00
array('userclass_id' => e_UC_MODS, 'userclass_name' => UC_LAN_7,
'userclass_description' => UCSLAN_78,
2007-12-22 12:39:27 +00:00
'userclass_editclass' => e_UC_MAINADMIN,
'userclass_parent' => e_UC_ADMINMOD,
'userclass_visibility' => e_UC_MEMBER
2008-03-23 10:11:17 +00:00
)
2007-12-22 12:39:27 +00:00
);
foreach ($init_list as $entry)
{
if ($this->sql_r->db_Select('userclass_classes','*',"userclass_id='".$entry['userclass_id']."' "))
{
$this->sql_r->db_Update('userclass_classes', "userclass_parent='".$entry['userclass_parent']."', userclass_visibility='".$entry['userclass_visibility']."' WHERE userclass_id='".$entry['userclass_id']."'");
}
else
{
$this->add_new_class($entry);
}
}
}
2006-12-02 04:36:16 +00:00
}
2007-12-22 12:39:27 +00:00
//========================================================================
// Legacy Admin Class handler - maybe add to admin class
//========================================================================
// class_add() - called only from userclass2.php
// class_remove() - called only from userclass2.php
// class_create() - called only from forum update routines - could probably go
class e_userclass
{
2006-12-02 04:36:16 +00:00
function class_add($cid, $uinfoArray)
{
global $tp;
$sql2 = new db;
foreach($uinfoArray as $uid => $curclass)
{
if ($curclass)
{
$newarray = array_unique(array_merge(explode(',', $curclass), array($cid)));
$new_userclass = implode(',', $newarray);
}
else
{
$new_userclass = $cid;
}
$sql2->db_Update('user', "user_class='".$tp -> toDB($new_userclass, true)."' WHERE user_id=".intval($uid));
}
}
function class_remove($cid, $uinfoArray)
{
global $tp;
$sql2 = new db;
foreach($uinfoArray as $uid => $curclass)
{
$newarray = array_diff(explode(',', $curclass), array('', $cid));
$new_userclass = implode(',', $newarray);
2006-12-02 04:36:16 +00:00
$sql2->db_Update('user', "user_class='".$tp -> toDB($new_userclass, true)."' WHERE user_id=".intval($uid));
}
}
2007-12-22 12:39:27 +00:00
// Mostly for upgrades?
// Create a new user class, with a specified prefix to the name
// $ulist - comma separated list of user names to be added
2006-12-02 04:36:16 +00:00
function class_create($ulist, $class_prefix = "NEW_CLASS_", $num = 0)
{
global $sql;
$varname = "uc_".$ulist;
if($ret = getcachedvars($varname))
{
return $ret;
}
$ul = explode(",", $ulist);
array_walk($ul, array($this, 'munge'));
$qry = "
SELECT user_id, user_class from #user AS u
WHERE user_name = ".implode(" OR user_name = ", $ul);
if($sql->db_Select_gen($qry))
{
while($row = $sql->db_Fetch())
{
$idList[$row['user_id']] = $row['user_class'];
}
while($sql->db_Count("userclass_classes","(*)","WHERE userclass_name = '".strtoupper($class_prefix.$num)."'"))
{
$num++;
}
$newname = strtoupper($class_prefix.$num);
$i = 1;
2007-12-22 12:39:27 +00:00
while ($sql->db_Select('userclass_classes', '*', "userclass_id='".intval($i)."' ") && $i < 240)
2006-12-02 04:36:16 +00:00
{
$i++;
}
2007-12-22 12:39:27 +00:00
if ($i < 240) // Give a bit of headroom - we're allocating 'system' classes downwards from 255
2006-12-02 04:36:16 +00:00
{
$sql->db_Insert("userclass_classes", "{$i}, '{$newname}', 'Auto_created_class', 254");
2007-12-22 12:39:27 +00:00
$this->class_add($i, $idList); // Add users
2006-12-02 04:36:16 +00:00
cachevars($varname, $i);
return $i;
}
}
}
function munge(&$value, &$key)
{
$value = "'".trim($value)."'";
}
}
2007-12-22 12:39:27 +00:00
2006-12-02 04:36:16 +00:00
?>