sortable_table.class.php 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. require_once 'pear/HTML/Table.php';
  4. require_once 'pear/Pager/Pager.php';
  5. /**
  6. * This class allows you to display a sortable data-table. It is possible to
  7. * split the data in several pages.
  8. * Using this class you can:
  9. * - automatically create checkboxes of the first table column
  10. * - a "select all" and "deselect all" link is added
  11. * - only if you provide a list of actions for the selected items
  12. * - click on the table header to sort the data
  13. * - choose how many items you see per page
  14. * - navigate through all data-pages
  15. * @package chamilo.library
  16. */
  17. class SortableTable extends HTML_Table
  18. {
  19. /**
  20. * A name for this table
  21. */
  22. public $table_name;
  23. /**
  24. * The page to display
  25. */
  26. public $page_nr;
  27. /**
  28. * The column to sort the data
  29. */
  30. public $column;
  31. /**
  32. * The sorting direction (ASC or DESC)
  33. */
  34. public $direction;
  35. /**
  36. * Number of items to display per page
  37. */
  38. public $per_page;
  39. /**
  40. * The default number of items to display per page
  41. */
  42. public $default_items_per_page;
  43. /**
  44. * A prefix for the URL-parameters, can be used on pages with multiple
  45. * SortableTables
  46. */
  47. public $param_prefix;
  48. /**
  49. * The pager object to split the data in several pages
  50. */
  51. public $pager;
  52. /**
  53. * The total number of items in the table
  54. */
  55. public $total_number_of_items;
  56. /**
  57. * The function to get the total number of items
  58. */
  59. public $get_total_number_function;
  60. /**
  61. * The function to the the data to display
  62. */
  63. public $get_data_function;
  64. /**
  65. * An array with defined column-filters
  66. */
  67. public $column_filters;
  68. /**
  69. * A list of actions which will be available through a select list
  70. */
  71. public $form_actions;
  72. /**
  73. * Additional parameters to pass in the URL
  74. */
  75. public $additional_parameters;
  76. /**
  77. * Additional attributes for the th-tags
  78. */
  79. public $th_attributes;
  80. /**
  81. * Additional attributes for the td-tags
  82. */
  83. public $td_attributes;
  84. /**
  85. * Array with names of the other tables defined on the same page of this
  86. * table
  87. */
  88. public $other_tables;
  89. /**
  90. * Activates the odd even rows
  91. * **/
  92. public $odd_even_rows_enabled = true;
  93. public $use_jqgrid = false;
  94. public $table_id = null;
  95. public $headers = array();
  96. /**
  97. * @var array
  98. * Columns to hide
  99. */
  100. private $columnsToHide = array();
  101. /**
  102. * Create a new SortableTable
  103. * @param string $table_name A name for the table (default = 'table')
  104. * @param string $get_total_number_function A user defined function to get
  105. * the total number of items in the table
  106. * @param string $get_data_function A function to get the data to display on
  107. * the current page
  108. * @param int $default_column The default column on which the data should be
  109. * sorted
  110. * @param int $default_items_per_page The default number of items to show
  111. * on one page
  112. * @param string $default_order_direction The default order direction;
  113. * either the constant 'ASC' or 'DESC'
  114. * @param string $table_id
  115. */
  116. public function __construct(
  117. $table_name = 'table',
  118. $get_total_number_function = null,
  119. $get_data_function = null,
  120. $default_column = 1,
  121. $default_items_per_page = 20,
  122. $default_order_direction = 'ASC',
  123. $table_id = null
  124. ) {
  125. if (empty($table_id)) {
  126. $table_id = $table_name.uniqid();
  127. }
  128. $this->table_id = $table_id;
  129. parent::__construct(array('class' => 'data_table', 'id' => $table_id));
  130. $this->table_name = $table_name;
  131. $this->additional_parameters = array();
  132. $this->param_prefix = $table_name.'_';
  133. $this->page_nr = isset($_SESSION[$this->param_prefix.'page_nr']) ? intval($_SESSION[$this->param_prefix.'page_nr']) : 1;
  134. $this->page_nr = isset($_GET[$this->param_prefix.'page_nr']) ? intval($_GET[$this->param_prefix.'page_nr']) : $this->page_nr;
  135. $this->column = isset($_SESSION[$this->param_prefix.'column']) ? intval($_SESSION[$this->param_prefix.'column']) : $default_column;
  136. $this->column = isset($_GET[$this->param_prefix.'column']) ? intval($_GET[$this->param_prefix.'column']) : $this->column;
  137. // Default direction.
  138. if (in_array(strtoupper($default_order_direction), array('ASC', 'DESC'))) {
  139. $this->direction = $default_order_direction;
  140. }
  141. if (isset($_SESSION[$this->param_prefix.'direction'])) {
  142. $my_session_direction = $_SESSION[$this->param_prefix.'direction'];
  143. if (!in_array($my_session_direction, array('ASC', 'DESC'))) {
  144. $this->direction = 'ASC';
  145. } else {
  146. if ($my_session_direction == 'ASC') {
  147. $this->direction = 'ASC';
  148. } elseif ($my_session_direction == 'DESC') {
  149. $this->direction = 'DESC';
  150. }
  151. }
  152. }
  153. if (isset($_GET[$this->param_prefix.'direction'])) {
  154. $my_get_direction = $_GET[$this->param_prefix.'direction'];
  155. if (!in_array($my_get_direction, array('ASC', 'DESC'))) {
  156. $this->direction = 'ASC';
  157. } else {
  158. if ($my_get_direction == 'ASC') {
  159. $this->direction = 'ASC';
  160. } elseif ($my_get_direction == 'DESC') {
  161. $this->direction = 'DESC';
  162. }
  163. }
  164. }
  165. // Allow to change paginate in multiples tabs
  166. unset($_SESSION[$this->param_prefix.'per_page']);
  167. $this->per_page = isset($_SESSION[$this->param_prefix.'per_page']) ? intval($_SESSION[$this->param_prefix.'per_page']) : $default_items_per_page;
  168. $this->per_page = isset($_GET[$this->param_prefix.'per_page']) ? intval($_GET[$this->param_prefix.'per_page']) : $this->per_page;
  169. $_SESSION[$this->param_prefix.'per_page'] = $this->per_page;
  170. $_SESSION[$this->param_prefix.'direction'] = $this->direction ;
  171. $_SESSION[$this->param_prefix.'page_nr'] = $this->page_nr;
  172. $_SESSION[$this->param_prefix.'column'] = $this->column;
  173. $this->pager = null;
  174. $this->default_items_per_page = $default_items_per_page;
  175. $this->total_number_of_items = -1;
  176. $this->get_total_number_function = $get_total_number_function;
  177. $this->get_data_function = $get_data_function;
  178. $this->column_filters = array();
  179. $this->form_actions = array();
  180. $this->checkbox_name = null;
  181. $this->td_attributes = array();
  182. $this->th_attributes = array();
  183. $this->other_tables = array();
  184. }
  185. /**
  186. * Get the Pager object to split the showed data in several pages
  187. */
  188. public function get_pager()
  189. {
  190. if (is_null($this->pager)) {
  191. $total_number_of_items = $this->get_total_number_of_items();
  192. $params['mode'] = 'Sliding';
  193. $params['perPage'] = $this->per_page;
  194. $params['totalItems'] = $total_number_of_items;
  195. $params['urlVar'] = $this->param_prefix.'page_nr';
  196. $params['currentPage'] = $this->page_nr;
  197. $icon_attributes = array('style' => 'vertical-align: middle;');
  198. $params['prevImg'] = Display :: return_icon('action_prev.png', get_lang('PreviousPage'), $icon_attributes);
  199. $params['nextImg'] = Display :: return_icon('action_next.png', get_lang('NextPage'), $icon_attributes);
  200. $params['firstPageText'] = Display :: return_icon('action_first.png', get_lang('FirstPage'), $icon_attributes);
  201. $params['lastPageText'] = Display :: return_icon('action_last.png', get_lang('LastPage'), $icon_attributes);
  202. $params['firstPagePre'] = '';
  203. $params['lastPagePre'] = '';
  204. $params['firstPagePost'] = '';
  205. $params['lastPagePost'] = '';
  206. $params['spacesBeforeSeparator'] = '';
  207. $params['spacesAfterSeparator'] = '';
  208. $query_vars = array_keys($_GET);
  209. $query_vars_needed = array ($this->param_prefix.'column', $this->param_prefix.'direction', $this->param_prefix.'per_page');
  210. if (count($this->additional_parameters) > 0) {
  211. $query_vars_needed = array_merge($query_vars_needed, array_keys($this->additional_parameters));
  212. }
  213. $query_vars_exclude = array_diff($query_vars, $query_vars_needed);
  214. $params['excludeVars'] = $query_vars_exclude;
  215. $this->pager = & Pager::factory($params);
  216. }
  217. return $this->pager;
  218. }
  219. /**
  220. * Display the table
  221. */
  222. public function display()
  223. {
  224. echo $this->return_table();
  225. }
  226. /**
  227. * Displays the table, complete with navigation buttons to browse through
  228. * the data-pages.
  229. */
  230. public function return_table()
  231. {
  232. $empty_table = false;
  233. $content = $this->get_table_html();
  234. if ($this->get_total_number_of_items() == 0) {
  235. $cols = $this->getColCount();
  236. $this->setCellAttributes(1, 0, 'style="font-style: italic;text-align:center;" colspan='.$cols);
  237. $message_empty = api_xml_http_response_encode(get_lang('TheListIsEmpty'));
  238. $this->setCellContents(1, 0, $message_empty);
  239. $empty_table = true;
  240. }
  241. $html = '';
  242. if (!$empty_table) {
  243. $table_id = 'form_'.$this->table_name.'_id';
  244. $form = $this->get_page_select_form();
  245. $nav = $this->get_navigation_html();
  246. // Only show pagination info when there are items to paginate
  247. if ($this->get_total_number_of_items() > $this->default_items_per_page) {
  248. $html = '<table class="data_table_pagination">';
  249. $html .= '<tr>';
  250. $html .= '<td style="width:25%;">';
  251. $html .= $form;
  252. $html .= '</td>';
  253. $html .= '<td style="text-align:center;">';
  254. $html .= $this->get_table_title();
  255. $html .= '</td>';
  256. $html .= '<td style="text-align:right;width:25%;">';
  257. $html .= $nav;
  258. $html .= '</td>';
  259. $html .= '</tr>';
  260. $html .= '</table>';
  261. }
  262. if (count($this->form_actions) > 0) {
  263. $params = $this->get_sortable_table_param_string().'&amp;'.$this->get_additional_url_paramstring();
  264. $html .= '<form id ="'.$table_id.'" class="form-search" method="post" action="'.api_get_self().'?'.$params.'" name="form_'.$this->table_name.'">';
  265. }
  266. }
  267. $html .= $content;
  268. if (!$empty_table) {
  269. if (!empty($this->additional_parameters)) {
  270. foreach($this->additional_parameters as $key => $value) {
  271. $html .= '<input type="hidden" name ="'.Security::remove_XSS($key).'" value ="'.Security::remove_XSS($value).'" />';
  272. }
  273. }
  274. $html .= '<input type="hidden" name="action">';
  275. $html .= '<table style="width:100%;">';
  276. $html .= '<tr>';
  277. $html .= '<td>';
  278. if (count($this->form_actions) > 0) {
  279. $html .= '<div class="btn-toolbar">';
  280. $html .= '<div class="btn-group">';
  281. $html .= '<a class="btn" href="?'.$params.'&amp;'.$this->param_prefix.'selectall=1" onclick="javascript: setCheckbox(true, \''.$table_id.'\'); return false;">'.get_lang('SelectAll').'</a>';
  282. $html .= '<a class="btn" href="?'.$params.'" onclick="javascript: setCheckbox(false, \''.$table_id.'\'); return false;">'.get_lang('UnSelectAll').'</a> ';
  283. $html .= '</div>';
  284. $html .= '<div class="btn-group">
  285. <button class="btn" onclick="javascript:return false;">'.get_lang('Actions').'</button>
  286. <button class="btn dropdown-toggle" data-toggle="dropdown">
  287. <span class="caret"></span>
  288. </button>';
  289. $html .= '<ul class="dropdown-menu">';
  290. foreach ($this->form_actions as $action => & $label) {
  291. $html .= '<li><a data-action ="'.$action.'" href="#" onclick="javascript:action_click(this, \''.$table_id.'\');">'.$label.'</a></li>';
  292. }
  293. $html .= '</ul>';
  294. $html .= '</div>';//btn-group
  295. $html .= '</div>'; //toolbar
  296. } else {
  297. $html .= $form;
  298. }
  299. $html .= '</td>';
  300. // Pagination
  301. if ($this->get_total_number_of_items() > $this->default_items_per_page) {
  302. $html .= '<td style="text-align:right;">';
  303. $html .= $nav;
  304. $html .= '</td>';
  305. } else {
  306. $html .= '<td> ';
  307. $html .= '</td>';
  308. }
  309. $html .= '</tr>';
  310. $html .= '</table>';
  311. if (count($this->form_actions) > 0) {
  312. $html .= '</form>';
  313. }
  314. }
  315. return $html;
  316. }
  317. /**
  318. * This function shows the content of a table in a grid.
  319. * Should not be use to edit information (edit/delete rows) only.
  320. **/
  321. public function display_grid()
  322. {
  323. $empty_table = false;
  324. if ($this->get_total_number_of_items() == 0) {
  325. $cols = $this->getColCount();
  326. //$this->setCellAttributes(1, 0, 'style="font-style: italic;text-align:center;" colspan='.$cols);
  327. $message_empty = api_xml_http_response_encode(get_lang('TheListIsEmpty'));
  328. $this->setCellContents(1, 0, $message_empty);
  329. $empty_table = true;
  330. }
  331. $html = '';
  332. if (!$empty_table) {
  333. $form = $this->get_page_select_form();
  334. $nav = $this->get_navigation_html();
  335. // @todo This style css must be moved to default.css only for dev
  336. echo '<style>
  337. .main-grid { width:100%;}
  338. .sub-header { width:100%; padding-top: 10px; padding-right: 10px; padding-left: 10px; height:30px;}
  339. .grid_container { width:100%;}
  340. .grid_item { height: 120px; width:98px; float:left; padding:5px; margin:8px;}
  341. .grid_element_0 { width:100px; height: 100px; float:left; text-align:center; margin-bottom:5px;}
  342. .grid_element_1 { width:100px; float:left; text-align:center;margin-bottom:5px;}
  343. .grid_element_2 { width:150px; float:left;}
  344. .grid_selectbox { width:30%; float:left;}
  345. .grid_title { width:30%; float:left;}
  346. .grid_nav { }
  347. </style>';
  348. // @todo This also must be moved
  349. // Show only navigations if there are more than 1 page
  350. $my_pager = $this->get_pager();
  351. $html .= '<div class="main-grid">';
  352. if ($my_pager->numPages() > 1) {
  353. $html .= '<div class="sub-header">';
  354. $html .= '<div class="grid_selectbox">'.$form.'</div>';
  355. $html .= '<div class="grid_title">'.$this->get_table_title().'</div>';
  356. $html .= '<div class="grid_nav">'.$nav.'</div>';
  357. $html .= '</div>';
  358. }
  359. $html .= '<div class="clear"></div>';
  360. if (count($this->form_actions) > 0) {
  361. $script= '<script>
  362. /*<![CDATA[*/
  363. function setCheckbox(value) {
  364. d = document.form_'.$this->table_name.';
  365. for (i = 0; i < d.elements.length; i++) {
  366. if (d.elements[i].type == "checkbox") {
  367. d.elements[i].checked = value;
  368. }
  369. if (value) {
  370. $(d.elements[i]).parentsUntil("tr").parent().addClass("row_selected");
  371. } else {
  372. $(d.elements[i]).parentsUntil("tr").parent().removeClass("row_selected");
  373. }
  374. }
  375. }
  376. /*]]>*/
  377. </script>';
  378. $params = $this->get_sortable_table_param_string().'&amp;'.$this->get_additional_url_paramstring();
  379. $html .= '<form method="post" action="'.api_get_self().'?'.$params.'" name="form_'.$this->table_name.'">';
  380. }
  381. }
  382. // Getting the items of the table
  383. $items = $this->get_clean_html(false); //no sort
  384. // Generation of style classes must be improved. Maybe we need a a table name to create style on the fly:
  385. // i.e: .whoisonline_table_grid_container instead of .grid_container
  386. // where whoisonline is the table's name like drupal's template engine
  387. $html .= '<div class="grid_container">';
  388. if (is_array($items) && count($items) > 0) {
  389. foreach ($items as & $row) {
  390. $html .= '<div class="grid_item">';
  391. $i = 0;
  392. foreach ($row as & $element) {
  393. $html .= '<div class="grid_element_'.$i.'">'.$element.'</div>';
  394. $i++;
  395. }
  396. $html .= '</div>';
  397. }
  398. }
  399. $html .= '</div>'; //close grid_container
  400. $html .= '</div>'; //close main grid
  401. $html .= '<div class="clear"></div>';
  402. echo $html;
  403. }
  404. /**
  405. * This function returns the content of a table in a grid
  406. * Should not be use to edit information (edit/delete rows) only.
  407. * @param array options of visibility
  408. * @param bool hide navigation optionally
  409. * @param int content per page when show navigation (optional)
  410. * @param bool sort data optionally
  411. * @return string grid html
  412. */
  413. public function display_simple_grid($visibility_options, $hide_navigation = true, $per_page = 20, $sort_data = true, $grid_class = array())
  414. {
  415. $empty_table = false;
  416. $total = $this->get_total_number_of_items();
  417. if ($this->get_total_number_of_items() == 0) {
  418. $cols = $this->getColCount();
  419. //$this->setCellAttributes(1, 0, 'style="font-style: italic;text-align:center;" colspan='.$cols);
  420. $message_empty = api_xml_http_response_encode(get_lang('TheListIsEmpty'));
  421. $this->setCellContents(1, 0, $message_empty);
  422. $empty_table = true;
  423. }
  424. $html = '';
  425. if (!$empty_table) {
  426. // If we show the pagination
  427. if (!$hide_navigation) {
  428. $form = '&nbsp;';
  429. if ($this->get_total_number_of_items() > $per_page) {
  430. if ($per_page > 10) {
  431. $form = $this->get_page_select_form();
  432. }
  433. $nav = $this->get_navigation_html();
  434. // This also must be moved
  435. $html = '<div class="sub-header">';
  436. $html .= '<div class="grid_selectbox">'.$form.'</div>';
  437. $html .= '<div class="grid_title">'.$this->get_table_title().'</div>';
  438. $html .= '<div class="grid_nav">'.$nav.'</div>';
  439. $html .= '</div>';
  440. }
  441. }
  442. $html .= '<div class="clear"></div>';
  443. if (count($this->form_actions) > 0) {
  444. $script= '<script>
  445. /*<![CDATA[*/
  446. function setCheckbox(value) {
  447. d = document.form_'.$this->table_name.';
  448. for (i = 0; i < d.elements.length; i++) {
  449. if (d.elements[i].type == "checkbox") {
  450. d.elements[i].checked = value;
  451. }
  452. if (value) {
  453. $(d.elements[i]).parentsUntil("tr").parent().addClass("row_selected");
  454. } else {
  455. $(d.elements[i]).parentsUntil("tr").parent().removeClass("row_selected");
  456. }
  457. }
  458. }
  459. /*]]>*/
  460. </script>';
  461. $params = $this->get_sortable_table_param_string().'&amp;'.$this->get_additional_url_paramstring();
  462. $html .= '<form method="post" action="'.api_get_self().'?'.$params.'" name="form_'.$this->table_name.'">';
  463. }
  464. }
  465. if ($hide_navigation) {
  466. $items = $this->table_data; // This is a faster way to get what we want
  467. } else {
  468. // The normal way
  469. $items = $this->get_clean_html($sort_data); // Getting the items of the table
  470. }
  471. // Generation of style classes must be improved. Maybe we need a a table name to create style on the fly:
  472. // i.e: .whoisonline_table_grid_container instead of .grid_container
  473. // where whoisonline is the table's name like drupal's template engine
  474. if (is_array($visibility_options)) {
  475. $filter = false; // The 2nd condition of the if will be loaded
  476. } else {
  477. $filter = $visibility_options !== false;
  478. }
  479. $item_css_class = $item_css_style = $grid_css_class = $grid_css_style = '';
  480. if (!empty($grid_class)) {
  481. $grid_css_class = $grid_class['main']['class'];
  482. $item_css_class = $grid_class['item']['class'];
  483. $grid_css_style = isset($grid_class['main']['style']) ? $grid_class['main']['style'] : null;
  484. $item_css_style = isset($grid_class['item']['style']) ? $grid_class['item']['style'] : null;
  485. }
  486. $div = '';
  487. if (is_array($items) && count($items) > 0) {
  488. foreach ($items as & $row) {
  489. $i = 0;
  490. $rows = '';
  491. foreach ($row as & $element) {
  492. if ($filter || $visibility_options[$i]) {
  493. $rows .= '<div class="'.$this->table_name.'_grid_element_'.$i.'">'.$element.'</div>';
  494. }
  495. $i++;
  496. }
  497. $div .= Display::div($rows, array('class'=>$item_css_class.' '.$this->table_name.'_grid_item', 'style' => $item_css_style));
  498. }
  499. }
  500. $html .= Display::div($div, array('class'=>$grid_css_class.' '.$this->table_name.'_grid_container', 'style' => $grid_css_style));
  501. $html .= '<div class="clear"></div>';
  502. return $html;
  503. }
  504. /**
  505. * Get the HTML-code with the navigational buttons to browse through the
  506. * data-pages.
  507. */
  508. public function get_navigation_html()
  509. {
  510. $pager = $this->get_pager();
  511. $pager_links = $pager->getLinks();
  512. $nav = $pager_links['first'].' '.$pager_links['back'];
  513. $nav .= ' '.$pager->getCurrentPageId().' / '.$pager->numPages().' ';
  514. $nav .= $pager_links['next'].' '.$pager_links['last'];
  515. return $nav;
  516. }
  517. /**
  518. * Get the HTML-code with the data-table.
  519. */
  520. public function get_table_html()
  521. {
  522. $pager = $this->get_pager();
  523. $offset = $pager->getOffsetByPageId();
  524. $from = $offset[0] - 1;
  525. $table_data = $this->get_table_data($from);
  526. $this->processHeaders();
  527. if (is_array($table_data)) {
  528. $count = 1;
  529. foreach ($table_data as & $row) {
  530. $row = $this->filter_data($row);
  531. $newRow = array();
  532. if (!empty($this->columnsToHide)) {
  533. $counter = 0;
  534. foreach ($row as $index => $rowInfo) {
  535. if (!isset($this->columnsToHide[$index])) {
  536. $newRow[$counter] = $rowInfo ;
  537. $counter++;
  538. }
  539. }
  540. $row = $newRow;
  541. }
  542. $this->addRow($row);
  543. if (isset($row['child_of'])) {
  544. $this->setRowAttributes($count, array('class' => 'hidden hidden_'.$row['child_of']), true);
  545. }
  546. $count++;
  547. }
  548. }
  549. if ($this->odd_even_rows_enabled == true) {
  550. $this->altRowAttributes(0, array ('class' => 'row_odd'), array ('class' => 'row_even'), true);
  551. }
  552. foreach ($this->th_attributes as $column => $attributes) {
  553. $this->setCellAttributes(0, $column, $attributes);
  554. }
  555. foreach ($this->td_attributes as $column => $attributes) {
  556. $this->setColAttributes($column, $attributes);
  557. }
  558. return $this->toHTML();
  559. }
  560. /**
  561. * This function return the items of the table
  562. * @param bool true for sorting table data or false otherwise
  563. * @return array table row items
  564. */
  565. public function get_clean_html($sort = true)
  566. {
  567. $pager = $this->get_pager();
  568. $offset = $pager->getOffsetByPageId();
  569. $from = $offset[0] - 1;
  570. $table_data = $this->get_table_data($from, null, null, null, $sort);
  571. $new_table_data = array();
  572. if (is_array($table_data)) {
  573. foreach ($table_data as $index => & $row) {
  574. $row = $this->filter_data($row);
  575. $new_table_data[] = $row;
  576. }
  577. }
  578. return $new_table_data;
  579. }
  580. /**
  581. * Get the HTML-code which represents a form to select how many items a page
  582. * should contain.
  583. */
  584. public function get_page_select_form()
  585. {
  586. $total_number_of_items = $this->get_total_number_of_items();
  587. if ($total_number_of_items <= $this->default_items_per_page) {
  588. return '';
  589. }
  590. $result[] = '<form method="GET" action="'.api_get_self().'" style="display:inline;">';
  591. $param[$this->param_prefix.'direction'] = $this->direction;
  592. $param[$this->param_prefix.'page_nr'] = $this->page_nr;
  593. $param[$this->param_prefix.'column'] = $this->column;
  594. if (is_array($this->additional_parameters)) {
  595. $param = array_merge($param, $this->additional_parameters);
  596. }
  597. foreach ($param as $key => & $value) {
  598. $result[] = '<input type="hidden" name="'.$key.'" value="'.$value.'"/>';
  599. }
  600. $result[] = '<select name="'.$this->param_prefix.'per_page" onchange="javascript: this.form.submit();">';
  601. for ($nr = 10; $nr <= min(50, $total_number_of_items); $nr += 10) {
  602. $result[] = '<option value="'.$nr.'" '. ($nr == $this->per_page ? 'selected="selected"' : '').'>'.$nr.'</option>';
  603. }
  604. // @todo no limits
  605. //if ($total_number_of_items < 500) {
  606. $result[] = '<option value="'.$total_number_of_items.'" '. ($total_number_of_items == $this->per_page ? 'selected="selected"' : '').'>'.api_ucfirst(get_lang('All')).'</option>';
  607. //}
  608. $result[] = '</select>';
  609. $result[] = '<noscript>';
  610. $result[] = '<button class="btn save" type="submit">'.get_lang('Save').'</button>';
  611. $result[] = '</noscript>';
  612. $result[] = '</form>';
  613. $result = implode("\n", $result);
  614. return $result;
  615. }
  616. /**
  617. * Get the table title.
  618. */
  619. public function get_table_title()
  620. {
  621. $pager = $this->get_pager();
  622. $showed_items = $pager->getOffsetByPageId();
  623. return $showed_items[0].' - '.$showed_items[1].' / '.$this->get_total_number_of_items();
  624. }
  625. /**
  626. * @return array
  627. */
  628. public function getHeaders()
  629. {
  630. return $this->headers;
  631. }
  632. /**
  633. * Process headers
  634. */
  635. public function processHeaders()
  636. {
  637. $counter = 0;
  638. foreach ($this->headers as $column => $columnInfo) {
  639. $label = $columnInfo['label'];
  640. $sortable = $columnInfo['sortable'];
  641. $th_attributes = $columnInfo['th_attributes'];
  642. $td_attributes = $columnInfo['td_attributes'];
  643. if (!empty($this->columnsToHide)) {
  644. if (isset($this->columnsToHide[$column])) {
  645. continue;
  646. }
  647. }
  648. $column = $counter;
  649. $param['direction'] = 'ASC';
  650. if ($this->column == $column && $this->direction == 'ASC') {
  651. $param['direction'] = 'DESC';
  652. }
  653. $param['page_nr'] = $this->page_nr;
  654. $param['per_page'] = $this->per_page;
  655. $param['column'] = $column;
  656. if ($sortable) {
  657. $link = '<a href="'.api_get_self().'?'.api_get_cidreq().'&amp;';
  658. foreach ($param as $key => & $value) {
  659. $link .= $this->param_prefix.$key.'='.urlencode($value).'&amp;';
  660. }
  661. $link .= $this->get_additional_url_paramstring();
  662. $link .= '">'.$label.'</a>';
  663. if ($this->column == $column) {
  664. $link .= $this->direction == 'ASC' ? ' &#8595;' : ' &#8593;';
  665. }
  666. } else {
  667. $link = $label;
  668. }
  669. $this->setHeaderContents(0, $column, $link);
  670. if (!is_null($td_attributes)) {
  671. $this->td_attributes[$column] = $td_attributes;
  672. }
  673. if (!is_null($th_attributes)) {
  674. $this->th_attributes[$column] = $th_attributes;
  675. }
  676. $counter++;
  677. }
  678. }
  679. /**
  680. * Set the header-label
  681. * @param int $column The column number
  682. * @param string $label The label
  683. * @param boolean $sortable Is the table sortable by this column? (defatult
  684. * = true)
  685. * @param string $th_attributes Additional attributes for the th-tag of the
  686. * table header
  687. * @param string $td_attributes Additional attributes for the td-tags of the
  688. * column
  689. */
  690. public function set_header($column, $label, $sortable = true, $th_attributes = null, $td_attributes = null)
  691. {
  692. $this->headers[$column] = array(
  693. 'label' => $label,
  694. 'sortable' => $sortable,
  695. 'th_attributes' => $th_attributes,
  696. 'td_attributes' => $td_attributes,
  697. );
  698. }
  699. /**
  700. * Get the parameter-string with additional parameters to use in the URLs
  701. * generated by this SortableTable
  702. */
  703. public function get_additional_url_paramstring()
  704. {
  705. $param_string_parts = array ();
  706. if (is_array($this->additional_parameters) && count($this->additional_parameters) > 0) {
  707. foreach ($this->additional_parameters as $key => & $value) {
  708. $param_string_parts[] = urlencode($key).'='.urlencode($value);
  709. }
  710. }
  711. $result = implode('&amp;', $param_string_parts);
  712. foreach ($this->other_tables as $index => & $tablename) {
  713. $param = array();
  714. if (isset($_GET[$tablename.'_direction'])) {
  715. //$param[$tablename.'_direction'] = $_GET[$tablename.'_direction'];
  716. $my_get_direction = $_GET[$tablename.'_direction'];
  717. if (!in_array($my_get_direction, array('ASC', 'DESC'))) {
  718. $param[$tablename.'_direction'] = 'ASC';
  719. } else {
  720. $param[$tablename.'_direction'] = $my_get_direction;
  721. }
  722. }
  723. if (isset($_GET[$tablename.'_page_nr'])) {
  724. $param[$tablename.'_page_nr'] = intval($_GET[$tablename.'_page_nr']);
  725. }
  726. if (isset($_GET[$tablename.'_per_page'])) {
  727. $param[$tablename.'_per_page'] = intval($_GET[$tablename.'_per_page']);
  728. }
  729. if (isset($_GET[$tablename.'_column'])) {
  730. $param[$tablename.'_column'] = intval($_GET[$tablename.'_column']);
  731. }
  732. $param_string_parts = array ();
  733. foreach ($param as $key => & $value) {
  734. $param_string_parts[] = urlencode($key).'='.urlencode($value);
  735. }
  736. if (count($param_string_parts) > 0)
  737. $result .= '&amp;'.implode('&amp;', $param_string_parts);
  738. }
  739. return $result;
  740. }
  741. /**
  742. * Get the parameter-string with the SortableTable-related parameters to use
  743. * in URLs
  744. */
  745. public function get_sortable_table_param_string()
  746. {
  747. $param[$this->param_prefix.'direction'] = $this->direction;
  748. $param[$this->param_prefix.'page_nr'] = $this->page_nr;
  749. $param[$this->param_prefix.'per_page'] = $this->per_page;
  750. $param[$this->param_prefix.'column'] = $this->column;
  751. $param_string_parts = array ();
  752. foreach ($param as $key => & $value) {
  753. $param_string_parts[] = urlencode($key).'='.urlencode($value);
  754. }
  755. $res = implode('&amp;', $param_string_parts);
  756. return $res;
  757. }
  758. /**
  759. * Add a filter to a column. If another filter was allready defined for the
  760. * given column, it will be overwritten.
  761. * @param int $column The number of the column
  762. * @param string $function The name of the filter-function. This should be a
  763. * function wich requires 1 parameter and returns the filtered value.
  764. */
  765. public function set_column_filter($column, $function)
  766. {
  767. $this->column_filters[$column] = $function;
  768. }
  769. /**
  770. * List of columns to hide
  771. * @param int $column
  772. */
  773. public function setHideColumn($column)
  774. {
  775. $this->columnsToHide[$column] = $column;
  776. }
  777. /**
  778. * Define a list of actions which can be performed on the table-date.
  779. * If you define a list of actions, the first column of the table will be
  780. * converted into checkboxes.
  781. * @param array $actions A list of actions. The key is the name of the
  782. * action. The value is the label to show in the select-box
  783. * @param string $checkbox_name The name of the generated checkboxes. The
  784. * value of the checkbox will be the value of the first column.
  785. */
  786. public function set_form_actions($actions, $checkbox_name = 'id')
  787. {
  788. $this->form_actions = $actions;
  789. $this->checkbox_name = $checkbox_name;
  790. }
  791. /**
  792. * Define a list of additional parameters to use in the generated URLs
  793. * <code>$parameters['action'] = 'test'; will be convert in <input type="hidden" name="action" value="test"></code>
  794. * @param array $parameters
  795. */
  796. public function set_additional_parameters($parameters)
  797. {
  798. $this->additional_parameters = $parameters;
  799. }
  800. /**
  801. * Set other tables on the same page.
  802. * If you have other sortable tables on the page displaying this sortable
  803. * tables, you can define those other tables with this function. If you
  804. * don't define the other tables, there sorting and pagination will return
  805. * to their default state when sorting this table.
  806. * @param array $tablenames An array of table names.
  807. */
  808. public function set_other_tables($tablenames)
  809. {
  810. $this->other_tables = $tablenames;
  811. }
  812. /**
  813. * Transform all data in a table-row, using the filters defined by the
  814. * function set_column_filter(...) defined elsewhere in this class.
  815. * If you've defined actions, the first element of the given row will be
  816. * converted into a checkbox
  817. * @param array $row A row from the table.
  818. */
  819. public function filter_data($row)
  820. {
  821. $url_params = $this->get_sortable_table_param_string().'&amp;'.$this->get_additional_url_paramstring();
  822. foreach ($this->column_filters as $column => & $function) {
  823. $row[$column] = call_user_func($function, $row[$column], $url_params, $row);
  824. }
  825. if (count($this->form_actions) > 0) {
  826. if (strlen($row[0]) > 0) {
  827. $row[0] = '<input type="checkbox" name="'.$this->checkbox_name.'[]" value="'.$row[0].'"';
  828. if (isset($_GET[$this->param_prefix.'selectall'])) {
  829. $row[0] .= ' checked="checked"';
  830. }
  831. $row[0] .= '/>';
  832. }
  833. }
  834. if (is_array($row)) {
  835. foreach ($row as & $value) {
  836. if (empty($value)) {
  837. $value = '-';
  838. }
  839. }
  840. }
  841. return $row;
  842. }
  843. /**
  844. * Get the total number of items. This function calls the function given as
  845. * 2nd argument in the constructor of a SortableTable. Make sure your
  846. * function has the same parameters as defined here.
  847. */
  848. public function get_total_number_of_items()
  849. {
  850. if ($this->total_number_of_items == -1 && !is_null($this->get_total_number_function)) {
  851. $this->total_number_of_items = call_user_func($this->get_total_number_function);
  852. }
  853. return $this->total_number_of_items;
  854. }
  855. /**
  856. * Get the data to display. This function calls the function given as
  857. * 2nd argument in the constructor of a SortableTable. Make sure your
  858. * function has the same parameters as defined here.
  859. * @param int $from Index of the first item to return.
  860. * @param int $per_page The number of items to return
  861. * @param int $column The number of the column on which the data should be
  862. * @param bool $sort Whether to sort or not
  863. * sorted
  864. * @param string $direction In which order should the data be sorted (ASC
  865. * or DESC)
  866. */
  867. public function get_table_data($from = null, $per_page = null, $column = null, $direction = null, $sort = null)
  868. {
  869. if (!is_null($this->get_data_function)) {
  870. return call_user_func($this->get_data_function, $from, $this->per_page, $this->column, $this->direction);
  871. }
  872. return array();
  873. }
  874. }
  875. /**
  876. * Sortable table which can be used for data available in an array
  877. * @package chamilo.library
  878. */
  879. class SortableTableFromArray extends SortableTable
  880. {
  881. /**
  882. * The array containing all data for this table
  883. */
  884. public $table_data;
  885. /**
  886. * Constructor
  887. * @param array $table_data
  888. * @param int $default_column
  889. * @param int $default_items_per_page
  890. */
  891. public function __construct(
  892. $table_data,
  893. $default_column = 1,
  894. $default_items_per_page = 20,
  895. $tablename = 'tablename',
  896. $get_total_number_function = null
  897. ) {
  898. parent:: __construct(
  899. $tablename,
  900. $get_total_number_function,
  901. null,
  902. $default_column,
  903. $default_items_per_page
  904. );
  905. $this->table_data = $table_data;
  906. }
  907. /**
  908. * Get table data to show on current page
  909. * @see SortableTable#get_table_data
  910. */
  911. public function get_table_data($from = 1, $per_page = null, $column = null, $direction = null, $sort = true)
  912. {
  913. if ($sort) {
  914. $content = TableSort::sort_table($this->table_data, $this->column, $this->direction == 'ASC' ? SORT_ASC : SORT_DESC);
  915. } else {
  916. $content = $this->table_data;
  917. }
  918. return array_slice($content, $from, $this->per_page);
  919. }
  920. /**
  921. * Get total number of items
  922. * @see SortableTable#get_total_number_of_items
  923. */
  924. public function get_total_number_of_items()
  925. {
  926. if (isset($this->total_number_of_items) && !empty($this->total_number_of_items)) {
  927. return $this->total_number_of_items;
  928. } else {
  929. return count($this->table_data);
  930. }
  931. }
  932. }
  933. /**
  934. * Sortable table which can be used for data available in an array
  935. *
  936. * Is a variation of SortableTableFromArray because we add 2 new arrays $column_show and $column_order
  937. * $column_show is an array that lets us decide which are going to be the columns to show
  938. * $column_order is an array that lets us decide the ordering of the columns
  939. * i.e: $column_header=array('a','b','c','d','e'); $column_order=array(1,2,5,4,5);
  940. * These means that the 3th column (letter "c") will be sort like the order we use in the 5th column
  941. * @package chamilo.library
  942. */
  943. class SortableTableFromArrayConfig extends SortableTable
  944. {
  945. /**
  946. * The array containing the columns that will be show i.e $column_show=array('1','0','0'); we will show only the 1st column
  947. */
  948. private $column_show;
  949. /**
  950. *The array containing the real sort column $column_order=array('1''4','3','4'); The 2nd column will be order like the 4th column
  951. */
  952. private $column_order;
  953. /**
  954. * The array containing all data for this table
  955. */
  956. private $table_data;
  957. private $doc_filter;
  958. /**
  959. * Constructor
  960. * @param array $table_data All the information of the table
  961. * @param int $default_column Default column that will be use in the sorts functions
  962. * @param int $default_items_per_page quantity of pages that we are going to see
  963. * @param int $tablename Name of the table
  964. * @param array $column_show An array with binary values 1: we show the column 2: we don't show it
  965. * @param array $column_order An array of integers that let us decide how the columns are going to be sort.
  966. * @param string $direction
  967. * @param bool $doc_filter special modification to fix the document name order
  968. */
  969. public function __construct(
  970. $table_data,
  971. $default_column = 1,
  972. $default_items_per_page = 20,
  973. $tablename = 'tablename',
  974. $column_show = array(),
  975. $column_order = array(),
  976. $direction = 'ASC',
  977. $doc_filter = false
  978. ) {
  979. $this->column_show = $column_show;
  980. $this->column_order = $column_order;
  981. $this->doc_filter = $doc_filter;
  982. parent::__construct($tablename, null, null, $default_column, $default_items_per_page, $direction);
  983. $this->table_data = $table_data;
  984. }
  985. /**
  986. * Get table data to show on current page
  987. * @see SortableTable#get_table_data
  988. */
  989. public function get_table_data(
  990. $from = 1,
  991. $per_page = null,
  992. $column = null,
  993. $direction = null,
  994. $sort = true
  995. ) {
  996. $content = TableSort::sort_table_config(
  997. $this->table_data,
  998. $this->column,
  999. $this->direction == 'ASC' ? SORT_ASC : SORT_DESC,
  1000. $this->column_show,
  1001. $this->column_order,
  1002. SORT_REGULAR,
  1003. $this->doc_filter
  1004. );
  1005. return array_slice($content, $from, $this->per_page);
  1006. }
  1007. /**
  1008. * Get total number of items
  1009. * @see SortableTable#get_total_number_of_items
  1010. */
  1011. public function get_total_number_of_items()
  1012. {
  1013. return count($this->table_data);
  1014. }
  1015. }