dolibarr 24.0.0-beta
modStock.class.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2003-2006 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3 * Copyright (C) 2004-2008 Laurent Destailleur <eldy@users.sourceforge.net>
4 * Copyright (C) 2005-2009 Regis Houssin <regis.houssin@inodbox.com>
5 * Copyright (C) 2012 Juanjo Menent <jmenent@2byte.es>
6 * Copyright (C) 2021 Ferran Marcet <fmarcet@2byte.es>
7 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
8 * Copyright (C) 2024-2026 Frédéric France <frederic.france@free.fr>
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 3 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program. If not, see <https://www.gnu.org/licenses/>.
22 */
23
32include_once DOL_DOCUMENT_ROOT.'/core/modules/DolibarrModules.class.php';
33
34
39{
45 public function __construct($db)
46 {
47 global $conf, $langs; // $conf is used by inc.php
48
49 $this->db = $db;
50 $this->numero = 52;
51
52 $this->family = "products";
53 $this->module_position = '39';
54 // Module label (no space allowed), used if translation string 'ModuleXXXName' not found (where XXX is value of numeric property 'numero' of module)
55 $this->name = preg_replace('/^mod/i', '', get_class($this));
56 $this->description = "Gestion des stocks";
57
58 // Possible values for version are: 'development', 'experimental', 'dolibarr' or version
59 $this->version = 'dolibarr';
60
61 $this->const_name = 'MAIN_MODULE_'.strtoupper($this->name);
62 $this->picto = 'stock';
63
64 // Data directories to create when module is enabled
65 $this->dirs = array("/stock/temp");
66
67 $this->config_page_url = array("stock.php");
68
69 // Dependencies
70 $this->hidden = false; // A condition to hide module
71 $this->depends = array("modProduct"); // List of module class names as string that must be enabled if this module is enabled
72 $this->requiredby = array("modProductBatch"); // List of module ids to disable if this one is disabled
73 $this->conflictwith = []; // List of module class names as string this module is in conflict with
74 $this->langfiles = array("stocks");
75
76 // Constants
77 $this->const = [
78 [
79 'STOCK_DISALLOW_NEGATIVE_TRANSFER',
80 'chaine',
81 '1',
82 '',
83 0,
84 ],
85 [
86 "STOCK_ADDON_PDF",
87 "chaine",
88 "standard_stock",
89 'Name of PDF model of stock',
90 0,
91 ],
92 [
93 "MOUVEMENT_ADDON_PDF",
94 "chaine",
95 "standard_movement_stock",
96 'Name of PDF model of stock movement',
97 0,
98 ],
99 [
100 "STOCK_ADDON_PDF_ODT_PATH",
101 "chaine",
102 "DOL_DATA_ROOT".($conf->entity > 1 ? '/'.$conf->entity : '')."/doctemplates/stocks",
103 "",
104 0,
105 ],
106 [
107 "MOUVEMENT_ADDON_PDF_ODT_PATH",
108 "chaine",
109 "DOL_DATA_ROOT".($conf->entity > 1 ? '/'.$conf->entity : '')."/doctemplates/stocks/movements",
110 "",
111 0,
112 ],
113 ];
114
115 // Boxes
116 $this->boxes = [];
117
118 // Permissions
119 $this->rights = [];
120 $this->rights_class = 'stock';
121
122 $r = 0;
123
124 $this->rights[$r][0] = 1001;
125 $this->rights[$r][1] = 'Read stocks';
126 $this->rights[$r][2] = 'r';
127 $this->rights[$r][3] = 0;
128 $this->rights[$r][4] = 'lire';
129 $this->rights[$r][5] = '';
130
131 $r++;
132 $this->rights[$r][0] = 1002;
133 $this->rights[$r][1] = 'Create/Modify stocks';
134 $this->rights[$r][2] = 'w';
135 $this->rights[$r][3] = 0;
136 $this->rights[$r][4] = 'creer';
137 $this->rights[$r][5] = '';
138
139 $r++;
140 $this->rights[$r][0] = 1003;
141 $this->rights[$r][1] = 'Delete stock';
142 $this->rights[$r][2] = 'd';
143 $this->rights[$r][3] = 0;
144 $this->rights[$r][4] = 'supprimer';
145 $this->rights[$r][5] = '';
146
147 $r++;
148 $this->rights[$r][0] = 1004;
149 $this->rights[$r][1] = 'Read stock movements';
150 $this->rights[$r][2] = 'r';
151 $this->rights[$r][3] = 0;
152 $this->rights[$r][4] = 'mouvement';
153 $this->rights[$r][5] = 'lire';
154
155 $r++;
156 $this->rights[$r][0] = 1005;
157 $this->rights[$r][1] = 'Create/modify stock movements';
158 $this->rights[$r][2] = 'w';
159 $this->rights[$r][3] = 0;
160 $this->rights[$r][4] = 'mouvement';
161 $this->rights[$r][5] = 'creer';
162
163 if (getDolGlobalInt('MAIN_FEATURES_LEVEL') >= 2) {
164 // Not yet implemented. TODO
165 $r++;
166 $this->rights[$r][0] = 1008;
167 $this->rights[$r][1] = 'Read stock value';
168 $this->rights[$r][2] = 'w';
169 $this->rights[$r][3] = 0;
170 $this->rights[$r][4] = 'value_advance';
171 $this->rights[$r][5] = 'read';
172 }
173
174 $r++;
175 $this->rights[$r][0] = 1011;
176 $this->rights[$r][1] = 'inventoryReadPermission'; // Permission label
177 $this->rights[$r][3] = 0; // Permission by default for new user (0/1)
178 $this->rights[$r][4] = 'inventory_advance'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
179 $this->rights[$r][5] = 'read'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
180
181 $r++;
182 $this->rights[$r][0] = 1012;
183 $this->rights[$r][1] = 'inventoryCreatePermission'; // Permission label
184 $this->rights[$r][3] = 0; // Permission by default for new user (0/1)
185 $this->rights[$r][4] = 'inventory_advance'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
186 $this->rights[$r][5] = 'write'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
187
188 $r++;
189 $this->rights[$r][0] = 1013;
190 $this->rights[$r][1] = 'inventoryDeletePermission'; // Permission label
191 $this->rights[$r][3] = 0; // Permission by default for new user (0/1)
192 $this->rights[$r][4] = 'inventory_advance'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
193 $this->rights[$r][5] = 'delete'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
194
195 if (getDolGlobalInt('MAIN_FEATURES_LEVEL') >= 2) {
196 $r++;
197 $this->rights[$r][0] = 1014;
198 $this->rights[$r][1] = 'inventoryValidatePermission'; // Permission label
199 $this->rights[$r][3] = 0; // Permission by default for new user (0/1)
200 $this->rights[$r][4] = 'inventory_advance'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
201 $this->rights[$r][5] = 'validate'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
202
203 $r++;
204 $this->rights[$r][0] = 1015;
205 $this->rights[$r][1] = 'inventoryChangePMPPermission'; // Permission label
206 $this->rights[$r][3] = 0; // Permission by default for new user (0/1)
207 $this->rights[$r][4] = 'inventory_advance'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
208 $this->rights[$r][5] = 'changePMP'; // In php code, permission will be checked by test if ($user->rights->permkey->level1->level2)
209 }
210
211 // Main menu entries
212 $this->menu = []; // List of menus to add
213 $r = 0;
214
215 // Menus
216 //-------
217 $this->menu = 1; // This module add menu entries. They are coded into menu manager.
218
219
220 // Exports
221 //--------
222 $r = 0;
223
224 // Export warehouses
225 $r++;
226 $this->export_code[$r] = $this->rights_class.'_emplacement';
227 $this->export_label[$r] = "Warehouses"; // Translation key (used only if key ExportDataset_xxx_z not found)
228 $this->export_icon[$r] = "warehouse";
229 $this->export_permission[$r] = array(array("stock", "lire"));
230 $this->export_fields_array[$r] = array(
231 'e.rowid' => 'IdWarehouse', 'e.ref' => 'LocationSummary', 'e.description' => 'DescWareHouse', 'e.lieu' => 'LieuWareHouse', 'e.address' => 'Address', 'e.zip' => 'Zip', 'e.town' => 'Town',
232 'd.code_departement' => 'Departement', 'c.code' => 'CountryCode',
233 'e.phone' => 'Phone', 'e.fax' => 'Fax', 'e.statut' => 'Status', 'pe.rowid' => 'ParentWarehouse', 'pe.ref' => 'LocationSummary'
234 );
235 $this->export_TypeFields_array[$r] = array(
236 'e.ref' => 'Text', 'e.description' => 'Text', 'e.lieu' => 'Text', 'e.address' => 'Text', 'e.zip' => 'Text', 'e.town' => 'Text',
237 'd.code_departement' => 'List:c_departements:code_departement:code_departement:', 'c.code' => 'List:c_country:code:code:',
238 'e.phone' => 'Text', 'e.fax' => 'Text', 'e.statut' => 'Text', 'pe.rowid' => 'List:entrepot:ref:rowid:stock', 'pe.ref' => 'Text'
239 );
240 $this->export_entities_array[$r] = []; // We define here only fields that use another icon that the one defined into export_icon
241 $this->export_aggregate_array[$r] = []; // TODO Not used yet
242 $keyforselect = 'entrepot';
243 $keyforelement = 'entrepot';
244 $keyforaliasextra = 'extra';
245 include DOL_DOCUMENT_ROOT.'/core/extrafieldsinexport.inc.php';
246
247 $this->export_sql_start[$r] = 'SELECT DISTINCT ';
248 $this->export_sql_end[$r] = ' FROM '.MAIN_DB_PREFIX.'entrepot as e';
249 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_departements as d ON d.rowid = e.fk_departement';
250 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_country as c ON c.rowid = e.fk_pays';
251 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'entrepot as pe ON pe.rowid = e.fk_parent';
252 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'entrepot_extrafields as extra ON extra.fk_object = e.rowid';
253 $this->export_sql_end[$r] .= ' WHERE e.entity IN ('.getEntity('stock').')';
254
255 // Export stock (without batch number)
256 $r++;
257 $this->export_code[$r] = $this->rights_class.'_emplacement_product';
258 $this->export_label[$r] = "Stocks"; // Translation key (used only if key ExportDataset_xxx_z not found)
259 $this->export_icon[$r] = "warehouse";
260 $this->export_permission[$r] = array(array("stock", "lire"));
261 $this->export_fields_array[$r] = array(
262 'e.rowid' => 'IdWarehouse', 'e.ref' => 'LocationSummary', 'e.description' => 'DescWareHouse', 'e.lieu' => 'LieuWareHouse', 'e.address' => 'Address', 'e.zip' => 'Zip', 'e.town' => 'Town',
263 'p.rowid' => "ProductId", 'p.ref' => "Ref", 'p.fk_product_type' => "Type", 'p.label' => "Label", 'p.description' => "Description", 'p.note' => "Note",
264 'p.price' => "Price", 'p.tva_tx' => 'VAT', 'p.tosell' => "OnSell", 'p.tobuy' => 'OnBuy', 'p.duration' => "Duration",
265 'p.datec' => 'DateCreation', 'p.tms' => 'DateModification', 'p.pmp' => 'PMPValue', 'p.cost_price' => 'CostPrice',
266 'p.seuil_stock_alerte' => 'StockLimit', 'p.barcode' => 'BarCode', 'bt.libelle' => 'BarcodeType',
267 );
268 if (isModEnabled('barcode')) {
269 $this->export_fields_array[$r] = array_merge($this->export_fields_array[$r], array('p.barcode' => 'BarCode'));
270 }
271 $this->export_TypeFields_array[$r] = array(
272 'e.rowid' => 'List:entrepot:ref::stock', 'e.ref' => 'Text', 'e.lieu' => 'Text', 'e.address' => 'Text', 'e.zip' => 'Text', 'e.town' => 'Text',
273 'p.rowid' => "Numeric", 'p.ref' => "Text", 'p.fk_product_type' => "Text", 'p.label' => "Text", 'p.description' => "Text", 'p.note' => "Text",
274 'p.price' => "Numeric", 'p.tva_tx' => 'Numeric', 'p.tosell' => "Boolean", 'p.tobuy' => "Boolean", 'p.duration' => "Duree",
275 'p.datec' => 'Date', 'p.tms' => 'Date', 'p.pmp' => 'Numeric', 'p.cost_price' => 'Numeric',
276 'ps.reel' => 'Numeric',
277 'p.seuil_stock_alerte' => 'Numeric', 'p.barcode' => 'Text', 'bt.libelle' => 'List:c_barcode_type:libelle',
278 );
279 if (isModEnabled('barcode')) {
280 $this->export_TypeFields_array[$r] = array_merge($this->export_TypeFields_array[$r], array('p.barcode' => 'Text'));
281 }
282 $this->export_entities_array[$r] = array(
283 'p.rowid' => "product", 'p.ref' => "product", 'p.fk_product_type' => "product", 'p.label' => "product", 'p.description' => "product", 'p.note' => "product",
284 'p.price' => "product", 'p.tva_tx' => 'product', 'p.tosell' => "product", 'p.tobuy' => "product", 'p.duration' => "product",
285 'p.datec' => 'product', 'p.tms' => 'product', 'p.pmp' => 'product', 'p.cost_price' => 'product',
286 'ps.reel' => 'stock',
287 'p.seuil_stock_alerte' => 'product', 'p.barcode' => 'product', 'bt.libelle' => 'product',
288 ); // We define here only fields that use another icon that the one defined into export_icon
289 if (isModEnabled('barcode')) {
290 $this->export_entities_array[$r] = array_merge($this->export_entities_array[$r], array('p.barcode' => 'product'));
291 }
292 $this->export_aggregate_array[$r] = array('ps.reel' => 'SUM'); // TODO Not used yet
293 $this->export_dependencies_array[$r] = array('stock' => array('p.rowid', 'e.rowid')); // We must keep this until the aggregate_array is used. To have a unique key, if we ask a field of a child, to avoid the DISTINCT to discard them.
294 $keyforselect = 'product';
295 $keyforelement = 'product';
296 $keyforaliasextra = 'extra';
297 include DOL_DOCUMENT_ROOT.'/core/extrafieldsinexport.inc.php';
298 $this->export_fields_array[$r] = array_merge($this->export_fields_array[$r], array('ps.reel' => 'Stock'));
299
300 $this->export_sql_start[$r] = 'SELECT DISTINCT ';
301 $this->export_sql_end[$r] = ' FROM '.MAIN_DB_PREFIX.'product as p';
302 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product_extrafields as extra ON extra.fk_object = p.rowid';
303 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_barcode_type as bt ON bt.rowid = p.fk_barcode_type';
304 $this->export_sql_end[$r] .= ', '.MAIN_DB_PREFIX.'product_stock as ps, '.MAIN_DB_PREFIX.'entrepot as e';
305 $this->export_sql_end[$r] .= ' WHERE p.rowid = ps.fk_product AND ps.fk_entrepot = e.rowid';
306 $this->export_sql_end[$r] .= ' AND e.entity IN ('.getEntity('stock').')';
307
308 // Export stock including batch number
309 if (isModEnabled('productbatch')) {
310 $langs->load("productbatch");
311
312 // This request is same than previous but without field ps.stock (real stock in warehouse) and with link to subtable productbatch
313 $r++;
314
315 $this->export_code[$r] = $this->rights_class.'_emplacement_product_lot';
316 $this->export_label[$r] = "StocksWithBatch"; // Translation key (used only if key ExportDataset_xxx_z not found)
317 $this->export_icon[$r] = "warehouse";
318 $this->export_permission[$r] = array(array("stock", "lire"));
319 $this->export_fields_array[$r] = array(
320 'e.rowid' => 'IdWarehouse', 'e.ref' => 'LocationSummary', 'e.description' => 'DescWareHouse', 'e.lieu' => 'LieuWareHouse', 'e.address' => 'Address', 'e.zip' => 'Zip', 'e.town' => 'Town',
321 'p.rowid' => "ProductId", 'p.ref' => "Ref", 'p.fk_product_type' => "Type", 'p.label' => "Label", 'p.description' => "Description", 'p.note' => "Note",
322 'p.price' => "Price", 'p.tva_tx' => 'VAT', 'p.tosell' => "OnSell", 'p.tobuy' => 'OnBuy', 'p.duration' => "Duration",
323 'p.datec' => 'DateCreation', 'p.tms' => 'DateModification', 'p.pmp' => 'PMPValue', 'p.cost_price' => 'CostPrice', 'lcpn.label'=>'Nature',
324 'pb.rowid' => 'Id', 'pb.batch' => 'Batch', 'pb.qty' => 'Qty',
325 'pl.eatby' => 'EatByDate', 'pl.sellby' => 'SellByDate', 'none.dateLastMovement' => 'LastMovement'
326 );
327 if (isModEnabled('barcode')) {
328 $this->export_fields_array[$r] = array_merge($this->export_fields_array[$r], array('p.barcode' => 'BarCode'));
329 }
330 $this->export_TypeFields_array[$r] = array(
331 'e.rowid' => 'List:entrepot:ref::stock', 'e.ref' => 'Text', 'e.lieu' => 'Text', 'e.description' => 'Text', 'e.address' => 'Text', 'e.zip' => 'Text', 'e.town' => 'Text',
332 'p.rowid' => "Numeric", 'p.ref' => "Text", 'p.fk_product_type' => "Text", 'p.label' => "Text", 'p.description' => "Text", 'p.note' => "Text",
333 'p.price' => "Numeric", 'p.tva_tx' => 'Numeric', 'p.tosell' => "Boolean", 'p.tobuy' => "Boolean", 'p.duration' => "Duree",
334 'p.datec' => 'Date', 'p.tms' => 'Date', 'p.pmp' => 'PMPValue', 'p.cost_price' => 'CostPrice', 'lcpn.label'=>'Text',
335 'pb.batch' => 'Text', 'pb.qty' => 'Numeric',
336 'pl.eatby' => 'Date', 'pl.sellby' => 'Date', 'none.dateLastMovement' => 'Date'
337 );
338 if (isModEnabled('barcode')) {
339 $this->export_TypeFields_array[$r] = array_merge($this->export_TypeFields_array[$r], array('p.barcode' => 'Text'));
340 }
341 $this->export_entities_array[$r] = array(
342 'p.rowid' => "product", 'p.ref' => "product", 'p.fk_product_type' => "product", 'p.label' => "product", 'p.description' => "product", 'p.note' => "product",
343 'p.price' => "product", 'p.tva_tx' => 'product', 'p.tosell' => "product", 'p.tobuy' => "product", 'p.duration' => "product",
344 'p.datec' => 'product', 'p.tms' => 'product', 'p.pmp' => 'product', 'p.cost_price' => 'product', 'lcpn.label'=>'product',
345 'pb.rowid' => 'batch', 'pb.batch' => 'batch', 'pb.qty' => 'batch', 'none.dateLastMovement' => 'movement',
346 'pl.eatby' => 'batch', 'pl.sellby' => 'batch'
347 ); // We define here only fields that use another icon that the one defined into export_icon
348 $this->export_special_array[$r] = array(
349 'none.dateLastMovement'=>array('rule'=>'compute', 'classfile'=>'/product/stock/class/mouvementstock.class.php', 'class'=>'MouvementStock', 'method'=>'getDateLastMovementProductBatch', 'method_params'=>['e_rowid', 'p_rowid', 'pb_batch']),
350 );
351 if (isModEnabled('barcode')) {
352 $this->export_entities_array[$r] = array_merge($this->export_entities_array[$r], array('p.barcode' => 'product'));
353 }
354 $this->export_aggregate_array[$r] = array('ps.reel' => 'SUM'); // TODO Not used yet
355 $this->export_dependencies_array[$r] = array('stockbatch' => array('pb.rowid'), 'batch' => array('pb.rowid')); // We must keep this until the aggregate_array is used. To add unique key if we ask a field of a child to avoid the DISTINCT to discard them.
356 $keyforselect = 'product_lot';
357 $keyforelement = 'batch';
358 $keyforaliasextra = 'extra';
359 include DOL_DOCUMENT_ROOT.'/core/extrafieldsinexport.inc.php';
360
361 $this->export_sql_start[$r] = 'SELECT DISTINCT ';
362 $this->export_sql_end[$r] = ' FROM '.MAIN_DB_PREFIX.'product_batch as pb';
363 $this->export_sql_end[$r] .= ' INNER JOIN '.MAIN_DB_PREFIX.'product_stock as ps ON ps.rowid = pb.fk_product_stock';
364 $this->export_sql_end[$r] .= ' INNER JOIN '.MAIN_DB_PREFIX.'product as p ON p.rowid = ps.fk_product';
365 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product_lot as pl ON pl.fk_product = p.rowid AND pl.batch = pb.batch';
366 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product_lot_extrafields as extra ON extra.fk_object = pl.rowid';
367 $this->export_sql_end[$r] .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_product_nature as lcpn on lcpn.code = p.finished,';
368 $this->export_sql_end[$r] .= ' '.MAIN_DB_PREFIX.'entrepot as e';
369 $this->export_sql_end[$r] .= ' WHERE ps.fk_entrepot = e.rowid';
370 $this->export_sql_end[$r] .= ' AND e.entity IN ('.getEntity('stock').')';
371 }
372
373 // Export of stock movements
374 $r++;
375 $this->export_code[$r] = $this->rights_class.'_movement';
376 $this->export_label[$r] = "StockMovements"; // Translation key (used only if key ExportDataset_xxx_z not found)
377 $this->export_icon[$r] = "movement";
378 $this->export_permission[$r] = array(array("stock", "lire"));
379 $this->export_fields_array[$r] = array(
380 'sm.rowid' => 'MovementId', 'sm.value' => 'Qty', 'sm.datem' => 'DateMovement', 'sm.label' => 'MovementLabel', 'sm.inventorycode' => 'InventoryCode',
381 'e.rowid' => 'IdWarehouse', 'e.ref' => 'LocationSummary', 'e.description' => 'DescWareHouse', 'e.lieu' => 'LieuWareHouse', 'e.address' => 'Address', 'e.zip' => 'Zip', 'e.town' => 'Town',
382 'p.rowid' => "ProductId", 'p.ref' => "Ref", 'p.fk_product_type' => "Type", 'p.label' => "Label", 'p.description' => "Description", 'p.note' => "Note",
383 'p.price' => "Price", 'p.tva_tx' => 'VAT', 'p.tosell' => "OnSell", 'p.tobuy' => 'OnBuy', 'p.duration' => "Duration", 'p.datec' => 'DateCreation', 'p.tms' => 'DateModification'
384 );
385 if (isModEnabled('barcode')) {
386 $this->export_fields_array[$r] = array_merge($this->export_fields_array[$r], array('p.barcode' => 'BarCode'));
387 }
388 $this->export_TypeFields_array[$r] = array(
389 'sm.rowid' => 'Numeric', 'sm.value' => 'Numeric', 'sm.datem' => 'Date', 'sm.batch' => 'Text', 'sm.label' => 'Text', 'sm.inventorycode' => 'Text',
390 'e.rowid' => 'List:entrepot:ref::stock', 'e.ref' => 'Text', 'e.description' => 'Text', 'e.lieu' => 'Text', 'e.address' => 'Text', 'e.zip' => 'Text', 'e.town' => 'Text',
391 'p.rowid' => "Numeric", 'p.ref' => "Text", 'p.fk_product_type' => "Text", 'p.label' => "Text", 'p.description' => "Text", 'p.note' => "Text",
392 'p.price' => "Numeric", 'p.tva_tx' => 'Numeric', 'p.tosell' => "Boolean", 'p.tobuy' => "Boolean", 'p.duration' => "Duree", 'p.datec' => 'Date', 'p.tms' => 'Date'
393 );
394 if (isModEnabled('barcode')) {
395 $this->export_TypeFields_array[$r] = array_merge($this->export_TypeFields_array[$r], array('p.barcode' => 'Text'));
396 }
397 $this->export_entities_array[$r] = array(
398 'e.rowid' => 'warehouse', 'e.ref' => 'warehouse', 'e.description' => 'warehouse', 'e.lieu' => 'warehouse', 'e.address' => 'warehouse', 'e.zip' => 'warehouse', 'e.town' => 'warehouse',
399 'p.rowid' => "product", 'p.ref' => "product", 'p.fk_product_type' => "product", 'p.label' => "product", 'p.description' => "product", 'p.note' => "product",
400 'p.price' => "product", 'p.tva_tx' => 'product', 'p.tosell' => "product", 'p.tobuy' => "product", 'p.duration' => "product", 'p.datec' => 'product', 'p.tms' => 'product'
401 ); // We define here only fields that use another icon that the one defined into export_icon
402 if (isModEnabled('productbatch')) {
403 $this->export_fields_array[$r]['sm.batch'] = 'Batch';
404 $this->export_TypeFields_array[$r]['sm.batch'] = 'Text';
405 $this->export_entities_array[$r]['sm.batch'] = 'movement';
406 }
407 if (isModEnabled('barcode')) {
408 $this->export_entities_array[$r] = array_merge($this->export_entities_array[$r], array('p.barcode' => 'product'));
409 }
410 $this->export_aggregate_array[$r] = array('sm.value' => 'SUM'); // TODO Not used yet
411 $this->export_dependencies_array[$r] = array('movement' => array('sm.rowid')); // We must keep this until the aggregate_array is used. To add unique key if we ask a field of a child to avoid the DISTINCT to discard them.
412
413 $this->export_sql_start[$r] = 'SELECT DISTINCT ';
414 $this->export_sql_end[$r] = ' FROM '.MAIN_DB_PREFIX.'product as p, '.MAIN_DB_PREFIX.'stock_mouvement as sm, '.MAIN_DB_PREFIX.'entrepot as e';
415 $this->export_sql_end[$r] .= ' WHERE p.rowid = sm.fk_product AND sm.fk_entrepot = e.rowid';
416 $this->export_sql_end[$r] .= ' AND e.entity IN ('.getEntity('stock').')';
417
418
419 // Export inventories
420 $r++;
421 $this->export_code[$r] = $this->rights_class.'_inventory';
422 $this->export_label[$r] = "Inventories"; // Translation key (used only if key ExportDataset_xxx_z not found)
423 $this->export_icon[$r] = "inventory";
424 $this->export_permission[$r] = array(array("stock", "lire"));
425 $this->export_fields_array[$r] = array(
426 'i.rowid' => 'InventoryId', 'i.ref' => 'InventoryRef', 'i.date_inventory' => 'DateInventory', 'i.status' => 'InventoryStatus', 'i.title' => 'InventoryTitle',
427 'id.rowid' => 'InventoryLineId', 'id.qty_view' => 'QtyViewed', 'id.qty_stock' => 'QtyStock', 'id.qty_regulated' => 'QtyRegulated',
428 'id.batch' => 'Lotserial',
429 'e.rowid' => 'IdWarehouse', 'e.ref' => 'LocationSummary', 'e.description' => 'DescWareHouse', 'e.lieu' => 'LieuWareHouse', 'e.address' => 'Address', 'e.zip' => 'Zip', 'e.town' => 'Town',
430 'p.rowid' => "ProductId", 'p.ref' => "Ref", 'p.fk_product_type' => "Type", 'p.label' => "Label", 'p.description' => "Description", 'p.note' => "Note",
431 'p.barcode' => "Barcode", 'p.price' => "Price", 'p.tva_tx' => 'VAT', 'p.tosell' => "OnSell", 'p.tobuy' => 'OnBuy', 'p.duration' => "Duration", 'p.datec' => 'DateCreation', 'p.tms' => 'DateModification'
432 );
433 if (isModEnabled('barcode')) {
434 $this->export_fields_array[$r] = array_merge($this->export_fields_array[$r], array('p.barcode' => 'BarCode'));
435 }
436 $this->export_TypeFields_array[$r] = array(
437 'id.rowid' => 'Numeric',
438 'e.rowid' => 'List:entrepot:ref::stock', 'e.ref' => 'Text', 'e.description' => 'Text', 'e.lieu' => 'Text', 'e.address' => 'Text', 'e.zip' => 'Text', 'e.town' => 'Text',
439 'p.rowid' => "Numeric", 'p.ref' => "Text", 'p.fk_product_type' => "Text", 'p.label' => "Text", 'p.description' => "Text", 'p.note' => "Text",
440 'p.barcode' => "Text", 'p.price' => "Numeric", 'p.tva_tx' => 'Numeric', 'p.tosell' => "Boolean", 'p.tobuy' => "Boolean", 'p.duration' => "Duree", 'p.datec' => 'Date', 'p.tms' => 'Date',
441 'i.rowid' => 'Numeric', 'i.ref' => 'Text', 'i.date_inventory' => 'Date', 'i.status' => 'Numeric', 'i.title' => 'Text',
442 'id.qty_view' => 'Numeric', 'id.qty_stock' => 'Numeric', 'id.batch' => 'Text',
443 'id.qty_regulated' => 'Numeric', 'id.fk_warehouse' => 'Numeric',
444 );
445 if (isModEnabled('barcode')) {
446 $this->export_TypeFields_array[$r] = array_merge($this->export_TypeFields_array[$r], array('p.barcode' => 'Text'));
447 }
448 $this->export_entities_array[$r] = array(
449 'id.qty_view' => 'inventory_line', 'id.qty_stock' => 'inventory_line', 'id.batch' => 'inventory_line', 'id.qty_regulated' => 'inventory_line', 'id.fk_warehouse' => 'inventory_line', 'id.rowid' => 'inventory_line', 'e.rowid' => 'warehouse', 'e.ref' => 'warehouse', 'e.description' => 'warehouse', 'e.lieu' => 'warehouse', 'e.address' => 'warehouse', 'e.zip' => 'warehouse', 'e.town' => 'warehouse',
450 'p.rowid' => "product", 'p.ref' => "product", 'p.fk_product_type' => "product", 'p.label' => "product", 'p.description' => "product", 'p.note' => "product",
451 'p.barcode' => "product", 'p.price' => "product", 'p.tva_tx' => 'product', 'p.tosell' => "product", 'p.tobuy' => "product", 'p.duration' => "product", 'p.datec' => 'product', 'p.tms' => 'product'
452 ); // We define here only fields that use another icon that the one defined into export_icon
453 if (isModEnabled('productbatch')) {
454 $this->export_fields_array[$r]['id.batch'] = 'Batch';
455 $this->export_TypeFields_array[$r]['id.batch'] = 'Text';
456 $this->export_entities_array[$r]['id.batch'] = 'inventory_line';
457 }
458 if (isModEnabled('barcode')) {
459 $this->export_entities_array[$r] = array_merge($this->export_entities_array[$r], array('p.barcode' => 'product'));
460 }
461 $this->export_aggregate_array[$r] = array('sm.value' => 'SUM'); // TODO Not used yet
462 $this->export_dependencies_array[$r] = array('movement' => array('sm.rowid')); // We must keep this until the aggregate_array is used. To add unique key if we ask a field of a child to avoid the DISTINCT to discard them.
463
464 $this->export_sql_start[$r] = 'SELECT DISTINCT ';
465 $this->export_sql_end[$r] = ' FROM '.MAIN_DB_PREFIX.'product as p, '.MAIN_DB_PREFIX.'inventory as i, '.MAIN_DB_PREFIX.'inventorydet as id, '.MAIN_DB_PREFIX.'entrepot as e';
466 $this->export_sql_end[$r] .= ' WHERE p.rowid = id.fk_product AND id.fk_inventory = i.rowid AND id.fk_warehouse = e.rowid';
467 $this->export_sql_end[$r] .= ' AND e.entity IN ('.getEntity('stock').')';
468
469
470 // Imports
471 //--------
472
473 $r = 0;
474
475 // Import warehouses
476 $r++;
477 $this->import_code[$r] = $this->rights_class.'_'.$r;
478 $this->import_label[$r] = "Warehouses"; // Translation key
479 $this->import_icon[$r] = "warehouse";
480 $this->import_entities_array[$r] = []; // We define here only fields that use another icon that the one defined into import_icon
481 $this->import_tables_array[$r] = array('e' => MAIN_DB_PREFIX.'entrepot', 'extra' => MAIN_DB_PREFIX.'entrepot_extrafields');
482 $this->import_tables_creator_array[$r] = array('e' => 'fk_user_author');
483 $this->import_fields_array[$r] = array('e.ref' => "LocationSummary*",
484 'e.description' => "DescWareHouse",
485 'e.lieu' => "LieuWareHouse",
486 'e.address' => "Address",
487 'e.zip' => 'Zip',
488 'e.fk_departement' => 'StateCode',
489 'e.fk_pays' => 'CountryCode',
490 'e.phone' => 'Phone',
491 'e.fax' => 'Fax',
492 'e.statut' => 'Status',
493 'e.fk_parent' => 'ParentWarehouse'
494 );
495
496 $this->import_convertvalue_array[$r] = array(
497 'e.fk_departement' => array('rule' => 'fetchidfromcodeid', 'classfile' => '/core/class/cstate.class.php', 'class' => 'Cstate', 'method' => 'fetch', 'dict' => 'DictionaryStateCode'),
498 'e.fk_pays' => array('rule' => 'fetchidfromcodeid', 'classfile' => '/core/class/ccountry.class.php', 'class' => 'Ccountry', 'method' => 'fetch', 'dict' => 'DictionaryCountry'),
499 'e.fk_parent' => array('rule' => 'fetchidfromref', 'classfile' => '/product/stock/class/entrepot.class.php', 'class' => 'Entrepot', 'method' => 'fetch', 'element' => 'ref')
500 );
501 $this->import_regex_array[$r] = array('e.statut' => '^[0|1]');
502 // Add extra fields
503 $import_extrafield_sample = [];
504 $sql = "SELECT name, label, fieldrequired FROM ".MAIN_DB_PREFIX."extrafields WHERE type <> 'separate' AND elementtype = 'entrepot' AND entity IN (0, ".$conf->entity.")";
505 $resql = $this->db->query($sql);
506 if ($resql) { // This can fail when class is used on old database (during migration for example)
507 while ($obj = $this->db->fetch_object($resql)) {
508 $fieldname = 'extra.'.$obj->name;
509 $fieldlabel = ucfirst($obj->label);
510 $this->import_fields_array[$r][$fieldname] = $fieldlabel.($obj->fieldrequired ? '*' : '');
511 $import_extrafield_sample[$fieldname] = $fieldlabel;
512 }
513 }
514 // End add extra fields
515 $this->import_fieldshidden_array[$r] = array('extra.fk_object' => 'lastrowid-'.MAIN_DB_PREFIX.'entrepot'); // aliastable.field => ('user->id' or 'lastrowid-'.tableparent)
516
517 $this->import_examplevalues_array[$r] = array_merge(array('e.ref' => "ALM001",
518 'e.description' => "Central Warehouse",
519 'e.lieu' => "Central",
520 'e.address' => "Route 66",
521 'e.zip' => '28080',
522 'e.fk_departement' => 'matches field "code_departement" in table "'.MAIN_DB_PREFIX.'c_departements"',
523 'e.fk_pays' => 'US/FR/DE etc. matches field "code" in table "'.MAIN_DB_PREFIX.'c_country"',
524 'e.phone' => '(+33)(0)123456789',
525 'e.fax' => '(+33)(0)123456790',
526 'e.statut' => '1',
527 'e.fk_parent' => 'id or ref of warehouse'
528 ), $import_extrafield_sample);
529 $this->import_updatekeys_array[$r] = array('p.ref' => 'Ref');
530
531 // Import stocks
532 $r++;
533 $this->import_code[$r] = $this->rights_class.'_'.$r;
534 $this->import_label[$r] = "Stocks"; // Translation key
535 $this->import_icon[$r] = "stock";
536 $this->import_entities_array[$r] = []; // We define here only fields that use another icon that the one defined into import_icon
537 $this->import_tables_array[$r] = array('ps' => MAIN_DB_PREFIX.'product_stock');
538 $this->import_fields_array[$r] = array('ps.fk_product' => "Product*", 'ps.fk_entrepot' => "Warehouse*", 'ps.reel' => "Stock*");
539
540 $this->import_convertvalue_array[$r] = array(
541 'ps.fk_product' => array('rule' => 'fetchidfromref', 'classfile' => '/product/class/product.class.php', 'class' => 'Product', 'method' => 'fetch', 'element' => 'product'),
542 'ps.fk_entrepot' => array('rule' => 'fetchidfromref', 'classfile' => '/product/stock/class/entrepot.class.php', 'class' => 'Entrepot', 'method' => 'fetch', 'element' => 'ref')
543 );
544 $this->import_examplevalues_array[$r] = array(
545 'ps.fk_product' => "id or ref of product", 'ps.fk_entrepot' => "id or ref of warehouse", 'ps.reel' => "10"
546 );
547 $this->import_updatekeys_array[$r] = array('ps.fk_product' => 'Product', 'ps.fk_entrepot' => "Warehouse");
548 $this->import_run_sql_after_array[$r] = array( // Because we may change data that are denormalized, we must update dernormalized data after.
549 'UPDATE '.MAIN_DB_PREFIX.'product as p SET stock = (SELECT SUM(ps.reel) FROM '.MAIN_DB_PREFIX.'product_stock ps WHERE ps.fk_product = p.rowid);'
550 );
551 }
552
553
562 public function init($options = '')
563 {
564 global $conf, $langs;
565
566 $result = $this->_load_tables('/install/mysql/', 'stock');
567 if ($result < 0) {
568 return -1; // Do not activate module if error 'not allowed' returned when loading module SQL queries (the _load_table run sql with run_sql with the error allowed parameter set to 'default')
569 }
570
571 // Permissions
572 $this->remove($options);
573
574 //ODT template
575 $src = DOL_DOCUMENT_ROOT.'/install/doctemplates/stocks/template_warehouse.odt';
576 $dirodt = DOL_DATA_ROOT.($conf->entity > 1 ? '/'.$conf->entity : '').'/doctemplates/stocks';
577 $dest = $dirodt.'/template_warehouse.odt';
578
579 if (file_exists($src) && !file_exists($dest)) {
580 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
581 dol_mkdir($dirodt);
582 $result = dol_copy($src, $dest, '0', 0);
583 if ($result < 0) {
584 $langs->load("errors");
585 $this->error = $langs->trans('ErrorFailToCopyFile', $src, $dest);
586 return 0;
587 }
588 }
589
590 $sql = [];
591
592 $sql = array(
593 "DELETE FROM ".MAIN_DB_PREFIX."document_model WHERE nom = '".$this->db->escape($this->const[1][2])."' AND type = 'stock' AND entity = ".((int) $conf->entity),
594 "INSERT INTO ".MAIN_DB_PREFIX."document_model (nom, type, entity) VALUES('".$this->db->escape($this->const[1][2])."','stock',".((int) $conf->entity).")",
595 "DELETE FROM ".MAIN_DB_PREFIX."document_model WHERE nom = '".$this->db->escape($this->const[2][2])."' AND type = 'mouvement' AND entity = ".((int) $conf->entity),
596 "INSERT INTO ".MAIN_DB_PREFIX."document_model (nom, type, entity) VALUES('".$this->db->escape($this->const[2][2])."','mouvement',".((int) $conf->entity).")",
597 );
598
599 return $this->_init($sql, $options);
600 }
601}
Class DolibarrModules.
_init($array_sql, $options='')
Enables a module.
_load_tables($reldir, $onlywithsuffix='')
Create tables and keys required by module:
Class to describe and enable module Stock.
init($options='')
Function called when module is enabled.
__construct($db)
Constructor.
print $script_file $mode $langs defaultlang(is_numeric($duration_value) ? " delay=". $duration_value :"").(is_numeric($duration_value2) ? " after cd cd cd description as description
Only used if Module[ID]Desc translation string is not found.
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $conf
The main.inc.php has been included so the following variable are now defined:
dol_copy($srcfile, $destfile, $newmask='0', $overwriteifexists=1, $testvirus=0, $indexdatabase=0)
Copy a file to another file.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
isModEnabled($module)
Is Dolibarr module enabled.
dol_mkdir($dir, $dataroot='', $newmask='')
Creation of a directory (this can create recursive subdir)
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:133