tasks.php 44 KB

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