dolibarr  x.y.z
facturerec.php
Go to the documentation of this file.
1 <?php
2 /* Copyright (C) 2003-2007 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3  * Copyright (C) 2004-2016 Laurent Destailleur <eldy@users.sourceforge.net>
4  * Copyright (C) 2005-2012 Regis Houssin <regis.houssin@inodbox.com>
5  * Copyright (C) 2014 Juanjo Menent <jmenent@2byte.es>
6  * Copyright (C) 2014 Florian Henry <florian.henry@open-concept.pro>
7  * Copyright (C) 2022 Eric Seigne <eric.seigne@cap-rel.fr>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 3 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program. If not, see <https://www.gnu.org/licenses/>.
21  */
22 
29 // Load Dolibarr environment
30 require '../../main.inc.php';
31 require_once DOL_DOCUMENT_ROOT.'/core/lib/product.lib.php';
32 require_once DOL_DOCUMENT_ROOT.'/compta/facture/class/facture-rec.class.php';
33 require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
34 require_once DOL_DOCUMENT_ROOT.'/core/class/html.formother.class.php';
35 
36 // Load translation files required by the page
37 $langs->loadLangs(array('companies', 'bills', 'products', 'supplier_proposal'));
38 
39 $id = GETPOST('id', 'int');
40 $ref = GETPOST('ref', 'alpha');
41 
42 // Security check
43 $fieldvalue = (!empty($id) ? $id : (!empty($ref) ? $ref : ''));
44 $fieldtype = (!empty($ref) ? 'ref' : 'rowid');
45 $socid = '';
46 if (!empty($user->socid)) {
47  $socid = $user->socid;
48 }
49 
50 // Initialize technical object to manage hooks of page. Note that conf->hooks_modules contains array of hook context
51 $hookmanager->initHooks(array('productstatsinvoice'));
52 
53 $showmessage = GETPOST('showmessage');
54 
55 // Load variable for pagination
56 $limit = GETPOST('limit', 'int') ?GETPOST('limit', 'int') : $conf->liste_limit;
57 $sortfield = GETPOST('sortfield', 'aZ09comma');
58 $sortorder = GETPOST('sortorder', 'aZ09comma');
59 $page = GETPOSTISSET('pageplusone') ? (GETPOST('pageplusone') - 1) : GETPOST("page", 'int');
60 if (empty($page) || $page == -1) {
61  $page = 0;
62 } // If $page is not defined, or '' or -1
63 $offset = $limit * $page;
64 $pageprev = $page - 1;
65 $pagenext = $page + 1;
66 if (!$sortorder) {
67  $sortorder = "DESC";
68 }
69 if (!$sortfield) {
70  $sortfield = "f.datec";
71 }
72 
73 $search_month = GETPOST('search_month', 'int');
74 $search_year = GETPOST('search_year', 'int');
75 
76 if (GETPOST('button_removefilter_x', 'alpha') || GETPOST('button_removefilter', 'alpha')) {
77  $search_month = '';
78  $search_year = '';
79 }
80 
81 $result = restrictedArea($user, 'produit|service', $fieldvalue, 'product&product', '', '', $fieldtype);
82 
83 
84 /*
85  * View
86  */
87 
88 $invoicestatic = new FactureRec($db);
89 $societestatic = new Societe($db);
90 
91 $form = new Form($db);
92 $formother = new FormOther($db);
93 
94 if ($id > 0 || !empty($ref)) {
95  $product = new Product($db);
96  $result = $product->fetch($id, $ref);
97 
98  $object = $product;
99 
100  $parameters = array('id'=>$id);
101  $reshook = $hookmanager->executeHooks('doActions', $parameters, $product, $action); // Note that $action and $object may have been modified by some hooks
102  if ($reshook < 0) {
103  setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
104  }
105 
106  $title = $langs->trans('ProductServiceCard');
107  $helpurl = '';
108  $shortlabel = dol_trunc($object->label, 16);
109  if (GETPOST("type") == '0' || ($object->type == Product::TYPE_PRODUCT)) {
110  $title = $langs->trans('Product')." ".$shortlabel." - ".$langs->trans('Referers');
111  $helpurl = 'EN:Module_Products|FR:Module_Produits|ES:M&oacute;dulo_Productos';
112  }
113  if (GETPOST("type") == '1' || ($object->type == Product::TYPE_SERVICE)) {
114  $title = $langs->trans('Service')." ".$shortlabel." - ".$langs->trans('Referers');
115  $helpurl = 'EN:Module_Services_En|FR:Module_Services|ES:M&oacute;dulo_Servicios';
116  }
117 
118  llxHeader('', $title, $helpurl);
119 
120  if ($result > 0) {
121  $head = product_prepare_head($product);
122  $titre = $langs->trans("CardProduct".$product->type);
123  $picto = ($product->type == Product::TYPE_SERVICE ? 'service' : 'product');
124  print dol_get_fiche_head($head, 'referers', $titre, -1, $picto);
125 
126  $reshook = $hookmanager->executeHooks('formObjectOptions', $parameters, $product, $action); // Note that $action and $object may have been modified by hook
127  print $hookmanager->resPrint;
128  if ($reshook < 0) {
129  setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
130  }
131 
132  $linkback = '<a href="'.DOL_URL_ROOT.'/product/list.php?restore_lastsearch_values=1">'.$langs->trans("BackToList").'</a>';
133 
134  $shownav = 1;
135  if ($user->socid && !in_array('product', explode(',', $conf->global->MAIN_MODULES_FOR_EXTERNAL))) {
136  $shownav = 0;
137  }
138 
139  dol_banner_tab($object, 'ref', $linkback, $shownav, 'ref');
140 
141  print '<div class="fichecenter">';
142 
143  print '<div class="underbanner clearboth"></div>';
144  print '<table class="border tableforfield" width="100%">';
145 
146  $nboflines = show_stats_for_company($product, $socid);
147 
148  print "</table>";
149 
150  print '</div>';
151  print '<div style="clear:both"></div>';
152 
153  print dol_get_fiche_end();
154 
155  if ($showmessage && $nboflines > 1) {
156  print '<span class="opacitymedium">'.$langs->trans("ClinkOnALinkOfColumn", $langs->transnoentitiesnoconv("Referers")).'</span>';
157  } elseif ($user->rights->facture->lire) {
158  $sql = "SELECT DISTINCT s.nom as name, s.rowid as socid, s.code_client,";
159  $sql .= "f.titre, f.datec, f.rowid as facid,";
160  $sql .= " d.rowid, d.total_ht as total_ht, d.qty"; // We must keep the d.rowid here to not loose record because of the distinct used to ignore duplicate line when link on societe_commerciaux is used
161  if (empty($user->rights->societe->client->voir) && !$socid) {
162  $sql .= ", sc.fk_soc, sc.fk_user ";
163  }
164  $sql .= " FROM ".MAIN_DB_PREFIX."societe as s";
165  $sql .= ", ".MAIN_DB_PREFIX."facture_rec as f";
166  $sql .= ", ".MAIN_DB_PREFIX."facturedet_rec as d";
167  if (empty($user->rights->societe->client->voir) && !$socid) {
168  $sql .= ", ".MAIN_DB_PREFIX."societe_commerciaux as sc";
169  }
170  $sql .= " WHERE f.fk_soc = s.rowid";
171  $sql .= " AND f.entity IN (".getEntity('invoice').")";
172  $sql .= " AND d.fk_facture = f.rowid";
173  $sql .= " AND d.fk_product = ".((int) $product->id);
174  if (!empty($search_month)) {
175  $sql .= ' AND MONTH(f.datec) IN ('.$db->sanitize($search_month).')';
176  }
177  if (!empty($search_year)) {
178  $sql .= ' AND YEAR(f.datec) IN ('.$db->sanitize($search_year).')';
179  }
180  if (empty($user->rights->societe->client->voir) && !$socid) {
181  $sql .= " AND s.rowid = sc.fk_soc AND sc.fk_user = ".((int) $user->id);
182  }
183  if ($socid) {
184  $sql .= " AND f.fk_soc = ".((int) $socid);
185  }
186  $sql .= $db->order($sortfield, $sortorder);
187 
188  // Calcul total qty and amount for global if full scan list
189  $total_ht = 0;
190  $total_qty = 0;
191 
192  // Count total nb of records
193  $totalofrecords = '';
194  if (empty($conf->global->MAIN_DISABLE_FULL_SCANLIST)) {
195  $result = $db->query($sql);
196  $totalofrecords = $db->num_rows($result);
197  }
198 
199  $sql .= $db->plimit($limit + 1, $offset);
200 
201  $result = $db->query($sql);
202  if ($result) {
203  $num = $db->num_rows($result);
204 
205  $option = '&id='.$product->id;
206 
207  if ($limit > 0 && $limit != $conf->liste_limit) {
208  $option .= '&limit='.urlencode($limit);
209  }
210  if (!empty($search_month)) {
211  $option .= '&search_month='.urlencode($search_month);
212  }
213  if (!empty($search_year)) {
214  $option .= '&search_year='.urlencode($search_year);
215  }
216 
217  print '<form method="post" action="'.$_SERVER ['PHP_SELF'].'?id='.$product->id.'" name="search_form">'."\n";
218  print '<input type="hidden" name="token" value="'.newToken().'">';
219  if (!empty($sortfield)) {
220  print '<input type="hidden" name="sortfield" value="'.$sortfield.'"/>';
221  }
222  if (!empty($sortorder)) {
223  print '<input type="hidden" name="sortorder" value="'.$sortorder.'"/>';
224  }
225 
226  print_barre_liste($langs->trans("CustomersInvoices"), $page, $_SERVER["PHP_SELF"], $option, $sortfield, $sortorder, '', $num, $totalofrecords, '', 0, '', '', $limit, 0, 0, 1);
227 
228  if (!empty($page)) {
229  $option .= '&page='.urlencode($page);
230  }
231 
232  print '<div class="liste_titre liste_titre_bydiv centpercent">';
233  print '<div class="divsearchfield">';
234  print $langs->trans('Period').' ('.$langs->trans("DateInvoice").') - ';
235  print $langs->trans('Month').':<input class="flat" type="text" size="4" name="search_month" value="'.$search_month.'"> ';
236  print $langs->trans('Year').':'.$formother->selectyear($search_year ? $search_year : - 1, 'search_year', 1, 20, 5);
237  print '<div style="vertical-align: middle; display: inline-block">';
238  print '<input type="image" class="liste_titre" name="button_search" src="'.img_picto($langs->trans("Search"), 'search.png', '', '', 1).'" value="'.dol_escape_htmltag($langs->trans("Search")).'" title="'.dol_escape_htmltag($langs->trans("Search")).'">';
239  print '<input type="image" class="liste_titre" name="button_removefilter" src="'.img_picto($langs->trans("Search"), 'searchclear.png', '', '', 1).'" value="'.dol_escape_htmltag($langs->trans("RemoveFilter")).'" title="'.dol_escape_htmltag($langs->trans("RemoveFilter")).'">';
240  print '</div>';
241  print '</div>';
242  print '</div>';
243 
244  $i = 0;
245  print '<div class="div-table-responsive">';
246  print '<table class="tagtable liste listwithfilterbefore" width="100%">';
247  print '<tr class="liste_titre">';
248  print_liste_field_titre("Ref", $_SERVER["PHP_SELF"], "s.rowid", "", $option, '', $sortfield, $sortorder);
249  print_liste_field_titre("Company", $_SERVER["PHP_SELF"], "s.nom", "", $option, '', $sortfield, $sortorder);
250  print_liste_field_titre("CustomerCode", $_SERVER["PHP_SELF"], "s.code_client", "", $option, '', $sortfield, $sortorder);
251  print_liste_field_titre("DateInvoice", $_SERVER["PHP_SELF"], "f.datec", "", $option, 'align="center"', $sortfield, $sortorder);
252  print_liste_field_titre("Qty", $_SERVER["PHP_SELF"], "d.qty", "", $option, 'align="center"', $sortfield, $sortorder);
253  print_liste_field_titre("AmountHT", $_SERVER["PHP_SELF"], "d.total_ht", "", $option, 'align="right"', $sortfield, $sortorder);
254  print_liste_field_titre("Status", $_SERVER["PHP_SELF"], "f.paye,f.fk_statut", "", $option, 'align="right"', $sortfield, $sortorder);
255  print "</tr>\n";
256 
257  if ($num > 0) {
258  while ($i < min($num, $limit)) {
259  $objp = $db->fetch_object($result);
260 
261  if ($objp->type == FactureRec::TYPE_CREDIT_NOTE) {
262  $objp->qty = -($objp->qty);
263  }
264 
265  $total_ht += $objp->total_ht;
266  $total_qty += $objp->qty;
267 
268  $invoicestatic->id = $objp->facid;
269  $invoicestatic->ref = $objp->titre;
270  $societestatic->fetch($objp->socid);
271  $paiement = $invoicestatic->getSommePaiement();
272 
273  print '<tr class="oddeven">';
274  print '<td>';
275  print $invoicestatic->getNomUrl(1);
276  print "</td>\n";
277  print '<td>'.$societestatic->getNomUrl(1).'</td>';
278  print "<td>".$objp->code_client."</td>\n";
279  print '<td class="center">';
280  print dol_print_date($db->jdate($objp->datec), 'dayhour')."</td>";
281  print '<td class="center">'.$objp->qty."</td>\n";
282  print '<td align="right">'.price($objp->total_ht)."</td>\n";
283  print '<td align="right">'.$invoicestatic->LibStatut($objp->paye, $objp->statut, 5, $paiement, $objp->type).'</td>';
284  print "</tr>\n";
285  $i++;
286  }
287  }
288  print '<tr class="liste_total">';
289  if ($num < $limit) {
290  print '<td class="left">'.$langs->trans("Total").'</td>';
291  } else {
292  print '<td class="left">'.$langs->trans("Totalforthispage").'</td>';
293  }
294  print '<td colspan="3"></td>';
295  print '<td class="center">'.$total_qty.'</td>';
296  print '<td align="right">'.price($total_ht).'</td>';
297  print '<td></td>';
298  print "</table>";
299  print '</div>';
300  print '</form>';
301  } else {
302  dol_print_error($db);
303  }
304  $db->free($result);
305  }
306  }
307 } else {
308  dol_print_error();
309 }
310 
311 // End of page
312 llxFooter();
313 $db->close();
if(!defined('NOREQUIRESOC')) if(!defined('NOREQUIRETRAN')) if(!defined('NOTOKENRENEWAL')) if(!defined('NOREQUIREMENU')) if(!defined('NOREQUIREHTML')) if(!defined('NOREQUIREAJAX')) llxHeader()
Empty header.
Definition: wrapper.php:56
llxFooter()
Empty footer.
Definition: wrapper.php:70
const TYPE_CREDIT_NOTE
Credit note invoice.
Class to manage invoice templates.
Class to manage generation of HTML components Only common components must be here.
Classe permettant la generation de composants html autre Only common components are here.
Class to manage products or services.
const TYPE_PRODUCT
Regular product.
const TYPE_SERVICE
Service.
Class to manage third parties objects (customers, suppliers, prospects...)
if($cancel &&! $id) if($action=='add' &&! $cancel) if($action=='delete') if($id) $form
Actions.
Definition: card.php:143
dol_banner_tab($object, $paramid, $morehtml='', $shownav=1, $fieldid='rowid', $fieldref='ref', $morehtmlref='', $moreparam='', $nodbprefix=0, $morehtmlleft='', $morehtmlstatus='', $onlybanner=0, $morehtmlright='')
Show tab footer of a card.
dol_get_fiche_head($links=array(), $active='', $title='', $notab=0, $picto='', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limittoshow=0, $moretabssuffix='')
Show tabs of a record.
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0)
Returns text escaped for inclusion in HTML alt or title tags, or into values of HTML input fields.
dol_print_error($db='', $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
dol_get_fiche_end($notab=0)
Return tab footer of a card.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='')
Set event messages in dol_events session object.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs='', $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
print_liste_field_titre($name, $file="", $field="", $begin="", $moreparam="", $moreattrib="", $sortfield="", $sortorder="", $prefix="", $tooltip="", $forcenowrapcolumntitle=0)
Show title line of an array.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
print_barre_liste($titre, $page, $file, $options='', $sortfield='', $sortorder='', $morehtmlcenter='', $num=-1, $totalnboflines='', $picto='generic', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limit=-1, $hideselectlimit=0, $hidenavigation=0, $pagenavastextinput=0, $morehtmlrightbeforearrow='')
Print a title with navigation controls for pagination.
dol_trunc($string, $size=40, $trunc='right', $stringencoding='UTF-8', $nodot=0, $display=0)
Truncate a string to a particular length adding '…' if string larger than length.
GETPOSTISSET($paramname)
Return true if we are in a context of submitting the parameter $paramname from a POST of a form.
product_prepare_head($object)
Prepare array with list of tabs.
Definition: product.lib.php:35
show_stats_for_company($product, $socid)
Show stats for company.
restrictedArea(User $user, $features, $object=0, $tableandshare='', $feature2='', $dbt_keyfield='fk_soc', $dbt_select='rowid', $isdraft=0, $mode=0)
Check permissions of a user to show a page and an object.