timesheet.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. /* Copyright (C) 2014 delcroip <delcroip@gmail.com>
  2. * Copyright (C) 2015-2017 Laurent Destailleur <eldy@users.sourceforge.net>
  3. * Copyright (C) 2021 Josep Lluís Amador <joseplluis@lliuretic.cat>
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  17. */
  18. /* Parse en input data for time entry into timesheet */
  19. function regexEvent(objet,evt,type)
  20. {
  21. console.log('regexEvent type='+type);
  22. switch(type)
  23. {
  24. case 'days':
  25. var regex= /^[0-9]{1}([.,]{1}[0-9]{1})?$/;
  26. if(regex.test(objet.value) )
  27. {
  28. var tmp=objet.value.replace(',','.');
  29. if(tmp<=1.5){
  30. var tmpint=parseInt(tmp);
  31. if(tmp-tmpint>=0.5){
  32. objet.value= tmpint+0.5;
  33. }else{
  34. objet.value= tmpint;
  35. }
  36. }else{
  37. objet.value= '1.5';
  38. }
  39. }else{
  40. objet.value= '0';
  41. }
  42. break;
  43. case 'hours':
  44. var regex= /^[0-9]{1,2}:[0-9]{2}$/;
  45. var regex2=/^[0-9]{1,2}$/;
  46. var regex3= /^[0-9]{1}([.,]{1}[0-9]{1,2})?$/;
  47. if(!regex.test(objet.value))
  48. {
  49. if(regex2.test(objet.value))
  50. objet.value=objet.value+':00';
  51. else if(regex3.test(objet.value)) {
  52. var tmp=parseFloat(objet.value.replace(',','.'));
  53. var rnd=Math.trunc(tmp);
  54. objet.value=rnd+':'+ Math.round(60*(tmp-rnd));
  55. } else
  56. objet.value='';
  57. }
  58. /* alert(jQuery("#"+id).val()); */
  59. break;
  60. case 'timeChar':
  61. //var regex= /^[0-9:]{1}$/;
  62. //alert(event.charCode);
  63. var charCode = (evt.which) ? evt.which : event.keyCode;
  64. if(((charCode >= 48) && (charCode <= 57)) || //num
  65. (charCode===46) || (charCode===8)||// comma & periode
  66. (charCode === 58) || (charCode==44) )// : & all charcode
  67. {
  68. // ((charCode>=96) && (charCode<=105)) || //numpad
  69. return true;
  70. }else
  71. {
  72. return false;
  73. }
  74. break;
  75. default:
  76. break;
  77. }
  78. }
  79. function pad(n) {
  80. return (n < 10) ? ("0" + n) : n;
  81. }
  82. /* function from http://www.timlabonne.com/2013/07/parsing-a-time-string-with-javascript/ */
  83. /* timeStr must be a duration with format XX:YY (AM/PM not supported) */
  84. /* return: nbofextradays (0, 1, ...) */
  85. function parseTime(timeStr, dt)
  86. {
  87. if (!dt) {
  88. dt = new Date();
  89. }
  90. //var time = timeStr.match(/(\d+)(?::(\d\d))?\s*(p?)/i);
  91. var time = timeStr.match(/(\d+)(?::(\d\d))?/i);
  92. if (!time) {
  93. return -1;
  94. }
  95. var hours = parseInt(time[1], 10);
  96. dt.setHours(hours);
  97. dt.setMinutes(parseInt(time[2], 10) || 0);
  98. dt.setSeconds(0, 0);
  99. //console.log("hours="+hours+" => return nbofextradays="+Math.floor(hours / 24)+" hours="+dt.getHours());
  100. return Math.floor(hours / 24);
  101. }
  102. /* Update total. days = column nb starting from 0 */
  103. function updateTotal(days,mode)
  104. {
  105. console.log('updateTotal days='+days+' mode='+mode);
  106. if (mode=="hours")
  107. {
  108. var total = new Date(0);
  109. total.setHours(0);
  110. total.setMinutes(0);
  111. var nbline = document.getElementById('numberOfLines').value;
  112. var startline = 0;
  113. if (document.getElementById('numberOfFirstLine')) {
  114. startline = parseInt(document.getElementById('numberOfFirstLine').value);
  115. }
  116. var nbextradays = 0;
  117. for (var i=-1; i < nbline; i++)
  118. {
  119. /* get value into timespent cell */
  120. var id='timespent['+i+']['+days+']';
  121. var taskTime = new Date(0);
  122. var element = document.getElementById(id);
  123. if (element)
  124. {
  125. /* alert(element.value);*/
  126. if (element.value)
  127. {
  128. result=parseTime(element.value,taskTime);
  129. }
  130. else
  131. {
  132. result=parseTime(element.innerHTML,taskTime);
  133. }
  134. if (result >= 0)
  135. {
  136. nbextradays = nbextradays + Math.floor((total.getHours()+taskTime.getHours() + result*24) / 24);
  137. //console.log("i="+i+" result="+result);
  138. total.setHours(total.getHours()+taskTime.getHours());
  139. total.setMinutes(total.getMinutes()+taskTime.getMinutes());
  140. //console.log("i="+i+" nbextradays cumul="+nbextradays+" h="+total.getHours()+" "+taskTime.getHours());
  141. }
  142. }
  143. /* get value into timeadded cell */
  144. var id='timeadded['+i+']['+days+']';
  145. var taskTime= new Date(0);
  146. var element=document.getElementById(id);
  147. if(element)
  148. {
  149. /* alert(element.value);*/
  150. if (element.value)
  151. {
  152. result=parseTime(element.value,taskTime);
  153. }
  154. else
  155. {
  156. result=parseTime(element.innerHTML,taskTime);
  157. }
  158. if (result >= 0)
  159. {
  160. nbextradays = nbextradays + Math.floor((total.getHours()+taskTime.getHours() + result*24) / 24);
  161. //console.log("i="+i+" result="+result);
  162. total.setHours(total.getHours()+taskTime.getHours());
  163. total.setMinutes(total.getMinutes()+taskTime.getMinutes());
  164. //console.log("i="+i+" nbextradays cumul="+nbextradays+" h="+total.getHours()+" "+taskTime.getHours());
  165. }
  166. }
  167. }
  168. // Add data on the perday view
  169. jQuery('.inputhour').each(function( index ) {
  170. if (this.value)
  171. {
  172. var taskTime= new Date(0);
  173. /*console.log(total.getHours())
  174. console.log(this.value)
  175. alert(element.value);*/
  176. if (this.value)
  177. {
  178. console.log(this.value+':00')
  179. result=parseTime(this.value+':00',taskTime);
  180. }
  181. else
  182. {
  183. result=parseTime(this.innerHTML+':00',taskTime);
  184. }
  185. if (result >= 0)
  186. {
  187. total.setHours(total.getHours()+taskTime.getHours());
  188. }
  189. console.log(total.getHours())
  190. }
  191. });
  192. // Add data on the perday view
  193. jQuery('.inputminute').each(function( index ) {
  194. if (this.value)
  195. {
  196. var taskTime= new Date(0);
  197. /* console.log(total.getHours())
  198. console.log(this.value)
  199. alert(element.value);*/
  200. if (this.value)
  201. {
  202. console.log('00:'+this.value)
  203. result=parseTime('00:'+"00".substring(0, 2 - this.value.length) + this.value,taskTime);
  204. }
  205. else
  206. {
  207. result=parseTime('00:'+"00".substring(0, 2 - this.innerHTML) + this.innerHTML,taskTime);
  208. }
  209. if (result >= 0)
  210. {
  211. total.setMinutes(total.getMinutes()+taskTime.getMinutes());
  212. }
  213. console.log(total.getMinutes())
  214. }
  215. });
  216. var stringdays = days;
  217. if (startline >= 1 && startline <= 9 && stringdays < 10) {
  218. stringdays = '0'+stringdays;
  219. }
  220. /* Output total in top of column */
  221. if (total.getHours() || total.getMinutes()) jQuery('.totalDay'+stringdays).addClass("bold");
  222. else jQuery('.totalDay'+stringdays).removeClass("bold");
  223. var texttoshow = pad(nbextradays * 24 + total.getHours())+':'+pad(total.getMinutes());
  224. jQuery('.totalDay'+stringdays).text(texttoshow);
  225. /* Output total of all total */
  226. var totalhour = 0;
  227. var totalmin = 0;
  228. for (var i=0; i<7; i++)
  229. {
  230. stringdays = (i + startline);
  231. if (startline >= 1 && startline <= 9 && stringdays < 10) {
  232. stringdays = '0'+stringdays;
  233. }
  234. var taskTime= new Date(0);
  235. result=parseTime(jQuery('.totalDay'+stringdays).text(),taskTime);
  236. if (result >= 0)
  237. {
  238. totalhour = totalhour + taskTime.getHours() + result*24;
  239. totalmin = totalmin + taskTime.getMinutes();
  240. }
  241. }
  242. morehours = Math.floor(totalmin / 60);
  243. totalmin = totalmin % 60;
  244. jQuery('.totalDayAll').text(pad(morehours + totalhour)+':'+pad(totalmin));
  245. }
  246. else
  247. {
  248. var total =0;
  249. var nbline = document.getElementById('numberOfLines').value;
  250. for (var i=-1; i<nbline; i++)
  251. {
  252. var id='timespent['+i+']['+days+']';
  253. var taskTime= new Date(0);
  254. var element=document.getElementById(id);
  255. if(element)
  256. {
  257. if (element.value)
  258. {
  259. total+=parseInt(element.value);
  260. }
  261. else
  262. {
  263. total+=parseInt(element.innerHTML);
  264. }
  265. }
  266. var id='timeadded['+i+']['+days+']';
  267. var taskTime= new Date(0);
  268. var element=document.getElementById(id);
  269. if(element)
  270. {
  271. if (element.value)
  272. {
  273. total+=parseInt(element.value);
  274. }
  275. else
  276. {
  277. total+=parseInt(element.innerHTML);
  278. }
  279. }
  280. }
  281. var stringdays = days;
  282. if (startline >= 1 && startline <= 9 && stringdays < 10) {
  283. stringdays = '0'+stringdays;
  284. console.log(stringdays);
  285. }
  286. if (total) jQuery('.totalDay'+stringdays).addClass("bold");
  287. else jQuery('.totalDay'+stringdays).removeClass("bold");
  288. jQuery('.totalDay'+stringdays).text(total);
  289. }
  290. }