card.php 116 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912
  1. <?php
  2. /* Copyright (C) 2004-2006 Rodolphe Quiedeville <rodolphe@quiedeville.org>
  3. * Copyright (C) 2004-2015 Laurent Destailleur <eldy@users.sourceforge.net>
  4. * Copyright (C) 2005 Eric Seigne <eric.seigne@ryxeo.com>
  5. * Copyright (C) 2005-2016 Regis Houssin <regis.houssin@inodbox.com>
  6. * Copyright (C) 2010-2015 Juanjo Menent <jmenent@2byte.es>
  7. * Copyright (C) 2011-2018 Philippe Grand <philippe.grand@atoo-net.com>
  8. * Copyright (C) 2012-2016 Marcos García <marcosgdf@gmail.com>
  9. * Copyright (C) 2013 Florian Henry <florian.henry@open-concept.pro>
  10. * Copyright (C) 2014 Ion Agorria <ion@agorria.com>
  11. * Copyright (C) 2018-2019 Frédéric France <frederic.france@netlogic.fr>
  12. * Copyright (C) 2022 Charlene Benke <charlene@patas-monkey.com>
  13. *
  14. * This program is free software; you can redistribute it and/or modify
  15. * it under the terms of the GNU General Public License as published by
  16. * the Free Software Foundation; either version 2 of the License, or
  17. * (at your option) any later version.
  18. *
  19. * This program is distributed in the hope that it will be useful,
  20. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  21. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  22. * GNU General Public License for more details.
  23. *
  24. * You should have received a copy of the GNU General Public License
  25. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  26. * or see https://www.gnu.org/
  27. */
  28. /**
  29. * \file htdocs/fourn/commande/card.php
  30. * \ingroup supplier, order
  31. * \brief Card supplier order
  32. */
  33. require '../../main.inc.php';
  34. require_once DOL_DOCUMENT_ROOT.'/core/class/html.formfile.class.php';
  35. require_once DOL_DOCUMENT_ROOT.'/core/class/html.formorder.class.php';
  36. require_once DOL_DOCUMENT_ROOT.'/core/modules/supplier_order/modules_commandefournisseur.php';
  37. require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.commande.class.php';
  38. require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.product.class.php';
  39. require_once DOL_DOCUMENT_ROOT.'/core/lib/fourn.lib.php';
  40. require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
  41. require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
  42. require_once DOL_DOCUMENT_ROOT.'/core/class/doleditor.class.php';
  43. if (!empty($conf->supplier_proposal->enabled)) {
  44. require_once DOL_DOCUMENT_ROOT.'/supplier_proposal/class/supplier_proposal.class.php';
  45. }
  46. if (!empty($conf->product->enabled)) {
  47. require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
  48. }
  49. if (!empty($conf->projet->enabled)) {
  50. require_once DOL_DOCUMENT_ROOT.'/projet/class/project.class.php';
  51. require_once DOL_DOCUMENT_ROOT.'/core/class/html.formprojet.class.php';
  52. }
  53. require_once NUSOAP_PATH.'/nusoap.php'; // Include SOAP
  54. if (!empty($conf->variants->enabled)) {
  55. require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductCombination.class.php';
  56. }
  57. $langs->loadLangs(array('admin', 'orders', 'sendings', 'companies', 'bills', 'propal', 'receptions', 'supplier_proposal', 'deliveries', 'products', 'stocks', 'productbatch'));
  58. if (!empty($conf->incoterm->enabled)) {
  59. $langs->load('incoterm');
  60. }
  61. $id = GETPOST('id', 'int');
  62. $ref = GETPOST('ref', 'alpha');
  63. $action = GETPOST('action', 'alpha');
  64. $confirm = GETPOST('confirm', 'alpha');
  65. $contextpage = GETPOST('contextpage', 'aZ') ? GETPOST('contextpage', 'aZ') : 'purchaseordercard'; // To manage different context of search
  66. $backtopage = GETPOST('backtopage', 'alpha');
  67. $backtopageforcancel = GETPOST('backtopageforcancel', 'alpha');
  68. $socid = GETPOST('socid', 'int');
  69. $projectid = GETPOST('projectid', 'int');
  70. $cancel = GETPOST('cancel', 'alpha');
  71. $lineid = GETPOST('lineid', 'int');
  72. $origin = GETPOST('origin', 'alpha');
  73. $originid = (GETPOST('originid', 'int') ? GETPOST('originid', 'int') : GETPOST('origin_id', 'int')); // For backward compatibility
  74. //PDF
  75. $hidedetails = (GETPOST('hidedetails', 'int') ? GETPOST('hidedetails', 'int') : (!empty($conf->global->MAIN_GENERATE_DOCUMENTS_HIDE_DETAILS) ? 1 : 0));
  76. $hidedesc = (GETPOST('hidedesc', 'int') ? GETPOST('hidedesc', 'int') : (!empty($conf->global->MAIN_GENERATE_DOCUMENTS_HIDE_DESC) ? 1 : 0));
  77. $hideref = (GETPOST('hideref', 'int') ? GETPOST('hideref', 'int') : (!empty($conf->global->MAIN_GENERATE_DOCUMENTS_HIDE_REF) ? 1 : 0));
  78. $datelivraison = dol_mktime(GETPOST('liv_hour', 'int'), GETPOST('liv_min', 'int'), GETPOST('liv_sec', 'int'), GETPOST('liv_month', 'int'), GETPOST('liv_day', 'int'), GETPOST('liv_year', 'int'));
  79. // Security check
  80. if ($user->socid) {
  81. $socid = $user->socid;
  82. }
  83. // Initialize technical object to manage hooks of page. Note that conf->hooks_modules contains array of hook context
  84. $hookmanager->initHooks(array('ordersuppliercard', 'globalcard'));
  85. $object = new CommandeFournisseur($db);
  86. $extrafields = new ExtraFields($db);
  87. // fetch optionals attributes and labels
  88. $extrafields->fetch_name_optionals_label($object->table_element);
  89. if ($user->socid) {
  90. $socid = $user->socid;
  91. }
  92. // Load object
  93. if ($id > 0 || !empty($ref)) {
  94. $ret = $object->fetch($id, $ref);
  95. if ($ret < 0) {
  96. dol_print_error($db, $object->error);
  97. }
  98. $ret = $object->fetch_thirdparty();
  99. if ($ret < 0) {
  100. dol_print_error($db, $object->error);
  101. }
  102. } elseif (!empty($socid) && $socid > 0) {
  103. $fourn = new Fournisseur($db);
  104. $ret = $fourn->fetch($socid);
  105. if ($ret < 0) {
  106. dol_print_error($db, $object->error);
  107. }
  108. $object->socid = $fourn->id;
  109. $ret = $object->fetch_thirdparty();
  110. if ($ret < 0) {
  111. dol_print_error($db, $object->error);
  112. }
  113. }
  114. // Security check
  115. $isdraft = (isset($object->statut) && ($object->statut == $object::STATUS_DRAFT) ? 1 : 0);
  116. $result = restrictedArea($user, 'fournisseur', $id, 'commande_fournisseur', 'commande', 'fk_soc', 'rowid', $isdraft);
  117. // Common permissions
  118. $usercanread = ($user->rights->fournisseur->commande->lire || $user->rights->supplier_order->lire);
  119. $usercancreate = ($user->rights->fournisseur->commande->creer || $user->rights->supplier_order->creer);
  120. $usercandelete = (($user->rights->fournisseur->commande->supprimer || $user->rights->supplier_order->supprimer) || ($usercancreate && isset($object->statut) && $object->statut == $object::STATUS_DRAFT));
  121. // Advanced permissions
  122. $usercanvalidate = ((empty($conf->global->MAIN_USE_ADVANCED_PERMS) && !empty($usercancreate)) || (!empty($conf->global->MAIN_USE_ADVANCED_PERMS) && !empty($user->rights->fournisseur->supplier_order_advance->validate)));
  123. // Additional area permissions
  124. $usercanapprove = $user->rights->fournisseur->commande->approuver;
  125. $usercanapprovesecond = $user->rights->fournisseur->commande->approve2;
  126. $usercanorder = $user->rights->fournisseur->commande->commander;
  127. if (empty($conf->reception->enabled)) {
  128. $usercanreceive = $user->rights->fournisseur->commande->receptionner;
  129. } else {
  130. $usercanreceive = $user->rights->reception->creer;
  131. }
  132. // Permissions for includes
  133. $permissionnote = $usercancreate; // Used by the include of actions_setnotes.inc.php
  134. $permissiondellink = $usercancreate; // Used by the include of actions_dellink.inc.php
  135. $permissiontoedit = $usercancreate; // Used by the include of actions_lineupdown.inc.php
  136. $permissiontoadd = $usercancreate; // Used by the include of actions_addupdatedelete.inc.php
  137. // Project permission
  138. $caneditproject = false;
  139. if (!empty($conf->projet->enabled)) {
  140. $caneditproject = empty($conf->global->SUPPLIER_ORDER_FORBID_EDIT_PROJECT) || ($object->statut == CommandeFournisseur::STATUS_DRAFT && preg_match('/^[\(]?PROV/i', $object->ref));
  141. }
  142. /*
  143. * Actions
  144. */
  145. $parameters = array('socid'=>$socid);
  146. $reshook = $hookmanager->executeHooks('doActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks
  147. if ($reshook < 0) {
  148. setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
  149. }
  150. if (empty($reshook)) {
  151. $backurlforlist = DOL_URL_ROOT.'/fourn/commande/list.php'.($socid > 0 ? '?socid='.((int) $socid) : '');
  152. if (empty($backtopage) || ($cancel && empty($id))) {
  153. if (empty($backtopage) || ($cancel && strpos($backtopage, '__ID__'))) {
  154. if (empty($id) && (($action != 'add' && $action != 'create') || $cancel)) {
  155. $backtopage = $backurlforlist;
  156. } else {
  157. $backtopage = DOL_URL_ROOT.'/fourn/commande/card.php?id='.($id > 0 ? $id : '__ID__');
  158. }
  159. }
  160. }
  161. if ($cancel) {
  162. if (!empty($backtopageforcancel)) {
  163. header("Location: ".$backtopageforcancel);
  164. exit;
  165. } elseif (!empty($backtopage)) {
  166. header("Location: ".$backtopage);
  167. exit;
  168. }
  169. $action = '';
  170. }
  171. include DOL_DOCUMENT_ROOT.'/core/actions_setnotes.inc.php'; // Must be include, not include_once
  172. include DOL_DOCUMENT_ROOT.'/core/actions_dellink.inc.php'; // Must be include, not include_once
  173. include DOL_DOCUMENT_ROOT.'/core/actions_lineupdown.inc.php'; // Must be include, not include_once
  174. if ($action == 'setref_supplier' && $usercancreate) {
  175. $result = $object->setValueFrom('ref_supplier', GETPOST('ref_supplier', 'alpha'), '', null, 'text', '', $user, 'ORDER_SUPPLIER_MODIFY');
  176. if ($result < 0) {
  177. setEventMessages($object->error, $object->errors, 'errors');
  178. }
  179. }
  180. // Set incoterm
  181. if ($action == 'set_incoterms' && $usercancreate) {
  182. $result = $object->setIncoterms(GETPOST('incoterm_id', 'int'), GETPOST('location_incoterms', 'alpha'));
  183. if ($result < 0) {
  184. setEventMessages($object->error, $object->errors, 'errors');
  185. }
  186. }
  187. // payment conditions
  188. if ($action == 'setconditions' && $usercancreate) {
  189. $result = $object->setPaymentTerms(GETPOST('cond_reglement_id', 'int'));
  190. if ($result < 0) {
  191. setEventMessages($object->error, $object->errors, 'errors');
  192. }
  193. }
  194. // payment mode
  195. if ($action == 'setmode' && $usercancreate) {
  196. $result = $object->setPaymentMethods(GETPOST('mode_reglement_id', 'int'));
  197. if ($result < 0) {
  198. setEventMessages($object->error, $object->errors, 'errors');
  199. }
  200. } elseif ($action == 'setmulticurrencycode' && $usercancreate) {
  201. // Multicurrency Code
  202. $result = $object->setMulticurrencyCode(GETPOST('multicurrency_code', 'alpha'));
  203. } elseif ($action == 'setmulticurrencyrate' && $usercancreate) {
  204. // Multicurrency rate
  205. $result = $object->setMulticurrencyRate(price2num(GETPOST('multicurrency_tx')), GETPOST('calculation_mode', 'int'));
  206. }
  207. // bank account
  208. if ($action == 'setbankaccount' && $usercancreate) {
  209. $result = $object->setBankAccount(GETPOST('fk_account', 'int'));
  210. if ($result < 0) {
  211. setEventMessages($object->error, $object->errors, 'errors');
  212. }
  213. }
  214. // date of delivery
  215. if ($action == 'setdate_livraison' && $usercancreate) {
  216. $result = $object->setDeliveryDate($user, $datelivraison);
  217. if ($result < 0) {
  218. setEventMessages($object->error, $object->errors, 'errors');
  219. }
  220. }
  221. // Set project
  222. if ($action == 'classin' && $usercancreate && $caneditproject) {
  223. $result = $object->setProject($projectid);
  224. if ($result < 0) {
  225. setEventMessages($object->error, $object->errors, 'errors');
  226. }
  227. }
  228. // Edit Thirdparty
  229. if (!empty($conf->global->MAIN_CAN_EDIT_SUPPLIER_ON_SUPPLIER_ORDER) && $action == 'set_thirdparty' && $usercancreate && $object->statut == CommandeFournisseur::STATUS_DRAFT) {
  230. $new_socid = GETPOST('new_socid', 'int');
  231. if (!empty($new_socid) && $new_socid != $object->thirdparty->id) {
  232. $db->begin();
  233. // Update supplier
  234. $sql = 'UPDATE '.MAIN_DB_PREFIX.'commande_fournisseur';
  235. $sql .= ' SET fk_soc = '.((int) $new_socid);
  236. $sql .= ' WHERE fk_soc = '.((int) $object->thirdparty->id);
  237. $sql .= ' AND rowid = '.((int) $object->id);
  238. $res = $db->query($sql);
  239. if (!$res) {
  240. $db->rollback();
  241. } else {
  242. $db->commit();
  243. // Replace prices for each lines by new supplier prices
  244. foreach ($object->lines as $l) {
  245. $sql = 'SELECT price, unitprice, tva_tx, ref_fourn';
  246. $sql .= ' FROM '.MAIN_DB_PREFIX.'product_fournisseur_price';
  247. $sql .= ' WHERE fk_product = '.((int) $l->fk_product);
  248. $sql .= ' AND fk_soc = '.((int) $new_socid);
  249. $sql .= ' ORDER BY unitprice ASC';
  250. $resql = $db->query($sql);
  251. if ($resql) {
  252. $num_row = $db->num_rows($resql);
  253. if (empty($num_row)) {
  254. // No product price for this supplier !
  255. $l->subprice = 0;
  256. $l->total_ht = 0;
  257. $l->total_tva = 0;
  258. $l->total_ttc = 0;
  259. $l->ref_supplier = '';
  260. $l->update();
  261. } else {
  262. // No need for loop to keep best supplier price
  263. $obj = $db->fetch_object($resql);
  264. $l->subprice = $obj->unitprice;
  265. $l->total_ht = $obj->price;
  266. $l->tva_tx = $obj->tva_tx;
  267. $l->total_tva = $l->total_ht * ($obj->tva_tx / 100);
  268. $l->total_ttc = $l->total_ht + $l->total_tva;
  269. $l->ref_supplier = $obj->ref_fourn;
  270. $l->update();
  271. }
  272. } else {
  273. dol_print_error($db);
  274. }
  275. $db->free($resql);
  276. }
  277. $object->update_price();
  278. }
  279. }
  280. header('Location: '.$_SERVER['PHP_SELF'].'?id='.$object->id);
  281. exit;
  282. }
  283. if ($action == 'setremisepercent' && $usercancreate) {
  284. $result = $object->set_remise($user, price2num(GETPOST('remise_percent')));
  285. if ($result < 0) {
  286. setEventMessages($object->error, $object->errors, 'errors');
  287. }
  288. }
  289. if ($action == 'reopen') { // no test on permission here, permission to use will depends on status
  290. if (in_array($object->statut, array(1, 2, 3, 4, 5, 6, 7, 9))) {
  291. if ($object->statut == 1) {
  292. $newstatus = 0; // Validated->Draft
  293. } elseif ($object->statut == 2) {
  294. $newstatus = 0; // Approved->Draft
  295. } elseif ($object->statut == 3) {
  296. $newstatus = 2; // Ordered->Approved
  297. } elseif ($object->statut == 4) {
  298. $newstatus = 3;
  299. } elseif ($object->statut == 5) {
  300. //$newstatus=2; // Ordered
  301. // TODO Can we set it to submited ?
  302. //$newstatus=3; // Submited
  303. // TODO If there is at least one reception, we can set to Received->Received partially
  304. $newstatus = 4; // Received partially
  305. } elseif ($object->statut == 6) {
  306. $newstatus = 2; // Canceled->Approved
  307. } elseif ($object->statut == 7) {
  308. $newstatus = 3; // Canceled->Process running
  309. } elseif ($object->statut == 9) {
  310. $newstatus = 1; // Refused->Validated
  311. } else {
  312. $newstatus = 2;
  313. }
  314. //print "old status = ".$object->statut.' new status = '.$newstatus;
  315. $db->begin();
  316. $result = $object->setStatus($user, $newstatus);
  317. if ($result > 0) {
  318. // Currently the "Re-open" also remove the billed flag because there is no button "Set unpaid" yet.
  319. $sql = 'UPDATE '.MAIN_DB_PREFIX.'commande_fournisseur';
  320. $sql .= ' SET billed = 0';
  321. $sql .= ' WHERE rowid = '.((int) $object->id);
  322. $resql = $db->query($sql);
  323. if ($newstatus == 0) {
  324. $sql = 'UPDATE '.MAIN_DB_PREFIX.'commande_fournisseur';
  325. $sql .= ' SET fk_user_approve = null, fk_user_approve2 = null, date_approve = null, date_approve2 = null';
  326. $sql .= ' WHERE rowid = '.((int) $object->id);
  327. $resql = $db->query($sql);
  328. }
  329. $db->commit();
  330. header('Location: '.$_SERVER["PHP_SELF"].'?id='.$object->id);
  331. exit;
  332. } else {
  333. $db->rollback();
  334. setEventMessages($object->error, $object->errors, 'errors');
  335. }
  336. }
  337. }
  338. /*
  339. * Classify supplier order as billed
  340. */
  341. if ($action == 'classifybilled' && $usercancreate) {
  342. $ret = $object->classifyBilled($user);
  343. if ($ret < 0) {
  344. setEventMessages($object->error, $object->errors, 'errors');
  345. }
  346. }
  347. // Add a product line
  348. if ($action == 'addline' && $usercancreate) {
  349. $db->begin();
  350. $langs->load('errors');
  351. $error = 0;
  352. // Set if we used free entry or predefined product
  353. $predef = '';
  354. $product_desc = (GETPOSTISSET('dp_desc') ? GETPOST('dp_desc', 'restricthtml') : '');
  355. $date_start = dol_mktime(GETPOST('date_start'.$predef.'hour'), GETPOST('date_start'.$predef.'min'), GETPOST('date_start'.$predef.'sec'), GETPOST('date_start'.$predef.'month'), GETPOST('date_start'.$predef.'day'), GETPOST('date_start'.$predef.'year'));
  356. $date_end = dol_mktime(GETPOST('date_end'.$predef.'hour'), GETPOST('date_end'.$predef.'min'), GETPOST('date_end'.$predef.'sec'), GETPOST('date_end'.$predef.'month'), GETPOST('date_end'.$predef.'day'), GETPOST('date_end'.$predef.'year'));
  357. $prod_entry_mode = GETPOST('prod_entry_mode');
  358. if ($prod_entry_mode == 'free') {
  359. $idprod = 0;
  360. $price_ht = price2num(GETPOST('price_ht'), 'MU', 2);
  361. $tva_tx = (GETPOST('tva_tx') ? GETPOST('tva_tx') : 0);
  362. } else {
  363. $idprod = GETPOST('idprod', 'int');
  364. $price_ht = price2num(GETPOST('price_ht'), 'MU', 2);
  365. $tva_tx = '';
  366. }
  367. $qty = price2num(GETPOST('qty'.$predef, 'alpha'), 'MS');
  368. $remise_percent = price2num(GETPOST('remise_percent'.$predef), 2);
  369. $price_ht_devise = price2num(GETPOST('multicurrency_price_ht'), 'CU', 2);
  370. // Extrafields
  371. $extralabelsline = $extrafields->fetch_name_optionals_label($object->table_element_line);
  372. $array_options = $extrafields->getOptionalsFromPost($object->table_element_line, $predef);
  373. // Unset extrafield
  374. if (is_array($extralabelsline)) {
  375. // Get extra fields
  376. foreach ($extralabelsline as $key => $value) {
  377. unset($_POST["options_".$key]);
  378. }
  379. }
  380. if ($prod_entry_mode == 'free' && GETPOST('price_ht') < 0 && $qty < 0) {
  381. setEventMessages($langs->trans('ErrorBothFieldCantBeNegative', $langs->transnoentitiesnoconv('UnitPrice'), $langs->transnoentitiesnoconv('Qty')), null, 'errors');
  382. $error++;
  383. }
  384. if ($prod_entry_mode == 'free' && !GETPOST('idprodfournprice') && GETPOST('type') < 0) {
  385. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv('Type')), null, 'errors');
  386. $error++;
  387. }
  388. if ($prod_entry_mode == 'free' && GETPOST('price_ht') === '' && GETPOST('price_ttc') === '' && $price_ht_devise === '') { // Unit price can be 0 but not ''
  389. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv('UnitPrice')), null, 'errors');
  390. $error++;
  391. }
  392. if ($prod_entry_mode == 'free' && !GETPOST('dp_desc')) {
  393. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv('Description')), null, 'errors');
  394. $error++;
  395. }
  396. if (GETPOST('qty', 'int') == '') {
  397. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv('Qty')), null, 'errors');
  398. $error++;
  399. }
  400. if (!$error && !empty($conf->variants->enabled) && $prod_entry_mode != 'free') {
  401. if ($combinations = GETPOST('combinations', 'array')) {
  402. //Check if there is a product with the given combination
  403. $prodcomb = new ProductCombination($db);
  404. if ($res = $prodcomb->fetchByProductCombination2ValuePairs($idprod, $combinations)) {
  405. $idprod = $res->fk_product_child;
  406. } else {
  407. setEventMessages($langs->trans('ErrorProductCombinationNotFound'), null, 'errors');
  408. $error++;
  409. }
  410. }
  411. }
  412. if ($prod_entry_mode != 'free' && empty($error)) { // With combolist mode idprodfournprice is > 0 or -1. With autocomplete, idprodfournprice is > 0 or ''
  413. $productsupplier = new ProductFournisseur($db);
  414. $idprod = 0;
  415. if (GETPOST('idprodfournprice', 'alpha') == -1 || GETPOST('idprodfournprice', 'alpha') == '') {
  416. $idprod = -99; // Same behaviour than with combolist. When not select idprodfournprice is now -99 (to avoid conflict with next action that may return -1, -2, ...)
  417. }
  418. $reg = array();
  419. if (preg_match('/^idprod_([0-9]+)$/', GETPOST('idprodfournprice', 'alpha'), $reg)) {
  420. $idprod = $reg[1];
  421. $res = $productsupplier->fetch($idprod); // Load product from its id
  422. // Call to init some price properties of $productsupplier
  423. // So if a supplier price already exists for another thirdparty (first one found), we use it as reference price
  424. if (!empty($conf->global->SUPPLIER_TAKE_FIRST_PRICE_IF_NO_PRICE_FOR_CURRENT_SUPPLIER)) {
  425. $fksoctosearch = 0;
  426. $productsupplier->get_buyprice(0, -1, $idprod, 'none', $fksoctosearch); // We force qty to -1 to be sure to find if a supplier price exist
  427. if ($productsupplier->fourn_socid != $socid) { // The price we found is for another supplier, so we clear supplier price
  428. $productsupplier->ref_supplier = '';
  429. }
  430. } else {
  431. $fksoctosearch = $object->thirdparty->id;
  432. $productsupplier->get_buyprice(0, -1, $idprod, 'none', $fksoctosearch); // We force qty to -1 to be sure to find if a supplier price exist
  433. }
  434. } elseif (GETPOST('idprodfournprice', 'alpha') > 0) {
  435. $qtytosearch = $qty; // Just to see if a price exists for the quantity. Not used to found vat.
  436. //$qtytosearch=-1; // We force qty to -1 to be sure to find if a supplier price exist
  437. $idprod = $productsupplier->get_buyprice(GETPOST('idprodfournprice', 'alpha'), $qtytosearch);
  438. $res = $productsupplier->fetch($idprod);
  439. }
  440. if ($idprod > 0) {
  441. $label = $productsupplier->label;
  442. // Define output language
  443. if (!empty($conf->global->MAIN_MULTILANGS) && !empty($conf->global->PRODUIT_TEXTS_IN_THIRDPARTY_LANGUAGE)) {
  444. $outputlangs = $langs;
  445. $newlang = '';
  446. if (empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  447. $newlang = GETPOST('lang_id', 'aZ09');
  448. }
  449. if (empty($newlang)) {
  450. $newlang = $object->thirdparty->default_lang;
  451. }
  452. if (!empty($newlang)) {
  453. $outputlangs = new Translate("", $conf);
  454. $outputlangs->setDefaultLang($newlang);
  455. }
  456. $desc = (!empty($productsupplier->multilangs[$outputlangs->defaultlang]["description"])) ? $productsupplier->multilangs[$outputlangs->defaultlang]["description"] : $productsupplier->description;
  457. } else {
  458. $desc = $productsupplier->description;
  459. }
  460. // if we use supplier description of the products
  461. if (!empty($productsupplier->desc_supplier) && !empty($conf->global->PRODUIT_FOURN_TEXTS)) {
  462. $desc = $productsupplier->desc_supplier;
  463. }
  464. //If text set in desc is the same as product descpription (as now it's preloaded) whe add it only one time
  465. if ($product_desc==$desc && !empty($conf->global->PRODUIT_AUTOFILL_DESC)) {
  466. $product_desc='';
  467. }
  468. if (!empty($product_desc) && !empty($conf->global->MAIN_NO_CONCAT_DESCRIPTION)) {
  469. $desc = $product_desc;
  470. }
  471. if (!empty($product_desc) && trim($product_desc) != trim($desc)) {
  472. $desc = dol_concatdesc($desc, $product_desc, '', !empty($conf->global->MAIN_CHANGE_ORDER_CONCAT_DESCRIPTION));
  473. }
  474. $type = $productsupplier->type;
  475. if ($price_ht != '' || $price_ht_devise != '') {
  476. $price_base_type = 'HT';
  477. $pu = price2num($price_ht, 'MU');
  478. $pu_ht_devise = price2num($price_ht_devise, 'MU');
  479. } else {
  480. $price_base_type = ($productsupplier->fourn_price_base_type ? $productsupplier->fourn_price_base_type : 'HT');
  481. if (empty($object->multicurrency_code) || ($productsupplier->fourn_multicurrency_code != $object->multicurrency_code)) { // If object is in a different currency and price not in this currency
  482. $pu = $productsupplier->fourn_pu;
  483. $pu_ht_devise = 0;
  484. } else {
  485. $pu = $productsupplier->fourn_pu;
  486. $pu_ht_devise = $productsupplier->fourn_multicurrency_unitprice;
  487. /*var_dump($pu);
  488. var_dump($pu_ht_devise);exit;*/
  489. }
  490. }
  491. $ref_supplier = $productsupplier->ref_supplier;
  492. $tva_tx = get_default_tva($object->thirdparty, $mysoc, $productsupplier->id, GETPOST('idprodfournprice', 'alpha'));
  493. $tva_npr = get_default_npr($object->thirdparty, $mysoc, $productsupplier->id, GETPOST('idprodfournprice', 'alpha'));
  494. if (empty($tva_tx)) {
  495. $tva_npr = 0;
  496. }
  497. $localtax1_tx = get_localtax($tva_tx, 1, $mysoc, $object->thirdparty, $tva_npr);
  498. $localtax2_tx = get_localtax($tva_tx, 2, $mysoc, $object->thirdparty, $tva_npr);
  499. if (empty($pu)) {
  500. $pu = 0; // If pu is '' or null, we force to have a numeric value
  501. }
  502. $result = $object->addline(
  503. $desc,
  504. $pu,
  505. $qty,
  506. $tva_tx,
  507. $localtax1_tx,
  508. $localtax2_tx,
  509. $idprod,
  510. 0, // We already have the $idprod always defined
  511. $ref_supplier,
  512. $remise_percent,
  513. $price_base_type,
  514. $pu_ttc,
  515. $type,
  516. $tva_npr,
  517. '',
  518. $date_start,
  519. $date_end,
  520. $array_options,
  521. $productsupplier->fk_unit,
  522. $pu_ht_devise,
  523. '',
  524. 0
  525. );
  526. }
  527. if ($idprod == -99 || $idprod == 0) {
  528. // Product not selected
  529. $error++;
  530. $langs->load("errors");
  531. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("ProductOrService")), null, 'errors');
  532. }
  533. if ($idprod == -1) {
  534. // Quantity too low
  535. $error++;
  536. $langs->load("errors");
  537. setEventMessages($langs->trans("ErrorQtyTooLowForThisSupplier"), null, 'errors');
  538. }
  539. } elseif (empty($error)) { // $price_ht is already set
  540. $tva_npr = (preg_match('/\*/', $tva_tx) ? 1 : 0);
  541. $tva_tx = str_replace('*', '', $tva_tx);
  542. $label = (GETPOST('product_label') ? GETPOST('product_label') : '');
  543. $desc = $product_desc;
  544. $type = GETPOST('type');
  545. $ref_supplier = GETPOST('fourn_ref', 'alpha');
  546. $fk_unit = GETPOST('units', 'alpha');
  547. if (!preg_match('/\((.*)\)/', $tva_tx)) {
  548. $tva_tx = price2num($tva_tx); // When vat is text input field
  549. }
  550. // Local Taxes
  551. $localtax1_tx = get_localtax($tva_tx, 1, $mysoc, $object->thirdparty);
  552. $localtax2_tx = get_localtax($tva_tx, 2, $mysoc, $object->thirdparty);
  553. if ($price_ht !== '') {
  554. $pu_ht = price2num($price_ht, 'MU'); // $pu_ht must be rounded according to settings
  555. } else {
  556. $pu_ttc = price2num(GETPOST('price_ttc'), 'MU');
  557. $pu_ht = price2num($pu_ttc / (1 + ($tva_tx / 100)), 'MU'); // $pu_ht must be rounded according to settings
  558. }
  559. $price_base_type = 'HT';
  560. $pu_ht_devise = price2num($price_ht_devise, 'MU');
  561. $result = $object->addline($desc, $pu_ht, $qty, $tva_tx, $localtax1_tx, $localtax2_tx, 0, 0, $ref_supplier, $remise_percent, $price_base_type, $pu_ttc, $type, '', '', $date_start, $date_end, $array_options, $fk_unit, $pu_ht_devise);
  562. }
  563. //print "xx".$tva_tx; exit;
  564. if (!$error && $result > 0) {
  565. $db->commit();
  566. $ret = $object->fetch($object->id); // Reload to get new records
  567. // Define output language
  568. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  569. $outputlangs = $langs;
  570. $newlang = '';
  571. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  572. $newlang = GETPOST('lang_id', 'aZ09');
  573. }
  574. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  575. $newlang = $object->thirdparty->default_lang;
  576. }
  577. if (!empty($newlang)) {
  578. $outputlangs = new Translate("", $conf);
  579. $outputlangs->setDefaultLang($newlang);
  580. }
  581. $model = $object->model_pdf;
  582. $ret = $object->fetch($id); // Reload to get new records
  583. $result = $object->generateDocument($model, $outputlangs, $hidedetails, $hidedesc, $hideref);
  584. if ($result < 0) {
  585. dol_print_error($db, $result);
  586. }
  587. }
  588. unset($_POST ['prod_entry_mode']);
  589. unset($_POST['qty']);
  590. unset($_POST['type']);
  591. unset($_POST['remise_percent']);
  592. unset($_POST['pu']);
  593. unset($_POST['price_ht']);
  594. unset($_POST['multicurrency_price_ht']);
  595. unset($_POST['price_ttc']);
  596. unset($_POST['fourn_ref']);
  597. unset($_POST['tva_tx']);
  598. unset($_POST['label']);
  599. unset($localtax1_tx);
  600. unset($localtax2_tx);
  601. unset($_POST['np_marginRate']);
  602. unset($_POST['np_markRate']);
  603. unset($_POST['dp_desc']);
  604. unset($_POST['idprodfournprice']);
  605. unset($_POST['date_starthour']);
  606. unset($_POST['date_startmin']);
  607. unset($_POST['date_startsec']);
  608. unset($_POST['date_startday']);
  609. unset($_POST['date_startmonth']);
  610. unset($_POST['date_startyear']);
  611. unset($_POST['date_endhour']);
  612. unset($_POST['date_endmin']);
  613. unset($_POST['date_endsec']);
  614. unset($_POST['date_endday']);
  615. unset($_POST['date_endmonth']);
  616. unset($_POST['date_endyear']);
  617. } else {
  618. $db->rollback();
  619. setEventMessages($object->error, $object->errors, 'errors');
  620. }
  621. $action = '';
  622. }
  623. /*
  624. * Updating a line in the order
  625. */
  626. if ($action == 'updateline' && $usercancreate && !GETPOST('cancel', 'alpha')) {
  627. $db->begin();
  628. $vat_rate = (GETPOST('tva_tx') ? GETPOST('tva_tx') : 0);
  629. if ($lineid) {
  630. $line = new CommandeFournisseurLigne($db);
  631. $res = $line->fetch($lineid);
  632. if (!$res) {
  633. dol_print_error($db);
  634. }
  635. }
  636. $productsupplier = new ProductFournisseur($db);
  637. if (!empty($conf->global->SUPPLIER_ORDER_WITH_PREDEFINED_PRICES_ONLY)) {
  638. if ($line->fk_product > 0 && $productsupplier->get_buyprice(0, price2num(GETPOST('qty', 'int')), $line->fk_product, 'none', GETPOST('socid', 'int')) < 0) {
  639. setEventMessages($langs->trans("ErrorQtyTooLowForThisSupplier"), null, 'warnings');
  640. }
  641. }
  642. $date_start = dol_mktime(GETPOST('date_starthour'), GETPOST('date_startmin'), GETPOST('date_startsec'), GETPOST('date_startmonth'), GETPOST('date_startday'), GETPOST('date_startyear'));
  643. $date_end = dol_mktime(GETPOST('date_endhour'), GETPOST('date_endmin'), GETPOST('date_endsec'), GETPOST('date_endmonth'), GETPOST('date_endday'), GETPOST('date_endyear'));
  644. // Define info_bits
  645. $info_bits = 0;
  646. if (preg_match('/\*/', $vat_rate)) {
  647. $info_bits |= 0x01;
  648. }
  649. // Define vat_rate
  650. $vat_rate = str_replace('*', '', $vat_rate);
  651. $localtax1_rate = get_localtax($vat_rate, 1, $mysoc, $object->thirdparty);
  652. $localtax2_rate = get_localtax($vat_rate, 2, $mysoc, $object->thirdparty);
  653. if (GETPOST('price_ht') != '') {
  654. $price_base_type = 'HT';
  655. $ht = price2num(GETPOST('price_ht'), '', 2);
  656. } else {
  657. $vatratecleaned = $vat_rate;
  658. if (preg_match('/^(.*)\s*\((.*)\)$/', $vat_rate, $reg)) { // If vat is "xx (yy)"
  659. $vatratecleaned = trim($reg[1]);
  660. $vatratecode = $reg[2];
  661. }
  662. $ttc = price2num(GETPOST('price_ttc'), '', 2);
  663. $ht = $ttc / (1 + ($vatratecleaned / 100));
  664. $price_base_type = 'HT';
  665. }
  666. $pu_ht_devise = price2num(GETPOST('multicurrency_subprice'), '', 2);
  667. // Extrafields Lines
  668. $extralabelsline = $extrafields->fetch_name_optionals_label($object->table_element_line);
  669. $array_options = $extrafields->getOptionalsFromPost($object->table_element_line);
  670. // Unset extrafield POST Data
  671. if (is_array($extralabelsline)) {
  672. foreach ($extralabelsline as $key => $value) {
  673. unset($_POST["options_".$key]);
  674. }
  675. }
  676. $result = $object->updateline(
  677. $lineid,
  678. GETPOST('product_desc', 'restricthtml'),
  679. $ht,
  680. price2num(GETPOST('qty'), 'MS'),
  681. price2num(GETPOST('remise_percent'), '', 2),
  682. $vat_rate,
  683. $localtax1_rate,
  684. $localtax2_rate,
  685. $price_base_type,
  686. 0,
  687. GETPOSTISSET("type") ? GETPOST("type") : $line->product_type,
  688. false,
  689. $date_start,
  690. $date_end,
  691. $array_options,
  692. GETPOST('units'),
  693. $pu_ht_devise,
  694. GETPOST('fourn_ref', 'alpha')
  695. );
  696. unset($_POST['qty']);
  697. unset($_POST['type']);
  698. unset($_POST['idprodfournprice']);
  699. unset($_POST['remmise_percent']);
  700. unset($_POST['dp_desc']);
  701. unset($_POST['np_desc']);
  702. unset($_POST['pu']);
  703. unset($_POST['fourn_ref']);
  704. unset($_POST['tva_tx']);
  705. unset($_POST['date_start']);
  706. unset($_POST['date_end']);
  707. unset($_POST['units']);
  708. unset($localtax1_tx);
  709. unset($localtax2_tx);
  710. unset($_POST['date_starthour']);
  711. unset($_POST['date_startmin']);
  712. unset($_POST['date_startsec']);
  713. unset($_POST['date_startday']);
  714. unset($_POST['date_startmonth']);
  715. unset($_POST['date_startyear']);
  716. unset($_POST['date_endhour']);
  717. unset($_POST['date_endmin']);
  718. unset($_POST['date_endsec']);
  719. unset($_POST['date_endday']);
  720. unset($_POST['date_endmonth']);
  721. unset($_POST['date_endyear']);
  722. if ($result >= 0) {
  723. // Define output language
  724. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  725. $outputlangs = $langs;
  726. $newlang = '';
  727. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  728. $newlang = GETPOST('lang_id', 'aZ09');
  729. }
  730. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  731. $newlang = $object->thirdparty->default_lang;
  732. }
  733. if (!empty($newlang)) {
  734. $outputlangs = new Translate("", $conf);
  735. $outputlangs->setDefaultLang($newlang);
  736. }
  737. $model = $object->model_pdf;
  738. $ret = $object->fetch($id); // Reload to get new records
  739. $result = $object->generateDocument($model, $outputlangs, $hidedetails, $hidedesc, $hideref);
  740. if ($result < 0) {
  741. dol_print_error($db, $result);
  742. }
  743. }
  744. $db->commit();
  745. } else {
  746. $db->rollback();
  747. setEventMessages($object->error, $object->errors, 'errors');
  748. }
  749. }
  750. // Remove a product line
  751. if ($action == 'confirm_deleteline' && $confirm == 'yes' && $usercancreate) {
  752. $db->begin();
  753. $result = $object->deleteline($lineid);
  754. if ($result > 0) {
  755. // reorder lines
  756. $object->line_order(true);
  757. // Define output language
  758. $outputlangs = $langs;
  759. $newlang = '';
  760. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  761. $newlang = GETPOST('lang_id', 'aZ09');
  762. }
  763. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  764. $newlang = $object->thirdparty->default_lang;
  765. }
  766. if (!empty($newlang)) {
  767. $outputlangs = new Translate("", $conf);
  768. $outputlangs->setDefaultLang($newlang);
  769. }
  770. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  771. $ret = $object->fetch($object->id); // Reload to get new records
  772. $object->generateDocument($object->model_pdf, $outputlangs, $hidedetails, $hidedesc, $hideref);
  773. }
  774. } else {
  775. $error++;
  776. setEventMessages($object->error, $object->errors, 'errors');
  777. // Reset action to avoid asking again confirmation on failure
  778. $action = '';
  779. }
  780. if (!$error) {
  781. $db->commit();
  782. header('Location: '.$_SERVER["PHP_SELF"].'?id='.$object->id);
  783. exit;
  784. } else {
  785. $db->rollback();
  786. }
  787. }
  788. // Validate
  789. if ($action == 'confirm_valid' && $confirm == 'yes' && $usercanvalidate) {
  790. $db->begin();
  791. $object->date_commande = dol_now();
  792. $result = $object->valid($user);
  793. if ($result >= 0) {
  794. // Define output language
  795. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  796. $outputlangs = $langs;
  797. $newlang = '';
  798. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  799. $newlang = GETPOST('lang_id', 'aZ09');
  800. }
  801. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  802. $newlang = $object->thirdparty->default_lang;
  803. }
  804. if (!empty($newlang)) {
  805. $outputlangs = new Translate("", $conf);
  806. $outputlangs->setDefaultLang($newlang);
  807. }
  808. $model = $object->model_pdf;
  809. $ret = $object->fetch($id); // Reload to get new records
  810. $result = $object->generateDocument($model, $outputlangs, $hidedetails, $hidedesc, $hideref);
  811. if ($result < 0) {
  812. $error++;
  813. dol_print_error($db, $result);
  814. }
  815. }
  816. } else {
  817. $error++;
  818. setEventMessages($object->error, $object->errors, 'errors');
  819. }
  820. // If we have permission, and if we don't need to provide the idwarehouse, we go directly on approved step
  821. if (!$error && empty($conf->global->SUPPLIER_ORDER_NO_DIRECT_APPROVE) && $usercanapprove && !(!empty($conf->global->STOCK_CALCULATE_ON_SUPPLIER_VALIDATE_ORDER) && $object->hasProductsOrServices(1))) {
  822. $action = 'confirm_approve'; // can make standard or first level approval also if permission is set
  823. }
  824. if (!$error) {
  825. $db->commit();
  826. } else {
  827. $db->rollback();
  828. }
  829. }
  830. if (($action == 'confirm_approve' || $action == 'confirm_approve2') && $confirm == 'yes' && $usercanapprove) {
  831. $db->begin();
  832. $idwarehouse = GETPOST('idwarehouse', 'int');
  833. $qualified_for_stock_change = 0;
  834. if (empty($conf->global->STOCK_SUPPORTS_SERVICES)) {
  835. $qualified_for_stock_change = $object->hasProductsOrServices(2);
  836. } else {
  837. $qualified_for_stock_change = $object->hasProductsOrServices(1);
  838. }
  839. // Check parameters
  840. if (!empty($conf->stock->enabled) && !empty($conf->global->STOCK_CALCULATE_ON_SUPPLIER_VALIDATE_ORDER) && $qualified_for_stock_change) { // warning name of option should be STOCK_CALCULATE_ON_SUPPLIER_APPROVE_ORDER
  841. if (!$idwarehouse || $idwarehouse == -1) {
  842. $error++;
  843. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentitiesnoconv("Warehouse")), null, 'errors');
  844. $action = '';
  845. }
  846. }
  847. if (!$error) {
  848. $result = $object->approve($user, $idwarehouse, ($action == 'confirm_approve2' ? 1 : 0));
  849. if ($result > 0) {
  850. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  851. $outputlangs = $langs;
  852. $newlang = '';
  853. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  854. $newlang = GETPOST('lang_id', 'aZ09');
  855. }
  856. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  857. $newlang = $object->thirdparty->default_lang;
  858. }
  859. if (!empty($newlang)) {
  860. $outputlangs = new Translate("", $conf);
  861. $outputlangs->setDefaultLang($newlang);
  862. }
  863. $object->generateDocument($object->model_pdf, $outputlangs, $hidedetails, $hidedesc, $hideref);
  864. }
  865. } else {
  866. $error++;
  867. setEventMessages($object->error, $object->errors, 'errors');
  868. }
  869. }
  870. if (!$error) {
  871. $db->commit();
  872. header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
  873. exit;
  874. } else {
  875. $db->rollback();
  876. }
  877. }
  878. if ($action == 'confirm_refuse' && $confirm == 'yes' && $usercanapprove) {
  879. if (GETPOST('refuse_note')) {
  880. $object->refuse_note = GETPOST('refuse_note');
  881. }
  882. $result = $object->refuse($user);
  883. if ($result > 0) {
  884. header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
  885. exit;
  886. } else {
  887. setEventMessages($object->error, $object->errors, 'errors');
  888. }
  889. }
  890. // Force mandatory order method
  891. if ($action == 'commande') {
  892. $methodecommande = GETPOST('methodecommande', 'int');
  893. if ($cancel) {
  894. $action = '';
  895. } elseif ($methodecommande <= 0) {
  896. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("OrderMode")), null, 'errors');
  897. $action = 'makeorder';
  898. }
  899. }
  900. if ($action == 'confirm_commande' && $confirm == 'yes' && $usercanorder) {
  901. $db->begin();
  902. $result = $object->commande($user, GETPOST("datecommande"), GETPOST("methode", 'int'), GETPOST('comment', 'alphanohtml'));
  903. if ($result > 0) {
  904. if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
  905. $outputlangs = $langs;
  906. $newlang = '';
  907. if ($conf->global->MAIN_MULTILANGS && empty($newlang) && GETPOST('lang_id', 'aZ09')) {
  908. $newlang = GETPOST('lang_id', 'aZ09');
  909. }
  910. if ($conf->global->MAIN_MULTILANGS && empty($newlang)) {
  911. $newlang = $object->thirdparty->default_lang;
  912. }
  913. if (!empty($newlang)) {
  914. $outputlangs = new Translate("", $conf);
  915. $outputlangs->setDefaultLang($newlang);
  916. }
  917. $object->generateDocument($object->model_pdf, $outputlangs, $hidedetails, $hidedesc, $hideref);
  918. }
  919. $action = '';
  920. } else {
  921. $error++;
  922. setEventMessages($object->error, $object->errors, 'errors');
  923. }
  924. if (!$error) {
  925. $db->commit();
  926. header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
  927. exit;
  928. } else {
  929. $db->rollback();
  930. }
  931. }
  932. if ($action == 'confirm_delete' && $confirm == 'yes' && $usercandelete) {
  933. $result = $object->delete($user);
  934. if ($result > 0) {
  935. header("Location: ".DOL_URL_ROOT.'/fourn/commande/list.php?restore_lastsearch_values=1');
  936. exit;
  937. } else {
  938. setEventMessages($object->error, $object->errors, 'errors');
  939. }
  940. }
  941. // Action clone object
  942. if ($action == 'confirm_clone' && $confirm == 'yes' && $usercancreate) {
  943. if (1 == 0 && !GETPOST('clone_content') && !GETPOST('clone_receivers')) {
  944. setEventMessages($langs->trans("NoCloneOptionsSpecified"), null, 'errors');
  945. } else {
  946. if ($object->id > 0) {
  947. $orig = clone $object;
  948. $result = $object->createFromClone($user, $socid);
  949. if ($result > 0) {
  950. header("Location: ".$_SERVER['PHP_SELF'].'?id='.$result);
  951. exit;
  952. } else {
  953. setEventMessages($object->error, $object->errors, 'errors');
  954. $object = $orig;
  955. $action = '';
  956. }
  957. }
  958. }
  959. }
  960. // Set status of reception (complete, partial, ...)
  961. if ($action == 'livraison' && $usercanreceive) {
  962. if ($cancel) {
  963. $action = '';
  964. } else {
  965. $db->begin();
  966. if (GETPOST("type") != '') {
  967. $date_liv = dol_mktime(GETPOST('rehour'), GETPOST('remin'), GETPOST('resec'), GETPOST("remonth"), GETPOST("reday"), GETPOST("reyear"));
  968. $result = $object->Livraison($user, $date_liv, GETPOST("type"), GETPOST("comment")); // GETPOST("type") is 'tot', 'par', 'nev', 'can'
  969. if ($result > 0) {
  970. $langs->load("deliveries");
  971. setEventMessages($langs->trans("DeliveryStateSaved"), null);
  972. $action = '';
  973. } elseif ($result == -3) {
  974. $error++;
  975. setEventMessages($object->error, $object->errors, 'errors');
  976. } else {
  977. $error++;
  978. setEventMessages($object->error, $object->errors, 'errors');
  979. }
  980. } else {
  981. $error++;
  982. setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentities("Delivery")), null, 'errors');
  983. }
  984. if (!$error) {
  985. $db->commit();
  986. } else {
  987. $db->rollback();
  988. }
  989. }
  990. }
  991. if ($action == 'confirm_cancel' && $confirm == 'yes' && $usercanorder) {
  992. if (GETPOST('cancel_note')) {
  993. $object->cancel_note = GETPOST('cancel_note');
  994. }
  995. $result = $object->cancel($user);
  996. if ($result > 0) {
  997. header("Location: ".$_SERVER["PHP_SELF"]."?id=".$object->id);
  998. exit;
  999. } else {
  1000. setEventMessages($object->error, $object->errors, 'errors');
  1001. }
  1002. }
  1003. // Actions when printing a doc from card
  1004. include DOL_DOCUMENT_ROOT.'/core/actions_printing.inc.php';
  1005. // Actions to send emails
  1006. $triggersendname = 'ORDER_SUPPLIER_SENTBYMAIL';
  1007. $autocopy = 'MAIN_MAIL_AUTOCOPY_SUPPLIER_ORDER_TO';
  1008. $trackid = 'sord'.$object->id;
  1009. include DOL_DOCUMENT_ROOT.'/core/actions_sendmails.inc.php';
  1010. // Actions to build doc
  1011. $upload_dir = $conf->fournisseur->commande->dir_output;
  1012. $permissiontoadd = $usercancreate;
  1013. include DOL_DOCUMENT_ROOT.'/core/actions_builddoc.inc.php';
  1014. if ($action == 'update_extras') {
  1015. $object->oldcopy = dol_clone($object);
  1016. // Fill array 'array_options' with data from add form
  1017. $ret = $extrafields->setOptionalsFromPost(null, $object, GETPOST('attribute', 'restricthtml'));
  1018. if ($ret < 0) {
  1019. $error++;
  1020. }
  1021. if (!$error) {
  1022. // Actions on extra fields
  1023. if (!$error) {
  1024. $result = $object->insertExtraFields('ORDER_SUPPLIER_MODIFY');
  1025. if ($result < 0) {
  1026. $error++;
  1027. setEventMessages($object->error, $object->errors, 'errors');
  1028. }
  1029. }
  1030. }
  1031. if ($error) {
  1032. $action = 'edit_extras';
  1033. }
  1034. }
  1035. /*
  1036. * Create an order
  1037. */
  1038. if ($action == 'add' && $usercancreate) {
  1039. $error = 0;
  1040. $selectedLines = GETPOST('toselect', 'array');
  1041. if ($socid < 1) {
  1042. setEventMessages($langs->trans('ErrorFieldRequired', $langs->transnoentities('Supplier')), null, 'errors');
  1043. $action = 'create';
  1044. $error++;
  1045. }
  1046. if (!$error) {
  1047. $db->begin();
  1048. // Creation commande
  1049. $object->ref_supplier = GETPOST('refsupplier');
  1050. $object->socid = $socid;
  1051. $object->cond_reglement_id = GETPOST('cond_reglement_id', 'int');
  1052. $object->mode_reglement_id = GETPOST('mode_reglement_id', 'int');
  1053. $object->fk_account = GETPOST('fk_account', 'int');
  1054. $object->note_private = GETPOST('note_private', 'restricthtml');
  1055. $object->note_public = GETPOST('note_public', 'restricthtml');
  1056. $object->date_livraison = $datelivraison; // deprecated
  1057. $object->delivery_date = $datelivraison;
  1058. $object->fk_incoterms = GETPOST('incoterm_id', 'int');
  1059. $object->location_incoterms = GETPOST('location_incoterms', 'alpha');
  1060. $object->multicurrency_code = GETPOST('multicurrency_code', 'alpha');
  1061. $object->multicurrency_tx = GETPOST('originmulticurrency_tx', 'int');
  1062. $object->fk_project = GETPOST('projectid', 'int');
  1063. // Fill array 'array_options' with data from add form
  1064. if (!$error) {
  1065. $ret = $extrafields->setOptionalsFromPost(null, $object);
  1066. if ($ret < 0) {
  1067. $error++;
  1068. }
  1069. }
  1070. if (!$error) {
  1071. // If creation from another object of another module (Example: origin=propal, originid=1)
  1072. if (!empty($origin) && !empty($originid)) {
  1073. $element = $subelement = $origin;
  1074. $classname = ucfirst($subelement);
  1075. if ($origin == 'propal' || $origin == 'proposal') {
  1076. $element = 'comm/propal'; $subelement = 'propal';
  1077. $classname = 'Propal';
  1078. }
  1079. if ($origin == 'order' || $origin == 'commande') {
  1080. $element = $subelement = 'commande';
  1081. $classname = 'Commande';
  1082. }
  1083. if ($origin == 'supplier_proposal') {
  1084. $classname = 'SupplierProposal';
  1085. $element = 'supplier_proposal';
  1086. $subelement = 'supplier_proposal';
  1087. }
  1088. $object->origin = $origin;
  1089. $object->origin_id = $originid;
  1090. // Possibility to add external linked objects with hooks
  1091. $object->linked_objects [$object->origin] = $object->origin_id;
  1092. $other_linked_objects = GETPOST('other_linked_objects', 'array');
  1093. if (!empty($other_linked_objects)) {
  1094. $object->linked_objects = array_merge($object->linked_objects, $other_linked_objects);
  1095. }
  1096. $id = $object->create($user);
  1097. if ($id > 0) {
  1098. dol_include_once('/'.$element.'/class/'.$subelement.'.class.php');
  1099. $srcobject = new $classname($db);
  1100. dol_syslog("Try to find source object origin=".$object->origin." originid=".$object->origin_id." to add lines");
  1101. $result = $srcobject->fetch($object->origin_id);
  1102. if ($result > 0) {
  1103. $tmpdate = ($srcobject->delivery_date ? $srcobject->delivery_date : $srcobject->date_livraison);
  1104. $object->setDeliveryDate($user, $tmpdate);
  1105. $object->set_id_projet($user, $srcobject->fk_project);
  1106. $lines = $srcobject->lines;
  1107. if (empty($lines) && method_exists($srcobject, 'fetch_lines')) {
  1108. $srcobject->fetch_lines();
  1109. $lines = $srcobject->lines;
  1110. }
  1111. $fk_parent_line = 0;
  1112. $num = count($lines);
  1113. for ($i = 0; $i < $num; $i++) {
  1114. if (empty($lines[$i]->subprice) || $lines[$i]->qty <= 0 || !in_array($lines[$i]->id, $selectedLines)) {
  1115. continue;
  1116. }
  1117. $label = (!empty($lines[$i]->label) ? $lines[$i]->label : '');
  1118. $desc = (!empty($lines[$i]->desc) ? $lines[$i]->desc : $lines[$i]->product_desc);
  1119. $product_type = (!empty($lines[$i]->product_type) ? $lines[$i]->product_type : 0);
  1120. // Reset fk_parent_line for no child products and special product
  1121. if (($lines[$i]->product_type != 9 && empty($lines[$i]->fk_parent_line)) || $lines[$i]->product_type == 9) {
  1122. $fk_parent_line = 0;
  1123. }
  1124. // Extrafields
  1125. if (method_exists($lines[$i], 'fetch_optionals')) { // For avoid conflicts if
  1126. $lines[$i]->fetch_optionals();
  1127. $array_option = $lines[$i]->array_options;
  1128. }
  1129. $ref_supplier = '';
  1130. $product_fourn_price_id = 0;
  1131. if ($origin == "commande") {
  1132. $productsupplier = new ProductFournisseur($db);
  1133. $result = $productsupplier->find_min_price_product_fournisseur($lines[$i]->fk_product, $lines[$i]->qty, $srcobject->socid);
  1134. if ($result > 0) {
  1135. $ref_supplier = $productsupplier->ref_supplier;
  1136. $product_fourn_price_id = $productsupplier->product_fourn_price_id;
  1137. }
  1138. } else {
  1139. $ref_supplier = $lines[$i]->ref_fourn;
  1140. $product_fourn_price_id = 0;
  1141. }
  1142. $tva_tx = $lines[$i]->tva_tx;
  1143. if ($origin == "commande") {
  1144. $soc = new societe($db);
  1145. $soc->fetch($socid);
  1146. $tva_tx = get_default_tva($soc, $mysoc, $lines[$i]->fk_product, $product_fourn_price_id);
  1147. }
  1148. $object->special_code = $lines[$i]->special_code;
  1149. $result = $object->addline(
  1150. $desc,
  1151. $lines[$i]->subprice,
  1152. $lines[$i]->qty,
  1153. $tva_tx,
  1154. $lines[$i]->localtax1_tx,
  1155. $lines[$i]->localtax2_tx,
  1156. $lines[$i]->fk_product > 0 ? $lines[$i]->fk_product : 0,
  1157. $product_fourn_price_id,
  1158. $ref_supplier,
  1159. $lines[$i]->remise_percent,
  1160. 'HT',
  1161. 0,
  1162. $lines[$i]->product_type,
  1163. '',
  1164. '',
  1165. null,
  1166. null,
  1167. $array_option,
  1168. $lines[$i]->fk_unit,
  1169. 0,
  1170. $element,
  1171. !empty($lines[$i]->id) ? $lines[$i]->id : $lines[$i]->rowid
  1172. );
  1173. if ($result < 0) {
  1174. $error++;
  1175. break;
  1176. }
  1177. // Defined the new fk_parent_line
  1178. if ($result > 0 && $lines[$i]->product_type == 9) {
  1179. $fk_parent_line = $result;
  1180. }
  1181. }
  1182. // Add link between elements
  1183. // Hooks
  1184. $parameters = array('objFrom' => $srcobject);
  1185. $reshook = $hookmanager->executeHooks('createFrom', $parameters, $object, $action); // Note that $action and $object may have been
  1186. if ($reshook < 0) {
  1187. $error++;
  1188. }
  1189. } else {
  1190. setEventMessages($srcobject->error, $srcobject->errors, 'errors');
  1191. $error++;
  1192. }
  1193. } else {
  1194. setEventMessages($object->error, $object->errors, 'errors');
  1195. $error++;
  1196. }
  1197. } else {
  1198. $id = $object->create($user);
  1199. if ($id < 0) {
  1200. $error++;
  1201. setEventMessages($object->error, $object->errors, 'errors');
  1202. }
  1203. }
  1204. }
  1205. if ($error) {
  1206. $langs->load("errors");
  1207. $db->rollback();
  1208. $action = 'create';
  1209. $_GET['socid'] = $_POST['socid'];
  1210. } else {
  1211. $db->commit();
  1212. header("Location: ".$_SERVER['PHP_SELF']."?id=".urlencode($id));
  1213. exit;
  1214. }
  1215. }
  1216. }
  1217. if ($action == 'webservice' && GETPOST('mode', 'alpha') == "send" && !GETPOST('cancel', 'alpha')) {
  1218. $ws_url = $object->thirdparty->webservices_url;
  1219. $ws_key = $object->thirdparty->webservices_key;
  1220. $ws_user = GETPOST('ws_user', 'alpha');
  1221. $ws_password = GETPOST('ws_password', 'alpha');
  1222. $ws_entity = GETPOST('ws_entity', 'int');
  1223. $ws_thirdparty = GETPOST('ws_thirdparty', 'int');
  1224. // NS and Authentication parameters
  1225. $ws_ns = 'http://www.dolibarr.org/ns/';
  1226. $ws_authentication = array(
  1227. 'dolibarrkey'=>$ws_key,
  1228. 'sourceapplication'=>'DolibarrWebServiceClient',
  1229. 'login'=>$ws_user,
  1230. 'password'=>$ws_password,
  1231. 'entity'=>$ws_entity
  1232. );
  1233. //Is sync supplier web services module activated? and everything filled?
  1234. if (empty($conf->syncsupplierwebservices->enabled)) {
  1235. setEventMessages($langs->trans("WarningModuleNotActive", $langs->transnoentities("Module2650Name")), null, 'mesgs');
  1236. } elseif (empty($ws_url) || empty($ws_key)) {
  1237. setEventMessages($langs->trans("ErrorWebServicesFieldsRequired"), null, 'errors');
  1238. } elseif (empty($ws_user) || empty($ws_password) || empty($ws_thirdparty)) {
  1239. setEventMessages($langs->trans("ErrorFieldsRequired"), null, 'errors');
  1240. } else {
  1241. //Create SOAP client and connect it to order
  1242. $soapclient_order = new nusoap_client($ws_url."/webservices/server_order.php");
  1243. $soapclient_order->soap_defencoding = 'UTF-8';
  1244. $soapclient_order->decodeUTF8(false);
  1245. //Create SOAP client and connect it to product/service
  1246. $soapclient_product = new nusoap_client($ws_url."/webservices/server_productorservice.php");
  1247. $soapclient_product->soap_defencoding = 'UTF-8';
  1248. $soapclient_product->decodeUTF8(false);
  1249. //Prepare the order lines from order
  1250. $order_lines = array();
  1251. foreach ($object->lines as $line) {
  1252. $ws_parameters = array('authentication' => $ws_authentication, 'id' => '', 'ref' => $line->ref_supplier);
  1253. $result_product = $soapclient_product->call("getProductOrService", $ws_parameters, $ws_ns, '');
  1254. if ($result_product["result"]["result_code"] == "OK") {
  1255. $order_lines[] = array(
  1256. 'desc' => $line->product_desc,
  1257. 'type' => $line->product_type,
  1258. 'product_id' => $result_product["product"]["id"],
  1259. 'vat_rate' => $line->tva_tx,
  1260. 'qty' => $line->qty,
  1261. 'price' => $line->price,
  1262. 'unitprice' => $line->subprice,
  1263. 'total_net' => $line->total_ht,
  1264. 'total_vat' => $line->total_tva,
  1265. 'total' => $line->total_ttc,
  1266. 'date_start' => $line->date_start,
  1267. 'date_end' => $line->date_end,
  1268. );
  1269. }
  1270. }
  1271. //Prepare the order header
  1272. $order = array(
  1273. 'thirdparty_id' => $ws_thirdparty,
  1274. 'date' => dol_print_date(dol_now(), 'dayrfc'),
  1275. 'total_net' => $object->total_ht,
  1276. 'total_var' => $object->total_tva,
  1277. 'total' => $object->total_ttc,
  1278. 'lines' => $order_lines
  1279. );
  1280. $ws_parameters = array('authentication'=>$ws_authentication, 'order' => $order);
  1281. $result_order = $soapclient_order->call("createOrder", $ws_parameters, $ws_ns, '');
  1282. if (empty($result_order["result"]["result_code"])) { //No result, check error str
  1283. setEventMessages($langs->trans("SOAPError")." '".$soapclient_order->error_str."'", null, 'errors');
  1284. } elseif ($result_order["result"]["result_code"] != "OK") { //Something went wrong
  1285. setEventMessages($langs->trans("SOAPError")." '".$result_order["result"]["result_code"]."' - '".$result_order["result"]["result_label"]."'", null, 'errors');
  1286. } else {
  1287. setEventMessages($langs->trans("RemoteOrderRef")." ".$result_order["ref"], null, 'mesgs');
  1288. }
  1289. }
  1290. }
  1291. if (!empty($conf->global->MAIN_DISABLE_CONTACTS_TAB) && $usercancreate) {
  1292. if ($action == 'addcontact') {
  1293. if ($object->id > 0) {
  1294. $contactid = (GETPOST('userid') ? GETPOST('userid') : GETPOST('contactid'));
  1295. $typeid = (GETPOST('typecontact') ? GETPOST('typecontact') : GETPOST('type'));
  1296. $result = $object->add_contact($contactid, $typeid, GETPOST("source", 'aZ09'));
  1297. }
  1298. if ($result >= 0) {
  1299. header("Location: ".$_SERVER['PHP_SELF']."?id=".$object->id);
  1300. exit;
  1301. } else {
  1302. if ($object->error == 'DB_ERROR_RECORD_ALREADY_EXISTS') {
  1303. $langs->load("errors");
  1304. setEventMessages($langs->trans("ErrorThisContactIsAlreadyDefinedAsThisType"), null, 'errors');
  1305. } else {
  1306. setEventMessages($object->error, $object->errors, 'errors');
  1307. }
  1308. }
  1309. } elseif ($action == 'swapstatut' && $object->id > 0) {
  1310. // bascule du statut d'un contact
  1311. $result = $object->swapContactStatus(GETPOST('ligne', 'int'));
  1312. } elseif ($action == 'deletecontact' && $object->id > 0) {
  1313. // Efface un contact
  1314. $result = $object->delete_contact(GETPOST("lineid", 'int'));
  1315. if ($result >= 0) {
  1316. header("Location: ".$_SERVER['PHP_SELF']."?id=".$object->id);
  1317. exit;
  1318. } else {
  1319. dol_print_error($db);
  1320. }
  1321. }
  1322. }
  1323. }
  1324. /*
  1325. * View
  1326. */
  1327. $form = new Form($db);
  1328. $formfile = new FormFile($db);
  1329. $formorder = new FormOrder($db);
  1330. $productstatic = new Product($db);
  1331. if (!empty($conf->projet->enabled)) {
  1332. $formproject = new FormProjets($db);
  1333. }
  1334. $title = $langs->trans('SupplierOrder')." - ".$langs->trans('Card');
  1335. $help_url = 'EN:Module_Suppliers_Orders|FR:CommandeFournisseur|ES:Módulo_Pedidos_a_proveedores';
  1336. llxHeader('', $title, $help_url);
  1337. $now = dol_now();
  1338. if ($action == 'create') {
  1339. print load_fiche_titre($langs->trans('NewOrderSupplier'), '', 'supplier_order');
  1340. dol_htmloutput_events();
  1341. $currency_code = $conf->currency;
  1342. $societe = '';
  1343. if ($socid > 0) {
  1344. $societe = new Societe($db);
  1345. $societe->fetch($socid);
  1346. }
  1347. if (!empty($origin) && !empty($originid)) {
  1348. // Parse element/subelement (ex: project_task)
  1349. $element = $subelement = $origin;
  1350. $classname = ucfirst($subelement);
  1351. $regs = array();
  1352. if (preg_match('/^([^_]+)_([^_]+)/i', $origin, $regs)) {
  1353. $element = $regs[1];
  1354. $subelement = $regs[2];
  1355. }
  1356. if ($origin == 'propal' || $origin == 'proposal') {
  1357. $classname = 'Propal';
  1358. $element = 'comm/propal'; $subelement = 'propal';
  1359. }
  1360. if ($origin == 'order' || $origin == 'commande') {
  1361. $classname = 'Commande';
  1362. $element = $subelement = 'commande';
  1363. }
  1364. if ($origin == 'supplier_proposal') {
  1365. $classname = 'SupplierProposal';
  1366. $element = 'supplier_proposal';
  1367. $subelement = 'supplier_proposal';
  1368. }
  1369. dol_include_once('/'.$element.'/class/'.$subelement.'.class.php');
  1370. $objectsrc = new $classname($db);
  1371. $objectsrc->fetch($originid);
  1372. if (empty($objectsrc->lines) && method_exists($objectsrc, 'fetch_lines')) {
  1373. $objectsrc->fetch_lines();
  1374. }
  1375. $objectsrc->fetch_thirdparty();
  1376. // Replicate extrafields
  1377. $objectsrc->fetch_optionals();
  1378. $object->array_options = $objectsrc->array_options;
  1379. $projectid = (!empty($objectsrc->fk_project) ? $objectsrc->fk_project : '');
  1380. $ref_client = (!empty($objectsrc->ref_client) ? $objectsrc->ref_client : '');
  1381. $soc = $objectsrc->client;
  1382. $cond_reglement_id = (!empty($objectsrc->cond_reglement_id) ? $objectsrc->cond_reglement_id : (!empty($soc->cond_reglement_id) ? $soc->cond_reglement_id : 0));
  1383. $mode_reglement_id = (!empty($objectsrc->mode_reglement_id) ? $objectsrc->mode_reglement_id : (!empty($soc->mode_reglement_id) ? $soc->mode_reglement_id : 0));
  1384. $fk_account = (!empty($objectsrc->fk_account) ? $objectsrc->fk_account : (!empty($soc->fk_account) ? $soc->fk_account : 0));
  1385. $availability_id = (!empty($objectsrc->availability_id) ? $objectsrc->availability_id : (!empty($soc->availability_id) ? $soc->availability_id : 0));
  1386. $shipping_method_id = (!empty($objectsrc->shipping_method_id) ? $objectsrc->shipping_method_id : (!empty($soc->shipping_method_id) ? $soc->shipping_method_id : 0));
  1387. $demand_reason_id = (!empty($objectsrc->demand_reason_id) ? $objectsrc->demand_reason_id : (!empty($soc->demand_reason_id) ? $soc->demand_reason_id : 0));
  1388. $remise_percent = (!empty($objectsrc->remise_percent) ? $objectsrc->remise_percent : (!empty($soc->remise_supplier_percent) ? $soc->remise_supplier_percent : 0));
  1389. $remise_absolue = (!empty($objectsrc->remise_absolue) ? $objectsrc->remise_absolue : (!empty($soc->remise_absolue) ? $soc->remise_absolue : 0));
  1390. $dateinvoice = empty($conf->global->MAIN_AUTOFILL_DATE) ?-1 : '';
  1391. $datedelivery = (!empty($objectsrc->date_livraison) ? $objectsrc->date_livraison : (!empty($objectsrc->delivery_date) ? $objectsrc->delivery_date : ''));
  1392. if (!empty($conf->multicurrency->enabled)) {
  1393. if (!empty($objectsrc->multicurrency_code)) {
  1394. $currency_code = $objectsrc->multicurrency_code;
  1395. }
  1396. if (!empty($conf->global->MULTICURRENCY_USE_ORIGIN_TX) && !empty($objectsrc->multicurrency_tx)) {
  1397. $currency_tx = $objectsrc->multicurrency_tx;
  1398. }
  1399. }
  1400. $note_private = $object->getDefaultCreateValueFor('note_private', (!empty($objectsrc->note_private) ? $objectsrc->note_private : null));
  1401. $note_public = $object->getDefaultCreateValueFor('note_public', (!empty($objectsrc->note_public) ? $objectsrc->note_public : null));
  1402. // Object source contacts list
  1403. $srccontactslist = $objectsrc->liste_contact(-1, 'external', 1);
  1404. } else {
  1405. $cond_reglement_id = $societe->cond_reglement_supplier_id;
  1406. $mode_reglement_id = $societe->mode_reglement_supplier_id;
  1407. if (!empty($conf->multicurrency->enabled) && !empty($societe->multicurrency_code)) {
  1408. $currency_code = $societe->multicurrency_code;
  1409. }
  1410. $note_private = $object->getDefaultCreateValueFor('note_private');
  1411. $note_public = $object->getDefaultCreateValueFor('note_public');
  1412. }
  1413. // If not defined, set default value from constant
  1414. if (empty($cond_reglement_id) && !empty($conf->global->SUPPLIER_ORDER_DEFAULT_PAYMENT_TERM_ID)) {
  1415. $cond_reglement_id = $conf->global->SUPPLIER_ORDER_DEFAULT_PAYMENT_TERM_ID;
  1416. }
  1417. if (empty($mode_reglement_id) && !empty($conf->global->SUPPLIER_ORDER_DEFAULT_PAYMENT_MODE_ID)) {
  1418. $mode_reglement_id = $conf->global->SUPPLIER_ORDER_DEFAULT_PAYMENT_MODE_ID;
  1419. }
  1420. print '<form name="add" action="'.$_SERVER["PHP_SELF"].'" method="post">';
  1421. print '<input type="hidden" name="token" value="'.newToken().'">';
  1422. print '<input type="hidden" name="action" value="add">';
  1423. print '<input type="hidden" name="remise_percent" value="'.(empty($soc->remise_supplier_percent) ? '' : $soc->remise_supplier_percent).'">';
  1424. print '<input type="hidden" name="origin" value="'.$origin.'">';
  1425. print '<input type="hidden" name="originid" value="'.$originid.'">';
  1426. if ($backtopage) {
  1427. print '<input type="hidden" name="backtopage" value="'.$backtopage.'">';
  1428. }
  1429. if ($backtopageforcancel) {
  1430. print '<input type="hidden" name="backtopageforcancel" value="'.$backtopageforcancel.'">';
  1431. }
  1432. if (!empty($currency_tx)) {
  1433. print '<input type="hidden" name="originmulticurrency_tx" value="'.$currency_tx.'">';
  1434. }
  1435. print dol_get_fiche_head('');
  1436. print '<table class="border centpercent">';
  1437. // Ref
  1438. print '<tr><td class="titlefieldcreate">'.$langs->trans('Ref').'</td><td>'.$langs->trans('Draft').'</td></tr>';
  1439. // Third party
  1440. print '<tr><td class="fieldrequired">'.$langs->trans('Supplier').'</td>';
  1441. print '<td>';
  1442. if ($societe->id > 0) {
  1443. print $societe->getNomUrl(1, 'supplier');
  1444. print '<input type="hidden" name="socid" value="'.$societe->id.'">';
  1445. } else {
  1446. print img_picto('', 'company').$form->select_company((empty($socid) ? '' : $socid), 'socid', 's.fournisseur=1', 'SelectThirdParty', 0, 0, null, 0, 'minwidth300');
  1447. // reload page to retrieve customer informations
  1448. if (!empty($conf->global->RELOAD_PAGE_ON_SUPPLIER_CHANGE)) {
  1449. print '<script>
  1450. $(document).ready(function() {
  1451. $("#socid").change(function() {
  1452. var socid = $(this).val();
  1453. var prjid = $("#projectid").val();
  1454. // reload page
  1455. window.location.href = "'.$_SERVER["PHP_SELF"].'?action=create&socid="+socid+"&projectid="+prjid
  1456. });
  1457. });
  1458. </script>';
  1459. }
  1460. print ' <a href="'.DOL_URL_ROOT.'/societe/card.php?action=create&client=0&fournisseur=1&backtopage='.urlencode($_SERVER["PHP_SELF"].'?action=create').'"><span class="fa fa-plus-circle valignmiddle paddingleft" title="'.$langs->trans("AddThirdParty").'"></span></a>';
  1461. }
  1462. print '</td>';
  1463. if ($societe->id > 0) {
  1464. // Discounts for third party
  1465. print '<tr><td>'.$langs->trans('Discounts').'</td><td>';
  1466. $absolute_discount = $societe->getAvailableDiscounts('', '', 0, 1);
  1467. $thirdparty = $societe;
  1468. $discount_type = 1;
  1469. $backtopage = urlencode($_SERVER["PHP_SELF"].'?socid='.$thirdparty->id.'&action='.$action.'&origin='.GETPOST('origin').'&originid='.GETPOST('originid'));
  1470. include DOL_DOCUMENT_ROOT.'/core/tpl/object_discounts.tpl.php';
  1471. print '</td></tr>';
  1472. }
  1473. // Ref supplier
  1474. print '<tr><td>'.$langs->trans('RefSupplier').'</td><td><input name="refsupplier" type="text"></td>';
  1475. print '</tr>';
  1476. // Payment term
  1477. print '<tr><td class="nowrap">'.$langs->trans('PaymentConditionsShort').'</td><td>';
  1478. $form->select_conditions_paiements(isset($_POST['cond_reglement_id']) ? $_POST['cond_reglement_id'] : $cond_reglement_id, 'cond_reglement_id');
  1479. print '</td></tr>';
  1480. // Payment mode
  1481. print '<tr><td>'.$langs->trans('PaymentMode').'</td><td>';
  1482. $form->select_types_paiements(isset($_POST['mode_reglement_id']) ? $_POST['mode_reglement_id'] : $mode_reglement_id, 'mode_reglement_id');
  1483. print '</td></tr>';
  1484. // Planned delivery date
  1485. print '<tr><td>';
  1486. print $langs->trans('DateDeliveryPlanned');
  1487. print '</td>';
  1488. print '<td>';
  1489. $usehourmin = 0;
  1490. if (!empty($conf->global->SUPPLIER_ORDER_USE_HOUR_FOR_DELIVERY_DATE)) {
  1491. $usehourmin = 1;
  1492. }
  1493. print $form->selectDate($datelivraison ? $datelivraison : -1, 'liv_', $usehourmin, $usehourmin, '', "set");
  1494. print '</td></tr>';
  1495. // Bank Account
  1496. if (!empty($conf->global->BANK_ASK_PAYMENT_BANK_DURING_SUPPLIER_ORDER) && !empty($conf->banque->enabled)) {
  1497. $langs->load("bank");
  1498. print '<tr><td>'.$langs->trans('BankAccount').'</td><td>';
  1499. print img_picto('', 'bank_account', 'class="paddingrightonly"');
  1500. $form->select_comptes($fk_account, 'fk_account', 0, '', 1);
  1501. print '</td></tr>';
  1502. }
  1503. // Project
  1504. if (!empty($conf->projet->enabled)) {
  1505. $formproject = new FormProjets($db);
  1506. $langs->load('projects');
  1507. print '<tr><td>'.$langs->trans('Project').'</td><td>';
  1508. print img_picto('', 'project').$formproject->select_projects((empty($conf->global->PROJECT_CAN_ALWAYS_LINK_TO_ALL_SUPPLIERS) ? $societe->id : -1), $projectid, 'projectid', 0, 0, 1, 1, 0, 0, 0, '', 1, 0, 'maxwidth500');
  1509. print ' &nbsp; <a href="'.DOL_URL_ROOT.'/projet/card.php?socid='.$societe->id.'&action=create&status=1&backtopage='.urlencode($_SERVER["PHP_SELF"].'?action=create&socid='.$societe->id).'"><span class="fa fa-plus-circle valignmiddle" title="'.$langs->trans("AddProject").'"></span></a>';
  1510. print '</td></tr>';
  1511. }
  1512. // Incoterms
  1513. if (!empty($conf->incoterm->enabled)) {
  1514. $fkincoterms = (!empty($object->fk_incoterms) ? $object->fk_incoterms : ($socid > 0 ? $societe->fk_incoterms : ''));
  1515. $locincoterms = (!empty($object->location_incoterms) ? $object->location_incoterms : ($socid > 0 ? $societe->location_incoterms : ''));
  1516. print '<tr>';
  1517. print '<td><label for="incoterm_id">'.$form->textwithpicto($langs->trans("IncotermLabel"), $object->label_incoterms, 1).'</label></td>';
  1518. print '<td class="maxwidthonsmartphone">';
  1519. print $form->select_incoterms($fkincoterms, $locincoterms);
  1520. print '</td></tr>';
  1521. }
  1522. // Multicurrency
  1523. if (!empty($conf->multicurrency->enabled)) {
  1524. print '<tr>';
  1525. print '<td>'.$form->editfieldkey('Currency', 'multicurrency_code', '', $object, 0).'</td>';
  1526. print '<td class="maxwidthonsmartphone">';
  1527. print $form->selectMultiCurrency($currency_code, 'multicurrency_code');
  1528. print '</td></tr>';
  1529. }
  1530. print '<tr><td>'.$langs->trans('NotePublic').'</td>';
  1531. print '<td>';
  1532. $doleditor = new DolEditor('note_public', isset($note_public) ? $note_public : GETPOST('note_public', 'restricthtml'), '', 80, 'dolibarr_notes', 'In', 0, false, empty($conf->global->FCKEDITOR_ENABLE_NOTE_PUBLIC) ? 0 : 1, ROWS_3, '90%');
  1533. print $doleditor->Create(1);
  1534. print '</td>';
  1535. //print '<textarea name="note_public" wrap="soft" cols="60" rows="'.ROWS_5.'"></textarea>';
  1536. print '</tr>';
  1537. print '<tr><td>'.$langs->trans('NotePrivate').'</td>';
  1538. print '<td>';
  1539. $doleditor = new DolEditor('note_private', isset($note_private) ? $note_private : GETPOST('note_private', 'restricthtml'), '', 80, 'dolibarr_notes', 'In', 0, false, empty($conf->global->FCKEDITOR_ENABLE_NOTE_PRIVATE) ? 0 : 1, ROWS_3, '90%');
  1540. print $doleditor->Create(1);
  1541. print '</td>';
  1542. //print '<td><textarea name="note_private" wrap="soft" cols="60" rows="'.ROWS_5.'"></textarea></td>';
  1543. print '</tr>';
  1544. if (!empty($origin) && !empty($originid) && is_object($objectsrc)) {
  1545. print "\n<!-- ".$classname." info -->";
  1546. print "\n";
  1547. print '<input type="hidden" name="amount" value="'.$objectsrc->total_ht.'">'."\n";
  1548. print '<input type="hidden" name="total" value="'.$objectsrc->total_ttc.'">'."\n";
  1549. print '<input type="hidden" name="tva" value="'.$objectsrc->total_tva.'">'."\n";
  1550. print '<input type="hidden" name="origin" value="'.$objectsrc->element.'">';
  1551. print '<input type="hidden" name="originid" value="'.$objectsrc->id.'">';
  1552. $newclassname = $classname;
  1553. print '<tr><td>'.$langs->trans($newclassname).'</td><td>'.$objectsrc->getNomUrl(1, 'supplier').'</td></tr>';
  1554. print '<tr><td>'.$langs->trans('AmountHT').'</td><td>'.price($objectsrc->total_ht).'</td></tr>';
  1555. print '<tr><td>'.$langs->trans('AmountVAT').'</td><td>'.price($objectsrc->total_tva)."</td></tr>";
  1556. if ($mysoc->localtax1_assuj == "1" || $objectsrc->total_localtax1 != 0) { // Localtax1 RE
  1557. print '<tr><td>'.$langs->transcountry("AmountLT1", $mysoc->country_code).'</td><td>'.price($objectsrc->total_localtax1)."</td></tr>";
  1558. }
  1559. if ($mysoc->localtax2_assuj == "1" || $objectsrc->total_localtax2 != 0) { // Localtax2 IRPF
  1560. print '<tr><td>'.$langs->transcountry("AmountLT2", $mysoc->country_code).'</td><td>'.price($objectsrc->total_localtax2)."</td></tr>";
  1561. }
  1562. print '<tr><td>'.$langs->trans('AmountTTC').'</td><td>'.price($objectsrc->total_ttc)."</td></tr>";
  1563. if (!empty($conf->multicurrency->enabled)) {
  1564. print '<tr><td>'.$langs->trans('MulticurrencyAmountHT').'</td><td>'.price($objectsrc->multicurrency_total_ht).'</td></tr>';
  1565. print '<tr><td>'.$langs->trans('MulticurrencyAmountVAT').'</td><td>'.price($objectsrc->multicurrency_total_tva).'</td></tr>';
  1566. print '<tr><td>'.$langs->trans('MulticurrencyAmountTTC').'</td><td>'.price($objectsrc->multicurrency_total_ttc).'</td></tr>';
  1567. }
  1568. }
  1569. // Other options
  1570. $parameters = array();
  1571. $reshook = $hookmanager->executeHooks('formObjectOptions', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
  1572. print $hookmanager->resPrint;
  1573. if (empty($reshook)) {
  1574. print $object->showOptionals($extrafields, 'create');
  1575. }
  1576. // Bouton "Create Draft"
  1577. print "</table>\n";
  1578. print dol_get_fiche_end();
  1579. print $form->buttonsSaveCancel("CreateDraft");
  1580. // Show origin lines
  1581. if (!empty($origin) && !empty($originid) && is_object($objectsrc)) {
  1582. $title = $langs->trans('ProductsAndServices');
  1583. print load_fiche_titre($title);
  1584. print '<table class="noborder centpercent">';
  1585. $objectsrc->printOriginLinesList('', $selectedLines);
  1586. print '</table>';
  1587. }
  1588. print "</form>\n";
  1589. } elseif (!empty($object->id)) {
  1590. $result = $object->fetch($id, $ref);
  1591. $societe = new Fournisseur($db);
  1592. $result = $societe->fetch($object->socid);
  1593. if ($result < 0) {
  1594. dol_print_error($db);
  1595. }
  1596. $author = new User($db);
  1597. $author->fetch($object->user_author_id);
  1598. $res = $object->fetch_optionals();
  1599. $head = ordersupplier_prepare_head($object);
  1600. $title = $langs->trans("SupplierOrder");
  1601. print dol_get_fiche_head($head, 'card', $title, -1, 'order');
  1602. $formconfirm = '';
  1603. // Confirmation de la suppression de la commande
  1604. if ($action == 'delete') {
  1605. $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('DeleteOrder'), $langs->trans('ConfirmDeleteOrder'), 'confirm_delete', '', 0, 2);
  1606. }
  1607. // Clone confirmation
  1608. if ($action == 'clone') {
  1609. // Create an array for form
  1610. $formquestion = array(
  1611. array('type' => 'other', 'name' => 'socid', 'label' => $langs->trans("SelectThirdParty"), 'value' => $form->select_company(GETPOST('socid', 'int'), 'socid', '(s.fournisseur=1)'))
  1612. );
  1613. // Paiement incomplet. On demande si motif = escompte ou autre
  1614. $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('ToClone'), $langs->trans('ConfirmCloneOrder', $object->ref), 'confirm_clone', $formquestion, 'yes', 1);
  1615. }
  1616. // Confirmation de la validation
  1617. if ($action == 'valid') {
  1618. $object->date_commande = dol_now();
  1619. // We check if number is temporary number
  1620. if (preg_match('/^[\(]?PROV/i', $object->ref) || empty($object->ref)) { // empty should not happened, but when it occurs, the test save life
  1621. $newref = $object->getNextNumRef($object->thirdparty);
  1622. } else {
  1623. $newref = $object->ref;
  1624. }
  1625. if ($newref < 0) {
  1626. setEventMessages($object->error, $object->errors, 'errors');
  1627. $action = '';
  1628. } else {
  1629. $text = $langs->trans('ConfirmValidateOrder', $newref);
  1630. if (!empty($conf->notification->enabled)) {
  1631. require_once DOL_DOCUMENT_ROOT.'/core/class/notify.class.php';
  1632. $notify = new Notify($db);
  1633. $text .= '<br>';
  1634. $text .= $notify->confirmMessage('ORDER_SUPPLIER_VALIDATE', $object->socid, $object);
  1635. }
  1636. $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('ValidateOrder'), $text, 'confirm_valid', '', 0, 1);
  1637. }
  1638. }
  1639. // Confirm approval
  1640. if ($action == 'approve' || $action == 'approve2') {
  1641. $qualified_for_stock_change = 0;
  1642. if (empty($conf->global->STOCK_SUPPORTS_SERVICES)) {
  1643. $qualified_for_stock_change = $object->hasProductsOrServices(2);
  1644. } else {
  1645. $qualified_for_stock_change = $object->hasProductsOrServices(1);
  1646. }
  1647. $formquestion = array();
  1648. if (!empty($conf->stock->enabled) && !empty($conf->global->STOCK_CALCULATE_ON_SUPPLIER_VALIDATE_ORDER) && $qualified_for_stock_change) {
  1649. $langs->load("stocks");
  1650. require_once DOL_DOCUMENT_ROOT.'/product/class/html.formproduct.class.php';
  1651. $formproduct = new FormProduct($db);
  1652. $forcecombo = 0;
  1653. if ($conf->browser->name == 'ie') {
  1654. $forcecombo = 1; // There is a bug in IE10 that make combo inside popup crazy
  1655. }
  1656. $formquestion = array(
  1657. //'text' => $langs->trans("ConfirmClone"),
  1658. //array('type' => 'checkbox', 'name' => 'clone_content', 'label' => $langs->trans("CloneMainAttributes"), 'value' => 1),
  1659. //array('type' => 'checkbox', 'name' => 'update_prices', 'label' => $langs->trans("PuttingPricesUpToDate"), 'value' => 1),
  1660. array('type' => 'other', 'name' => 'idwarehouse', 'label' => $langs->trans("SelectWarehouseForStockIncrease"), 'value' => $formproduct->selectWarehouses(GETPOST('idwarehouse', 'int'), 'idwarehouse', '', 1, 0, 0, '', 0, $forcecombo))
  1661. );
  1662. }
  1663. $text = $langs->trans("ConfirmApproveThisOrder", $object->ref);
  1664. if (!empty($conf->notification->enabled)) {
  1665. require_once DOL_DOCUMENT_ROOT.'/core/class/notify.class.php';
  1666. $notify = new Notify($db);
  1667. $text .= '<br>';
  1668. $text .= $notify->confirmMessage('ORDER_SUPPLIER_APPROVE', $object->socid, $object);
  1669. }
  1670. $formconfirm = $form->formconfirm($_SERVER['PHP_SELF']."?id=".$object->id, $langs->trans("ApproveThisOrder"), $text, "confirm_".$action, $formquestion, 1, 1, 240);
  1671. }
  1672. // Confirmation of disapproval
  1673. if ($action == 'refuse') {
  1674. $formquestion = array(
  1675. array(
  1676. 'type' => 'text',
  1677. 'name' => 'refuse_note',
  1678. 'label' => $langs->trans("Reason"),
  1679. 'value' => '',
  1680. 'morecss' => 'minwidth300'
  1681. )
  1682. );
  1683. $formconfirm = $form->formconfirm($_SERVER['PHP_SELF']."?id=$object->id", $langs->trans("DenyingThisOrder"), $langs->trans("ConfirmDenyingThisOrder", $object->ref), "confirm_refuse", $formquestion, 0, 1);
  1684. }
  1685. // Confirmation of cancellation
  1686. if ($action == 'cancel') {
  1687. $formquestion = array(
  1688. array(
  1689. 'type' => 'text',
  1690. 'name' => 'cancel_note',
  1691. 'label' => $langs->trans("Reason"),
  1692. 'value' => '',
  1693. 'morecss' => 'minwidth300'
  1694. )
  1695. );
  1696. $formconfirm = $form->formconfirm($_SERVER['PHP_SELF']."?id=$object->id", $langs->trans("Cancel"), $langs->trans("ConfirmCancelThisOrder", $object->ref), "confirm_cancel", $formquestion, 0, 1);
  1697. }
  1698. // Confirmation de l'envoi de la commande
  1699. if ($action == 'commande') {
  1700. $date_com = dol_mktime(GETPOST('rehour'), GETPOST('remin'), GETPOST('resec'), GETPOST("remonth"), GETPOST("reday"), GETPOST("reyear"));
  1701. $formconfirm = $form->formconfirm($_SERVER['PHP_SELF']."?id=".$object->id."&datecommande=".$date_com."&methode=".GETPOST("methodecommande")."&comment=".urlencode(GETPOST("comment")), $langs->trans("MakeOrder"), $langs->trans("ConfirmMakeOrder", dol_print_date($date_com, 'day')), "confirm_commande", '', 0, 2);
  1702. }
  1703. // Confirmation to delete line
  1704. if ($action == 'ask_deleteline') {
  1705. $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id.'&lineid='.$lineid, $langs->trans('DeleteProductLine'), $langs->trans('ConfirmDeleteProductLine'), 'confirm_deleteline', '', 0, 1);
  1706. }
  1707. $parameters = array('formConfirm' => $formconfirm, 'lineid'=>$lineid);
  1708. $reshook = $hookmanager->executeHooks('formConfirm', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
  1709. if (empty($reshook)) {
  1710. $formconfirm .= $hookmanager->resPrint;
  1711. } elseif ($reshook > 0) {
  1712. $formconfirm = $hookmanager->resPrint;
  1713. }
  1714. // Print form confirm
  1715. print $formconfirm;
  1716. // Supplier order card
  1717. $linkback = '<a href="'.DOL_URL_ROOT.'/fourn/commande/list.php?restore_lastsearch_values=1'.(!empty($socid) ? '&socid='.$socid : '').'">'.$langs->trans("BackToList").'</a>';
  1718. $morehtmlref = '<div class="refidno">';
  1719. // Ref supplier
  1720. $morehtmlref .= $form->editfieldkey("RefSupplier", 'ref_supplier', $object->ref_supplier, $object, $usercancreate, 'string', '', 0, 1);
  1721. $morehtmlref .= $form->editfieldval("RefSupplier", 'ref_supplier', $object->ref_supplier, $object, $usercancreate, 'string', '', null, null, '', 1);
  1722. // Thirdparty
  1723. $morehtmlref .= '<br>'.$langs->trans('ThirdParty');
  1724. if (!empty($conf->global->MAIN_CAN_EDIT_SUPPLIER_ON_SUPPLIER_ORDER) && !empty($usercancreate) && $action == 'edit_thirdparty') {
  1725. $morehtmlref .= ' : ';
  1726. $morehtmlref .= '<form method="post" action="'.$_SERVER['PHP_SELF'].'?id='.$object->id.'">';
  1727. $morehtmlref .= '<input type="hidden" name="action" value="set_thirdparty">';
  1728. $morehtmlref .= '<input type="hidden" name="token" value="'.newToken().'">';
  1729. $morehtmlref .= $form->select_company($object->thirdparty->id, 'new_socid', 's.fournisseur=1', '', 0, 0, array(), 0, 'minwidth300');
  1730. $morehtmlref .= '<input type="submit" class="button valignmiddle" value="'.$langs->trans("Modify").'">';
  1731. $morehtmlref .= '</form>';
  1732. }
  1733. if (empty($conf->global->MAIN_CAN_EDIT_SUPPLIER_ON_SUPPLIER_ORDER) || $action != 'edit_thirdparty') {
  1734. if (!empty($conf->global->MAIN_CAN_EDIT_SUPPLIER_ON_SUPPLIER_ORDER) && $object->statut == CommandeFournisseur::STATUS_DRAFT) {
  1735. $morehtmlref .= '<a class="editfielda" href="'.$_SERVER['PHP_SELF'].'?action=edit_thirdparty&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->transnoentitiesnoconv('SetThirdParty')).'</a>';
  1736. }
  1737. $morehtmlref .= ' : '.$object->thirdparty->getNomUrl(1, 'supplier');
  1738. if (empty($conf->global->MAIN_DISABLE_OTHER_LINK) && $object->thirdparty->id > 0) {
  1739. $morehtmlref .= ' (<a href="'.DOL_URL_ROOT.'/fourn/commande/list.php?socid='.$object->thirdparty->id.'&search_company='.urlencode($object->thirdparty->name).'">'.$langs->trans("OtherOrders").'</a>)';
  1740. }
  1741. }
  1742. // Project
  1743. if (!empty($conf->projet->enabled)) {
  1744. $langs->load("projects");
  1745. $morehtmlref .= '<br>'.$langs->trans('Project').' ';
  1746. if ($usercancreate) {
  1747. if ($action != 'classify' && $caneditproject) {
  1748. $morehtmlref .= '<a class="editfielda" href="'.$_SERVER['PHP_SELF'].'?action=classify&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->transnoentitiesnoconv('SetProject')).'</a> : ';
  1749. }
  1750. if ($action == 'classify') {
  1751. //$morehtmlref.=$form->form_project($_SERVER['PHP_SELF'] . '?id=' . $object->id, $object->socid, $object->fk_project, 'projectid', 0, 0, 1, 1);
  1752. $morehtmlref .= '<form method="post" action="'.$_SERVER['PHP_SELF'].'?id='.$object->id.'">';
  1753. $morehtmlref .= '<input type="hidden" name="action" value="classin">';
  1754. $morehtmlref .= '<input type="hidden" name="token" value="'.newToken().'">';
  1755. $morehtmlref .= $formproject->select_projects((empty($conf->global->PROJECT_CAN_ALWAYS_LINK_TO_ALL_SUPPLIERS) ? $object->socid : -1), $object->fk_project, 'projectid', 0, 0, 1, 1, 1, 0, 0, '', 1, 0, 'maxwidth500');
  1756. $morehtmlref .= '<input type="submit" class="button valignmiddle" value="'.$langs->trans("Modify").'">';
  1757. $morehtmlref .= '</form>';
  1758. } else {
  1759. $morehtmlref .= $form->form_project($_SERVER['PHP_SELF'].'?id='.$object->id, $object->socid, $object->fk_project, 'none', 0, 0, 0, 1);
  1760. }
  1761. } else {
  1762. if (!empty($object->fk_project)) {
  1763. $proj = new Project($db);
  1764. $proj->fetch($object->fk_project);
  1765. $morehtmlref .= ' : '.$proj->getNomUrl(1);
  1766. if ($proj->title) {
  1767. $morehtmlref .= ' - '.$proj->title;
  1768. }
  1769. } else {
  1770. $morehtmlref .= '';
  1771. }
  1772. }
  1773. }
  1774. $morehtmlref .= '</div>';
  1775. dol_banner_tab($object, 'ref', $linkback, 1, 'ref', 'ref', $morehtmlref);
  1776. print '<div class="fichecenter">';
  1777. print '<div class="fichehalfleft">';
  1778. print '<div class="underbanner clearboth"></div>';
  1779. print '<table class="border tableforfield centpercent">';
  1780. // Date
  1781. if ($object->methode_commande_id > 0) {
  1782. print '<tr><td class="titlefield">'.$langs->trans("Date").'</td><td>';
  1783. print $object->date_commande ? dol_print_date($object->date_commande, $usehourmin) : '';
  1784. if ($object->hasDelay() && !empty($object->date_delivery) && !empty($object->date_commande)) {
  1785. print ' '.img_picto($langs->trans("Late").' : '.$object->showDelay(), "warning");
  1786. }
  1787. print "</td></tr>";
  1788. if ($object->methode_commande) {
  1789. print '<tr><td>'.$langs->trans("Method").'</td><td>'.$object->getInputMethod().'</td></tr>';
  1790. }
  1791. }
  1792. // Author
  1793. print '<tr><td class="titlefield">'.$langs->trans("AuthorRequest").'</td>';
  1794. print '<td>'.$author->getNomUrl(1, '', 0, 0, 0).'</td>';
  1795. print '</tr>';
  1796. // Relative and absolute discounts
  1797. if (!empty($conf->global->FACTURE_DEPOSITS_ARE_JUST_PAYMENTS)) {
  1798. $filterabsolutediscount = "fk_invoice_supplier_source IS NULL"; // If we want deposit to be substracted to payments only and not to total of final invoice
  1799. $filtercreditnote = "fk_invoice_supplier_source IS NOT NULL"; // If we want deposit to be substracted to payments only and not to total of final invoice
  1800. } else {
  1801. $filterabsolutediscount = "fk_invoice_supplier_source IS NULL OR (description LIKE '(DEPOSIT)%' AND description NOT LIKE '(EXCESS PAID)%')";
  1802. $filtercreditnote = "fk_invoice_supplier_source IS NOT NULL AND (description NOT LIKE '(DEPOSIT)%' OR description LIKE '(EXCESS PAID)%')";
  1803. }
  1804. $absolute_discount = $societe->getAvailableDiscounts('', $filterabsolutediscount, 0, 1);
  1805. $absolute_creditnote = $societe->getAvailableDiscounts('', $filtercreditnote, 0, 1);
  1806. $absolute_discount = price2num($absolute_discount, 'MT');
  1807. $absolute_creditnote = price2num($absolute_creditnote, 'MT');
  1808. print '<tr><td class="titlefield">'.$langs->trans('Discounts').'</td><td>';
  1809. $thirdparty = $societe;
  1810. $discount_type = 1;
  1811. $backtopage = urlencode($_SERVER["PHP_SELF"].'?id='.$object->id);
  1812. include DOL_DOCUMENT_ROOT.'/core/tpl/object_discounts.tpl.php';
  1813. print '</td></tr>';
  1814. // Default terms of the settlement
  1815. $langs->load('bills');
  1816. print '<tr><td class="nowrap">';
  1817. print '<table class="nobordernopadding centpercent"><tr><td class="nowrap">';
  1818. print $langs->trans('PaymentConditions');
  1819. print '<td>';
  1820. if ($action != 'editconditions') {
  1821. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editconditions&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->trans('SetConditions'), 1).'</a></td>';
  1822. }
  1823. print '</tr></table>';
  1824. print '</td><td>';
  1825. if ($action == 'editconditions') {
  1826. $form->form_conditions_reglement($_SERVER['PHP_SELF'].'?id='.$object->id, $object->cond_reglement_id, 'cond_reglement_id');
  1827. } else {
  1828. $form->form_conditions_reglement($_SERVER['PHP_SELF'].'?id='.$object->id, $object->cond_reglement_id, 'none');
  1829. }
  1830. print "</td>";
  1831. print '</tr>';
  1832. // Mode of payment
  1833. $langs->load('bills');
  1834. print '<tr><td class="nowrap">';
  1835. print '<table class="nobordernopadding centpercent"><tr><td class="nowrap">';
  1836. print $langs->trans('PaymentMode');
  1837. print '</td>';
  1838. if ($action != 'editmode') {
  1839. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editmode&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->trans('SetMode'), 1).'</a></td>';
  1840. }
  1841. print '</tr></table>';
  1842. print '</td><td>';
  1843. if ($action == 'editmode') {
  1844. $form->form_modes_reglement($_SERVER['PHP_SELF'].'?id='.$object->id, $object->mode_reglement_id, 'mode_reglement_id', 'DBIT', 1, 1);
  1845. } else {
  1846. $form->form_modes_reglement($_SERVER['PHP_SELF'].'?id='.$object->id, $object->mode_reglement_id, 'none');
  1847. }
  1848. print '</td></tr>';
  1849. // Multicurrency
  1850. if (!empty($conf->multicurrency->enabled)) {
  1851. // Multicurrency code
  1852. print '<tr>';
  1853. print '<td>';
  1854. print '<table class="nobordernopadding centpercent"><tr><td>';
  1855. print $form->editfieldkey('Currency', 'multicurrency_code', '', $object, 0);
  1856. print '</td>';
  1857. if ($action != 'editmulticurrencycode' && $object->statut == $object::STATUS_DRAFT) {
  1858. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editmulticurrencycode&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->transnoentitiesnoconv('SetMultiCurrencyCode'), 1).'</a></td>';
  1859. }
  1860. print '</tr></table>';
  1861. print '</td><td>';
  1862. if ($action == 'editmulticurrencycode') {
  1863. $form->form_multicurrency_code($_SERVER['PHP_SELF'].'?id='.$object->id, $object->multicurrency_code, 'multicurrency_code');
  1864. } else {
  1865. $form->form_multicurrency_code($_SERVER['PHP_SELF'].'?id='.$object->id, $object->multicurrency_code, 'none');
  1866. }
  1867. print '</td></tr>';
  1868. // Multicurrency rate
  1869. if ($object->multicurrency_code != $conf->currency || $object->multicurrency_tx != 1) {
  1870. print '<tr>';
  1871. print '<td>';
  1872. print '<table class="nobordernopadding centpercent"><tr>';
  1873. print '<td>';
  1874. print $form->editfieldkey('CurrencyRate', 'multicurrency_tx', '', $object, 0);
  1875. print '</td>';
  1876. if ($action != 'editmulticurrencyrate' && $object->statut == $object::STATUS_DRAFT && $object->multicurrency_code && $object->multicurrency_code != $conf->currency) {
  1877. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editmulticurrencyrate&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->transnoentitiesnoconv('SetMultiCurrencyCode'), 1).'</a></td>';
  1878. }
  1879. print '</tr></table>';
  1880. print '</td><td>';
  1881. if ($action == 'editmulticurrencyrate' || $action == 'actualizemulticurrencyrate') {
  1882. if ($action == 'actualizemulticurrencyrate') {
  1883. list($object->fk_multicurrency, $object->multicurrency_tx) = MultiCurrency::getIdAndTxFromCode($object->db, $object->multicurrency_code);
  1884. }
  1885. $form->form_multicurrency_rate($_SERVER['PHP_SELF'].'?id='.$object->id, $object->multicurrency_tx, 'multicurrency_tx', $object->multicurrency_code);
  1886. } else {
  1887. $form->form_multicurrency_rate($_SERVER['PHP_SELF'].'?id='.$object->id, $object->multicurrency_tx, 'none', $object->multicurrency_code);
  1888. if ($object->statut == $object::STATUS_DRAFT && $object->multicurrency_code && $object->multicurrency_code != $conf->currency) {
  1889. print '<div class="inline-block"> &nbsp; &nbsp; &nbsp; &nbsp; ';
  1890. print '<a href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=actualizemulticurrencyrate">'.$langs->trans("ActualizeCurrency").'</a>';
  1891. print '</div>';
  1892. }
  1893. }
  1894. print '</td></tr>';
  1895. }
  1896. }
  1897. // Bank Account
  1898. if (!empty($conf->global->BANK_ASK_PAYMENT_BANK_DURING_SUPPLIER_ORDER) && !empty($conf->banque->enabled)) {
  1899. print '<tr><td class="nowrap">';
  1900. print '<table class="nobordernopadding centpercent"><tr><td class="nowrap">';
  1901. print $langs->trans('BankAccount');
  1902. print '<td>';
  1903. if ($action != 'editbankaccount' && $usercancreate) {
  1904. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editbankaccount&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->trans('SetBankAccount'), 1).'</a></td>';
  1905. }
  1906. print '</tr></table>';
  1907. print '</td><td>';
  1908. if ($action == 'editbankaccount') {
  1909. $form->formSelectAccount($_SERVER['PHP_SELF'].'?id='.$object->id, $object->fk_account, 'fk_account', 1);
  1910. } else {
  1911. $form->formSelectAccount($_SERVER['PHP_SELF'].'?id='.$object->id, $object->fk_account, 'none');
  1912. }
  1913. print '</td>';
  1914. print '</tr>';
  1915. }
  1916. // Delivery delay (in days)
  1917. print '<tr>';
  1918. print '<td>'.$langs->trans('NbDaysToDelivery').'&nbsp;'.img_picto($langs->trans('DescNbDaysToDelivery'), 'info', 'style="cursor:help"').'</td>';
  1919. print '<td>'.$object->getMaxDeliveryTimeDay($langs).'</td>';
  1920. print '</tr>';
  1921. // Delivery date planed
  1922. print '<tr><td>';
  1923. print '<table class="nobordernopadding centpercent"><tr><td>';
  1924. print $langs->trans('DateDeliveryPlanned');
  1925. print '</td>';
  1926. if ($action != 'editdate_livraison') {
  1927. print '<td class="right"><a class="editfielda" href="'.$_SERVER["PHP_SELF"].'?action=editdate_livraison&token='.newToken().'&id='.$object->id.'">'.img_edit($langs->trans('SetDeliveryDate'), 1).'</a></td>';
  1928. }
  1929. print '</tr></table>';
  1930. print '</td><td>';
  1931. if ($action == 'editdate_livraison') {
  1932. print '<form name="setdate_livraison" action="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'" method="post">';
  1933. print '<input type="hidden" name="token" value="'.newToken().'">';
  1934. print '<input type="hidden" name="action" value="setdate_livraison">';
  1935. $usehourmin = 0;
  1936. if (!empty($conf->global->SUPPLIER_ORDER_USE_HOUR_FOR_DELIVERY_DATE)) {
  1937. $usehourmin = 1;
  1938. }
  1939. print $form->selectDate($object->delivery_date ? $object->delivery_date : -1, 'liv_', $usehourmin, $usehourmin, '', "setdate_livraison");
  1940. print '<input type="submit" class="button button-edit" value="'.$langs->trans('Modify').'">';
  1941. print '</form>';
  1942. } else {
  1943. $usehourmin = 'day';
  1944. if (!empty($conf->global->SUPPLIER_ORDER_USE_HOUR_FOR_DELIVERY_DATE)) {
  1945. $usehourmin = 'dayhour';
  1946. }
  1947. print $object->delivery_date ? dol_print_date($object->delivery_date, $usehourmin) : '&nbsp;';
  1948. if ($object->hasDelay() && !empty($object->delivery_date)) {
  1949. print ' '.img_picto($langs->trans("Late").' : '.$object->showDelay(), "warning");
  1950. }
  1951. }
  1952. print '</td></tr>';
  1953. // Incoterms
  1954. if (!empty($conf->incoterm->enabled)) {
  1955. print '<tr><td>';
  1956. print '<table class="nobordernopadding centpercent"><tr><td>';
  1957. print $langs->trans('IncotermLabel');
  1958. print '<td><td class="right">';
  1959. if ($usercancreate) {
  1960. print '<a class="editfielda" href="'.DOL_URL_ROOT.'/fourn/commande/card.php?id='.$object->id.'&action=editincoterm&token='.newToken().'">'.img_edit().'</a>';
  1961. } else {
  1962. print '&nbsp;';
  1963. }
  1964. print '</td></tr></table>';
  1965. print '</td>';
  1966. print '<td>';
  1967. if ($action != 'editincoterm') {
  1968. print $form->textwithpicto(dol_escape_htmltag($object->display_incoterms()), $object->label_incoterms, 1);
  1969. } else {
  1970. print $form->select_incoterms((!empty($object->fk_incoterms) ? $object->fk_incoterms : ''), (!empty($object->location_incoterms) ? $object->location_incoterms : ''), $_SERVER['PHP_SELF'].'?id='.$object->id);
  1971. }
  1972. print '</td></tr>';
  1973. }
  1974. // Other attributes
  1975. include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_view.tpl.php';
  1976. print '</table>';
  1977. print '</div>';
  1978. print '<div class="fichehalfright">';
  1979. print '<div class="underbanner clearboth"></div>';
  1980. print '<table class="border tableforfield centpercent">';
  1981. if ($object->multicurrency_code != $conf->currency || $object->multicurrency_tx != 1) {
  1982. // Multicurrency Amount HT
  1983. print '<tr><td class="titlefieldmiddle">'.$form->editfieldkey('MulticurrencyAmountHT', 'multicurrency_total_ht', '', $object, 0).'</td>';
  1984. print '<td class="nowrap">'.price($object->multicurrency_total_ht, '', $langs, 0, - 1, - 1, (!empty($object->multicurrency_code) ? $object->multicurrency_code : $conf->currency)).'</td>';
  1985. print '</tr>';
  1986. // Multicurrency Amount VAT
  1987. print '<tr><td>'.$form->editfieldkey('MulticurrencyAmountVAT', 'multicurrency_total_tva', '', $object, 0).'</td>';
  1988. print '<td class="nowrap">'.price($object->multicurrency_total_tva, '', $langs, 0, - 1, - 1, (!empty($object->multicurrency_code) ? $object->multicurrency_code : $conf->currency)).'</td>';
  1989. print '</tr>';
  1990. // Multicurrency Amount TTC
  1991. print '<tr><td>'.$form->editfieldkey('MulticurrencyAmountTTC', 'multicurrency_total_ttc', '', $object, 0).'</td>';
  1992. print '<td class="nowrap">'.price($object->multicurrency_total_ttc, '', $langs, 0, - 1, - 1, (!empty($object->multicurrency_code) ? $object->multicurrency_code : $conf->currency)).'</td>';
  1993. print '</tr>';
  1994. }
  1995. // Total
  1996. $alert = '';
  1997. if (!empty($conf->global->ORDER_MANAGE_MIN_AMOUNT) && $object->total_ht < $object->thirdparty->supplier_order_min_amount) {
  1998. $alert = ' '.img_warning($langs->trans('OrderMinAmount').': '.price($object->thirdparty->supplier_order_min_amount));
  1999. }
  2000. print '<tr><td class="titlefieldmiddle">'.$langs->trans("AmountHT").'</td>';
  2001. print '<td>'.price($object->total_ht, '', $langs, 1, -1, -1, $conf->currency).$alert.'</td>';
  2002. print '</tr>';
  2003. // Total VAT
  2004. print '<tr><td>'.$langs->trans("AmountVAT").'</td><td>'.price($object->total_tva, '', $langs, 1, -1, -1, $conf->currency).'</td>';
  2005. print '</tr>';
  2006. // Amount Local Taxes
  2007. if ($mysoc->localtax1_assuj == "1" || $object->total_localtax1 != 0) { //Localtax1
  2008. print '<tr><td>'.$langs->transcountry("AmountLT1", $mysoc->country_code).'</td>';
  2009. print '<td>'.price($object->total_localtax1, '', $langs, 1, -1, -1, $conf->currency).'</td>';
  2010. print '</tr>';
  2011. }
  2012. if ($mysoc->localtax2_assuj == "1" || $object->total_localtax2 != 0) { //Localtax2
  2013. print '<tr><td>'.$langs->transcountry("AmountLT2", $mysoc->country_code).'</td>';
  2014. print '<td>'.price($object->total_localtax2, '', $langs, 1, -1, -1, $conf->currency).'</td>';
  2015. print '</tr>';
  2016. }
  2017. // Total TTC
  2018. print '<tr><td>'.$langs->trans("AmountTTC").'</td><td>'.price($object->total_ttc, '', $langs, 1, -1, -1, $conf->currency).'</td>';
  2019. print '</tr>';
  2020. print '</table>';
  2021. // Margin Infos
  2022. /*if (! empty($conf->margin->enabled)) {
  2023. $formmargin->displayMarginInfos($object);
  2024. }*/
  2025. print '</div>';
  2026. print '</div>';
  2027. print '<div class="clearboth"></div><br>';
  2028. if (!empty($conf->global->MAIN_DISABLE_CONTACTS_TAB)) {
  2029. $blocname = 'contacts';
  2030. $title = $langs->trans('ContactsAddresses');
  2031. include DOL_DOCUMENT_ROOT.'/core/tpl/bloc_showhide.tpl.php';
  2032. }
  2033. if (!empty($conf->global->MAIN_DISABLE_NOTES_TAB)) {
  2034. $blocname = 'notes';
  2035. $title = $langs->trans('Notes');
  2036. include DOL_DOCUMENT_ROOT.'/core/tpl/bloc_showhide.tpl.php';
  2037. }
  2038. /*
  2039. * Lines
  2040. */
  2041. //$result = $object->getLinesArray();
  2042. print ' <form name="addproduct" id="addproduct" action="'.$_SERVER["PHP_SELF"].'?id='.$object->id.(($action != 'editline') ? '' : '#line_'.GETPOST('lineid', 'int')).'" method="POST">
  2043. <input type="hidden" name="token" value="'.newToken().'">
  2044. <input type="hidden" name="action" value="' . (($action != 'editline') ? 'addline' : 'updateline').'">
  2045. <input type="hidden" name="mode" value="">
  2046. <input type="hidden" name="page_y" value="">
  2047. <input type="hidden" name="id" value="'.$object->id.'">
  2048. <input type="hidden" name="socid" value="'.$societe->id.'">
  2049. ';
  2050. if (!empty($conf->use_javascript_ajax) && $object->statut == 0) {
  2051. include DOL_DOCUMENT_ROOT.'/core/tpl/ajaxrow.tpl.php';
  2052. }
  2053. print '<div class="div-table-responsive-no-min">';
  2054. print '<table id="tablelines" class="noborder noshadow centpercent">';
  2055. // Add free products/services form
  2056. global $forceall, $senderissupplier, $dateSelector, $inputalsopricewithtax;
  2057. $forceall = 1; $dateSelector = 0; $inputalsopricewithtax = 1;
  2058. $senderissupplier = 2; // $senderissupplier=2 is same than 1 but disable test on minimum qty and disable autofill qty with minimum.
  2059. if (!empty($conf->global->SUPPLIER_ORDER_WITH_PREDEFINED_PRICES_ONLY)) {
  2060. $senderissupplier = 1;
  2061. }
  2062. // Show object lines
  2063. if (!empty($object->lines)) {
  2064. $ret = $object->printObjectLines($action, $societe, $mysoc, $lineid, 1);
  2065. }
  2066. $num = count($object->lines);
  2067. // Form to add new line
  2068. if ($object->statut == CommandeFournisseur::STATUS_DRAFT && $usercancreate) {
  2069. if ($action != 'editline') {
  2070. // Add free products/services
  2071. $parameters = array();
  2072. $reshook = $hookmanager->executeHooks('formAddObjectLine', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
  2073. if ($reshook < 0) setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
  2074. if (empty($reshook))
  2075. $object->formAddObjectLine(1, $societe, $mysoc);
  2076. }
  2077. }
  2078. print '</table>';
  2079. print '</div>';
  2080. print '</form>';
  2081. print dol_get_fiche_end();
  2082. /**
  2083. * Buttons for actions
  2084. */
  2085. if ($user->socid == 0 && $action != 'editline' && $action != 'delete') {
  2086. print '<div class="tabsAction">';
  2087. $parameters = array();
  2088. $reshook = $hookmanager->executeHooks('addMoreActionsButtons', $parameters, $object, $action); // Note that $action and $object may have been
  2089. // modified by hook
  2090. if (empty($reshook)) {
  2091. $object->fetchObjectLinked(); // Links are used to show or not button, so we load them now.
  2092. // Validate
  2093. if ($object->statut == 0 && $num > 0) {
  2094. if ($usercanvalidate) {
  2095. $tmpbuttonlabel = $langs->trans('Validate');
  2096. if ($usercanapprove && empty($conf->global->SUPPLIER_ORDER_NO_DIRECT_APPROVE)) {
  2097. $tmpbuttonlabel = $langs->trans("ValidateAndApprove");
  2098. }
  2099. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=valid">';
  2100. print $tmpbuttonlabel;
  2101. print '</a>';
  2102. }
  2103. }
  2104. // Create event
  2105. /*if ($conf->agenda->enabled && ! empty($conf->global->MAIN_ADD_EVENT_ON_ELEMENT_CARD)) // Add hidden condition because this is not a "workflow" action so should appears somewhere else on page.
  2106. {
  2107. print '<div class="inline-block divButAction"><a class="butAction" href="' . DOL_URL_ROOT . '/comm/action/card.php?action=create&amp;origin=' . $object->element . '&amp;originid=' . $object->id . '&amp;socid=' . $object->socid . '">' . $langs->trans("AddAction") . '</a></div>';
  2108. }*/
  2109. // Modify
  2110. if ($object->statut == CommandeFournisseur::STATUS_VALIDATED) {
  2111. if ($usercanorder) {
  2112. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=reopen&token='.newToken().'">'.$langs->trans("Modify").'</a>';
  2113. }
  2114. }
  2115. // Approve
  2116. if ($object->statut == CommandeFournisseur::STATUS_VALIDATED) {
  2117. if ($usercanapprove) {
  2118. if (!empty($conf->global->SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED) && $object->total_ht >= $conf->global->SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED && !empty($object->user_approve_id)) {
  2119. print '<a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("FirstApprovalAlreadyDone")).'">'.$langs->trans("ApproveOrder").'</a>';
  2120. } else {
  2121. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=approve">'.$langs->trans("ApproveOrder").'</a>';
  2122. }
  2123. } else {
  2124. print '<a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("NotAllowed")).'">'.$langs->trans("ApproveOrder").'</a>';
  2125. }
  2126. }
  2127. // Second approval (if option SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED is set)
  2128. if (!empty($conf->global->SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED) && $object->total_ht >= $conf->global->SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED) {
  2129. if ($object->statut == CommandeFournisseur::STATUS_VALIDATED) {
  2130. if ($usercanapprovesecond) {
  2131. if (!empty($object->user_approve_id2)) {
  2132. print '<a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("SecondApprovalAlreadyDone")).'">'.$langs->trans("Approve2Order").'</a>';
  2133. } else {
  2134. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=approve2">'.$langs->trans("Approve2Order").'</a>';
  2135. }
  2136. } else {
  2137. print '<a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("NotAllowed")).'">'.$langs->trans("Approve2Order").'</a>';
  2138. }
  2139. }
  2140. }
  2141. // Refuse
  2142. if ($object->statut == CommandeFournisseur::STATUS_VALIDATED) {
  2143. if ($usercanapprove || $usercanapprovesecond) {
  2144. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=refuse">'.$langs->trans("RefuseOrder").'</a>';
  2145. } else {
  2146. print '<a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("NotAllowed")).'">'.$langs->trans("RefuseOrder").'</a>';
  2147. }
  2148. }
  2149. // Send
  2150. if (empty($user->socid)) {
  2151. if (in_array($object->statut, array(CommandeFournisseur::STATUS_ACCEPTED, 3, 4, 5)) || !empty($conf->global->SUPPLIER_ORDER_SENDBYEMAIL_FOR_ALL_STATUS)) {
  2152. if ($usercanorder) {
  2153. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=presend&mode=init#formmailbeforetitle">'.$langs->trans('SendMail').'</a>';
  2154. }
  2155. }
  2156. }
  2157. // Reopen
  2158. if (in_array($object->statut, array(CommandeFournisseur::STATUS_ACCEPTED))) {
  2159. $buttonshown = 0;
  2160. if (!$buttonshown && $usercanapprove) {
  2161. if (empty($conf->global->SUPPLIER_ORDER_REOPEN_BY_APPROVER_ONLY)
  2162. || (!empty($conf->global->SUPPLIER_ORDER_REOPEN_BY_APPROVER_ONLY) && $user->id == $object->user_approve_id)) {
  2163. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=reopen&token='.newToken().'">'.$langs->trans("Disapprove").'</a>';
  2164. $buttonshown++;
  2165. }
  2166. }
  2167. if (!$buttonshown && $usercanapprovesecond && !empty($conf->global->SUPPLIER_ORDER_3_STEPS_TO_BE_APPROVED)) {
  2168. if (empty($conf->global->SUPPLIER_ORDER_REOPEN_BY_APPROVER2_ONLY)
  2169. || (!empty($conf->global->SUPPLIER_ORDER_REOPEN_BY_APPROVER2_ONLY) && $user->id == $object->user_approve_id2)) {
  2170. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=reopen&token='.newToken().'">'.$langs->trans("Disapprove").'</a>';
  2171. }
  2172. }
  2173. }
  2174. if (in_array($object->statut, array(3, 4, 5, 6, 7, 9))) {
  2175. if ($usercanorder) {
  2176. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=reopen&token='.newToken().'">'.$langs->trans("ReOpen").'</a>';
  2177. }
  2178. }
  2179. // Ship
  2180. $hasreception = 0;
  2181. if (!empty($conf->stock->enabled) && (!empty($conf->global->STOCK_CALCULATE_ON_SUPPLIER_DISPATCH_ORDER) || !empty($conf->global->STOCK_CALCULATE_ON_RECEPTION) || !empty($conf->global->STOCK_CALCULATE_ON_RECEPTION_CLOSE))) {
  2182. $labelofbutton = $langs->trans('ReceiveProducts');
  2183. if ($conf->reception->enabled) {
  2184. $labelofbutton = $langs->trans("CreateReception");
  2185. if (!empty($object->linkedObjects['reception'])) {
  2186. foreach ($object->linkedObjects['reception'] as $element) {
  2187. if ($element->statut >= 0) {
  2188. $hasreception = 1;
  2189. break;
  2190. }
  2191. }
  2192. }
  2193. }
  2194. if (in_array($object->statut, array(3, 4, 5))) {
  2195. if (((!empty($conf->fournisseur->enabled) && empty($conf->global->MAIN_USE_NEW_SUPPLIERMOD)) || !empty($conf->supplier_order->enabled)) && $usercanreceive) {
  2196. print '<div class="inline-block divButAction"><a class="butAction" href="'.DOL_URL_ROOT.'/fourn/commande/dispatch.php?id='.$object->id.'">'.$labelofbutton.'</a></div>';
  2197. } else {
  2198. print '<div class="inline-block divButAction"><a class="butActionRefused classfortooltip" href="#" title="'.dol_escape_htmltag($langs->trans("NotAllowed")).'">'.$labelofbutton.'</a></div>';
  2199. }
  2200. }
  2201. }
  2202. if ($object->statut == CommandeFournisseur::STATUS_ACCEPTED) {
  2203. if ($usercanorder) {
  2204. print '<div class="inline-block divButAction"><a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=makeorder#makeorder">'.$langs->trans("MakeOrder").'</a></div>';
  2205. } else {
  2206. print '<div class="inline-block divButAction"><a class="butActionRefused classfortooltip" href="#">'.$langs->trans("MakeOrder").'</a></div>';
  2207. }
  2208. }
  2209. // Classify received (this does not record reception)
  2210. if ($object->statut == CommandeFournisseur::STATUS_ORDERSENT || $object->statut == CommandeFournisseur::STATUS_RECEIVED_PARTIALLY) {
  2211. if ($usercanreceive) {
  2212. print '<div class="inline-block divButAction"><a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&token='.newToken().'&action=classifyreception#classifyreception">'.$langs->trans("ClassifyReception").'</a></div>';
  2213. }
  2214. }
  2215. // Create bill
  2216. //if (! empty($conf->facture->enabled))
  2217. //{
  2218. if (((!empty($conf->fournisseur->enabled) && empty($conf->global->MAIN_USE_NEW_SUPPLIERMOD)) || !empty($conf->supplier_invoice->enabled)) && ($object->statut >= 2 && $object->statut != 7 && $object->billed != 1)) { // statut 2 means approved, 7 means canceled
  2219. if ($user->rights->fournisseur->facture->creer || $user->rights->supplier_invoice->creer) {
  2220. print '<a class="butAction" href="'.DOL_URL_ROOT.'/fourn/facture/card.php?action=create&amp;origin='.$object->element.'&amp;originid='.$object->id.'&amp;socid='.$object->socid.'">'.$langs->trans("CreateBill").'</a>';
  2221. }
  2222. }
  2223. //}
  2224. // Classify billed manually (need one invoice if module invoice is on, no condition on invoice if not)
  2225. if ($usercancreate && $object->statut >= 2 && $object->statut != 7 && $object->billed != 1) { // statut 2 means approved
  2226. if (empty($conf->facture->enabled)) {
  2227. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=classifybilled&token='.newToken().'">'.$langs->trans("ClassifyBilled").'</a>';
  2228. } else {
  2229. if (!empty($object->linkedObjectsIds['invoice_supplier'])) {
  2230. if ($user->rights->fournisseur->facture->creer || $user->rights->supplier_invoice->creer) {
  2231. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=classifybilled&token='.newToken().'">'.$langs->trans("ClassifyBilled").'</a>';
  2232. }
  2233. } else {
  2234. print '<a class="butActionRefused" href="#" title="'.dol_escape_htmltag($langs->trans("NeedAtLeastOneInvoice")).'">'.$langs->trans("ClassifyBilled").'</a>';
  2235. }
  2236. }
  2237. }
  2238. // Create a remote order using WebService only if module is activated
  2239. if (!empty($conf->syncsupplierwebservices->enabled) && $object->statut >= 2) { // 2 means accepted
  2240. print '<a class="butAction" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=webservice&token='.newToken().'&mode=init">'.$langs->trans('CreateRemoteOrder').'</a>';
  2241. }
  2242. // Clone
  2243. if ($usercancreate) {
  2244. print '<a class="butAction" href="'.$_SERVER['PHP_SELF'].'?id='.$object->id.'&amp;socid='.$object->socid.'&amp;action=clone&amp;object=order">'.$langs->trans("ToClone").'</a>';
  2245. }
  2246. // Cancel
  2247. if ($object->statut == CommandeFournisseur::STATUS_ACCEPTED) {
  2248. if ($usercanorder) {
  2249. print '<a class="butActionDelete" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&amp;action=cancel">'.$langs->trans("CancelOrder").'</a>';
  2250. }
  2251. }
  2252. // Delete
  2253. if (!empty($usercandelete)) {
  2254. if ($hasreception) {
  2255. print '<a class="butActionRefused classfortooltip" href="#" title="'.$langs->trans("ReceptionExist").'">'.$langs->trans("Delete").'</a>';
  2256. } else {
  2257. print '<a class="butActionDelete" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=delete&token='.newToken().'">'.$langs->trans("Delete").'</a>';
  2258. }
  2259. }
  2260. }
  2261. print "</div>";
  2262. if ($usercanorder && $object->statut == CommandeFournisseur::STATUS_ACCEPTED && $action == 'makeorder') {
  2263. // Set status to ordered (action=commande)
  2264. print '<!-- form to record supplier order -->'."\n";
  2265. print '<form name="commande" id="makeorder" action="card.php?id='.$object->id.'&amp;action=commande" method="POST">';
  2266. print '<input type="hidden" name="token" value="'.newToken().'">';
  2267. print '<input type="hidden" name="action" value="commande">';
  2268. print load_fiche_titre($langs->trans("ToOrder"), '', '');
  2269. print '<table class="noborder centpercent">';
  2270. //print '<tr class="liste_titre"><td colspan="2">'.$langs->trans("ToOrder").'</td></tr>';
  2271. print '<tr><td class="fieldrequired">'.$langs->trans("OrderDate").'</td><td>';
  2272. $date_com = dol_mktime(GETPOST('rehour', 'int'), GETPOST('remin', 'int'), GETPOST('resec', 'int'), GETPOST('remonth', 'int'), GETPOST('reday', 'int'), GETPOST('reyear', 'int'));
  2273. if (empty($date_com)) {
  2274. $date_com = dol_now();
  2275. }
  2276. print $form->selectDate($date_com, '', 1, 1, '', "commande", 1, 1);
  2277. print '</td></tr>';
  2278. // Force mandatory order method
  2279. print '<tr><td class="fieldrequired">'.$langs->trans("OrderMode").'</td><td>';
  2280. $formorder->selectInputMethod(GETPOST('methodecommande'), "methodecommande", 1);
  2281. print '</td></tr>';
  2282. print '<tr><td>'.$langs->trans("Comment").'</td><td><input size="40" type="text" name="comment" value="'.GETPOST('comment').'"></td></tr>';
  2283. print '<tr><td class="center" colspan="2">';
  2284. print '<input type="submit" name="makeorder" class="button" value="'.$langs->trans("ToOrder").'">';
  2285. print ' &nbsp; &nbsp; ';
  2286. print '<input type="submit" name="cancel" class="button button-cancel" value="'.$langs->trans("Cancel").'">';
  2287. print '</td></tr>';
  2288. print '</table>';
  2289. print '</form>';
  2290. print "<br>";
  2291. }
  2292. if ($action != 'makeorder') {
  2293. print '<div class="fichecenter"><div class="fichehalfleft">';
  2294. // Generated documents
  2295. $objref = dol_sanitizeFileName($object->ref);
  2296. $file = $conf->fournisseur->dir_output.'/commande/'.$objref.'/'.$objref.'.pdf';
  2297. $relativepath = $objref.'/'.$objref.'.pdf';
  2298. $filedir = $conf->fournisseur->dir_output.'/commande/'.$objref;
  2299. $urlsource = $_SERVER["PHP_SELF"]."?id=".$object->id;
  2300. $genallowed = $usercanread;
  2301. $delallowed = $usercancreate;
  2302. $modelpdf = (!empty($object->model_pdf) ? $object->model_pdf : (empty($conf->global->COMMANDE_SUPPLIER_ADDON_PDF) ? '' : $conf->global->COMMANDE_SUPPLIER_ADDON_PDF));
  2303. print $formfile->showdocuments('commande_fournisseur', $objref, $filedir, $urlsource, $genallowed, $delallowed, $modelpdf, 1, 0, 0, 0, 0, '', '', '', $object->thirdparty->default_lang);
  2304. $somethingshown = $formfile->numoffiles;
  2305. // Show links to link elements
  2306. $linktoelem = $form->showLinkToObjectBlock($object, null, array('supplier_order', 'order_supplier'));
  2307. $somethingshown = $form->showLinkedObjectBlock($object, $linktoelem);
  2308. print '</div><div class="fichehalfright">';
  2309. if ($action == 'classifyreception') {
  2310. if ($usercanreceive && ($object->statut == CommandeFournisseur::STATUS_ORDERSENT || $object->statut == CommandeFournisseur::STATUS_RECEIVED_PARTIALLY)) {
  2311. // Set status to received (action=livraison)
  2312. print '<!-- form to record purchase order received -->'."\n";
  2313. print '<form id="classifyreception" action="card.php?id='.$object->id.'" method="post">';
  2314. print '<input type="hidden" name="token" value="'.newToken().'">';
  2315. print '<input type="hidden" name="action" value="livraison">';
  2316. print load_fiche_titre($langs->trans("Receive"), '', '');
  2317. print '<table class="noborder centpercent">';
  2318. //print '<tr class="liste_titre"><td colspan="2">'.$langs->trans("Receive").'</td></tr>';
  2319. print '<tr><td>'.$langs->trans("DeliveryDate").'</td><td>';
  2320. $datepreselected = dol_now();
  2321. print $form->selectDate($datepreselected, '', 1, 1, '', "commande", 1, 1);
  2322. print "</td></tr>\n";
  2323. print '<tr><td class="fieldrequired">'.$langs->trans("Delivery")."</td><td>\n";
  2324. $liv = array();
  2325. $liv[''] = '&nbsp;';
  2326. $liv['tot'] = $langs->trans("CompleteOrNoMoreReceptionExpected");
  2327. $liv['par'] = $langs->trans("PartialWoman");
  2328. $liv['nev'] = $langs->trans("NeverReceived");
  2329. $liv['can'] = $langs->trans("Canceled");
  2330. print $form->selectarray("type", $liv);
  2331. print '</td></tr>';
  2332. print '<tr><td>'.$langs->trans("Comment").'</td><td><input size="40" type="text" name="comment"></td></tr>';
  2333. print '<tr><td class="center" colspan="2">';
  2334. print '<input type="submit" name="receive" class="button" value="'.$langs->trans("Receive").'">';
  2335. print ' &nbsp; &nbsp; ';
  2336. print '<input type="submit" name="cancel" class="button button-cancel" value="'.$langs->trans("Cancel").'">';
  2337. print '</td></tr>';
  2338. print "</table>\n";
  2339. print "</form>\n";
  2340. print "<br>";
  2341. }
  2342. }
  2343. // List of actions on element
  2344. include_once DOL_DOCUMENT_ROOT.'/core/class/html.formactions.class.php';
  2345. $formactions = new FormActions($db);
  2346. $somethingshown = $formactions->showactions($object, 'order_supplier', $socid, 1, 'listaction'.($genallowed ? 'largetitle' : ''));
  2347. print '</div></div>';
  2348. }
  2349. /*
  2350. * Action webservice
  2351. */
  2352. if ($action == 'webservice' && GETPOST('mode', 'alpha') != "send" && !GETPOST('cancel', 'alpha')) {
  2353. $mode = GETPOST('mode', 'alpha');
  2354. $ws_url = $object->thirdparty->webservices_url;
  2355. $ws_key = $object->thirdparty->webservices_key;
  2356. $ws_user = GETPOST('ws_user', 'alpha');
  2357. $ws_password = GETPOST('ws_password', 'alpha');
  2358. // NS and Authentication parameters
  2359. $ws_ns = 'http://www.dolibarr.org/ns/';
  2360. $ws_authentication = array(
  2361. 'dolibarrkey'=>$ws_key,
  2362. 'sourceapplication'=>'DolibarrWebServiceClient',
  2363. 'login'=>$ws_user,
  2364. 'password'=>$ws_password,
  2365. 'entity'=>''
  2366. );
  2367. print load_fiche_titre($langs->trans('CreateRemoteOrder'), '');
  2368. //Is everything filled?
  2369. if (empty($ws_url) || empty($ws_key)) {
  2370. setEventMessages($langs->trans("ErrorWebServicesFieldsRequired"), null, 'errors');
  2371. $mode = "init";
  2372. $error_occurred = true; //Don't allow to set the user/pass if thirdparty fields are not filled
  2373. } elseif ($mode != "init" && (empty($ws_user) || empty($ws_password))) {
  2374. setEventMessages($langs->trans("ErrorFieldsRequired"), null, 'errors');
  2375. $mode = "init";
  2376. }
  2377. if ($mode == "init") {
  2378. //Table/form header
  2379. print '<table class="border centpercent">';
  2380. print '<form action="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'" method="post">';
  2381. print '<input type="hidden" name="token" value="'.newToken().'">';
  2382. print '<input type="hidden" name="action" value="webservice">';
  2383. print '<input type="hidden" name="mode" value="check">';
  2384. if ($error_occurred) {
  2385. print "<br>".$langs->trans("ErrorOccurredReviseAndRetry")."<br>";
  2386. print '<input class="button button-cancel" type="submit" id="cancel" name="cancel" value="'.$langs->trans("Cancel").'">';
  2387. } else {
  2388. $textinput_size = "50";
  2389. // Webservice url
  2390. print '<tr><td>'.$langs->trans("WebServiceURL").'</td><td colspan="3">'.dol_print_url($ws_url).'</td></tr>';
  2391. //Remote User
  2392. print '<tr><td>'.$langs->trans("User").'</td><td><input size="'.$textinput_size.'" type="text" name="ws_user"></td></tr>';
  2393. //Remote Password
  2394. print '<tr><td>'.$langs->trans("Password").'</td><td><input size="'.$textinput_size.'" type="text" name="ws_password"></td></tr>';
  2395. //Submit button
  2396. print '<tr><td class="center" colspan="2">';
  2397. print '<input type="submit" class="button" id="ws_submit" name="ws_submit" value="'.$langs->trans("CreateRemoteOrder").'">';
  2398. print ' &nbsp; &nbsp; ';
  2399. //Cancel button
  2400. print '<input class="button button-cancel" type="submit" id="cancel" name="cancel" value="'.$langs->trans("Cancel").'">';
  2401. print '</td></tr>';
  2402. }
  2403. //End table/form
  2404. print '</form>';
  2405. print '</table>';
  2406. } elseif ($mode == "check") {
  2407. $ws_entity = '';
  2408. $ws_thirdparty = '';
  2409. $error_occurred = false;
  2410. //Create SOAP client and connect it to user
  2411. $soapclient_user = new nusoap_client($ws_url."/webservices/server_user.php");
  2412. $soapclient_user->soap_defencoding = 'UTF-8';
  2413. $soapclient_user->decodeUTF8(false);
  2414. //Get the thirdparty associated to user
  2415. $ws_parameters = array('authentication'=>$ws_authentication, 'id' => '', 'ref'=>$ws_user);
  2416. $result_user = $soapclient_user->call("getUser", $ws_parameters, $ws_ns, '');
  2417. $user_status_code = $result_user["result"]["result_code"];
  2418. if ($user_status_code == "OK") {
  2419. //Fill the variables
  2420. $ws_entity = $result_user["user"]["entity"];
  2421. $ws_authentication['entity'] = $ws_entity;
  2422. $ws_thirdparty = $result_user["user"]["fk_thirdparty"];
  2423. if (empty($ws_thirdparty)) {
  2424. setEventMessages($langs->trans("RemoteUserMissingAssociatedSoc"), null, 'errors');
  2425. $error_occurred = true;
  2426. } else {
  2427. //Create SOAP client and connect it to product/service
  2428. $soapclient_product = new nusoap_client($ws_url."/webservices/server_productorservice.php");
  2429. $soapclient_product->soap_defencoding = 'UTF-8';
  2430. $soapclient_product->decodeUTF8(false);
  2431. // Iterate each line and get the reference that uses the supplier of that product/service
  2432. $i = 0;
  2433. foreach ($object->lines as $line) {
  2434. $i = $i + 1;
  2435. $ref_supplier = $line->ref_supplier;
  2436. $line_id = $i."º) ".$line->product_ref.": ";
  2437. if (empty($ref_supplier)) {
  2438. continue;
  2439. }
  2440. $ws_parameters = array('authentication' => $ws_authentication, 'id' => '', 'ref' => $ref_supplier);
  2441. $result_product = $soapclient_product->call("getProductOrService", $ws_parameters, $ws_ns, '');
  2442. if (!$result_product) {
  2443. setEventMessages($line_id.$langs->trans("SOAPError")." ".$soapclient_product->error_str." - ".$soapclient_product->response, null, 'errors');
  2444. $error_occurred = true;
  2445. break;
  2446. }
  2447. // Check the result code
  2448. $status_code = $result_product["result"]["result_code"];
  2449. if (empty($status_code)) { //No result, check error str
  2450. setEventMessages($langs->trans("SOAPError")." '".$soapclient_order->error_str."'", null, 'errors');
  2451. } elseif ($status_code != "OK") { //Something went wrong
  2452. if ($status_code == "NOT_FOUND") {
  2453. setEventMessages($line_id.$langs->trans("SupplierMissingRef")." '".$ref_supplier."'", null, 'warnings');
  2454. } else {
  2455. setEventMessages($line_id.$langs->trans("ResponseNonOK")." '".$status_code."' - '".$result_product["result"]["result_label"]."'", null, 'errors');
  2456. $error_occurred = true;
  2457. break;
  2458. }
  2459. }
  2460. // Ensure that price is equal and warn user if it's not
  2461. $supplier_price = price($result_product["product"]["price_net"]); //Price of client tab in supplier dolibarr
  2462. $local_price = null; //Price of supplier as stated in product suppliers tab on this dolibarr, NULL if not found
  2463. $product_fourn = new ProductFournisseur($db);
  2464. $product_fourn_list = $product_fourn->list_product_fournisseur_price($line->fk_product);
  2465. if (count($product_fourn_list) > 0) {
  2466. foreach ($product_fourn_list as $product_fourn_line) {
  2467. //Only accept the line where the supplier is the same at this order and has the same ref
  2468. if ($product_fourn_line->fourn_id == $object->socid && $product_fourn_line->fourn_ref == $ref_supplier) {
  2469. $local_price = price($product_fourn_line->fourn_price);
  2470. }
  2471. }
  2472. }
  2473. if ($local_price != null && $local_price != $supplier_price) {
  2474. setEventMessages($line_id.$langs->trans("RemotePriceMismatch")." ".$supplier_price." - ".$local_price, null, 'warnings');
  2475. }
  2476. // Check if is in sale
  2477. if (empty($result_product["product"]["status_tosell"])) {
  2478. setEventMessages($line_id.$langs->trans("ProductStatusNotOnSellShort")." '".$ref_supplier."'", null, 'warnings');
  2479. }
  2480. }
  2481. }
  2482. } elseif ($user_status_code == "PERMISSION_DENIED") {
  2483. setEventMessages($langs->trans("RemoteUserNotPermission"), null, 'errors');
  2484. $error_occurred = true;
  2485. } elseif ($user_status_code == "BAD_CREDENTIALS") {
  2486. setEventMessages($langs->trans("RemoteUserBadCredentials"), null, 'errors');
  2487. $error_occurred = true;
  2488. } else {
  2489. setEventMessages($langs->trans("ResponseNonOK")." '".$user_status_code."'", null, 'errors');
  2490. $error_occurred = true;
  2491. }
  2492. //Form
  2493. print '<form action="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'" method="post">';
  2494. print '<input type="hidden" name="token" value="'.newToken().'">';
  2495. print '<input type="hidden" name="action" value="webservice">';
  2496. print '<input type="hidden" name="mode" value="send">';
  2497. print '<input type="hidden" name="ws_user" value="'.$ws_user.'">';
  2498. print '<input type="hidden" name="ws_password" value="'.$ws_password.'">';
  2499. print '<input type="hidden" name="ws_entity" value="'.$ws_entity.'">';
  2500. print '<input type="hidden" name="ws_thirdparty" value="'.$ws_thirdparty.'">';
  2501. if ($error_occurred) {
  2502. print "<br>".$langs->trans("ErrorOccurredReviseAndRetry")."<br>";
  2503. } else {
  2504. print '<input type="submit" class="button" id="ws_submit" name="ws_submit" value="'.$langs->trans("Confirm").'">';
  2505. print ' &nbsp; &nbsp; ';
  2506. }
  2507. print '<input class="button button-cancel" type="submit" id="cancel" name="cancel" value="'.$langs->trans("Cancel").'">';
  2508. print '</form>';
  2509. }
  2510. }
  2511. // Select mail models is same action as presend
  2512. if (GETPOST('modelselected')) {
  2513. $action = 'presend';
  2514. }
  2515. // Presend form
  2516. $modelmail = 'order_supplier_send';
  2517. $defaulttopic = 'SendOrderRef';
  2518. $diroutput = $conf->fournisseur->commande->dir_output;
  2519. $autocopy = 'MAIN_MAIL_AUTOCOPY_SUPPLIER_ORDER_TO';
  2520. $trackid = 'sord'.$object->id;
  2521. include DOL_DOCUMENT_ROOT.'/core/tpl/card_presend.tpl.php';
  2522. }
  2523. }
  2524. // End of page
  2525. llxFooter();
  2526. $db->close();