| [ Index ] |
|
Code source de eGroupWare 1.2.106-2 |
1 <?php 2 /**************************************************************************\ 3 * eGroupWare - ProjectManager - UI list and edit projects-elements * 4 * http://www.egroupware.org * 5 * Written and (c) 2005 by Ralf Becker <RalfBecker@outdoor-training.de> * 6 * -------------------------------------------- * 7 * This program is free software; you can redistribute it and/or modify it * 8 * under the terms of the GNU General Public License as published by the * 9 * Free Software Foundation; either version 2 of the License, or (at your * 10 * option) any later version. * 11 \**************************************************************************/ 12 13 /* $Id: class.uiprojectelements.inc.php 20774 2006-03-21 17:16:45Z ralfbecker $ */ 14 15 include_once (EGW_INCLUDE_ROOT.'/projectmanager/inc/class.boprojectelements.inc.php'); 16 17 /** 18 * ProjectManage UI: list and edit projects-elements 19 * 20 * @package projectmanager 21 * @author RalfBecker-AT-outdoor-training.de 22 * @copyright (c) 2005 by RalfBecker-AT-outdoor-training.de 23 * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License 24 */ 25 class uiprojectelements extends boprojectelements 26 { 27 /** 28 * @var array $public_functions Functions to call via menuaction 29 */ 30 var $public_functions = array( 31 'index' => true, 32 'edit' => true, 33 'view' => true, 34 ); 35 /** 36 * @var etemplate-object $tpl instance of the etemplate class 37 */ 38 var $tpl; 39 /** 40 * @var array $status_labels labels for status-filter 41 */ 42 var $status_labels; 43 /** 44 * @var array $config config-settings for projectmanager 45 */ 46 var $config; 47 48 /** 49 * Constructor, calls the constructor of the extended class 50 */ 51 function uiprojectelements() 52 { 53 $this->tpl =& CreateObject('etemplate.etemplate'); 54 55 if ((int) $_REQUEST['pm_id']) 56 { 57 $pm_id = (int) $_REQUEST['pm_id']; 58 // store the current project (only for index, as popups may be called by other parent-projects) 59 } 60 else 61 { 62 $pm_id = $GLOBALS['egw']->session->appsession('pm_id','projectmanager'); 63 } 64 if (!$pm_id) 65 { 66 $this->tpl->location(array( 67 'menuaction' => 'projectmanager.uiprojectmanager.index', 68 'msg' => lang('You need to select a project first'), 69 )); 70 } 71 $this->boprojectelements($pm_id); 72 73 // check if we have at least read-access to this project 74 if (!$this->project->check_acl(EGW_ACL_READ)) 75 { 76 $this->tpl->location(array( 77 'menuaction' => 'projectmanager.uiprojectmanager.index', 78 'msg' => lang('Permission denied !!!'), 79 )); 80 } 81 82 $this->status_labels = array( 83 'all' => lang('all'), 84 'used' => lang('used'), 85 'new' => lang('new'), 86 'ignored' => lang('ignored'), 87 ); 88 } 89 90 91 /** 92 * View a project-element, just calls edit with view-param set 93 */ 94 function view() 95 { 96 $this->edit(null,true); 97 } 98 99 /** 100 * Edit or view a project-element 101 * 102 * @var array $content content-array if called by process-exec 103 * @var boolean $view only view project, default false, only used on first call !is_array($content) 104 */ 105 function edit($content=null,$view=false) 106 { 107 if ((int) $this->debug >= 1 || $this->debug == 'edit') $this->debug_message("uiprojectelements::edit(".print_r($content,true).",$view)"); 108 109 if (is_array($content)) 110 { 111 $this->data = $content['data']; 112 $update_necessary = $save_necessary = 0; 113 $datasource = $this->datasource($this->data['pe_app']); 114 if (($ds_read_from_element = !($ds = $datasource->read($this->data['pe_app_id'],$this->data)))) 115 { 116 $ds = $datasource->element_values($this->data); 117 } 118 if (!$content['view']) 119 { 120 if ($content['pe_completion'] !== '') $content['pe_completion'] .= '%'; 121 122 foreach($datasource->name2id as $name => $id) 123 { 124 //echo "checking $name=$id (overwrite={$this->data['pe_overwrite']}&$id == ".($this->data['pe_overwrite']&$id?'true':'false')."), content='{$content[$name]}'<br>\n"; 125 // check if update is necessary, because a field has be set or changed 126 if ($content[$name] && ($content[$name] != $this->data[$name] || !($this->data['pe_overwrite'] & $id))) 127 { 128 //echo "need to update $name as content[$name] changed to '".$content[$name]."' != '".$this->data[$name]."'<br>\n"; 129 $this->data[$name] = $content[$name]; 130 $this->data['pe_overwrite'] |= $id; 131 $update_necessary |= $id; 132 } 133 // check if a field is no longer set, or it's not set and datasource changed 134 // => set it from the datasource 135 elseif (($this->data['pe_overwrite'] & $id) && !$content[$name] || 136 !($this->data['pe_overwrite'] & $id) && (int)$this->data[$name] != (int)$ds[$name]) 137 { 138 //echo "need to update $name as content[$name] is unset or datasource changed cont='".$content[$name]."', data='".$this->data[$name]."', ds='".$ds[$name]."'<br>\n"; 139 // if we have a change in the datasource, set pe_synced 140 if ($this->data[$name] != $ds[$name]) 141 { 142 $this->data['pe_synced'] = $this->now_su; 143 } 144 $this->data[$name] = $ds[$name]; 145 $this->data['pe_overwrite'] &= ~$id; 146 $update_necessary |= $id; 147 } 148 } 149 $content['cat_id'] = (int) $content['cat_id']; // as All='' and cat_id column is int 150 151 // calculate the new summary and if a percentage give the share in hours 152 //echo "<p>project_summary[pe_total_shares]={$this->project_summary['pe_total_shares']}, old_pe_share={$content['old_pe_share']}, old_default_share=$content[old_default_share], content[pe_share]={$content['pe_share']}</p>\n"; 153 $planned_time = $content['pe_planned_time'] ? $content['pe_planned_time'] : $ds['pe_planned_time']; 154 $default_share = $planned_time && $this->project->data['pm_accounting_type'] != 'status' ? $planned_time : $this->default_share; 155 156 $this->project_summary['pe_total_shares'] -= round((string) $content['old_pe_share'] !== '' ? $content['old_pe_share'] : $content['old_default_share']); 157 if (substr($content['pe_share'],-1) == '%' || $this->project->data['pm_accounting_type'] == 'status') 158 { 159 //echo "<p>project_summary[pe_total_shares]={$this->project_summary['pe_total_shares']}</p>\n"; 160 if ((float) $content['pe_share'] == 100 || !$this->project_summary['pe_total_shares']) 161 { 162 $content['pe_share'] = $this->default_share; 163 } 164 else 165 { 166 $content['pe_share'] = round($this->project_summary['pe_total_shares'] * (float) $content['pe_share'] / 167 (100 - (float) $content['pe_share']),1); 168 } 169 //echo "<p>pe_share={$content['pe_share']}</p>\n"; 170 } 171 $this->project_summary['pe_total_shares'] += round((string) $content['pe_share'] !== '' ? $content['pe_share'] : $default_share); 172 //echo "<p>project_summary[pe_total_shares]={$this->project_summary['pe_total_shares']}, default_share=$default_share, content[pe_share]={$content['pe_share']}</p>\n"; 173 174 foreach(array('pe_status','cat_id','pe_remark','pe_constraints','pe_share') as $name) 175 { 176 if ($name == 'pe_constraints') 177 { 178 foreach($content['pe_constraints'] as $type => $data) 179 { 180 if (!$data) 181 { 182 unset($content['pe_constraints'][$type]); // ignore not set constraints 183 } 184 elseif (!is_array($data)) 185 { 186 $content['pe_constraints'][$type] = explode(',',$data); // otherwise it's detected as change 187 } 188 } 189 } 190 if ($content[$name] != $this->data[$name] || 191 $name == 'pe_share' && $content[$name] !== $this->data[$name]) // for pe_share we differ between 0 and empty! 192 { 193 //echo "need to update $name as content[$name] changed to '".print_r($content[$name],true)."' != '".print_r($this->data[$name],true)."'<br>\n"; 194 $this->data[$name] = $content[$name]; 195 $save_necessary = true; 196 197 if ($name == 'pe_remark') $this->data['update_remark'] = true; 198 } 199 } 200 } 201 //echo "uiprojectelements::edit(): save_necessary=".(int)$save_necessary.", update_necessary=$update_necessary, data="; _debug_array($this->data); 202 203 $view = $content['view'] && !($content['edit'] && $this->check_acl(EGW_ACL_EDIT)); 204 205 if (($content['save'] || $content['apply']) && $this->check_acl(EGW_ACL_EDIT)) 206 { 207 if ($update_necessary || $save_necessary) 208 { 209 if ($this->save(null,true,$update_necessary) != 0) 210 { 211 $msg = lang('Error: saving the project-element (%1) !!!',$this->db->Error); 212 unset($content['save']); // dont exit 213 } 214 else 215 { 216 $msg = lang('Project-Element saved'); 217 $js = "opener.location.href='".$GLOBALS['egw']->link('/index.php',array( 218 'menuaction' => $content['caller'] ? $content['caller'] : 'projectmanager.uiprojectelements.index', 219 'msg' => $msg, 220 ))."';"; 221 } 222 } 223 else 224 { 225 $msg = lang('no save necessary'); 226 } 227 } 228 if ($content['delete'] && $this->check_acl(EGW_ACL_DELETE)) 229 { 230 // all delete are done by index 231 $js = "opener.location.href='".$GLOBALS['egw']->link('/index.php',array( 232 'menuaction' => $content['caller'] ? $content['caller'] : 'projectmanager.uiprojectelements.index', 233 'delete' => $this->data['pe_id'], 234 ))."';"; 235 /* 236 return $this->index(array('nm'=>array('rows'=>array( 237 'delete' => array($this->data['pe_id']=>true) 238 )))); 239 */ 240 } 241 if ($content['save'] || $content['cancel'] || $content['delete']) 242 { 243 $js .= 'window.close();'; 244 echo '<html><body onload="'.$js.'"></body></html>'; 245 $GLOBALS['egw']->common->egw_exit(); 246 /* 247 $this->tpl->location(array( 248 'menuaction' => 'projectmanager.uiprojectelements.index', 249 'msg' => $msg, 250 )); 251 */ 252 } 253 } 254 else 255 { 256 if ((int) $_GET['pe_id']) 257 { 258 $this->read((int) $_GET['pe_id']); 259 } 260 if ($this->data['pe_id']) 261 { 262 if (!$this->check_acl(EGW_ACL_READ)) 263 { 264 $this->tpl->location(array( 265 'menuaction' => 'projectmanager.uiprojectelements.index', 266 'msg' => lang('Permission denied !!!'), 267 )); 268 } 269 if (!$this->check_acl(EGW_ACL_EDIT)) $view = true; 270 } 271 $js = 'window.focus();'; 272 273 $datasource = $this->datasource($this->data['pe_app']); 274 if (($ds_read_from_element = !($ds = $datasource->read($this->data['pe_app_id'],$this->data)))) 275 { 276 $ds = $datasource->element_values($this->data); 277 } 278 else 279 { 280 $this->data['pe_title'] = $ds['pe_title']; // updating the title, not all datasources do it automatic 281 } 282 } 283 $planned_time = $this->data['pe_planned_time'] ? $this->data['pe_planned_time'] : $ds['pe_planned_time']; 284 $default_share = $planned_time && $this->project->data['pm_accounting_type'] != 'status' ? $planned_time : $this->default_share; 285 286 if ($ds_read_from_element && !$view) 287 { 288 $msg .= lang('No READ access to the datasource: removing overwritten values will just empty them !!!'); 289 } 290 $preserv = $this->data + array( 291 'view' => $view, 292 'data' => $this->data, 293 'caller' => !$content['caller'] && preg_match('/menuaction=([^&]+)/',$_SERVER['HTTP_REFERER'],$matches) ? 294 $matches[1] : $content['caller'], 295 'old_pe_share' => $this->data['pe_share'], 296 'old_default_share' => $default_share, 297 ); 298 unset($preserv['pe_resources']); // otherwise we cant detect no more resources set 299 300 foreach($datasource->name2id as $name => $id) 301 { 302 if ($id != PM_TITLE && !($this->data['pe_overwrite'] & $id)) // empty not explicitly set values 303 { 304 $this->data[$name] = ''; 305 } 306 } 307 $js .= "\nfunction calc_budget(form) { 308 form['exec[pe_used_budget]'].value = form['exec[pe_used_quantity]'].value * form['exec[pe_unitprice]'].value; 309 if (form['exec[pe_used_budget]'].value == '0') form['exec[pe_used_budget]'].value = ''; 310 form['exec[pe_planned_budget]'].value = form['exec[pe_planned_quantity]'].value * form['exec[pe_unitprice]'].value; 311 if (form['exec[pe_planned_budget]'].value == '0') form['exec[pe_planned_budget]'].value = ''; 312 }"; 313 $tabs = 'dates|times|budget|constraints|resources|details'; 314 $content = $this->data + array( 315 'ds' => $ds, 316 'msg' => $msg, 317 'js' => '<script>'.$js.'</script>', 318 'default_share' => $default_share, 319 'duration_format' => $this->config['duration_format'], 320 'no_times' => $this->project->data['pm_accounting_type'] == 'status', 321 $tabs => $content[$tabs], 322 'no_pricelist' => $this->project->data['pm_accounting_type'] != 'pricelist', 323 'planned_quantity_blur' => $this->data['pe_planned_time'] ? $this->data['pe_planned_time'] / 60 : $ds['pe_planned_quantity'], 324 'used_quantity_blur' => $this->data['pe_used_time'] ? $this->data['pe_used_time'] / 60 : $ds['pe_used_quantity'], 325 ); 326 // calculate percentual shares 327 $content['default_total'] = $content['share_total'] = $this->project_summary['pe_total_shares']; 328 if ((string) $this->data['pe_share'] !== '') 329 { 330 if ($this->project_summary['pe_total_shares']) 331 { 332 $content['share_percentage'] = round(100.0 * $this->data['pe_share'] / $this->project_summary['pe_total_shares'],1) . '%'; 333 } 334 $content['default_total'] += $default_share - $this->data['pe_share']; 335 } 336 if ($content['default_total']) 337 { 338 $content['default_percentage'] = round(100.0 * $default_share / $content['default_total'],1) . '%'; 339 } 340 if ($this->project->data['pm_accounting_type'] == 'status') 341 { 342 $content['pe_share'] = $content['share_percentage']; 343 } 344 //_debug_array($content); 345 $sel_options = array( 346 'pe_constraints' => $this->titles(array( // only titles of elements displayed in a gantchart 347 "pe_status != 'ignore'", 348 // '(pe_planned_start IS NOT NULL OR pe_real_start IS NOT NULL)', 349 // '(pe_planned_end IS NOT NULL OR pe_real_end IS NOT NULL)', 350 'pe_id != '.(int)$this->data['pe_id'], // dont show own title 351 )), 352 'milestone' => $this->milestones->titles(array('pm_id' => $this->data['pm_id'])), 353 ); 354 $readonlys = array( 355 'delete' => !$this->data['pe_id'] || !$this->check_acl(EGW_ACL_DELETE), 356 'edit' => !$view || !$this->check_acl(EGW_ACL_EDIT), 357 ); 358 // disable the times tab, if accounting-type status 359 $readonlys[$tabs]['times'] = $this->project->data['pm_accounting_type'] == 'status'; 360 // check if user has the necessary rights to view or edit the budget 361 $readonlys[$tabs]['budget'] = !$this->check_acl(EGW_ACL_BUDGET); 362 if (!$this->check_acl(EGW_ACL_EDIT_BUDGET)) 363 { 364 foreach(array('pe_planned_budget','pe_used_budget','pl_id','pe_unitprice','pe_planned_quantity','pe_used_quantity') as $key) 365 { 366 $readonlys[$key] = true; 367 } 368 } 369 if ($view) 370 { 371 foreach($this->db_cols as $name) 372 { 373 $readonlys[$name] = true; 374 } 375 $readonlys['pe_remark'] = true; 376 $readonlys['save'] = $readonlys['apply'] = true; 377 $readonlys['pe_constraints[start]'] = $readonlys['pe_constraints[end]'] = $readonlys['pe_constraints[milestone]'] = true; 378 } 379 $GLOBALS['egw_info']['flags']['app_header'] = lang('projectmanager') . ' - ' . 380 ($this->data['pm_id'] ? ($view ? lang('View project-elements') : lang('Edit project-elements')) : lang('Add project-elements')); 381 $this->tpl->read('projectmanager.elements.edit'); 382 $this->tpl->exec('projectmanager.uiprojectelements.edit',$content,$sel_options,$readonlys,$preserv,2); 383 } 384 385 /** 386 * query projects for nextmatch in the projects-list 387 * 388 * reimplemented from so_sql to disable action-buttons based on the acl and make some modification on the data 389 * 390 * @param array $query 391 * @param array &$rows returned rows/cups 392 * @param array &$readonlys eg. to disable buttons based on acl 393 */ 394 function get_rows(&$query,&$rows,&$readonlys) 395 { 396 $GLOBALS['egw']->session->appsession('projectelements_list','projectmanager',$query); 397 398 if ($this->status_filter[$query['filter']]) 399 { 400 $query['col_filter']['pe_status'] = $this->status_filter[$query['filter']]; 401 } 402 else 403 { 404 unset($query['col_filter']['pe_status']); 405 } 406 if ($query['cat_id']) 407 { 408 $query['col_filter']['cat_id'] = $query['cat_id']; 409 $query['link_add']['extra'] = array('cat_id' => $query['cat_id']); 410 } 411 else 412 { 413 unset($query['col_filter']['cat_id']); 414 unset($query['link_add']['extra']); 415 } 416 if (!$query['col_filter']['pe_resources']) 417 { 418 unset($query['col_filter']['pe_resources']); 419 } 420 $total = parent::get_rows($query,$rows,$readonlys,true); 421 422 // adding the project itself always as first line 423 $self = $this->update('projectmanager',$this->pm_id); 424 $self['pe_app'] = 'projectmanager'; 425 $self['pe_app_id'] = $this->pm_id; 426 $self['pe_icon'] = 'projectmanager/navbar'; 427 $self['pe_modified'] = $this->project->data['pm_modified']; 428 $self['pe_modifier'] = $this->project->data['pm_modifier']; 429 $rows = array_merge(array($self),$rows); 430 431 $readonlys = array(); 432 foreach($rows as $n => $val) 433 { 434 $row =& $rows[$n]; 435 if ($n && !$this->check_acl(EGW_ACL_EDIT,$row)) 436 { 437 $readonlys["edit[$row[pe_id]]"] = true; 438 } 439 if ($n && !$this->check_acl(EGW_ACL_DELETE,$row)) 440 { 441 $readonlys["delete[$row[pe_id]]"] = true; 442 } 443 if (!$n) 444 { 445 // no link for own project 446 if (!$this->project->check_acl(EGW_ACL_EDIT,$this->project->data)) 447 { 448 $readonlys['edit'] = true; 449 } 450 } 451 else 452 { 453 $row['link'] = array( 454 'app' => $row['pe_app'], 455 'id' => $row['pe_app_id'], 456 'title'=> $row['pe_title'], 457 'help' => $row['pe_app'] == 'projectmanager' ? lang("Select this project and show it's elements") : 458 lang('View this element in %1',lang($row['pe_app'])), 459 ); 460 } 461 if (!$query['filter2']) unset($row['pe_details']); 462 463 $row['pe_completion_icon'] = $row['pe_completion'] == 100 ? 'done' : $row['pe_completion']; 464 465 $custom_app_icons[$row['pe_app']][] = $row['pe_app_id']; 466 } 467 if ($GLOBALS['egw_info']['user']['preferences']['projectmanager']['show_custom_app_icons']) 468 { 469 $custom_app_icons['location'] = 'pm_custom_app_icons'; 470 $custom_app_icons = $GLOBALS['egw']->hooks->process($custom_app_icons); 471 unset($row); // it's used as reference before !!! 472 foreach($rows as $n => $row) 473 { 474 if (isset($custom_app_icons[$row['pe_app']][$row['pe_app_id']])) 475 { 476 $rows[$n]['pe_completion_icon'] = $custom_app_icons[$row['pe_app']][$row['pe_app_id']]; 477 } 478 } 479 } 480 $rows['no_budget'] = !$this->project->check_acl(EGW_ACL_BUDGET); 481 $rows['no_times'] = $this->project->data['pm_accounting_type'] == 'status'; 482 $rows['duration_format'] = ','.$this->config['duration_format'].',,1'; 483 484 // calculate the filter-specific summary if we have a filter, beside the default pe_status=used=array(new,regular) 485 if (count($query['col_filter']) > (int)isset($query['col_filter']['pe_status']) || !is_array($query['col_filter']['pe_status'])) 486 { 487 $rows += $this->summary(null,$query['col_filter']); 488 } 489 if ((int)$this->debug >= 2 || $this->debug == 'get_rows') 490 { 491 $this->debug_message("uiprojectelements::get_rows(".print_r($query,true).") total=$total, rows =".print_r($rows,true)."readonlys=".print_r($readonlys,true)); 492 } 493 return $total; 494 } 495 496 /** 497 * List existing projects-elements 498 * 499 * @param array $content=null 500 * @param string $msg='' 501 */ 502 function index($content=null,$msg='') 503 { 504 if ((int) $this->debug >= 1 || $this->debug == 'index') $this->debug_message("uiprojectelements::index(".print_r($content,true).",$msg)"); 505 506 // store the current project (only for index, as popups may be called by other parent-projects) 507 $GLOBALS['egw']->session->appsession('pm_id','projectmanager',$this->project->data['pm_id']); 508 509 if ($_GET['msg']) $msg = $_GET['msg']; 510 511 if ($content['nm']['rows']['edit']) 512 { 513 $this->tpl->location(array( 514 'menuaction' => 'projectmanager.uiprojectmanager.edit', 515 'pm_id' => $this->pm_id, 516 )); 517 } 518 elseif ($content['sync_all'] && $this->project->check_acl(EGW_ACL_ADD)) 519 { 520 $msg = lang('%1 element(s) updated',$this->sync_all()); 521 } 522 elseif((int) $_GET['delete'] || $content['nm']['rows']['delete']) 523 { 524 if ($content['nm']['rows']['delete']) 525 { 526 list($pe_id) = each($content['nm']['rows']['delete']); 527 } 528 else 529 { 530 $pe_id = (int) $_GET['delete']; 531 } 532 if ($this->read($pe_id) && !$this->check_acl(EGW_ACL_DELETE)) 533 { 534 $msg = lang('Permission denied !!!'); 535 } 536 else 537 { 538 $msg = $this->delete($pe_id) ? lang('Project-Element deleted') : 539 lang('Error: deleting project-element !!!'); 540 } 541 } 542 $content = array( 543 'nm' => $GLOBALS['egw']->session->appsession('projectelements_list','projectmanager'), 544 'msg' => $msg, 545 ); 546 if (!is_array($content['nm'])) 547 { 548 $content['nm'] = array( 549 'get_rows' => 'projectmanager.uiprojectelements.get_rows', 550 'filter' => 'used',// I initial value for the filter 551 'filter_label' => lang('Filter'),// I label for filter (optional) 552 'options-filter' => $this->status_labels, 553 'filter_no_lang' => True,// I set no_lang for filter (=dont translate the options) 554 'options-filter2' => array('no details','details'), 555 // 'bottom_too' => True,// I show the nextmatch-line (arrows, filters, search, ...) again after the rows 556 'order' => 'pe_modified',// IO name of the column to sort after (optional for the sortheaders) 557 'sort' => 'DESC',// IO direction of the sort: 'ASC' or 'DESC' 558 ); 559 } 560 // add "buttons" only with add-rights 561 if ($this->project->check_acl(EGW_ACL_ADD)) 562 { 563 $content['nm']['header_right'] = 'projectmanager.elements.list.add'; 564 $content['nm']['header_left'] = 'projectmanager.elements.list.add-new'; 565 } 566 else 567 { 568 unset($content['nm']['header_right']); 569 unset($content['nm']['header_left']); 570 $readonlys['sync_all'] = true; 571 } 572 $content['nm']['link_to'] = array( 573 'to_id' => $this->pm_id, 574 'to_app' => 'projectmanager', 575 'no_files' => true, 576 'search_label' => 'Add existing', 577 'link_label' => 'Add', 578 ); 579 $content['nm']['link_add'] = array( 580 'to_id' => $this->pm_id, 581 'to_app' => 'projectmanager', 582 ); 583 $GLOBALS['egw_info']['flags']['app_header'] = lang('projectmanager').' - '.lang('Elementlist') . 584 ': ' . $this->project->data['pm_number'] . ': ' .$this->project->data['pm_title'] ; 585 $this->tpl->read('projectmanager.elements.list'); 586 $this->tpl->exec('projectmanager.uiprojectelements.index',$content,'',$readonlys); 587 } 588 }
titre
Description
Corps
titre
Description
Corps
titre
Description
Corps
titre
Corps
| Généré le : Sun Feb 25 17:20:01 2007 | par Balluche grâce à PHPXref 0.7 |