tasks.php 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092
  1. <?php
  2. /* Copyright (C) 2005 Rodolphe Quiedeville <rodolphe@quiedeville.org>
  3. * Copyright (C) 2004-2019 Laurent Destailleur <eldy@users.sourceforge.net>
  4. * Copyright (C) 2005-2017 Regis Houssin <regis.houssin@inodbox.com>
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 3 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  18. */
  19. /**
  20. * \file htdocs/projet/tasks.php
  21. * \ingroup project
  22. * \brief List all tasks of a project
  23. */
  24. require "../main.inc.php";
  25. require_once DOL_DOCUMENT_ROOT.'/projet/class/project.class.php';
  26. require_once DOL_DOCUMENT_ROOT.'/projet/class/task.class.php';
  27. require_once DOL_DOCUMENT_ROOT.'/core/lib/project.lib.php';
  28. require_once DOL_DOCUMENT_ROOT.'/core/lib/date.lib.php';
  29. require_once DOL_DOCUMENT_ROOT.'/core/class/html.formother.class.php';
  30. require_once DOL_DOCUMENT_ROOT.'/core/class/extrafields.class.php';
  31. require_once DOL_DOCUMENT_ROOT.'/contact/class/contact.class.php';
  32. if (!empty($conf->categorie->enabled)) {
  33. require_once DOL_DOCUMENT_ROOT.'/categories/class/categorie.class.php';
  34. }
  35. // Load translation files required by the page
  36. $langsLoad=array('projects', 'users', 'companies');
  37. if (!empty($conf->eventorganization->enabled)) {
  38. $langsLoad[]='eventorganization';
  39. }
  40. $langs->loadLangs($langsLoad);
  41. $action = GETPOST('action', 'aZ09');
  42. $massaction = GETPOST('massaction', 'alpha');
  43. $show_files = GETPOST('show_files', 'int');
  44. $confirm = GETPOST('confirm', 'alpha');
  45. $toselect = GETPOST('toselect', 'array');
  46. $id = GETPOST('id', 'int');
  47. $ref = GETPOST('ref', 'alpha');
  48. $taskref = GETPOST('taskref', 'alpha');
  49. // Load variable for pagination
  50. $limit = GETPOST('limit', 'int') ? GETPOST('limit', 'int') : $conf->liste_limit;
  51. $sortfield = GETPOST('sortfield', 'aZ09comma');
  52. $sortorder = GETPOST('sortorder', 'aZ09comma');
  53. $page = GETPOSTISSET('pageplusone') ? (GETPOST('pageplusone') - 1) : GETPOST("page", 'int');
  54. if (empty($page) || $page < 0 || GETPOST('button_search', 'alpha') || GETPOST('button_removefilter', 'alpha')) {
  55. $page = 0;
  56. } // If $page is not defined, or '' or -1 or if we click on clear filters
  57. $offset = $limit * $page;
  58. $pageprev = $page - 1;
  59. $pagenext = $page + 1;
  60. $backtopage = GETPOST('backtopage', 'alpha');
  61. $cancel = GETPOST('cancel', 'alpha');
  62. $search_user_id = GETPOST('search_user_id', 'int');
  63. $search_taskref = GETPOST('search_taskref');
  64. $search_tasklabel = GETPOST('search_tasklabel');
  65. $search_taskdescription = GETPOST('search_taskdescription');
  66. $search_dtstartday = GETPOST('search_dtstartday');
  67. $search_dtstartmonth = GETPOST('search_dtstartmonth');
  68. $search_dtstartyear = GETPOST('search_dtstartyear');
  69. $search_dtendday = GETPOST('search_dtendday');
  70. $search_dtendmonth = GETPOST('search_dtendmonth');
  71. $search_dtendyear = GETPOST('search_dtendyear');
  72. $search_planedworkload = GETPOST('search_planedworkload');
  73. $search_timespend = GETPOST('search_timespend');
  74. $search_progresscalc = GETPOST('search_progresscalc');
  75. $search_progressdeclare = GETPOST('search_progressdeclare');
  76. $search_task_budget_amount = GETPOST('search_task_budget_amount');
  77. $search_date_start_startmonth = GETPOST('search_date_start_startmonth', 'int');
  78. $search_date_start_startyear = GETPOST('search_date_start_startyear', 'int');
  79. $search_date_start_startday = GETPOST('search_date_start_startday', 'int');
  80. $search_date_start_start = dol_mktime(0, 0, 0, $search_date_start_startmonth, $search_date_start_startday, $search_date_start_startyear); // Use tzserver
  81. $search_date_start_endmonth = GETPOST('search_date_start_endmonth', 'int');
  82. $search_date_start_endyear = GETPOST('search_date_start_endyear', 'int');
  83. $search_date_start_endday = GETPOST('search_date_start_endday', 'int');
  84. $search_date_start_end = dol_mktime(23, 59, 59, $search_date_start_endmonth, $search_date_start_endday, $search_date_start_endyear); // Use tzserver
  85. $search_date_end_startmonth = GETPOST('search_date_end_startmonth', 'int');
  86. $search_date_end_startyear = GETPOST('search_date_end_startyear', 'int');
  87. $search_date_end_startday = GETPOST('search_date_end_startday', 'int');
  88. $search_date_end_start = dol_mktime(0, 0, 0, $search_date_end_startmonth, $search_date_end_startday, $search_date_end_startyear); // Use tzserver
  89. $search_date_end_endmonth = GETPOST('search_date_end_endmonth', 'int');
  90. $search_date_end_endyear = GETPOST('search_date_end_endyear', 'int');
  91. $search_date_end_endday = GETPOST('search_date_end_endday', 'int');
  92. $search_date_end_end = dol_mktime(23, 59, 59, $search_date_end_endmonth, $search_date_end_endday, $search_date_end_endyear); // Use tzserver
  93. $contextpage = GETPOST('contextpage', 'aZ') ?GETPOST('contextpage', 'aZ') : 'projecttasklist';
  94. //if (! $user->rights->projet->all->lire) $mine=1; // Special for projects
  95. $object = new Project($db);
  96. $taskstatic = new Task($db);
  97. $extrafields = new ExtraFields($db);
  98. include DOL_DOCUMENT_ROOT.'/core/actions_fetchobject.inc.php'; // Must be include, not include_once
  99. if (!empty($conf->global->PROJECT_ALLOW_COMMENT_ON_PROJECT) && method_exists($object, 'fetchComments') && empty($object->comments)) {
  100. $object->fetchComments();
  101. }
  102. if ($id > 0 || !empty($ref)) {
  103. // fetch optionals attributes and labels
  104. $extrafields->fetch_name_optionals_label($object->table_element);
  105. }
  106. $extrafields->fetch_name_optionals_label($taskstatic->table_element);
  107. $search_array_options = $extrafields->getOptionalsFromPost($taskstatic->table_element, '', 'search_');
  108. // Default sort order (if not yet defined by previous GETPOST)
  109. if (!$sortfield) {
  110. reset($object->fields); $sortfield="t.".key($object->fields);
  111. } // Set here default search field. By default 1st field in definition. Reset is required to avoid key() to return null.
  112. if (!$sortorder) {
  113. $sortorder = "ASC";
  114. }
  115. // Security check
  116. $socid = 0;
  117. //if ($user->socid > 0) $socid = $user->socid; // For external user, no check is done on company because readability is managed by public status of project and assignement.
  118. $result = restrictedArea($user, 'projet', $id, 'projet&project');
  119. $diroutputmassaction = $conf->projet->dir_output.'/tasks/temp/massgeneration/'.$user->id;
  120. // Initialize technical object to manage hooks of page. Note that conf->hooks_modules contains array of hook context
  121. $hookmanager->initHooks(array('projecttaskscard', 'globalcard'));
  122. $progress = GETPOST('progress', 'int');
  123. $budget_amount = GETPOST('budget_amount', 'int');
  124. $label = GETPOST('label', 'alpha');
  125. $description = GETPOST('description', 'restricthtml');
  126. $planned_workloadhour = (GETPOST('planned_workloadhour', 'int') ?GETPOST('planned_workloadhour', 'int') : 0);
  127. $planned_workloadmin = (GETPOST('planned_workloadmin', 'int') ?GETPOST('planned_workloadmin', 'int') : 0);
  128. $planned_workload = $planned_workloadhour * 3600 + $planned_workloadmin * 60;
  129. // Definition of fields for list
  130. $arrayfields = array(
  131. 't.ref'=>array('label'=>$langs->trans("RefTask"), 'checked'=>1, 'position'=>1),
  132. 't.label'=>array('label'=>$langs->trans("LabelTask"), 'checked'=>1, 'position'=>2),
  133. 't.description'=>array('label'=>$langs->trans("Description"), 'checked'=>0, 'position'=>3),
  134. 't.dateo'=>array('label'=>$langs->trans("DateStart"), 'checked'=>1, 'position'=>4),
  135. 't.datee'=>array('label'=>$langs->trans("Deadline"), 'checked'=>1, 'position'=>5),
  136. 't.planned_workload'=>array('label'=>$langs->trans("PlannedWorkload"), 'checked'=>1, 'position'=>6),
  137. 't.duration_effective'=>array('label'=>$langs->trans("TimeSpent"), 'checked'=>1, 'position'=>7),
  138. 't.progress_calculated'=>array('label'=>$langs->trans("ProgressCalculated"), 'checked'=>1, 'position'=>8),
  139. 't.progress'=>array('label'=>$langs->trans("ProgressDeclared"), 'checked'=>1, 'position'=>9),
  140. 't.progress_summary'=>array('label'=>$langs->trans("TaskProgressSummary"), 'checked'=>1, 'position'=>10),
  141. 't.budget_amount'=>array('label'=>"Budget", 'checked'=>1, 'position'=>11),
  142. 'c.assigned'=>array('label'=>$langs->trans("TaskRessourceLinks"), 'checked'=>1, 'position'=>12),
  143. );
  144. if ($object->usage_bill_time) {
  145. $arrayfields['t.tobill'] = array('label'=>$langs->trans("TimeToBill"), 'checked'=>0, 'position'=>11);
  146. $arrayfields['t.billed'] = array('label'=>$langs->trans("TimeBilled"), 'checked'=>0, 'position'=>12);
  147. }
  148. // Extra fields
  149. $extrafieldsobjectkey = $taskstatic->table_element;
  150. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_list_array_fields.tpl.php';
  151. $arrayfields = dol_sort_array($arrayfields, 'position');
  152. $varpage = empty($contextpage) ? $_SERVER["PHP_SELF"] : $contextpage;
  153. /*
  154. * Actions
  155. */
  156. $parameters = array('id'=>$id);
  157. $reshook = $hookmanager->executeHooks('doActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks
  158. if ($reshook < 0) {
  159. setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
  160. }
  161. if (empty($reshook)) {
  162. // Selection of new fields
  163. include DOL_DOCUMENT_ROOT.'/core/actions_changeselectedfields.inc.php';
  164. // Purge search criteria
  165. if (GETPOST('button_removefilter_x', 'alpha') || GETPOST('button_removefilter.x', 'alpha') || GETPOST('button_removefilter', 'alpha')) { // All tests are required to be compatible with all browsers
  166. $search_user_id = "";
  167. $search_taskref = '';
  168. $search_tasklabel = '';
  169. $search_dtstartday = '';
  170. $search_dtstartmonth = '';
  171. $search_dtstartyear = '';
  172. $search_dtendday = '';
  173. $search_dtendmonth = '';
  174. $search_dtendyear = '';
  175. $search_planedworkload = '';
  176. $search_timespend = '';
  177. $search_progresscalc = '';
  178. $search_progressdeclare = '';
  179. $search_task_budget_amount = '';
  180. $toselect = array();
  181. $search_array_options = array();
  182. $search_date_start_startmonth = "";
  183. $search_date_start_startyear = "";
  184. $search_date_start_startday = "";
  185. $search_date_start_start = "";
  186. $search_date_start_endmonth = "";
  187. $search_date_start_endyear = "";
  188. $search_date_start_endday = "";
  189. $search_date_start_end = "";
  190. $search_date_end_startmonth = "";
  191. $search_date_end_startyear = "";
  192. $search_date_end_startday = "";
  193. $search_date_end_start = "";
  194. $search_date_end_endmonth = "";
  195. $search_date_end_endyear = "";
  196. $search_date_end_endday = "";
  197. $search_date_end_end = "";
  198. }
  199. // Mass actions
  200. $objectclass = 'Task';
  201. $objectlabel = 'Tasks';
  202. $permissiontoread = $user->rights->projet->lire;
  203. $permissiontodelete = $user->rights->projet->supprimer;
  204. $uploaddir = $conf->projet->dir_output.'/tasks';
  205. include DOL_DOCUMENT_ROOT.'/core/actions_massactions.inc.php';
  206. }
  207. $morewherefilterarray = array();
  208. if (!empty($search_taskref)) {
  209. $morewherefilterarray[] = natural_search('t.ref', $search_taskref, 0, 1);
  210. }
  211. if (!empty($search_tasklabel)) {
  212. $morewherefilterarray[] = natural_search('t.label', $search_tasklabel, 0, 1);
  213. }
  214. $moresql = dolSqlDateFilter('t.dateo', $search_dtstartday, $search_dtstartmonth, $search_dtstartyear, 1);
  215. if ($moresql) {
  216. $morewherefilterarray[] = $moresql;
  217. }
  218. $moresql = dolSqlDateFilter('t.datee', $search_dtendday, $search_dtendmonth, $search_dtendyear, 1);
  219. if ($moresql) {
  220. $morewherefilterarray[] = $moresql;
  221. }
  222. if ($search_date_start_start) {
  223. $morewherefilterarray[] = " t.dateo >= '".$db->idate($search_date_start_start)."'";
  224. }
  225. if ($search_date_start_end) {
  226. $morewherefilterarray[] = " t.dateo <= '".$db->idate($search_date_start_end)."'";
  227. }
  228. if ($search_date_end_start) {
  229. $morewherefilterarray[] = " t.datee >= '".$db->idate($search_date_end_start)."'";
  230. }
  231. if ($search_date_end_end) {
  232. $morewherefilterarray[] = " t.datee <= '".$db->idate($search_date_end_end)."'";
  233. }
  234. if (!empty($search_planedworkload)) {
  235. $morewherefilterarray[] = natural_search('t.planned_workload', $search_planedworkload, 1, 1);
  236. }
  237. if (!empty($search_timespend)) {
  238. $morewherefilterarray[] = natural_search('t.duration_effective', $search_timespend, 1, 1);
  239. }
  240. if (!empty($search_progresscalc)) {
  241. $filterprogresscalc = 'if '.natural_search('round(100 * $line->duration / $line->planned_workload,2)', $search_progresscalc, 1, 1).'{return 1;} else {return 0;}';
  242. } else {
  243. $filterprogresscalc = '';
  244. }
  245. if (!empty($search_progressdeclare)) {
  246. $morewherefilterarray[] = natural_search('t.progress', $search_progressdeclare, 1, 1);
  247. }
  248. if ($search_task_budget_amount) {
  249. $morewherefilterarray[]= natural_search('t.budget_amount', $search_task_budget_amount, 1, 1);
  250. }
  251. $morewherefilter = '';
  252. if (count($morewherefilterarray) > 0) {
  253. $morewherefilter = ' AND '.implode(' AND ', $morewherefilterarray);
  254. }
  255. if ($action == 'createtask' && $user->rights->projet->creer) {
  256. $error = 0;
  257. // If we use user timezone, we must change also view/list to use user timezone everywhere
  258. $date_start = dol_mktime(GETPOST('dateohour', 'int'), GETPOST('dateomin', 'int'), 0, GETPOST('dateomonth', 'int'), GETPOST('dateoday', 'int'), GETPOST('dateoyear', 'int'));
  259. $date_end = dol_mktime(GETPOST('dateehour', 'int'), GETPOST('dateemin', 'int'), 0, GETPOST('dateemonth', 'int'), GETPOST('dateeday', 'int'), GETPOST('dateeyear', 'int'));
  260. if (!$cancel) {
  261. if (empty($taskref)) {
  262. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("Ref")), null, 'errors');
  263. $action = 'create';
  264. $error++;
  265. }
  266. if (empty($label)) {
  267. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("Label")), null, 'errors');
  268. $action = 'create';
  269. $error++;
  270. } elseif (!GETPOST('task_parent')) {
  271. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("ChildOfProjectTask")), null, 'errors');
  272. $action = 'create';
  273. $error++;
  274. }
  275. if (!$error) {
  276. $tmparray = explode('_', GETPOST('task_parent'));
  277. $projectid = $tmparray[0];
  278. if (empty($projectid)) {
  279. $projectid = $id; // If projectid is ''
  280. }
  281. $task_parent = $tmparray[1];
  282. if (empty($task_parent)) {
  283. $task_parent = 0; // If task_parent is ''
  284. }
  285. $task = new Task($db);
  286. $task->fk_project = $projectid;
  287. $task->ref = $taskref;
  288. $task->label = $label;
  289. $task->description = $description;
  290. $task->planned_workload = $planned_workload;
  291. $task->fk_task_parent = $task_parent;
  292. $task->date_c = dol_now();
  293. $task->date_start = $date_start;
  294. $task->date_end = $date_end;
  295. $task->progress = $progress;
  296. $task->budget_amount = $budget_amount;
  297. // Fill array 'array_options' with data from add form
  298. $ret = $extrafields->setOptionalsFromPost(null, $task);
  299. $taskid = $task->create($user);
  300. if ($taskid > 0) {
  301. $result = $task->add_contact(GETPOST("userid", 'int'), 'TASKEXECUTIVE', 'internal');
  302. } else {
  303. if ($db->lasterrno() == 'DB_ERROR_RECORD_ALREADY_EXISTS') {
  304. $langs->load("projects");
  305. setEventMessages($langs->trans('NewTaskRefSuggested'), '', 'warnings');
  306. $duplicate_code_error = true;
  307. } else {
  308. setEventMessages($task->error, $task->errors, 'errors');
  309. }
  310. $action = 'create';
  311. $error++;
  312. }
  313. }
  314. if (!$error) {
  315. if (!empty($backtopage)) {
  316. header("Location: ".$backtopage);
  317. exit;
  318. } elseif (empty($projectid)) {
  319. header("Location: ".DOL_URL_ROOT.'/projet/tasks/list.php'.(empty($mode) ? '' : '?mode='.$mode));
  320. exit;
  321. }
  322. $id = $projectid;
  323. }
  324. } else {
  325. if (!empty($backtopage)) {
  326. header("Location: ".$backtopage);
  327. exit;
  328. } elseif (empty($id)) {
  329. // We go back on task list
  330. header("Location: ".DOL_URL_ROOT.'/projet/tasks/list.php'.(empty($mode) ? '' : '?mode='.$mode));
  331. exit;
  332. }
  333. }
  334. }
  335. /*
  336. * View
  337. */
  338. $now = dol_now();
  339. $form = new Form($db);
  340. $formother = new FormOther($db);
  341. $socstatic = new Societe($db);
  342. $projectstatic = new Project($db);
  343. $taskstatic = new Task($db);
  344. $userstatic = new User($db);
  345. $title = $langs->trans("Project").' - '.$langs->trans("Tasks").' - '.$object->ref.' '.$object->name;
  346. if (!empty($conf->global->MAIN_HTML_TITLE) && preg_match('/projectnameonly/', $conf->global->MAIN_HTML_TITLE) && $object->name) {
  347. $title = $object->ref.' '.$object->name.' - '.$langs->trans("Tasks");
  348. }
  349. $help_url = "EN:Module_Projects|FR:Module_Projets|ES:M&oacute;dulo_Proyectos";
  350. llxHeader("", $title, $help_url);
  351. if ($id > 0 || !empty($ref)) {
  352. $result = $object->fetch($id, $ref);
  353. if ($result < 0) {
  354. setEventMessages(null, $object->errors, 'errors');
  355. }
  356. $result = $object->fetch_thirdparty();
  357. if ($result < 0) {
  358. setEventMessages(null, $object->errors, 'errors');
  359. }
  360. $result = $object->fetch_optionals();
  361. if ($result < 0) {
  362. setEventMessages(null, $object->errors, 'errors');
  363. }
  364. // To verify role of users
  365. //$userAccess = $object->restrictedProjectArea($user,'read');
  366. $userWrite = $object->restrictedProjectArea($user, 'write');
  367. //$userDelete = $object->restrictedProjectArea($user,'delete');
  368. //print "userAccess=".$userAccess." userWrite=".$userWrite." userDelete=".$userDelete;
  369. $tab = (GETPOSTISSET('tab') ? GETPOST('tab') : 'tasks');
  370. $head = project_prepare_head($object);
  371. print dol_get_fiche_head($head, $tab, $langs->trans("Project"), -1, ($object->public ? 'projectpub' : 'project'));
  372. $param = '&id='.$object->id;
  373. if (!empty($contextpage) && $contextpage != $_SERVER["PHP_SELF"]) {
  374. $param .= '&contextpage='.urlencode($contextpage);
  375. }
  376. if ($search_user_id) {
  377. $param .= '&search_user_id='.urlencode($search_user_id);
  378. }
  379. if ($search_taskref) {
  380. $param .= '&search_taskref='.urlencode($search_taskref);
  381. }
  382. if ($search_tasklabel) {
  383. $param .= '&search_tasklabel='.urlencode($search_tasklabel);
  384. }
  385. if ($search_taskdescription) {
  386. $param .= '&search_taskdescription='.urlencode($search_taskdescription);
  387. }
  388. if ($search_dtstartday) {
  389. $param .= '&search_dtstartday='.urlencode($search_dtstartday);
  390. }
  391. if ($search_dtstartmonth) {
  392. $param .= '&search_dtstartmonth='.urlencode($search_dtstartmonth);
  393. }
  394. if ($search_dtstartyear) {
  395. $param .= '&search_dtstartyear='.urlencode($search_dtstartyear);
  396. }
  397. if ($search_dtendday) {
  398. $param .= '&search_dtendday='.urlencode($search_dtendday);
  399. }
  400. if ($search_dtendmonth) {
  401. $param .= '&search_dtendmonth='.urlencode($search_dtendmonth);
  402. }
  403. if ($search_dtendyear) {
  404. $param .= '&search_dtendyear='.urlencode($search_dtendyear);
  405. }
  406. if ($search_date_start_startmonth) {
  407. $param .= '&search_date_start_startmonth='.urlencode($search_date_start_startmonth);
  408. }
  409. if ($search_date_start_startyear) {
  410. $param .= '&search_date_start_startyear='.urlencode($search_date_start_startyear);
  411. }
  412. if ($search_date_start_startday) {
  413. $param .= '&search_date_start_startday='.urlencode($search_date_start_startday);
  414. }
  415. if ($search_date_start_start) {
  416. $param .= '&search_date_start_start='.urlencode($search_date_start_start);
  417. }
  418. if ($search_date_start_endmonth) {
  419. $param .= '&search_date_start_endmonth='.urlencode($search_date_start_endmonth);
  420. }
  421. if ($search_date_start_endyear) {
  422. $param .= '&search_date_start_endyear='.urlencode($search_date_start_endyear);
  423. }
  424. if ($search_date_start_endday) {
  425. $param .= '&search_date_start_endday='.urlencode($search_date_start_endday);
  426. }
  427. if ($search_date_start_end) {
  428. $param .= '&search_date_start_end='.urlencode($search_date_start_end);
  429. }
  430. if ($search_date_end_startmonth) {
  431. $param .= '&search_date_end_startmonth='.urlencode($search_date_end_startmonth);
  432. }
  433. if ($search_date_end_startyear) {
  434. $param .= '&search_date_end_startyear='.urlencode($search_date_end_startyear);
  435. }
  436. if ($search_date_end_startday) {
  437. $param .= '&search_date_end_startday='.urlencode($search_date_end_startday);
  438. }
  439. if ($search_date_end_start) {
  440. $param .= '&search_date_end_start='.urlencode($search_date_end_start);
  441. }
  442. if ($search_date_end_endmonth) {
  443. $param .= '&search_date_end_endmonth='.urlencode($search_date_end_endmonth);
  444. }
  445. if ($search_date_end_endyear) {
  446. $param .= '&search_date_end_endyear='.urlencode($search_date_end_endyear);
  447. }
  448. if ($search_date_end_endday) {
  449. $param .= '&search_date_end_endday='.urlencode($search_date_end_endday);
  450. }
  451. if ($search_date_end_end) {
  452. $param .= '&search_date_end_end=' . urlencode($search_date_end_end);
  453. }
  454. if ($search_planedworkload) {
  455. $param .= '&search_planedworkload='.urlencode($search_planedworkload);
  456. }
  457. if ($search_timespend) {
  458. $param .= '&search_timespend='.urlencode($search_timespend);
  459. }
  460. if ($search_progresscalc) {
  461. $param .= '&search_progresscalc='.urlencode($search_progresscalc);
  462. }
  463. if ($search_progressdeclare) {
  464. $param .= '&search_progressdeclare='.urlencode($search_progressdeclare);
  465. }
  466. if ($search_task_budget_amount) {
  467. $param .= '&search_task_budget_amount='.urlencode($search_task_budget_amount);
  468. }
  469. if ($optioncss != '') {
  470. $param .= '&optioncss='.urlencode($optioncss);
  471. }
  472. // Add $param from extra fields
  473. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_list_search_param.tpl.php';
  474. // Project card
  475. $linkback = '<a href="'.DOL_URL_ROOT.'/projet/list.php?restore_lastsearch_values=1">'.$langs->trans("BackToList").'</a>';
  476. $morehtmlref = '<div class="refidno">';
  477. // Title
  478. $morehtmlref .= $object->title;
  479. // Thirdparty
  480. if (!empty($object->thirdparty->id) && $object->thirdparty->id > 0) {
  481. $morehtmlref .= '<br>'.$langs->trans('ThirdParty').' : '.$object->thirdparty->getNomUrl(1, 'project');
  482. }
  483. $morehtmlref .= '</div>';
  484. // Define a complementary filter for search of next/prev ref.
  485. if (empty($user->rights->projet->all->lire)) {
  486. $objectsListId = $object->getProjectsAuthorizedForUser($user, 0, 0);
  487. $object->next_prev_filter = " rowid IN (".$db->sanitize(count($objectsListId) ?join(',', array_keys($objectsListId)) : '0').")";
  488. }
  489. dol_banner_tab($object, 'ref', $linkback, 1, 'ref', 'ref', $morehtmlref);
  490. print '<div class="fichecenter">';
  491. print '<div class="fichehalfleft">';
  492. print '<div class="underbanner clearboth"></div>';
  493. print '<table class="border tableforfield centpercent">';
  494. // Usage
  495. if (!empty($conf->global->PROJECT_USE_OPPORTUNITIES) || empty($conf->global->PROJECT_HIDE_TASKS) || !empty($conf->eventorganization->enabled)) {
  496. print '<tr><td class="tdtop">';
  497. print $langs->trans("Usage");
  498. print '</td>';
  499. print '<td>';
  500. if (!empty($conf->global->PROJECT_USE_OPPORTUNITIES)) {
  501. print '<input type="checkbox" disabled name="usage_opportunity"'.(GETPOSTISSET('usage_opportunity') ? (GETPOST('usage_opportunity', 'alpha') != '' ? ' checked="checked"' : '') : ($object->usage_opportunity ? ' checked="checked"' : '')).'"> ';
  502. $htmltext = $langs->trans("ProjectFollowOpportunity");
  503. print $form->textwithpicto($langs->trans("ProjectFollowOpportunity"), $htmltext);
  504. print '<br>';
  505. }
  506. if (empty($conf->global->PROJECT_HIDE_TASKS)) {
  507. print '<input type="checkbox" disabled name="usage_task"'.(GETPOSTISSET('usage_task') ? (GETPOST('usage_task', 'alpha') != '' ? ' checked="checked"' : '') : ($object->usage_task ? ' checked="checked"' : '')).'"> ';
  508. $htmltext = $langs->trans("ProjectFollowTasks");
  509. print $form->textwithpicto($langs->trans("ProjectFollowTasks"), $htmltext);
  510. print '<br>';
  511. }
  512. if (empty($conf->global->PROJECT_HIDE_TASKS) && !empty($conf->global->PROJECT_BILL_TIME_SPENT)) {
  513. print '<input type="checkbox" disabled name="usage_bill_time"'.(GETPOSTISSET('usage_bill_time') ? (GETPOST('usage_bill_time', 'alpha') != '' ? ' checked="checked"' : '') : ($object->usage_bill_time ? ' checked="checked"' : '')).'"> ';
  514. $htmltext = $langs->trans("ProjectBillTimeDescription");
  515. print $form->textwithpicto($langs->trans("BillTime"), $htmltext);
  516. print '<br>';
  517. }
  518. if (!empty($conf->eventorganization->enabled)) {
  519. print '<input type="checkbox" disabled name="usage_organize_event"'.(GETPOSTISSET('usage_organize_event') ? (GETPOST('usage_organize_event', 'alpha') != '' ? ' checked="checked"' : '') : ($object->usage_organize_event ? ' checked="checked"' : '')).'"> ';
  520. $htmltext = $langs->trans("EventOrganizationDescriptionLong");
  521. print $form->textwithpicto($langs->trans("ManageOrganizeEvent"), $htmltext);
  522. }
  523. print '</td></tr>';
  524. }
  525. // Visibility
  526. print '<tr><td class="titlefield">'.$langs->trans("Visibility").'</td><td>';
  527. if ($object->public) {
  528. print $langs->trans('SharedProject');
  529. } else {
  530. print $langs->trans('PrivateProject');
  531. }
  532. print '</td></tr>';
  533. // Date start - end
  534. print '<tr><td>'.$langs->trans("DateStart").' - '.$langs->trans("DateEnd").'</td><td>';
  535. $start = dol_print_date($object->date_start, 'day');
  536. print ($start ? $start : '?');
  537. $end = dol_print_date($object->date_end, 'day');
  538. print ' - ';
  539. print ($end ? $end : '?');
  540. if ($object->hasDelay()) {
  541. print img_warning("Late");
  542. }
  543. print '</td></tr>';
  544. // Budget
  545. print '<tr><td>'.$langs->trans("Budget").'</td><td>';
  546. if (strcmp($object->budget_amount, '')) {
  547. print '<span class="amount">'.price($object->budget_amount, '', $langs, 1, 0, 0, $conf->currency).'</span>';
  548. }
  549. print '</td></tr>';
  550. // Other attributes
  551. $cols = 2;
  552. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_view.tpl.php';
  553. print '</table>';
  554. print '</div>';
  555. print '<div class="fichehalfright">';
  556. print '<div class="underbanner clearboth"></div>';
  557. print '<table class="border tableforfield centpercent">';
  558. // Description
  559. print '<td class="titlefield tdtop">'.$langs->trans("Description").'</td><td>';
  560. print nl2br($object->description);
  561. print '</td></tr>';
  562. // Categories
  563. if ($conf->categorie->enabled) {
  564. print '<tr><td class="valignmiddle">'.$langs->trans("Categories").'</td><td>';
  565. print $form->showCategories($object->id, Categorie::TYPE_PROJECT, 1);
  566. print "</td></tr>";
  567. }
  568. print '</table>';
  569. print '</div>';
  570. print '</div>';
  571. print '<div class="clearboth"></div>';
  572. print dol_get_fiche_end();
  573. }
  574. if ($action == 'create' && $user->rights->projet->creer && (empty($object->thirdparty->id) || $userWrite > 0)) {
  575. if ($id > 0 || !empty($ref)) {
  576. print '<br>';
  577. }
  578. print load_fiche_titre($langs->trans("NewTask"), '', 'projecttask');
  579. if ($object->id > 0 && $object->statut == Project::STATUS_CLOSED) {
  580. print '<div class="warning">';
  581. $langs->load("errors");
  582. print $langs->trans("WarningProjectClosed");
  583. print '</div>';
  584. }
  585. if ($object->id > 0 && $object->statut == Project::STATUS_DRAFT) {
  586. print '<div class="warning">';
  587. $langs->load("errors");
  588. print $langs->trans("WarningProjectDraft");
  589. print '</div>';
  590. }
  591. print '<form action="'.$_SERVER['PHP_SELF'].'" method="POST">';
  592. print '<input type="hidden" name="token" value="'.newToken().'">';
  593. print '<input type="hidden" name="action" value="createtask">';
  594. print '<input type="hidden" name="backtopage" value="'.$backtopage.'">';
  595. if (!empty($object->id)) {
  596. print '<input type="hidden" name="id" value="'.$object->id.'">';
  597. }
  598. print dol_get_fiche_head('');
  599. print '<table class="border centpercent">';
  600. $defaultref = '';
  601. $obj = empty($conf->global->PROJECT_TASK_ADDON) ? 'mod_task_simple' : $conf->global->PROJECT_TASK_ADDON;
  602. if (!empty($conf->global->PROJECT_TASK_ADDON) && is_readable(DOL_DOCUMENT_ROOT."/core/modules/project/task/".$conf->global->PROJECT_TASK_ADDON.".php")) {
  603. require_once DOL_DOCUMENT_ROOT."/core/modules/project/task/".$conf->global->PROJECT_TASK_ADDON.'.php';
  604. $modTask = new $obj;
  605. $defaultref = $modTask->getNextValue($object->thirdparty, null);
  606. }
  607. if (is_numeric($defaultref) && $defaultref <= 0) {
  608. $defaultref = '';
  609. }
  610. // Ref
  611. print '<tr><td class="titlefieldcreate"><span class="fieldrequired">'.$langs->trans("Ref").'</span></td><td>';
  612. if (empty($duplicate_code_error)) {
  613. print (GETPOSTISSET("ref") ? GETPOST("ref", 'alpha') : $defaultref);
  614. } else {
  615. print $defaultref;
  616. }
  617. print '<input type="hidden" name="taskref" value="'.(GETPOSTISSET("ref") ? GETPOST("ref", 'alpha') : $defaultref).'">';
  618. print '</td></tr>';
  619. // Label
  620. print '<tr><td class="fieldrequired">'.$langs->trans("Label").'</td><td>';
  621. print '<input type="text" name="label" autofocus class="minwidth500 maxwidthonsmartphone" value="'.$label.'">';
  622. print '</td></tr>';
  623. // Project
  624. print '<tr><td class="fieldrequired">'.$langs->trans("ChildOfProjectTask").'</td><td>';
  625. print img_picto('', 'project');
  626. $formother->selectProjectTasks(GETPOST('task_parent'), !empty($projectid) ? $projectid : $object->id, 'task_parent', 0, 0, 1, 1, 0, '0,1', 'maxwidth500');
  627. print '</td></tr>';
  628. // Assigned to
  629. print '<tr><td>'.$langs->trans("AffectedTo").'</td><td>';
  630. $contactsofproject = (!empty($object->id) ? $object->getListContactId('internal') : '');
  631. if (is_array($contactsofproject) && count($contactsofproject)) {
  632. print $form->select_dolusers($user->id, 'userid', 0, '', 0, '', $contactsofproject, 0, 0, 0, '', 0, '', 'maxwidth300');
  633. } else {
  634. print '<span class="opacitymedium">'.$langs->trans("NoUserAssignedToTheProject").'</span>';
  635. }
  636. print '</td></tr>';
  637. // Date start
  638. print '<tr><td>'.$langs->trans("DateStart").'</td><td>';
  639. print $form->selectDate((!empty($date_start) ? $date_start : ''), 'dateo', 1, 1, 0, '', 1, 1);
  640. print '</td></tr>';
  641. // Date end
  642. print '<tr><td>'.$langs->trans("DateEnd").'</td><td>';
  643. print $form->selectDate((!empty($date_end) ? $date_end : -1), 'datee', -1, 1, 0, '', 1, 1);
  644. print '</td></tr>';
  645. // Planned workload
  646. print '<tr><td>'.$langs->trans("PlannedWorkload").'</td><td>';
  647. print $form->select_duration('planned_workload', !empty($planned_workload) ? $planned_workload : 0, 0, 'text');
  648. print '</td></tr>';
  649. // Progress
  650. print '<tr><td>'.$langs->trans("ProgressDeclared").'</td><td colspan="3">';
  651. print $formother->select_percent($progress, 'progress', 0, 5, 0, 100, 1);
  652. print '</td></tr>';
  653. // Description
  654. print '<tr><td class="tdtop">'.$langs->trans("Description").'</td>';
  655. print '<td>';
  656. if (empty($conf->global->FCKEDITOR_ENABLE_SOCIETE)) {
  657. print '<textarea name="description" class="quatrevingtpercent" rows="'.ROWS_4.'">'.$description.'</textarea>';
  658. } else {
  659. // WYSIWYG editor
  660. include_once DOL_DOCUMENT_ROOT.'/core/class/doleditor.class.php';
  661. $cked_enabled = (!empty($conf->global->FCKEDITOR_ENABLE_DETAILS) ? $conf->global->FCKEDITOR_ENABLE_DETAILS : 0);
  662. if (!empty($conf->global->MAIN_INPUT_DESC_HEIGHT)) {
  663. $nbrows = $conf->global->MAIN_INPUT_DESC_HEIGHT;
  664. }
  665. $doleditor = new DolEditor('description', $object->description, '', 80, 'dolibarr_details', '', false, true, $cked_enabled, $nbrows);
  666. print $doleditor->Create();
  667. }
  668. print '</td></tr>';
  669. print '<tr><td>'.$langs->trans("Budget").'</td>';
  670. print '<td><input size="5" type="text" name="budget_amount" value="'.dol_escape_htmltag(GETPOSTISSET('budget_amount') ? GETPOST('budget_amount') : '').'"></td>';
  671. print '</tr>';
  672. // Other options
  673. $parameters = array();
  674. $reshook = $hookmanager->executeHooks('formObjectOptions', $parameters, $taskstatic, $action); // Note that $action and $object may have been modified by hook
  675. print $hookmanager->resPrint;
  676. if (empty($reshook) && !empty($extrafields->attributes[$taskstatic->table_element]['label'])) {
  677. print $taskstatic->showOptionals($extrafields, 'edit'); // Do not use $object here that is object of project but use $taskstatic
  678. }
  679. print '</table>';
  680. print dol_get_fiche_end();
  681. print $form->buttonsSaveCancel("Add");
  682. print '</form>';
  683. } elseif ($id > 0 || !empty($ref)) {
  684. $selectedfields = $form->multiSelectArrayWithCheckbox('selectedfields', $arrayfields, $varpage); // This also change content of $arrayfields
  685. /*
  686. * Projet card in view mode
  687. */
  688. print '<br>';
  689. // Link to create task
  690. $linktocreatetaskParam = array();
  691. $linktocreatetaskUserRight = false;
  692. if ($user->rights->projet->all->creer || $user->rights->projet->creer) {
  693. if ($object->public || $userWrite > 0) {
  694. $linktocreatetaskUserRight = true;
  695. } else {
  696. $linktocreatetaskParam['attr']['title'] = $langs->trans("NotOwnerOfProject");
  697. }
  698. }
  699. $linktocreatetask = dolGetButtonTitle($langs->trans('AddTask'), '', 'fa fa-plus-circle', DOL_URL_ROOT.'/projet/tasks.php?action=create'.$param.'&backtopage='.urlencode($_SERVER['PHP_SELF'].'?id='.$object->id), '', $linktocreatetaskUserRight, $linktocreatetaskParam);
  700. print '<form method="POST" id="searchFormList" action="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'">';
  701. print '<input type="hidden" name="optioncss" value="'.$optioncss.'">';
  702. print '<input type="hidden" name="token" value="'.newToken().'">';
  703. print '<input type="hidden" name="action" value="list">';
  704. print '<input type="hidden" name="formfilteraction" id="formfilteraction" value="list">';
  705. print '<input type="hidden" name="sortfield" value="'.$sortfield.'">';
  706. print '<input type="hidden" name="sortorder" value="'.$sortorder.'">';
  707. print '<input type="hidden" name="page" value="'.$page.'">';
  708. print '<input type="hidden" name="contextpage" value="'.$contextpage.'">';
  709. $title = $langs->trans("ListOfTasks");
  710. $linktotasks = dolGetButtonTitle($langs->trans('ViewList'), '', 'fa fa-list-alt imgforviewmode', DOL_URL_ROOT.'/projet/tasks.php?id='.$object->id, '', 1, array('morecss'=>'reposition btnTitleSelected'));
  711. $linktotasks .= dolGetButtonTitle($langs->trans('ViewGantt'), '', 'fa fa-stream imgforviewmode', DOL_URL_ROOT.'/projet/ganttview.php?id='.$object->id.'&withproject=1', '', 1, array('morecss'=>'reposition marginleftonly'));
  712. //print_barre_liste($title, 0, $_SERVER["PHP_SELF"], '', $sortfield, $sortorder, $linktotasks, $num, $totalnboflines, 'generic', 0, '', '', 0, 1);
  713. print load_fiche_titre($title, $linktotasks.' &nbsp; '.$linktocreatetask, 'projecttask');
  714. // Get list of tasks in tasksarray and taskarrayfiltered
  715. // We need all tasks (even not limited to a user because a task to user can have a parent that is not affected to him).
  716. $filteronthirdpartyid = $socid;
  717. $tasksarray = $taskstatic->getTasksArray(0, 0, $object->id, $filteronthirdpartyid, 0, '', -1, $morewherefilter, 0, 0, $extrafields, 1, $search_array_options);
  718. // We load also tasks limited to a particular user
  719. $tmpuser = new User($db);
  720. if ($search_user_id > 0) {
  721. $tmpuser->fetch($search_user_id);
  722. }
  723. $tasksrole = ($tmpuser->id > 0 ? $taskstatic->getUserRolesForProjectsOrTasks(0, $tmpuser, $object->id, 0) : '');
  724. //var_dump($tasksarray);
  725. //var_dump($tasksrole);
  726. if (!empty($conf->use_javascript_ajax)) {
  727. include DOL_DOCUMENT_ROOT.'/core/tpl/ajaxrow.tpl.php';
  728. }
  729. // Filter on categories
  730. $moreforfilter = '';
  731. if (count($tasksarray) > 0) {
  732. $moreforfilter .= '<div class="divsearchfield">';
  733. $moreforfilter .= img_picto('', 'user', 'class="pictofixedwidth"');
  734. $moreforfilter .= $form->select_dolusers($tmpuser->id > 0 ? $tmpuser->id : '', 'search_user_id', $langs->trans("TasksAssignedTo"), null, 0, '', '');
  735. $moreforfilter .= '</div>';
  736. }
  737. if ($moreforfilter) {
  738. print '<div class="liste_titre liste_titre_bydiv centpercent">';
  739. print $moreforfilter;
  740. print '</div>';
  741. }
  742. print '<div class="div-table-responsive">';
  743. print '<table id="tablelines" class="tagtable nobottom liste'.($moreforfilter ? " listwithfilterbefore" : "").'">';
  744. // Fields title search
  745. print '<tr class="liste_titre_filter">';
  746. if (!empty($arrayfields['t.ref']['checked'])) {
  747. print '<td class="liste_titre">';
  748. print '<input class="flat searchstring maxwidth50" type="text" name="search_taskref" value="'.dol_escape_htmltag($search_taskref).'">';
  749. print '</td>';
  750. }
  751. if (!empty($arrayfields['t.label']['checked'])) {
  752. print '<td class="liste_titre">';
  753. print '<input class="flat searchstring maxwidth100" type="text" name="search_tasklabel" value="'.dol_escape_htmltag($search_tasklabel).'">';
  754. print '</td>';
  755. }
  756. if (!empty($arrayfields['t.description']['checked'])) {
  757. print '<td class="liste_titre">';
  758. print '<input class="flat searchstring maxwidth100" type="text" name="search_taskdescription" value="'.dol_escape_htmltag($search_taskdescription).'">';
  759. print '</td>';
  760. }
  761. if (!empty($arrayfields['t.dateo']['checked'])) {
  762. print '<td class="liste_titre center">';
  763. /*print '<span class="nowraponall"><input class="flat valignmiddle width20" type="text" maxlength="2" name="search_dtstartday" value="'.$search_dtstartday.'">';
  764. print '<input class="flat valignmiddle width20" type="text" maxlength="2" name="search_dtstartmonth" value="'.$search_dtstartmonth.'"></span>';
  765. $formother->select_year($search_dtstartyear ? $search_dtstartyear : -1, 'search_dtstartyear', 1, 20, 5);*/
  766. print '<div class="nowrap">';
  767. print $form->selectDate($search_date_start_start ? $search_date_start_start : -1, 'search_date_start_start', 0, 0, 1, '', 1, 0, 0, '', '', '', '', 1, '', $langs->trans('From'));
  768. print '</div>';
  769. print '<div class="nowrap">';
  770. print $form->selectDate($search_date_start_end ? $search_date_start_end : -1, 'search_date_start_end', 0, 0, 1, '', 1, 0, 0, '', '', '', '', 1, '', $langs->trans('to'));
  771. print '</div>';
  772. print '</td>';
  773. }
  774. if (!empty($arrayfields['t.datee']['checked'])) {
  775. print '<td class="liste_titre center">';
  776. /*print '<span class="nowraponall"><input class="flat valignmiddle width20" type="text" maxlength="2" name="search_dtendday" value="'.$search_dtendday.'">';
  777. print '<input class="flat valignmiddle width20" type="text" maxlength="2" name="search_dtendmonth" value="'.$search_dtendmonth.'"></span>';
  778. $formother->select_year($search_dtendyear ? $search_dtendyear : -1, 'search_dtendyear', 1, 20, 5);*/
  779. print '<div class="nowrap">';
  780. print $form->selectDate($search_date_end_start ? $search_date_end_start : -1, 'search_date_end_start', 0, 0, 1, '', 1, 0, 0, '', '', '', '', 1, '', $langs->trans('From'));
  781. print '</div>';
  782. print '<div class="nowrap">';
  783. print $form->selectDate($search_date_end_end ? $search_date_end_end : -1, 'search_date_end_end', 0, 0, 1, '', 1, 0, 0, '', '', '', '', 1, '', $langs->trans('to'));
  784. print '</div>';
  785. print '</td>';
  786. }
  787. if (!empty($arrayfields['t.planned_workload']['checked'])) {
  788. print '<td class="liste_titre right">';
  789. print '<input class="flat" type="text" size="4" name="search_planedworkload" value="'.$search_planedworkload.'">';
  790. print '</td>';
  791. }
  792. if (!empty($arrayfields['t.duration_effective']['checked'])) {
  793. print '<td class="liste_titre right">';
  794. print '<input class="flat" type="text" size="4" name="search_timespend" value="'.$search_timespend.'">';
  795. print '</td>';
  796. }
  797. if (!empty($arrayfields['t.progress_calculated']['checked'])) {
  798. print '<td class="liste_titre right">';
  799. print '<input class="flat" type="text" size="4" name="search_progresscalc" value="'.$search_progresscalc.'">';
  800. print '</td>';
  801. }
  802. if (!empty($arrayfields['t.progress']['checked'])) {
  803. print '<td class="liste_titre right">';
  804. print '<input class="flat" type="text" size="4" name="search_progressdeclare" value="'.$search_progressdeclare.'">';
  805. print '</td>';
  806. }
  807. // progress resume not searchable
  808. print '<td class="liste_titre right"></td>';
  809. if ($object->usage_bill_time) {
  810. if (!empty($arrayfields['t.tobill']['checked'])) {
  811. print '<td class="liste_titre right">';
  812. print '</td>';
  813. }
  814. if (!empty($arrayfields['t.billed']['checked'])) {
  815. print '<td class="liste_titre right">';
  816. print '</td>';
  817. }
  818. }
  819. // Contacts of task, disabled because available by default jsut after
  820. /*
  821. if (!empty($conf->global->PROJECT_SHOW_CONTACTS_IN_LIST)) {
  822. print '<td class="liste_titre"></td>';
  823. }
  824. */
  825. if (!empty($arrayfields['t.budget_amount']['checked'])) {
  826. print '<td class="liste_titre center">';
  827. print '<input type="text" class="flat" name="search_task_budget_amount" value="'.$search_task_budget_amount.'" size="4">';
  828. print '</td>';
  829. }
  830. if (!empty($arrayfields['c.assigned']['checked'])) {
  831. print '<td class="liste_titre right">';
  832. print '</td>';
  833. }
  834. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_list_search_input.tpl.php';
  835. // Action column
  836. print '<td class="liste_titre maxwidthsearch">';
  837. $searchpicto = $form->showFilterButtons();
  838. print $searchpicto;
  839. print '</td>';
  840. print "</tr>\n";
  841. print '<tr class="liste_titre nodrag nodrop">';
  842. // print '<td>'.$langs->trans("Project").'</td>';
  843. if (!empty($arrayfields['t.ref']['checked'])) {
  844. print_liste_field_titre($arrayfields['t.ref']['label'], $_SERVER["PHP_SELF"], '', '', $param, '', $sortfield, $sortorder, '');
  845. }
  846. if (!empty($arrayfields['t.label']['checked'])) {
  847. print_liste_field_titre($arrayfields['t.label']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, '');
  848. }
  849. if (!empty($arrayfields['t.description']['checked'])) {
  850. print_liste_field_titre($arrayfields['t.description']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, '');
  851. }
  852. if (!empty($arrayfields['t.dateo']['checked'])) {
  853. print_liste_field_titre($arrayfields['t.dateo']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'center ');
  854. }
  855. if (!empty($arrayfields['t.datee']['checked'])) {
  856. print_liste_field_titre($arrayfields['t.datee']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'center ');
  857. }
  858. if (!empty($arrayfields['t.planned_workload']['checked'])) {
  859. print_liste_field_titre($arrayfields['t.planned_workload']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ', '', 1);
  860. }
  861. if (!empty($arrayfields['t.duration_effective']['checked'])) {
  862. print_liste_field_titre($arrayfields['t.duration_effective']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ', '', 1);
  863. }
  864. if (!empty($arrayfields['t.progress_calculated']['checked'])) {
  865. print_liste_field_titre($arrayfields['t.progress_calculated']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ', '', 1);
  866. }
  867. if (!empty($arrayfields['t.progress']['checked'])) {
  868. print_liste_field_titre($arrayfields['t.progress']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ', '', 1);
  869. }
  870. if (!empty($arrayfields['t.progress_summary']['checked'])) {
  871. print_liste_field_titre($arrayfields['t.progress_summary']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'center ', '', 1);
  872. }
  873. if ($object->usage_bill_time) {
  874. if (!empty($arrayfields['t.tobill']['checked'])) {
  875. print_liste_field_titre($arrayfields['t.tobill']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ');
  876. }
  877. if (!empty($arrayfields['t.billed']['checked'])) {
  878. print_liste_field_titre($arrayfields['t.billed']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'right ');
  879. }
  880. }
  881. // Contacts of task, disabled because available by default jsut after
  882. /*
  883. if (!empty($conf->global->PROJECT_SHOW_CONTACTS_IN_LIST)) {
  884. print_liste_field_titre("TaskRessourceLinks", $_SERVER["PHP_SELF"], '', '', $param, $sortfield, $sortorder);
  885. }
  886. */
  887. if (!empty($arrayfields['t.budget_amount']['checked'])) {
  888. print_liste_field_titre($arrayfields['t.budget_amount']['label'], $_SERVER["PHP_SELF"], "", "", $param, '', $sortfield, $sortorder, 'center ');
  889. }
  890. if (!empty($arrayfields['c.assigned']['checked'])) {
  891. print_liste_field_titre($arrayfields['c.assigned']['label'], $_SERVER["PHP_SELF"], "", '', $param, '', $sortfield, $sortorder, 'center ', '');
  892. }
  893. // Extra fields
  894. $disablesortlink = 1;
  895. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_list_search_title.tpl.php';
  896. // Hook fields
  897. $parameters = array('arrayfields'=>$arrayfields, 'param'=>$param, 'sortfield'=>$sortfield, 'sortorder'=>$sortorder);
  898. $reshook = $hookmanager->executeHooks('printFieldListTitle', $parameters); // Note that $action and $object may have been modified by hook
  899. print $hookmanager->resPrint;
  900. print_liste_field_titre($selectedfields, $_SERVER["PHP_SELF"], "", '', '', '', $sortfield, $sortorder, 'center maxwidthsearch ');
  901. print "</tr>\n";
  902. if (count($tasksarray) > 0) {
  903. // Show all lines in taskarray (recursive function to go down on tree)
  904. $j = 0; $level = 0;
  905. $nboftaskshown = projectLinesa($j, 0, $tasksarray, $level, true, 0, $tasksrole, $object->id, 1, $object->id, $filterprogresscalc, ($object->usage_bill_time ? 1 : 0), $arrayfields);
  906. } else {
  907. $colspan = 10;
  908. if ($object->usage_bill_time) {
  909. $colspan += 2;
  910. }
  911. print '<tr class="oddeven nobottom"><td colspan="'.$colspan.'"><span class="opacitymedium">'.$langs->trans("NoTasks").'</span></td></tr>';
  912. }
  913. print "</table>";
  914. print '</div>';
  915. print '</form>';
  916. // Test if database is clean. If not we clean it.
  917. //print 'mode='.$_REQUEST["mode"].' $nboftaskshown='.$nboftaskshown.' count($tasksarray)='.count($tasksarray).' count($tasksrole)='.count($tasksrole).'<br>';
  918. if (!empty($user->rights->projet->all->lire)) { // We make test to clean only if user has permission to see all (test may report false positive otherwise)
  919. if ($search_user_id == $user->id) {
  920. if ($nboftaskshown < count($tasksrole)) {
  921. include_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
  922. cleanCorruptedTree($db, 'projet_task', 'fk_task_parent');
  923. }
  924. } else {
  925. if ($nboftaskshown < count($tasksarray) && !GETPOST('search_user_id', 'int')) {
  926. include_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
  927. cleanCorruptedTree($db, 'projet_task', 'fk_task_parent');
  928. }
  929. }
  930. }
  931. }
  932. // End of page
  933. llxFooter();
  934. $db->close();