1 #=====================================================================
 
   4 # Based on SQL-Ledger Version 2.1.9
 
   5 # Web http://www.lx-office.org
 
   7 #=====================================================================
 
   8 # SQL-Ledger, Accounting
 
  11 #  Author: Dieter Simader
 
  12 #   Email: dsimader@sql-ledger.org
 
  13 #     Web: http://www.sql-ledger.org
 
  16 # This program is free software; you can redistribute it and/or modify
 
  17 # it under the terms of the GNU General Public License as published by
 
  18 # the Free Software Foundation; either version 2 of the License, or
 
  19 # (at your option) any later version.
 
  21 # This program is distributed in the hope that it will be useful,
 
  22 # but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  23 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  24 # GNU General Public License for more details.
 
  25 # You should have received a copy of the GNU General Public License
 
  26 # along with this program; if not, write to the Free Software
 
  27 # Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 
  28 #======================================================================
 
  30 # Inventory Control module
 
  32 #======================================================================
 
  34 use POSIX qw(strftime);
 
  35 use List::Util qw(first max);
 
  36 use List::MoreUtils qw(any);
 
  41 use SL::ReportGenerator;
 
  49 our ($form, $locale, %myconfig, $lxdebug, $auth);
 
  51 require "bin/mozilla/io.pl";
 
  52 require "bin/mozilla/invoice_io.pl";
 
  53 require "bin/mozilla/common.pl";
 
  54 require "bin/mozilla/reportgenerator.pl";
 
  59 # type=submit $locale->text('Add Part')
 
  60 # type=submit $locale->text('Add Service')
 
  61 # type=submit $locale->text('Add Assembly')
 
  62 # type=submit $locale->text('Edit Part')
 
  63 # type=submit $locale->text('Edit Service')
 
  64 # type=submit $locale->text('Edit Assembly')
 
  65 # $locale->text('Parts')
 
  66 # $locale->text('Services')
 
  67 # $locale->text('Inventory quantity must be zero before you can set this part obsolete!')
 
  68 # $locale->text('Inventory quantity must be zero before you can set this assembly obsolete!')
 
  69 # $locale->text('Part Number missing!')
 
  70 # $locale->text('Service Number missing!')
 
  71 # $locale->text('Assembly Number missing!')
 
  72 # $locale->text('ea');
 
  77   $lxdebug->enter_sub();
 
  79   $auth->assert('part_service_assembly_edit');
 
  81   my $title                = 'Add ' . ucfirst $form->{item};
 
  82   $form->{title}           = $locale->text($title);
 
  83   $form->{callback}        = "$form->{script}?action=add&item=$form->{item}" unless $form->{callback};
 
  84   $form->{unit_changeable} = 1;
 
  86   IC->get_pricegroups(\%myconfig, \%$form);
 
  90   $lxdebug->leave_sub();
 
  94   $lxdebug->enter_sub();
 
  96   $auth->assert('part_service_assembly_edit');
 
  98   $form->{revers}       = 0;  # switch for backward sorting
 
  99   $form->{lastsort}     = ""; # memory for which table was sort at last time
 
 100   $form->{ndxs_counter} = 0;  # counter for added entries to top100
 
 102   my %is_xyz     = map { +"is_$_" => ($form->{searchitems} eq $_) } qw(part service assembly);
 
 104   $form->{title} = (ucfirst $form->{searchitems}) . "s";
 
 105   $form->{title} = $locale->text($form->{title});
 
 106   $form->{title} = $locale->text('Assemblies') if ($is_xyz{is_assembly});
 
 108   $form->{jsscript} = 1;
 
 110   $form->{CUSTOM_VARIABLES}                  = CVar->get_configs('module' => 'IC');
 
 111   ($form->{CUSTOM_VARIABLES_FILTER_CODE},
 
 112    $form->{CUSTOM_VARIABLES_INCLUSION_CODE}) = CVar->render_search_options('variables'      => $form->{CUSTOM_VARIABLES},
 
 113                                                                            'include_prefix' => 'l_',
 
 114                                                                            'include_value'  => 'Y');
 
 118   $form->get_lists('partsgroup'    => 'ALL_PARTSGROUPS');
 
 119   print $form->parse_html_template('ic/search', { %is_xyz,
 
 120                                                   dateformat => $myconfig{dateformat},
 
 121                                                   limit => $myconfig{vclimit}, });
 
 123   $lxdebug->leave_sub();
 
 126 sub search_update_prices {
 
 127   $lxdebug->enter_sub();
 
 129   $auth->assert('part_service_assembly_edit');
 
 131   my $pricegroups = IC->get_pricegroups(\%myconfig, \%$form);
 
 133   $form->{title} = $locale->text('Update Prices');
 
 137   print $form->parse_html_template('ic/search_update_prices', { PRICE_ROWS => $pricegroups });
 
 139   $lxdebug->leave_sub();
 
 142 sub confirm_price_update {
 
 143   $lxdebug->enter_sub();
 
 145   $auth->assert('part_service_assembly_edit');
 
 148   my $value_found = undef;
 
 150   foreach my $idx (qw(sellprice listprice), (1..$form->{price_rows})) {
 
 151     my $name      = $idx =~ m/\d/ ? $form->{"pricegroup_${idx}"}      : $idx eq 'sellprice' ? $locale->text('Sell Price') : $locale->text('List Price');
 
 152     my $type      = $idx =~ m/\d/ ? $form->{"pricegroup_type_${idx}"} : $form->{"${idx}_type"};
 
 153     my $value_idx = $idx =~ m/\d/ ? "price_${idx}" : $idx;
 
 154     my $value     = $form->parse_amount(\%myconfig, $form->{$value_idx});
 
 156     if ((0 > $value) && ($type eq 'percent')) {
 
 157       push @errors, $locale->text('You cannot adjust the price for pricegroup "#1" by a negative percentage.', $name);
 
 159     } elsif (!$value && ($form->{$value_idx} ne '')) {
 
 160       push @errors, $locale->text('No valid number entered for pricegroup "#1".', $name);
 
 162     } elsif (0 < $value) {
 
 167   push @errors, $locale->text('No prices will be updated because no prices have been entered.') if (!$value_found);
 
 169   my $num_matches = IC->get_num_matches_for_priceupdate();
 
 174     $form->show_generic_error(join('<br>', @errors), 'back_button' => 1);
 
 177   $form->{nextsub} = "update_prices";
 
 179   map { delete $form->{$_} } qw(action header);
 
 181   print $form->parse_html_template('ic/confirm_price_update', { HIDDENS     => [ map { name => $_, value => $form->{$_} }, keys %$form ],
 
 182                                                                 num_matches => $num_matches });
 
 184   $lxdebug->leave_sub();
 
 188   $lxdebug->enter_sub();
 
 190   $auth->assert('part_service_assembly_edit');
 
 192   my $num_updated = IC->update_prices(\%myconfig, \%$form);
 
 194   if (-1 != $num_updated) {
 
 195     $form->redirect($locale->text('#1 prices were updated.', $num_updated));
 
 197     $form->error($locale->text('Could not update prices!'));
 
 200   $lxdebug->leave_sub();
 
 204 #  $lxdebug->enter_sub();
 
 206 #  $auth->assert('part_service_assembly_edit');
 
 208 #  our ($j, $lastndx);
 
 211 #  $form->{title} = $locale->text('Top 100 hinzufuegen');
 
 215 #  push @custom_hiddens, qw(searchitems title bom titel revers lastsort sort ndxs_counter extras);
 
 216 #  push @custom_hiddens, qw(itemstatus l_linetotal l_partnumber l_description l_onhand l_unit l_sellprice l_linetotalsellprice);
 
 218 #        +{ name => 'row',     value => $j              },
 
 219 #        +{ name => 'nextsub', value => 'item_selected' },
 
 220 #        +{ name => 'test',    value => 'item_selected' },
 
 221 #        +{ name => 'lastndx', value => $lastndx        },
 
 222 #    map(+{ name => $_,        value => $form->{$_}     }, @custom_hiddens),
 
 225 #  my ($partnumber, $description, $unit, $sellprice, $soldtotal);
 
 226 #  # if choice set data
 
 227 ##  if ($form->{ndx}) {
 
 228 ##    for my $i (0 .. $form->{ndxs_counter}) {
 
 230 ##      # insert data into top100
 
 231 ##      push @{ $form->{parts} },
 
 233 ##          partnumber  => $form->{"totop100_partnumber_$j"},
 
 234 ##          description => $form->{"totop100_description_$j"},
 
 235 ##          unit        => $form->{"totop100_unit_$j"},
 
 236 ##          sellprice   => $form->{"totop100_sellprice_$j"},
 
 237 ##          soldtotal   => $form->{"totop100_soldtotal_$j"},
 
 244 #  # set data for next page
 
 245 #  for my $i (1 .. $form->{ndxs_counter}) {
 
 246 #    $partnumber  = $form->{"totop100_partnumber_$i"};
 
 247 #    $description = $form->{"totop100_description_$i"};
 
 248 #    $unit        = $form->{"totop100_unit_$i"};
 
 249 #    $sellprice   = $form->{"totop100_sellprice_$i"};
 
 250 #    $soldtotal   = $form->{"totop100_soldtotal_$i"};
 
 253 #    totop100_partnumber  => $form->{"totop100_partnumber_$i"},
 
 254 #    totop100_description => $form->{"totop100_description_$i"},
 
 255 #    totop100_unit        => $form->{"totop100_unit_$i"},
 
 256 #    totop100_sellprice   => $form->{"totop100_sellprice_$i"},
 
 257 #    totop100_soldtotal   => $form->{"totop100_soldtotal_$i"},
 
 261 ##<input type=hidden name=totop100_partnumber_$i value=$form->{"totop100_partnumber_$i"}>
 
 262 ##<input type=hidden name=totop100_description_$i value=$form->{"totop100_description_$i"}>
 
 263 ##<input type=hidden name=totop100_unit_$i value=$form->{"totop100_unit_$i"}>
 
 264 ##<input type=hidden name=totop100_sellprice_$i value=$form->{"totop100_sellprice_$i"}>
 
 265 ##<input type=hidden name=totop100_soldtotal_$i value=$form->{"totop100_soldtotal_$i"}>
 
 269 #  print $form->parse_html_template('ic/choice', +{ HIDDENS => \@HIDDENS, PARTS => \@PARTS });
 
 271 #  $lxdebug->leave_sub();
 
 275 #  $lxdebug->enter_sub();
 
 277 #  $auth->assert('part_service_assembly_edit');
 
 280 #  our ($partnumber, $description, $unit, $sellprice, $soldtotal);
 
 282 #  my @sortorders = ("", "partnumber", "description", "all");
 
 283 #  my $sortorder = $sortorders[($form->{description} ? 2 : 0) + ($form->{partnumber} ? 1 : 0)];
 
 284 #  IC->get_parts(\%myconfig, \%$form, $sortorder);
 
 286 #  $form->{title} = $locale->text('Top 100 hinzufuegen');
 
 291 #  <form method=post action=ic.pl>
 
 294 #      <th class=listtop colspan=6>| . $locale->text('choice part') . qq|</th>
 
 296 #        <tr height="5"></tr>
 
 297 #        <tr class=listheading>
 
 299 #          <th class=listheading>| . $locale->text('Part Number') . qq|</th>
 
 300 #          <th class=listheading>| . $locale->text('Part Description') . qq|</th>
 
 301 #          <th class=listheading>| . $locale->text('Unit of measure') . qq|</th>
 
 302 #          <th class=listheading>| . $locale->text('Sell Price') . qq|</th>
 
 303 #          <th class=listheading>| . $locale->text('soldtotal') . qq|</th>
 
 307 #  my $i = $form->{rows};
 
 309 #  for ($j = 1; $j <= $i; $j++) {
 
 312 #        <tr class=listrow| . ($j % 2) . qq|>|;
 
 315 #            <td><input name=ndx class=radio type=radio value=$j checked></td>|;
 
 318 #          <td><input name=ndx class=radio type=radio value=$j></td>|;
 
 321 #          <td><input name="new_partnumber_$j" type=hidden value="$form->{"partnumber_$j"}">$form->{"partnumber_$j"}</td>
 
 322 #          <td><input name="new_description_$j" type=hidden value="$form->{"description_$j"}">$form->{"description_$j"}</td>
 
 323 #          <td><input name="new_unit_$j" type=hidden value="$form->{"unit_$j"}">$form->{"unit_$j"}</td>
 
 324 #          <td><input name="new_sellprice_$j" type=hidden value="$form->{"sellprice_$j"}">$form->{"sellprice_$j"}</td>
 
 325 #          <td><input name="new_soldtotal_$j" type=hidden value="$form->{"soldtotal_$j"}">$form->{"soldtotal_$j"}</td>
 
 328 #        <input name="new_id_$j" type=hidden value="$form->{"id_$j"}">|;
 
 338 #<input type=hidden name=itemstatus value="$form->{itemstatus}">
 
 339 #<input type=hidden name=l_linetotal value="$form->{l_linetotal}">
 
 340 #<input type=hidden name=l_partnumber value="$form->{l_partnumber}">
 
 341 #<input type=hidden name=l_description value="$form->{l_description}">
 
 342 #<input type=hidden name=l_onhand value="$form->{l_onhand}">
 
 343 #<input type=hidden name=l_unit value="$form->{l_unit}">
 
 344 #<input type=hidden name=l_sellprice value="$form->{l_sellprice}">
 
 345 #<input type=hidden name=l_linetotalsellprice value="$form->{l_linetotalsellprice}">
 
 346 #<input type=hidden name=sort value="$form->{sort}">
 
 347 #<input type=hidden name=revers value="$form->{revers}">
 
 348 #<input type=hidden name=lastsort value="$form->{lastsort}">
 
 350 #<input type=hidden name=bom value="$form->{bom}">
 
 351 #<input type=hidden name=titel value="$form->{titel}">
 
 352 #<input type=hidden name=searchitems value="$form->{searchitems}">
 
 354 #<input type=hidden name=row value=$j>
 
 356 #<input type=hidden name=nextsub value=item_selected>
 
 358 #<input name=lastndx type=hidden value=$lastndx>
 
 360 #<input name=ndxs_counter type=hidden value=$form->{ndxs_counter}>|;
 
 364 #  if (($form->{ndxs_counter}) > 0) {
 
 365 #    for ($i = 1; ($i < $form->{ndxs_counter} + 1); $i++) {
 
 367 #      $partnumber  = $form->{"totop100_partnumber_$i"};
 
 368 #      $description = $form->{"totop100_description_$i"};
 
 369 #      $unit        = $form->{"totop100_unit_$i"};
 
 370 #      $sellprice   = $form->{"totop100_sellprice_$i"};
 
 371 #      $soldtotal   = $form->{"totop100_soldtotal_$i"};
 
 374 #<input type=hidden name=totop100_partnumber_$i value=$form->{"totop100_partnumber_$i"}>
 
 375 #<input type=hidden name=totop100_description_$i value=$form->{"totop100_description_$i"}>
 
 376 #<input type=hidden name=totop100_unit_$i value=$form->{"totop100_unit_$i"}>
 
 377 #<input type=hidden name=totop100_sellprice_$i value=$form->{"totop100_sellprice_$i"}>
 
 378 #<input type=hidden name=totop100_soldtotal_$i value=$form->{"totop100_soldtotal_$i"}>
 
 386 #<input class=submit type=submit name=action value="|
 
 387 #    . $locale->text('TOP100') . qq|">
 
 391 #  $lxdebug->leave_sub();
 
 395   $lxdebug->enter_sub();
 
 397   $auth->assert('part_service_assembly_edit');
 
 400     $form->{ndxs_counter}++;
 
 402     if ($form->{ndxs_counter} > 0) {
 
 404       my $index = $form->{ndx};
 
 406       $form->{"totop100_partnumber_$form->{ndxs_counter}"} = $form->{"new_partnumber_$index"};
 
 407       $form->{"totop100_description_$form->{ndxs_counter}"} = $form->{"new_description_$index"};
 
 408       $form->{"totop100_unit_$form->{ndxs_counter}"} = $form->{"new_unit_$index"};
 
 409       $form->{"totop100_sellprice_$form->{ndxs_counter}"} = $form->{"new_sellprice_$index"};
 
 410       $form->{"totop100_soldtotal_$form->{ndxs_counter}"} = $form->{"new_soldtotal_$index"};
 
 414   $lxdebug->leave_sub();
 
 418   $lxdebug->enter_sub();
 
 420   $auth->assert('part_service_assembly_edit');
 
 422   my ($revers, $lastsort, $callback, $option, $description, $sameitem,
 
 423       $partnumber, $unit, $sellprice, $soldtotal, $totop100, $onhand, $align);
 
 424   my (@column_index, %column_header, %column_data);
 
 425   my ($totalsellprice, $totallastcost, $totallistprice, $subtotalonhand, $subtotalsellprice, $subtotallastcost, $subtotallistprice);
 
 427   $form->{top100}      = "top100";
 
 428   $form->{l_soldtotal} = "Y";
 
 429   $form->{soldtotal}   = "soldtotal";
 
 430   $form->{sort}        = "soldtotal";
 
 431   $form->{l_qty}       = "N";
 
 432   $form->{l_linetotal} = "";
 
 434   $form->{number}      = "position";
 
 435   $form->{l_number}    = "Y";
 
 439   $form->{title} = $locale->text('Top 100');
 
 441   $revers   = $form->{revers};
 
 442   $lastsort = $form->{lastsort};
 
 444   if (($form->{lastsort} eq "") && ($form->{sort} eq undef)) {
 
 446     $form->{lastsort} = "partnumber";
 
 447     $form->{sort}     = "partnumber";
 
 451     "$form->{script}?action=top100&searchitems=$form->{searchitems}&itemstatus=$form->{itemstatus}&bom=$form->{bom}&l_linetotal=$form->{l_linetotal}&title="
 
 452     . $form->escape($form->{title}, 1);
 
 454   # if we have a serialnumber limit search
 
 455   if ($form->{serialnumber} || $form->{l_serialnumber}) {
 
 456     $form->{l_serialnumber} = "Y";
 
 457     unless (   $form->{bought}
 
 460             || $form->{quoted}) {
 
 461       $form->{bought} = $form->{sold} = 1;
 
 464   IC->all_parts(\%myconfig, \%$form);
 
 466   if ($form->{itemstatus} eq 'active') {
 
 467     $option .= $locale->text('Active') . " : ";
 
 469   if ($form->{itemstatus} eq 'obsolete') {
 
 470     $option .= $locale->text('Obsolete') . " : ";
 
 472   if ($form->{itemstatus} eq 'orphaned') {
 
 473     $option .= $locale->text('Orphaned') . " : ";
 
 475   if ($form->{itemstatus} eq 'onhand') {
 
 476     $option .= $locale->text('On Hand') . " : ";
 
 477     $form->{l_onhand} = "Y";
 
 479   if ($form->{itemstatus} eq 'short') {
 
 480     $option .= $locale->text('Short') . " : ";
 
 481     $form->{l_onhand} = "Y";
 
 483   if ($form->{onorder}) {
 
 484     $form->{l_ordnumber} = "Y";
 
 485     $callback .= "&onorder=$form->{onorder}";
 
 486     $option   .= $locale->text('On Order') . " : ";
 
 488   if ($form->{ordered}) {
 
 489     $form->{l_ordnumber} = "Y";
 
 490     $callback .= "&ordered=$form->{ordered}";
 
 491     $option   .= $locale->text('Ordered') . " : ";
 
 494     $form->{l_quonumber} = "Y";
 
 495     $callback .= "&rfq=$form->{rfq}";
 
 496     $option   .= $locale->text('RFQ') . " : ";
 
 498   if ($form->{quoted}) {
 
 499     $form->{l_quonumber} = "Y";
 
 500     $callback .= ""ed=$form->{quoted}";
 
 501     $option   .= $locale->text('Quoted') . " : ";
 
 503   if ($form->{bought}) {
 
 504     $form->{l_invnumber} = "Y";
 
 505     $callback .= "&bought=$form->{bought}";
 
 506     $option   .= $locale->text('Bought') . " : ";
 
 509     $form->{l_invnumber} = "Y";
 
 510     $callback .= "&sold=$form->{sold}";
 
 511     $option   .= $locale->text('Sold') . " : ";
 
 518       || $form->{quoted}) {
 
 520     $form->{l_lastcost} = "";
 
 521     $form->{l_name}     = "Y";
 
 522     if ($form->{transdatefrom}) {
 
 523       $callback .= "&transdatefrom=$form->{transdatefrom}";
 
 525         . $locale->text('From')
 
 527         . $locale->date(\%myconfig, $form->{transdatefrom}, 1);
 
 529     if ($form->{transdateto}) {
 
 530       $callback .= "&transdateto=$form->{transdateto}";
 
 532         . $locale->text('To')
 
 534         . $locale->date(\%myconfig, $form->{transdateto}, 1);
 
 540   if ($form->{partnumber}) {
 
 541     $callback .= "&partnumber=$form->{partnumber}";
 
 542     $option   .= $locale->text('Part Number') . qq| : $form->{partnumber}<br>|;
 
 545     $callback .= "&partnumber=$form->{ean}";
 
 546     $option   .= $locale->text('EAN') . qq| : $form->{ean}<br>|;
 
 548   if ($form->{partsgroup}) {
 
 549     $callback .= "&partsgroup=$form->{partsgroup}";
 
 550     $option   .= $locale->text('Group') . qq| : $form->{partsgroup}<br>|;
 
 552   if ($form->{serialnumber}) {
 
 553     $callback .= "&serialnumber=$form->{serialnumber}";
 
 554     $option   .= $locale->text('Serial Number') . qq| : $form->{serialnumber}<br>|;
 
 556   if ($form->{description}) {
 
 557     $callback   .= "&description=$form->{description}";
 
 558     $description = $form->{description};
 
 559     $description =~ s/\n/<br>/g;
 
 560     $option     .= $locale->text('Part Description') . qq| : $form->{description}<br>|;
 
 563     $callback .= "&make=$form->{make}";
 
 564     $option   .= $locale->text('Make') . qq| : $form->{make}<br>|;
 
 566   if ($form->{model}) {
 
 567     $callback .= "&model=$form->{model}";
 
 568     $option   .= $locale->text('Model') . qq| : $form->{model}<br>|;
 
 570   if ($form->{drawing}) {
 
 571     $callback .= "&drawing=$form->{drawing}";
 
 572     $option   .= $locale->text('Drawing') . qq| : $form->{drawing}<br>|;
 
 574   if ($form->{microfiche}) {
 
 575     $callback .= "µfiche=$form->{microfiche}";
 
 576     $option   .= $locale->text('Microfiche') . qq| : $form->{microfiche}<br>|;
 
 578   if ($form->{l_soldtotal}) {
 
 579     $callback .= "&soldtotal=$form->{soldtotal}";
 
 580     $option   .= $locale->text('soldtotal') . qq| : $form->{soldtotal}<br>|;
 
 583   my @columns = $form->sort_columns(
 
 584     qw(number partnumber ean description partsgroup bin onhand rop unit listprice linetotallistprice sellprice linetotalsellprice lastcost linetotallastcost priceupdate weight image drawing microfiche invnumber ordnumber quonumber name serialnumber soldtotal)
 
 587   if ($form->{l_linetotal}) {
 
 588     $form->{l_onhand} = "Y";
 
 589     $form->{l_linetotalsellprice} = "Y" if $form->{l_sellprice};
 
 590     if ($form->{l_lastcost}) {
 
 591       $form->{l_linetotallastcost} = "Y";
 
 592       if (($form->{searchitems} eq 'assembly') && !$form->{bom}) {
 
 593         $form->{l_linetotallastcost} = "";
 
 596     $form->{l_linetotallistprice} = "Y" if $form->{l_listprice};
 
 599   if ($form->{searchitems} eq 'service') {
 
 601     # remove bin, weight and rop from list
 
 602     map { $form->{"l_$_"} = "" } qw(bin weight rop);
 
 604     $form->{l_onhand} = "";
 
 606     # qty is irrelevant unless bought or sold
 
 612         || $form->{quoted}) {
 
 613       $form->{l_onhand} = "Y";
 
 615       $form->{l_linetotalsellprice} = "";
 
 616       $form->{l_linetotallastcost}  = "";
 
 620   foreach my $item (@columns) {
 
 621     if ($form->{"l_$item"} eq "Y") {
 
 622       push @column_index, $item;
 
 624       # add column to callback
 
 625       $callback .= "&l_$item=Y";
 
 629   if ($form->{l_subtotal} eq 'Y') {
 
 630     $callback .= "&l_subtotal=Y";
 
 633   $column_header{number} =
 
 634     qq|<th class=listheading nowrap>| . $locale->text('number') . qq|</th>|;
 
 635   $column_header{partnumber} =
 
 636     qq|<th nowrap><a class=listheading href=$callback&sort=partnumber&revers=$form->{revers}&lastsort=$form->{lastsort}>|
 
 637     . $locale->text('Part Number')
 
 639   $column_header{description} =
 
 640     qq|<th nowrap><a class=listheading href=$callback&sort=description&revers=$form->{revers}&lastsort=$form->{lastsort}>|
 
 641     . $locale->text('Part Description')
 
 643   $column_header{partsgroup} =
 
 644       qq|<th nowrap><a class=listheading href=$callback&sort=partsgroup>|
 
 645     . $locale->text('Group')
 
 647   $column_header{bin} =
 
 648       qq|<th><a class=listheading href=$callback&sort=bin>|
 
 649     . $locale->text('Bin')
 
 651   $column_header{priceupdate} =
 
 652       qq|<th nowrap><a class=listheading href=$callback&sort=priceupdate>|
 
 653     . $locale->text('Updated')
 
 655   $column_header{onhand} =
 
 656     qq|<th nowrap><a  class=listheading href=$callback&sort=onhand&revers=$form->{revers}&lastsort=$form->{lastsort}>|
 
 657     . $locale->text('Qty')
 
 659   $column_header{unit} =
 
 660     qq|<th class=listheading nowrap>| . $locale->text('Unit') . qq|</th>|;
 
 661   $column_header{listprice} =
 
 662       qq|<th class=listheading nowrap>|
 
 663     . $locale->text('List Price')
 
 665   $column_header{lastcost} =
 
 666     qq|<th class=listheading nowrap>| . $locale->text('Last Cost') . qq|</th>|;
 
 667   $column_header{rop} =
 
 668     qq|<th class=listheading nowrap>| . $locale->text('ROP') . qq|</th>|;
 
 669   $column_header{weight} =
 
 670     qq|<th class=listheading nowrap>| . $locale->text('Weight') . qq|</th>|;
 
 672   $column_header{invnumber} =
 
 673       qq|<th nowrap><a class=listheading href=$callback&sort=invnumber>|
 
 674     . $locale->text('Invoice Number')
 
 676   $column_header{ordnumber} =
 
 677       qq|<th nowrap><a class=listheading href=$callback&sort=ordnumber>|
 
 678     . $locale->text('Order Number')
 
 680   $column_header{quonumber} =
 
 681       qq|<th nowrap><a class=listheading href=$callback&sort=quonumber>|
 
 682     . $locale->text('Quotation')
 
 685   $column_header{name} =
 
 686       qq|<th nowrap><a class=listheading href=$callback&sort=name>|
 
 687     . $locale->text('Name')
 
 690   $column_header{sellprice} =
 
 691       qq|<th class=listheading nowrap>|
 
 692     . $locale->text('Sell Price')
 
 694   $column_header{linetotalsellprice} =
 
 695     qq|<th class=listheading nowrap>| . $locale->text('Extended') . qq|</th>|;
 
 696   $column_header{linetotallastcost} =
 
 697     qq|<th class=listheading nowrap>| . $locale->text('Extended') . qq|</th>|;
 
 698   $column_header{linetotallistprice} =
 
 699     qq|<th class=listheading nowrap>| . $locale->text('Extended') . qq|</th>|;
 
 701   $column_header{image} =
 
 702     qq|<th class=listheading nowrap>| . $locale->text('Image') . qq|</a></th>|;
 
 703   $column_header{drawing} =
 
 704       qq|<th nowrap><a class=listheading href=$callback&sort=drawing>|
 
 705     . $locale->text('Drawing')
 
 707   $column_header{microfiche} =
 
 708       qq|<th nowrap><a class=listheading href=$callback&sort=microfiche>|
 
 709     . $locale->text('Microfiche')
 
 712   $column_header{serialnumber} =
 
 713       qq|<th nowrap><a class=listheading href=$callback&sort=serialnumber>|
 
 714     . $locale->text('Serial Number')
 
 716   $column_header{soldtotal} =
 
 717     qq|<th nowrap><a class=listheading href=$callback&sort=soldtotal&revers=$form->{revers}&lastsort=$form->{lastsort}>|
 
 718     . $locale->text('soldtotal')
 
 722   my $colspan = $#column_index + 1;
 
 727     <th class=listtop colspan=$colspan>$form->{title}</th>
 
 731   <tr><td colspan=$colspan>$option</td></tr>
 
 733   <tr class=listheading>
 
 736   map { print "\n$column_header{$_}" } @column_index;
 
 742   # add order to callback
 
 743   $form->{callback} = $callback .= "&sort=$form->{sort}";
 
 745   # escape callback for href
 
 746   $callback = $form->escape($callback);
 
 748   if (@{ $form->{parts} }) {
 
 749     $sameitem = $form->{parts}->[0]->{ $form->{sort} };
 
 752   # insert numbers for top100
 
 754   foreach my $ref (@{ $form->{parts} }) {
 
 759   # if avaible -> insert choice here
 
 760   if (($form->{ndxs_counter}) > 0) {
 
 761     for (my $i = 1; ($i < $form->{ndxs_counter} + 1); $i++) {
 
 762       $partnumber  = $form->{"totop100_partnumber_$i"};
 
 763       $description = $form->{"totop100_description_$i"};
 
 764       $unit        = $form->{"totop100_unit_$i"};
 
 765       $sellprice   = $form->{"totop100_sellprice_$i"};
 
 766       $soldtotal   = $form->{"totop100_soldtotal_$i"};
 
 769 <input type=hidden name=totop100_partnumber_$i value=$form->{"totop100_partnumber_$i"}>
 
 770 <input type=hidden name=totop100_description_$i value=$form->{"totop100_description_$i"}>
 
 771 <input type=hidden name=totop100_unit_$i value=$form->{"totop100_unit_$i"}>
 
 772 <input type=hidden name=totop100_sellprice_$i value=$form->{"totop100_sellprice_$i"}>
 
 773 <input type=hidden name=totop100_soldtotal_$i value=$form->{"totop100_soldtotal_$i"}>
 
 777       push @{ $form->{parts} },
 
 779           partnumber  => "$partnumber",
 
 780           description => "$description",
 
 782           sellprice   => "$sellprice",
 
 783           soldtotal   => "$soldtotal" };
 
 786        # build data for columns
 
 788   foreach my $ref (@{ $form->{parts} }) {
 
 790     if ($form->{l_subtotal} eq 'Y' && !$ref->{assemblyitem}) {
 
 791       if ($sameitem ne $ref->{ $form->{sort} }) {
 
 792         parts_subtotal(\@column_index, \$subtotalonhand, \$subtotalsellprice, \$subtotallastcost, \$subtotallistprice);
 
 793         $sameitem = $ref->{ $form->{sort} };
 
 797     $ref->{exchangerate} = 1 unless $ref->{exchangerate};
 
 798     $ref->{sellprice} *= $ref->{exchangerate};
 
 799     $ref->{listprice} *= $ref->{exchangerate};
 
 800     $ref->{lastcost}  *= $ref->{exchangerate};
 
 802     # use this for assemblies
 
 803     $onhand = $ref->{onhand};
 
 806     if ($ref->{assemblyitem}) {
 
 808       $onhand = 0 if ($form->{sold});
 
 811     $ref->{description} =~ s/\n/<br>/g;
 
 813     $column_data{number} =
 
 815       . $form->format_amount(\%myconfig, $ref->{number})
 
 817     $column_data{partnumber} =
 
 818       "<td align=$align>$ref->{partnumber} </a></td>";
 
 819     $column_data{description} = "<td>$ref->{description} </td>";
 
 820     $column_data{partsgroup}  = "<td>$ref->{partsgroup} </td>";
 
 822     $column_data{onhand} =
 
 824       . $form->format_amount(\%myconfig, $ref->{onhand})
 
 826     $column_data{sellprice} =
 
 828       . $form->format_amount(\%myconfig, $ref->{sellprice})
 
 830     $column_data{listprice} =
 
 832       . $form->format_amount(\%myconfig, $ref->{listprice})
 
 834     $column_data{lastcost} =
 
 836       . $form->format_amount(\%myconfig, $ref->{lastcost})
 
 839     $column_data{linetotalsellprice} = "<td align=right>"
 
 840       . $form->format_amount(\%myconfig, $ref->{onhand} * $ref->{sellprice}, 2)
 
 842     $column_data{linetotallastcost} = "<td align=right>"
 
 843       . $form->format_amount(\%myconfig, $ref->{onhand} * $ref->{lastcost}, 2)
 
 845     $column_data{linetotallistprice} = "<td align=right>"
 
 846       . $form->format_amount(\%myconfig, $ref->{onhand} * $ref->{listprice}, 2)
 
 849     if (!$ref->{assemblyitem}) {
 
 850       $totalsellprice += $onhand * $ref->{sellprice};
 
 851       $totallastcost  += $onhand * $ref->{lastcost};
 
 852       $totallistprice += $onhand * $ref->{listprice};
 
 854       $subtotalonhand    += $onhand;
 
 855       $subtotalsellprice += $onhand * $ref->{sellprice};
 
 856       $subtotallastcost  += $onhand * $ref->{lastcost};
 
 857       $subtotallistprice += $onhand * $ref->{listprice};
 
 862       . $form->format_amount(\%myconfig, $ref->{rop}) . "</td>";
 
 863     $column_data{weight} =
 
 865       . $form->format_amount(\%myconfig, $ref->{weight})
 
 867     $column_data{unit}        = "<td>$ref->{unit} </td>";
 
 868     $column_data{bin}         = "<td>$ref->{bin} </td>";
 
 869     $column_data{priceupdate} = "<td>$ref->{priceupdate} </td>";
 
 871     $column_data{invnumber} =
 
 872       ($ref->{module} ne 'oe')
 
 873       ? "<td><a href=$ref->{module}.pl?action=edit&type=invoice&id=$ref->{trans_id}&callback=$callback>$ref->{invnumber}</a></td>"
 
 874       : "<td>$ref->{invnumber}</td>";
 
 875     $column_data{ordnumber} =
 
 876       ($ref->{module} eq 'oe')
 
 877       ? "<td><a href=$ref->{module}.pl?action=edit&type=$ref->{type}&id=$ref->{trans_id}&callback=$callback>$ref->{ordnumber}</a></td>"
 
 878       : "<td>$ref->{ordnumber}</td>";
 
 879     $column_data{quonumber} =
 
 880       ($ref->{module} eq 'oe' && !$ref->{ordnumber})
 
 881       ? "<td><a href=$ref->{module}.pl?action=edit&type=$ref->{type}&id=$ref->{trans_id}&callback=$callback>$ref->{quonumber}</a></td>"
 
 882       : "<td>$ref->{quonumber}</td>";
 
 884     $column_data{name} = "<td>$ref->{name}</td>";
 
 886     $column_data{image} =
 
 888       ? "<td><a href=$ref->{image}><img src=$ref->{image} height=32 border=0></a></td>"
 
 890     $column_data{drawing} =
 
 892       ? "<td><a href=$ref->{drawing}>$ref->{drawing}</a></td>"
 
 894     $column_data{microfiche} =
 
 896       ? "<td><a href=$ref->{microfiche}>$ref->{microfiche}</a></td>"
 
 899     $column_data{serialnumber} = "<td>$ref->{serialnumber}</td>";
 
 901     $column_data{soldtotal} = "<td  align=right>$ref->{soldtotal}</td>";
 
 905     print "<tr class=listrow$i>";
 
 907     map { print "\n$column_data{$_}" } @column_index;
 
 914   if ($form->{l_subtotal} eq 'Y') {
 
 915     parts_subtotal(\@column_index, \$subtotalonhand, \$subtotalsellprice, \$subtotallastcost, \$subtotallistprice);
 
 918   if ($form->{"l_linetotal"}) {
 
 919     map { $column_data{$_} = "<td> </td>" } @column_index;
 
 920     $column_data{linetotalsellprice} =
 
 921         "<th class=listtotal align=right>"
 
 922       . $form->format_amount(\%myconfig, $totalsellprice, 2)
 
 924     $column_data{linetotallastcost} =
 
 925         "<th class=listtotal align=right>"
 
 926       . $form->format_amount(\%myconfig, $totallastcost, 2)
 
 928     $column_data{linetotallistprice} =
 
 929         "<th class=listtotal align=right>"
 
 930       . $form->format_amount(\%myconfig, $totallistprice, 2)
 
 933     print "<tr class=listtotal>";
 
 935     map { print "\n$column_data{$_}" } @column_index;
 
 942   <tr><td colspan=$colspan><hr size=3 noshade></td></tr>
 
 951 <form method=post action=$form->{script}>
 
 953 <input type=hidden name=itemstatus value="$form->{itemstatus}">
 
 954 <input type=hidden name=l_linetotal value="$form->{l_linetotal}">
 
 955 <input type=hidden name=l_partnumber value="$form->{l_partnumber}">
 
 956 <input type=hidden name=l_description value="$form->{l_description}">
 
 957 <input type=hidden name=l_onhand value="$form->{l_onhand}">
 
 958 <input type=hidden name=l_unit value="$form->{l_unit}">
 
 959 <input type=hidden name=l_sellprice value="$form->{l_sellprice}">
 
 960 <input type=hidden name=l_linetotalsellprice value="$form->{l_linetotalsellprice}">
 
 961 <input type=hidden name=sort value="$form->{sort}">
 
 962 <input type=hidden name=revers value="$form->{revers}">
 
 963 <input type=hidden name=lastsort value="$form->{lastsort}">
 
 964 <input type=hidden name=parts value="$form->{parts}">
 
 966 <input type=hidden name=bom value="$form->{bom}">
 
 967 <input type=hidden name=titel value="$form->{titel}">
 
 968 <input type=hidden name=searchitems value="$form->{searchitems}">|;
 
 973 <!--    <input type=hidden name=ndxs_counter value="$form->{ndxs_counter}">-->
 
 975     <input class=submit type=submit name=action value="|
 
 976     . $locale->text('choice') . qq|">
 
 981   $lxdebug->leave_sub();
 
 986 # Warning, deep magic ahead.
 
 987 # This function parses the requested details, sanity checks them, and converts them into a format thats usable for IC->all_parts
 
 989 # flags coming from the form:
 
 991 #  searchitems=part revers=0 lastsort=''
 
 994 # partnumber ean description partsgroup serialnumber make model drawing microfiche
 
 995 # transdatefrom transdateto
 
 998 #  itemstatus = active | onhand | short | obsolete | orphaned
 
 999 #  action     = continue | top100
 
1002 #  bought sold onorder ordered rfq quoted
 
1003 #  l_partnumber l_description l_serialnumber l_unit l_listprice l_sellprice l_lastcost
 
1004 #  l_linetotal l_priceupdate l_bin l_rop l_weight l_image l_drawing l_microfiche
 
1005 #  l_partsgroup l_subtotal l_soldtotal l_deliverydate l_pricegroups
 
1008 #  nextsub revers lastsort sort ndxs_counter
 
1010 sub generate_report {
 
1011   $lxdebug->enter_sub();
 
1013   $auth->assert('part_service_assembly_edit');
 
1015   my ($revers, $lastsort, $description);
 
1017   my $cvar_configs = CVar->get_configs('module' => 'IC');
 
1019   $form->{title} = (ucfirst $form->{searchitems}) . "s";
 
1020   $form->{title} =~ s/ys$/ies/;
 
1021   $form->{title} = $locale->text($form->{title});
 
1024     'bin'                => { 'text' => $locale->text('Bin'), },
 
1025     'deliverydate'       => { 'text' => $locale->text('deliverydate'), },
 
1026     'description'        => { 'text' => $locale->text('Part Description'), },
 
1027     'notes'              => { 'text' => $locale->text('Notes'), },
 
1028     'drawing'            => { 'text' => $locale->text('Drawing'), },
 
1029     'ean'                => { 'text' => $locale->text('EAN'), },
 
1030     'image'              => { 'text' => $locale->text('Image'), },
 
1031     'invnumber'          => { 'text' => $locale->text('Invoice Number'), },
 
1032     'lastcost'           => { 'text' => $locale->text('Last Cost'), },
 
1033     'linetotallastcost'  => { 'text' => $locale->text('Extended'), },
 
1034     'linetotallistprice' => { 'text' => $locale->text('Extended'), },
 
1035     'linetotalsellprice' => { 'text' => $locale->text('Extended'), },
 
1036     'listprice'          => { 'text' => $locale->text('List Price'), },
 
1037     'microfiche'         => { 'text' => $locale->text('Microfiche'), },
 
1038     'name'               => { 'text' => $locale->text('Name'), },
 
1039     'onhand'             => { 'text' => $locale->text('Stocked Qty'), },
 
1040     'ordnumber'          => { 'text' => $locale->text('Order Number'), },
 
1041     'partnumber'         => { 'text' => $locale->text('Part Number'), },
 
1042     'partsgroup'         => { 'text' => $locale->text('Group'), },
 
1043     'priceupdate'        => { 'text' => $locale->text('Updated'), },
 
1044     'quonumber'          => { 'text' => $locale->text('Quotation'), },
 
1045     'rop'                => { 'text' => $locale->text('ROP'), },
 
1046     'sellprice'          => { 'text' => $locale->text('Sell Price'), },
 
1047     'serialnumber'       => { 'text' => $locale->text('Serial Number'), },
 
1048     'soldtotal'          => { 'text' => $locale->text('Qty in Selected Records'), },
 
1049     'transdate'          => { 'text' => $locale->text('Transdate'), },
 
1050     'unit'               => { 'text' => $locale->text('Unit'), },
 
1051     'weight'             => { 'text' => $locale->text('Weight'), },
 
1052     'projectnumber'      => { 'text' => $locale->text('Project Number'), },
 
1053     'projectdescription' => { 'text' => $locale->text('Project Description'), },
 
1056   $revers     = $form->{revers};
 
1057   $lastsort   = $form->{lastsort};
 
1059   # sorting and direction of sorting
 
1060   # ToDO: change this to the simpler field+direction method
 
1061   if (($form->{lastsort} eq "") && ($form->{sort} eq undef)) {
 
1062     $form->{revers}   = 0;
 
1063     $form->{lastsort} = "partnumber";
 
1064     $form->{sort}     = "partnumber";
 
1066     if ($form->{lastsort} eq $form->{sort}) {
 
1067       $form->{revers} = 1 - $form->{revers};
 
1069       $form->{revers} = 0;
 
1070       $form->{lastsort} = $form->{sort};
 
1074   # special case if we have a serialnumber limit search
 
1075   # serialnumbers are only given in invoices and orders,
 
1076   # so they can only pop up in bought, sold, rfq, and quoted stuff
 
1077   $form->{no_sn_joins} = 'Y' if (   !$form->{bought} && !$form->{sold}
 
1078                                  && !$form->{rfq}    && !$form->{quoted}
 
1079                                  && ($form->{l_serialnumber} || $form->{serialnumber}));
 
1081   # special case for any checkbox of bought | sold | onorder | ordered | rfq | quoted.
 
1082   # if any of these are ticked the behavior changes slightly for lastcost
 
1083   # since all those are aggregation checks for the legder tables this is an internal switch
 
1084   # refered to as ledgerchecks
 
1085   $form->{ledgerchecks} = 'Y' if (   $form->{bought} || $form->{sold} || $form->{onorder}
 
1086                                   || $form->{ordered} || $form->{rfq} || $form->{quoted});
 
1088   # if something should be activated if something else is active, enter it here
 
1089   my %dependencies = (
 
1090     onhand       => [ qw(l_onhand) ],
 
1091     short        => [ qw(l_onhand) ],
 
1092     onorder      => [ qw(l_ordnumber) ],
 
1093     ordered      => [ qw(l_ordnumber) ],
 
1094     rfq          => [ qw(l_quonumber) ],
 
1095     quoted       => [ qw(l_quonumber) ],
 
1096     bought       => [ qw(l_invnumber) ],
 
1097     sold         => [ qw(l_invnumber) ],
 
1098     ledgerchecks => [ qw(l_name) ],
 
1099     serialnumber => [ qw(l_serialnumber) ],
 
1100     no_sn_joins  => [ qw(bought sold) ],
 
1103   # get name of partsgroup if id is given
 
1105   if ($form->{partsgroup_id}) {
 
1106     my $pg = SL::DB::PartsGroup->new(id => $form->{partsgroup_id})->load;
 
1107     $pg_name = $pg->{'partsgroup'};
 
1110   # these strings get displayed at the top of the results to indicate the user which switches were used
 
1112     active        => $locale->text('Active'),
 
1113     obsolete      => $locale->text('Obsolete'),
 
1114     orphaned      => $locale->text('Orphaned'),
 
1115     onhand        => $locale->text('On Hand'),
 
1116     short         => $locale->text('Short'),
 
1117     onorder       => $locale->text('On Order'),
 
1118     ordered       => $locale->text('Ordered'),
 
1119     rfq           => $locale->text('RFQ'),
 
1120     quoted        => $locale->text('Quoted'),
 
1121     bought        => $locale->text('Bought'),
 
1122     sold          => $locale->text('Sold'),
 
1123     transdatefrom => $locale->text('From')       . " " . $locale->date(\%myconfig, $form->{transdatefrom}, 1),
 
1124     transdateto   => $locale->text('To (time)')  . " " . $locale->date(\%myconfig, $form->{transdateto}, 1),
 
1125     partnumber    => $locale->text('Part Number')      . ": '$form->{partnumber}'",
 
1126     partsgroup    => $locale->text('Group')            . ": '$form->{partsgroup}'",
 
1127     partsgroup_id => $locale->text('Group')            . ": '$pg_name'",
 
1128     serialnumber  => $locale->text('Serial Number')    . ": '$form->{serialnumber}'",
 
1129     description   => $locale->text('Part Description') . ": '$form->{description}'",
 
1130     make          => $locale->text('Make')             . ": '$form->{make}'",
 
1131     model         => $locale->text('Model')            . ": '$form->{model}'",
 
1132     drawing       => $locale->text('Drawing')          . ": '$form->{drawing}'",
 
1133     microfiche    => $locale->text('Microfiche')       . ": '$form->{microfiche}'",
 
1134     l_soldtotal   => $locale->text('Qty in Selected Records'),
 
1135     ean           => $locale->text('EAN')              . ": '$form->{ean}'",
 
1138   my @itemstatus_keys = qw(active obsolete orphaned onhand short);
 
1139   my @callback_keys   = qw(onorder ordered rfq quoted bought sold partnumber partsgroup partsgroup_id serialnumber description make model
 
1140                            drawing microfiche l_soldtotal l_deliverydate transdatefrom transdateto ean);
 
1142   # calculate dependencies
 
1143   for (@itemstatus_keys, @callback_keys) {
 
1144     next if ($form->{itemstatus} ne $_ && !$form->{$_});
 
1145     map { $form->{$_} = 'Y' } @{ $dependencies{$_} } if $dependencies{$_};
 
1148   # generate callback and optionstrings
 
1150   for my  $key (@itemstatus_keys, @callback_keys) {
 
1151     next if ($form->{itemstatus} ne $key && !$form->{$key});
 
1152     push @options, $optiontexts{$key};
 
1155   # special case for lastcost
 
1156   if ($form->{ledgerchecks}){
 
1157     # ledgerchecks don't know about sellprice or lastcost. they just return a
 
1158     # price. so rename sellprice to price, and drop lastcost.
 
1159     $column_defs{sellprice}{text} = $locale->text('Price');
 
1160     $form->{l_lastcost} = ""
 
1163   if ($form->{description}) {
 
1164     $description = $form->{description};
 
1165     $description =~ s/\n/<br>/g;
 
1168   if ($form->{l_linetotal}) {
 
1169     $form->{l_qty} = "Y";
 
1170     $form->{l_linetotalsellprice} = "Y" if $form->{l_sellprice};
 
1171     $form->{l_linetotallastcost}  = $form->{searchitems} eq 'assembly' && !$form->{bom} ? "" : 'Y' if  $form->{l_lastcost};
 
1172     $form->{l_linetotallistprice} = "Y" if $form->{l_listprice};
 
1175   if ($form->{searchitems} eq 'service') {
 
1177     # remove bin, weight and rop from list
 
1178     map { $form->{"l_$_"} = "" } qw(bin weight rop);
 
1180     $form->{l_onhand} = "";
 
1182     # qty is irrelevant unless bought or sold
 
1183     if (   $form->{bought}
 
1188         || $form->{quoted}) {
 
1189 #      $form->{l_onhand} = "Y";
 
1191       $form->{l_linetotalsellprice} = "";
 
1192       $form->{l_linetotallastcost}  = "";
 
1196   # soldtotal doesn't make sense with more than one bsooqr option.
 
1197   # so reset it to sold (the most common option), and issue a warning
 
1198   my @bsooqr = qw(sold bought onorder ordered rfq quoted);
 
1199   if ($form->{l_subtotal} && 1 < grep { $form->{$_} } @bsooqr) {
 
1200     my $enabled       = first { $form->{$_} } @bsooqr;
 
1201     $form->{$_}       = ''   for @bsooqr;
 
1202     $form->{$enabled} = 'Y';
 
1204     push @options, $::locale->text('Subtotal cannot distinguish betweens record types. Only one of the selected record types will be displayed: #1', $optiontexts{$enabled});
 
1207   IC->all_parts(\%myconfig, \%$form);
 
1210     partnumber description notes partsgroup bin onhand rop soldtotal unit listprice
 
1211     linetotallistprice sellprice linetotalsellprice lastcost linetotallastcost
 
1212     priceupdate weight image drawing microfiche invnumber ordnumber quonumber
 
1213     transdate name serialnumber deliverydate ean projectnumber projectdescription
 
1216   my $pricegroups = SL::DB::Manager::Pricegroup->get_all(sort => 'id');
 
1217   my @pricegroup_columns;
 
1218   my %column_defs_pricegroups;
 
1219   if ($form->{l_pricegroups}) {
 
1220     @pricegroup_columns      = map { "pricegroup_" . $_->id } @{ $pricegroups };
 
1221     %column_defs_pricegroups = map {
 
1222       "pricegroup_" . $_->id => {
 
1223         text    => $::locale->text('Pricegroup') . ' ' . $_->pricegroup,
 
1226     }  @{ $pricegroups };
 
1228   push @columns, @pricegroup_columns;
 
1230   my @includeable_custom_variables = grep { $_->{includeable} } @{ $cvar_configs };
 
1231   my @searchable_custom_variables  = grep { $_->{searchable} }  @{ $cvar_configs };
 
1232   my %column_defs_cvars            = map { +"cvar_$_->{name}" => { 'text' => $_->{description} } } @includeable_custom_variables;
 
1234   push @columns, map { "cvar_$_->{name}" } @includeable_custom_variables;
 
1236   %column_defs = (%column_defs, %column_defs_cvars, %column_defs_pricegroups);
 
1237   map { $column_defs{$_}->{visible} ||= $form->{"l_$_"} ? 1 : 0 } @columns;
 
1238   map { $column_defs{$_}->{align}   = 'right' } qw(onhand sellprice listprice lastcost linetotalsellprice linetotallastcost linetotallistprice rop weight soldtotal), @pricegroup_columns;
 
1240   my @hidden_variables = (qw(l_subtotal l_linetotal searchitems itemstatus bom l_pricegroups), @itemstatus_keys, @callback_keys,
 
1241                               map({ "cvar_$_->{name}" } @searchable_custom_variables), map { "l_$_" } @columns);
 
1243   my $callback         = build_std_url('action=generate_report', grep { $form->{$_} } @hidden_variables);
 
1245   my @sort_full        = qw(partnumber description onhand soldtotal deliverydate);
 
1246   my @sort_no_revers   = qw(partsgroup bin priceupdate invnumber ordnumber quonumber name image drawing serialnumber);
 
1248   foreach my $col (@sort_full) {
 
1249     $column_defs{$col}->{link} = join '&', $callback, "sort=$col", map { "$_=" . E($form->{$_}) } qw(revers lastsort);
 
1251   map { $column_defs{$_}->{link} = "${callback}&sort=$_" } @sort_no_revers;
 
1253   # add order to callback
 
1254   $form->{callback} = join '&', ($callback, map { "${_}=" . E($form->{$_}) } qw(sort revers));
 
1256   my $report = SL::ReportGenerator->new(\%myconfig, $form);
 
1258   my %attachment_basenames = (
 
1259     'part'     => $locale->text('part_list'),
 
1260     'service'  => $locale->text('service_list'),
 
1261     'assembly' => $locale->text('assembly_list'),
 
1264   $report->set_options('top_info_text'         => $locale->text('Options') . ': ' . join(', ', grep $_, @options),
 
1265                        'raw_bottom_info_text'  => $form->parse_html_template('ic/generate_report_bottom'),
 
1266                        'output_format'         => 'HTML',
 
1267                        'title'                 => $form->{title},
 
1268                        'attachment_basename'   => $attachment_basenames{$form->{searchitems}} . strftime('_%Y%m%d', localtime time),
 
1270   $report->set_options_from_form();
 
1271   $locale->set_numberformat_wo_thousands_separator(\%myconfig) if lc($report->{options}->{output_format}) eq 'csv';
 
1273   $report->set_columns(%column_defs);
 
1274   $report->set_column_order(@columns);
 
1276   $report->set_export_options('generate_report', @hidden_variables, qw(sort revers));
 
1278   $report->set_sort_indicator($form->{sort}, $form->{revers} ? 0 : 1);
 
1280   CVar->add_custom_variables_to_report('module'         => 'IC',
 
1281                                        'trans_id_field' => 'id',
 
1282                                        'configs'        => $cvar_configs,
 
1283                                        'column_defs'    => \%column_defs,
 
1284                                        'data'           => $form->{parts});
 
1286   CVar->add_custom_variables_to_report('module'         => 'IC',
 
1287                                        'sub_module'     => sub { $_[0]->{ioi} },
 
1288                                        'trans_id_field' => 'ioi_id',
 
1289                                        'configs'        => $cvar_configs,
 
1290                                        'column_defs'    => \%column_defs,
 
1291                                        'data'           => $form->{parts});
 
1293   my @subtotal_columns = qw(sellprice listprice lastcost);
 
1294   my %subtotals = map { $_ => 0 } ('onhand', @subtotal_columns);
 
1295   my %totals    = map { $_ => 0 } @subtotal_columns;
 
1297   my $same_item = @{ $form->{parts} } ? $form->{parts}[0]{ $form->{sort} } : undef;
 
1299   my $defaults  = AM->get_defaults();
 
1302   foreach my $ref (@{ $form->{parts} }) {
 
1304     # fresh row, for inserting later
 
1305     my $row = { map { $_ => { 'data' => $ref->{$_} } } @columns };
 
1307     $ref->{exchangerate} ||= 1;
 
1308     $ref->{price_factor} ||= 1;
 
1309     $ref->{sellprice}     *= $ref->{exchangerate} / $ref->{price_factor};
 
1310     $ref->{listprice}     *= $ref->{exchangerate} / $ref->{price_factor};
 
1311     $ref->{lastcost}      *= $ref->{exchangerate} / $ref->{price_factor};
 
1313     # use this for assemblies
 
1314     my $soldtotal = $ref->{soldtotal};
 
1316     if ($ref->{assemblyitem}) {
 
1317       $row->{partnumber}{align}   = 'right';
 
1318       $row->{soldtotal}{data}     = 0;
 
1319       $soldtotal                  = 0 if ($form->{sold});
 
1322     my $edit_link               = build_std_url('action=edit', 'id=' . E($ref->{id}), 'callback');
 
1323     $row->{partnumber}->{link}  = $edit_link;
 
1324     $row->{description}->{link} = $edit_link;
 
1326     foreach (qw(sellprice listprice lastcost)) {
 
1327       $row->{$_}{data}            = $form->format_amount(\%myconfig, $ref->{$_}, 2);
 
1328       $row->{"linetotal$_"}{data} = $form->format_amount(\%myconfig, $ref->{onhand} * $ref->{$_}, 2);
 
1330     foreach ( @pricegroup_columns ) {
 
1331       $row->{$_}{data}            = $form->format_amount(\%myconfig, $ref->{"$_"}, 2);
 
1335     map { $row->{$_}{data} = $form->format_amount(\%myconfig, $ref->{$_}); } qw(onhand rop weight soldtotal);
 
1337     $row->{weight}->{data} .= ' ' . $defaults->{weightunit};
 
1339     if (!$ref->{assemblyitem}) {
 
1340       foreach my $col (@subtotal_columns) {
 
1341         $totals{$col}    += $soldtotal * $ref->{$col};
 
1342         $subtotals{$col} += $soldtotal * $ref->{$col};
 
1345       $subtotals{soldtotal} += $soldtotal;
 
1349     if ($ref->{module} eq 'oe') {
 
1350       # für oe gibt es vier fälle, jeweils nach kunde oder lieferant unterschiedlich:
 
1352       # | ist bestellt  | Vom Kunde bestellt |  -> edit_oe_ord_link
 
1353       # | Anfrage       | Angebot            |  -> edit_oe_quo_link
 
1355       my $edit_oe_ord_link = build_std_url("script=oe.pl", 'action=edit', 'type=' . E($ref->{cv} eq 'vendor' ? 'purchase_order' : 'sales_order'), 'id=' . E($ref->{trans_id}), 'callback');
 
1356       my $edit_oe_quo_link = build_std_url("script=oe.pl", 'action=edit', 'type=' . E($ref->{cv} eq 'vendor' ? 'request_quotation' : 'sales_quotation'), 'id=' . E($ref->{trans_id}), 'callback');
 
1358       $row->{ordnumber}{link} = $edit_oe_ord_link;
 
1359       $row->{quonumber}{link} = $edit_oe_quo_link if (!$ref->{ordnumber});
 
1362       $row->{invnumber}{link} = build_std_url("script=$ref->{module}.pl", 'action=edit', 'type=invoice', 'id=' . E($ref->{trans_id}), 'callback');
 
1365     # set properties of images
 
1366     if ($ref->{image} && (lc $report->{options}->{output_format} eq 'html')) {
 
1367       $row->{image}{data}     = '';
 
1368       $row->{image}{raw_data} = '<a href="' . H($ref->{image}) . '"><img src="' . H($ref->{image}) . '" height="32" border="0"></a>';
 
1370     map { $row->{$_}{link} = $ref->{$_} } qw(drawing microfiche);
 
1372     $report->add_data($row);
 
1374     my $next_ref = $form->{parts}[$idx + 1];
 
1376     # insert subtotal rows
 
1377     if (($form->{l_subtotal} eq 'Y') &&
 
1379          (!$next_ref->{assemblyitem} && ($same_item ne $next_ref->{ $form->{sort} })))) {
 
1380       my $row = { map { $_ => { 'class' => 'listsubtotal', } } @columns };
 
1382       if (($form->{searchitems} ne 'assembly') || !$form->{bom}) {
 
1383         $row->{soldtotal}->{data} = $form->format_amount(\%myconfig, $subtotals{soldtotal});
 
1386       map { $row->{"linetotal$_"}->{data} = $form->format_amount(\%myconfig, $subtotals{$_}, 2) } @subtotal_columns;
 
1387       map { $subtotals{$_} = 0 } ('soldtotal', @subtotal_columns);
 
1389       $report->add_data($row);
 
1391       $same_item = $next_ref->{ $form->{sort} };
 
1397   if ($form->{"l_linetotal"} && !$form->{report_generator_csv_options_for_import}) {
 
1398     my $row = { map { $_ => { 'class' => 'listtotal', } } @columns };
 
1400     map { $row->{"linetotal$_"}->{data} = $form->format_amount(\%myconfig, $totals{$_}, 2) } @subtotal_columns;
 
1402     $report->add_separator();
 
1403     $report->add_data($row);
 
1406   $report->generate_with_headers();
 
1408   $lxdebug->leave_sub();
 
1409 }    #end generate_report
 
1411 sub parts_subtotal {
 
1412   $lxdebug->enter_sub();
 
1414   $auth->assert('part_service_assembly_edit');
 
1417   my ($column_index, $subtotalonhand, $subtotalsellprice, $subtotallastcost, $subtotallistprice) = @_;
 
1419   map { $column_data{$_} = "<td> </td>" } @{ $column_index };
 
1420   $$subtotalonhand = 0 if ($form->{searchitems} eq 'assembly' && $form->{bom});
 
1422   $column_data{onhand} =
 
1423       "<th class=listsubtotal align=right>"
 
1424     . $form->format_amount(\%myconfig, $$subtotalonhand)
 
1427   $column_data{linetotalsellprice} =
 
1428       "<th class=listsubtotal align=right>"
 
1429     . $form->format_amount(\%myconfig, $$subtotalsellprice, 2)
 
1431   $column_data{linetotallistprice} =
 
1432       "<th class=listsubtotal align=right>"
 
1433     . $form->format_amount(\%myconfig, $$subtotallistprice, 2)
 
1435   $column_data{linetotallastcost} =
 
1436       "<th class=listsubtotal align=right>"
 
1437     . $form->format_amount(\%myconfig, $$subtotallastcost, 2)
 
1440   $$subtotalonhand    = 0;
 
1441   $$subtotalsellprice = 0;
 
1442   $$subtotallistprice = 0;
 
1443   $$subtotallastcost  = 0;
 
1445   print "<tr class=listsubtotal>";
 
1447   map { print "\n$column_data{$_}" } @{ $column_index };
 
1453   $lxdebug->leave_sub();
 
1457   $lxdebug->enter_sub();
 
1459   $auth->assert('part_service_assembly_edit');
 
1461   # show history button
 
1462   $form->{javascript} = qq|<script type="text/javascript" src="js/show_history.js"></script>|;
 
1463   #/show hhistory button
 
1464   IC->get_part(\%myconfig, \%$form);
 
1466   $form->{"original_partnumber"} = $form->{"partnumber"};
 
1468   my $title      = 'Edit ' . ucfirst $form->{item};
 
1469   $form->{title} = $locale->text($title);
 
1474   $lxdebug->leave_sub();
 
1478   $lxdebug->enter_sub();
 
1480   $auth->assert('part_service_assembly_edit');
 
1482   IC->create_links("IC", \%myconfig, \%$form);
 
1485   map({ $form->{selectcurrency} .= "<option>$_\n" }
 
1486       split(/:/, $form->{currencies}));
 
1488   # parts and assemblies have the same links
 
1489   my $item = $form->{item};
 
1490   if ($form->{item} eq 'assembly') {
 
1494   # build the popup menus
 
1495   $form->{taxaccounts} = "";
 
1496   foreach my $key (keys %{ $form->{IC_links} }) {
 
1497     foreach my $ref (@{ $form->{IC_links}{$key} }) {
 
1499       # if this is a tax field
 
1500       if ($key =~ /IC_tax/) {
 
1501         if ($key =~ /\Q$item\E/) {
 
1502           $form->{taxaccounts} .= "$ref->{accno} ";
 
1503           $form->{"IC_tax_$ref->{accno}_description"} =
 
1504             "$ref->{accno}--$ref->{description}";
 
1507             if ($form->{amount}{ $ref->{accno} }) {
 
1508               $form->{"IC_tax_$ref->{accno}"} = "checked";
 
1511             $form->{"IC_tax_$ref->{accno}"} = "checked";
 
1516         $form->{"select$key"} .=
 
1517           "<option $ref->{selected}>$ref->{accno}--$ref->{description}\n";
 
1518         if ($form->{amount}{$key} eq $ref->{accno}) {
 
1519           $form->{$key} = "$ref->{accno}--$ref->{description}";
 
1525   chop $form->{taxaccounts};
 
1527   if (($form->{item} eq "part") || ($form->{item} eq "assembly")) {
 
1528     $form->{selectIC_income}  = $form->{selectIC_sale};
 
1529     $form->{selectIC_expense} = $form->{selectIC_cogs};
 
1530     $form->{IC_income}        = $form->{IC_sale};
 
1531     $form->{IC_expense}       = $form->{IC_cogs};
 
1534   delete $form->{IC_links};
 
1535   delete $form->{amount};
 
1537   $form->get_partsgroup(\%myconfig, { all => 1 });
 
1539   $form->{partsgroup} = "$form->{partsgroup}--$form->{partsgroup_id}";
 
1541   if (@{ $form->{all_partsgroup} }) {
 
1542     $form->{selectpartsgroup} = qq|<option>\n|;
 
1543     map { $form->{selectpartsgroup} .= qq|<option value="$_->{partsgroup}--$_->{id}">$_->{partsgroup}\n| } @{ $form->{all_partsgroup} };
 
1546   if ($form->{item} eq 'assembly') {
 
1548     foreach my $i (1 .. $form->{assembly_rows}) {
 
1549       if ($form->{"partsgroup_id_$i"}) {
 
1550         $form->{"partsgroup_$i"} =
 
1551           qq|$form->{"partsgroup_$i"}--$form->{"partsgroup_id_$i"}|;
 
1554     $form->get_partsgroup(\%myconfig);
 
1556     if (@{ $form->{all_partsgroup} }) {
 
1557       $form->{selectassemblypartsgroup} = qq|<option>\n|;
 
1560         $form->{selectassemblypartsgroup} .=
 
1561           qq|<option value="$_->{partsgroup}--$_->{id}">$_->{partsgroup}\n|
 
1562       } @{ $form->{all_partsgroup} };
 
1565   $lxdebug->leave_sub();
 
1569   $lxdebug->enter_sub();
 
1571   $auth->assert('part_service_assembly_edit');
 
1573   $form->{pg_keys}          = sub { "$_[0]->{partsgroup}--$_[0]->{id}" };
 
1574   $form->{description_area} = ($form->{rows} = $form->numtextrows($form->{description}, 40)) > 1;
 
1575   $form->{notes_rows}       =  max 4, $form->numtextrows($form->{notes}, 40), $form->numtextrows($form->{formel}, 40);
 
1577   map { $form->{"is_$_"}  = ($form->{item} eq $_) } qw(part service assembly);
 
1578   map { $form->{$_}       =~ s/"/"/g;        } qw(unit);
 
1580   $form->get_lists('price_factors' => 'ALL_PRICE_FACTORS',
 
1581                    'partsgroup'    => 'all_partsgroup',
 
1582                    'vendors'       => 'ALL_VENDORS',);
 
1585   IC->retrieve_buchungsgruppen(\%myconfig, $form);
 
1586   @{ $form->{BUCHUNGSGRUPPEN} } = grep { $_->{id} eq $form->{buchungsgruppen_id} || ($form->{id} && $form->{orphaned}) || !$form->{id} } @{ $form->{BUCHUNGSGRUPPEN} };
 
1588   # use JavaScript Calendar or not (yes!)
 
1589   $form->{jsscript} = 1;
 
1591   my $units = AM->retrieve_units(\%myconfig, $form);
 
1592   $form->{ALL_UNITS} = [ map +{ name => $_ }, sort { $units->{$a}{sortkey} <=> $units->{$b}{sortkey} } keys %$units ];
 
1594   $form->{defaults} = AM->get_defaults();
 
1596   $form->{fokus} = "ic.partnumber";
 
1598   $form->{CUSTOM_VARIABLES} = CVar->get_custom_variables('module' => 'IC', 'trans_id' => $form->{id});
 
1600   CVar->render_inputs('variables' => $form->{CUSTOM_VARIABLES}, show_disabled_message => 1)
 
1601     if (scalar @{ $form->{CUSTOM_VARIABLES} });
 
1604   #print $form->parse_html_template('ic/form_header', { ALL_PRICE_FACTORS => $form->{ALL_PRICE_FACTORS},
 
1605   #                                                     ALL_UNITS         => $form->{ALL_UNITS},
 
1606   #                                                     BUCHUNGSGRUPPEN   => $form->{BUCHUNGSGRUPPEN},
 
1607   #                                                     payment_terms     => $form->{payment_terms},
 
1608   #                                                     all_partsgroup    => $form->{all_partsgroup}});
 
1609   print $form->parse_html_template('ic/form_header');
 
1610   $lxdebug->leave_sub();
 
1614   $lxdebug->enter_sub();
 
1616   $auth->assert('part_service_assembly_edit');
 
1618   print $form->parse_html_template('ic/form_footer');
 
1620   $lxdebug->leave_sub();
 
1624   $lxdebug->enter_sub();
 
1627   my @mm_data = grep { any { $_ ne '' } @$_{qw(make model)} } map +{ make => $form->{"make_$_"}, model => $form->{"model_$_"}, lastcost => $form->{"lastcost_$_"}, lastupdate => $form->{"lastupdate_$_"}, sortorder => $form->{"sortorder_$_"} }, 1 .. $numrows;
 
1628   delete @{$form}{grep { m/^make_\d+/ || m/^model_\d+/ } keys %{ $form }};
 
1629   print $form->parse_html_template('ic/makemodel', { MM_DATA => [ @mm_data, {} ], mm_rows => scalar @mm_data + 1 });
 
1631   $lxdebug->leave_sub();
 
1635   $lxdebug->enter_sub();
 
1638   my ($nochange, $callback, $previousform, $linetotal, $line_purchase_price, $href);
 
1640   @column_index = qw(runningnumber qty unit bom partnumber description partsgroup lastcost total);
 
1642   if ($form->{previousform}) {
 
1644     @column_index = qw(qty unit bom partnumber description partsgroup total);
 
1648     $form->{old_callback} = $form->{callback};
 
1649     $callback             = $form->{callback};
 
1650     $form->{callback}     = "$form->{script}?action=display_form";
 
1653     map { delete $form->{$_} } qw(action header);
 
1655     # save form variables in a previousform variable
 
1656     my %form_to_save = map   { ($_ => m/^ (?: listprice | sellprice | lastcost ) $/x ? $form->format_amount(\%myconfig, $form->{$_}) : $form->{$_}) }
 
1658     $previousform    = $::auth->save_form_in_session(form => \%form_to_save);
 
1660     $form->{callback} = $callback;
 
1661     $form->{assemblytotal} = 0;
 
1662     $form->{assembly_purchase_price_total} = 0;
 
1663     $form->{weight}        = 0;
 
1667    runningnumber => { text =>  $locale->text('No.'),              nowrap => 1, width => '5%'  },
 
1668    qty           => { text =>  $locale->text('Qty'),              nowrap => 1, width => '10%' },
 
1669    unit          => { text =>  $locale->text('Unit'),             nowrap => 1, width => '5%'  },
 
1670    partnumber    => { text =>  $locale->text('Part Number'),      nowrap => 1, width => '20%' },
 
1671    description   => { text =>  $locale->text('Part Description'), nowrap => 1, width => '50%' },
 
1672    lastcost      => { text =>  $locale->text('Purchase Prices'),  nowrap => 1, width => '50%' },
 
1673    total         => { text =>  $locale->text('Sale Prices'),      nowrap => 1,                },
 
1674    bom           => { text =>  $locale->text('BOM'),                                          },
 
1675    partsgroup    => { text =>  $locale->text('Group'),                                        },
 
1680   for my $i (1 .. $numrows) {
 
1681     my (%row, @row_hiddens);
 
1683     $form->{"partnumber_$i"} =~ s/\"/"/g;
 
1685     $linetotal           = $form->round_amount($form->{"sellprice_$i"} * $form->{"qty_$i"} / ($form->{"price_factor_$i"} || 1), 4);
 
1686     $line_purchase_price = $form->round_amount($form->{"lastcost_$i"} *  $form->{"qty_$i"} / ($form->{"price_factor_$i"} || 1), 4);
 
1687     $form->{assemblytotal}                  += $linetotal;
 
1688     $form->{assembly_purchase_price_total}  += $line_purchase_price;
 
1689     $form->{"qty_$i"}    = $form->format_amount(\%myconfig, $form->{"qty_$i"});
 
1690     $linetotal           = $form->format_amount(\%myconfig, $linetotal, 2);
 
1691     $line_purchase_price = $form->format_amount(\%myconfig, $line_purchase_price, 2);
 
1692     $href                = qq|$form->{script}?action=edit&id=$form->{"id_$i"}&rowcount=$i&previousform=$previousform|;
 
1693     map { $row{$_}{data} = "" } qw(qty unit partnumber description bom partsgroup runningnumber);
 
1696     if (($i >= 1) && ($i == $numrows)) {
 
1697       if (!$form->{previousform}) {
 
1698         $row{partnumber}{data}  = qq|<input name="partnumber_$i" size=15 value="$form->{"partnumber_$i"}">|;
 
1699         $row{qty}{data}         = qq|<input name="qty_$i" size=5 value="$form->{"qty_$i"}">|;
 
1700         $row{description}{data} = qq|<input name="description_$i" size=40 value="$form->{"description_$i"}">|;
 
1701         $row{partsgroup}{data}  = qq|<input name="partsgroup_$i" size=10 value="$form->{"partsgroup_$i"}">|;
 
1705       if ($form->{previousform}) {
 
1706         push @row_hiddens,          qw(qty bom);
 
1707         $row{partnumber}{data}    = $form->{"partnumber_$i"};
 
1708         $row{qty}{data}           = $form->{"qty_$i"};
 
1709         $row{bom}{data}           = $form->{"bom_$i"} ? "x" : " ";
 
1710         $row{qty}{align}          = 'right';
 
1712         $row{partnumber}{data}    = qq|<a href=$href>$form->{"partnumber_$i"}</a>|;
 
1713         $row{qty}{data}           = qq|<input name="qty_$i" size=5 value="$form->{"qty_$i"}">|;
 
1714         $row{runningnumber}{data} = qq|<input name="runningnumber_$i" size=3 value="$i">|;
 
1715         $row{bom}{data}   = sprintf qq|<input name="bom_$i" type=checkbox class=checkbox value=1 %s>|,
 
1716                                        $form->{"bom_$i"} ? 'checked' : '';
 
1718       push @row_hiddens,        qw(unit description partnumber partsgroup);
 
1719       $row{unit}{data}        = $form->{"unit_$i"};
 
1720       #Bei der Artikelbeschreibung und Warengruppe können Sonderzeichen verwendet
 
1721       #werden, die den HTML Code stören. Daher sollen diese im Template escaped werden
 
1722       #dies geschieht, wenn die Variable escape gesetzt ist
 
1723       $row{description}{data}   = $form->{"description_$i"};
 
1724       $row{description}{escape} = 1;
 
1725       $row{partsgroup}{data}    = $form->{"partsgroup_$i"};
 
1726       $row{partsgroup}{escape}  = 1;
 
1727       $row{bom}{align}          = 'center';
 
1730     $row{lastcost}{data}      = $line_purchase_price;
 
1731     $row{total}{data}         = $linetotal;
 
1732     $row{lastcost}{align}     = 'right';
 
1733     $row{total}{align}        = 'right';
 
1734     $row{deliverydate}{align} = 'right';
 
1736     push @row_hiddens, qw(id sellprice lastcost weight price_factor_id price_factor);
 
1737     $row{hiddens} = [ map +{ name => "${_}_$i", value => $form->{"${_}_$i"} }, @row_hiddens ];
 
1742   print $form->parse_html_template('ic/assembly_row', { COLUMNS => \@column_index, ROWS => \@ROWS, HEADER => \%header });
 
1744   $lxdebug->leave_sub();
 
1748   $lxdebug->enter_sub();
 
1750   # parse pricegroups. and no, don't rely on check_form for this...
 
1751   map { $form->{"price_$_"} = $form->parse_amount(\%myconfig, $form->{"price_$_"}) } 1 .. $form->{price_rows};
 
1753   # same for makemodel lastcosts
 
1754   # but parse_amount not necessary for assembly component lastcosts
 
1755   unless ($form->{item} eq "assembly") {
 
1756     map { $form->{"lastcost_$_"} = $form->parse_amount(\%myconfig, $form->{"lastcost_$_"}) } 1 .. $form->{"makemodel_rows"};
 
1759   if ($form->{item} eq "assembly") {
 
1760     my $i = $form->{assembly_rows};
 
1762     # if last row is empty check the form otherwise retrieve item
 
1763     if (   ($form->{"partnumber_$i"} eq "")
 
1764         && ($form->{"description_$i"} eq "")
 
1765         && ($form->{"partsgroup_$i"}  eq "")) {
 
1771       IC->assembly_item(\%myconfig, \%$form);
 
1773       my $rows = scalar @{ $form->{item_list} };
 
1776         $form->{"qty_$i"} = 1 unless ($form->{"qty_$i"});
 
1779           $form->{makemodel_rows}--;
 
1780           select_item(mode => 'IC');
 
1783           map { $form->{item_list}[$i]{$_} =~ s/\"/"/g }
 
1784             qw(partnumber description unit partsgroup);
 
1785           map { $form->{"${_}_$i"} = $form->{item_list}[0]{$_} }
 
1786             keys %{ $form->{item_list}[0] };
 
1787           $form->{"runningnumber_$i"} = $form->{assembly_rows};
 
1788           $form->{assembly_rows}++;
 
1796         $form->{rowcount} = $i;
 
1797         $form->{assembly_rows}++;
 
1804   } elsif (($form->{item} eq 'part') || ($form->{item} eq 'service')) {
 
1808   $lxdebug->leave_sub();
 
1812   $lxdebug->enter_sub();
 
1814   $auth->assert('part_service_assembly_edit');
 
1816   my ($parts_id, %newform, $amount, $callback);
 
1818   # check if there is a part number - commented out, cause there is an automatic allocation of numbers
 
1819   # $form->isblank("partnumber", $locale->text(ucfirst $form->{item}." Part Number missing!"));
 
1821   # check if there is a description
 
1822   $form->isblank("description", $locale->text("Part Description missing!"));
 
1824   $form->error($locale->text("Inventory quantity must be zero before you can set this $form->{item} obsolete!"))
 
1825     if $form->{obsolete} && $form->{onhand} * 1 && $form->{item} ne 'service';
 
1827   if (!$form->{buchungsgruppen_id}) {
 
1828     $form->error($locale->text("Parts must have an entry type.") . " " .
 
1829      $locale->text("If you see this message, you most likely just setup your LX-Office and haven't added any entry types. If this is the case, the option is accessible for administrators in the System menu.")
 
1833   $form->error($locale->text('Description must not be empty!')) unless $form->{description};
 
1834   $form->error($locale->text('Partnumber must not be set to empty!')) if $form->{id} && !$form->{partnumber};
 
1837   if (IC->save(\%myconfig, \%$form) == 3) {
 
1838     $form->error($locale->text('Partnumber not unique!'));
 
1840   # saving the history
 
1841   if(!exists $form->{addition}) {
 
1842     $form->{snumbers} = qq|partnumber_| . $form->{partnumber};
 
1843     $form->{addition} = "SAVED";
 
1844     $form->save_history;
 
1846   # /saving the history
 
1847   $parts_id = $form->{id};
 
1850   # load previous variables
 
1851   if ($form->{previousform}) {
 
1853     # save the new form variables before splitting previousform
 
1854     map { $newform{$_} = $form->{$_} } keys %$form;
 
1856     # don't trample on previous variables
 
1857     map { delete $form->{$_} } keys %newform;
 
1859     my $ic_cvar_configs = CVar->get_configs(module => 'IC');
 
1860     my @ic_cvar_fields  = map { "cvar_$_->{name}" } @{ $ic_cvar_configs };
 
1862     # restore original values
 
1863     $::auth->restore_form_from_session($newform{previousform}, form => $form);
 
1864     $form->{taxaccounts} = $newform{taxaccount2};
 
1866     if ($form->{item} eq 'assembly') {
 
1868       # undo number formatting
 
1869       map { $form->{$_} = $form->parse_amount(\%myconfig, $form->{$_}) }
 
1870         qw(weight listprice sellprice rop);
 
1872       $form->{assembly_rows}--;
 
1873       $i = $form->{assembly_rows};
 
1874       $form->{"qty_$i"} = 1 unless ($form->{"qty_$i"});
 
1876       $form->{sellprice} -= $form->{"sellprice_$i"} * $form->{"qty_$i"};
 
1877       $form->{weight}    -= $form->{"weight_$i"} * $form->{"qty_$i"};
 
1879       # change/add values for assembly item
 
1880       map { $form->{"${_}_$i"} = $newform{$_} } qw(partnumber description bin unit weight listprice sellprice inventory_accno income_accno expense_accno price_factor_id);
 
1881       map { $form->{"ic_${_}_$i"} = $newform{$_} } @ic_cvar_fields;
 
1883       # das ist __voll__ bekloppt, dass so auszurechnen jb 22.5.09
 
1884       #$form->{sellprice} += $form->{"sellprice_$i"} * $form->{"qty_$i"};
 
1885       $form->{weight}    += $form->{"weight_$i"} * $form->{"qty_$i"};
 
1889       # set values for last invoice/order item
 
1890       $i = $form->{rowcount};
 
1891       $form->{"qty_$i"} = 1 unless ($form->{"qty_$i"});
 
1893       map { $form->{"${_}_$i"} = $newform{$_} } qw(partnumber description bin unit listprice inventory_accno income_accno expense_accno sellprice lastcost price_factor_id);
 
1894       map { $form->{"ic_${_}_$i"} = $newform{$_} } @ic_cvar_fields;
 
1896       $form->{"longdescription_$i"} = $newform{notes};
 
1898       $form->{"sellprice_$i"} = $newform{lastcost} if ($form->{vendor_id});
 
1900       if ($form->{exchangerate} != 0) {
 
1901         $form->{"sellprice_$i"} /= $form->{exchangerate};
 
1904       map { $form->{"taxaccounts_$i"} .= "$_ " } split / /, $newform{taxaccount};
 
1905       chop $form->{"taxaccounts_$i"};
 
1906       foreach my $item (qw(description rate taxnumber)) {
 
1907         my $index = $form->{"taxaccounts_$i"} . "_$item";
 
1908         $form->{$index} = $newform{$index};
 
1911       # credit remaining calculation
 
1912       $amount = $form->{"sellprice_$i"} * (1 - $form->{"discount_$i"} / 100) * $form->{"qty_$i"};
 
1914       map { $form->{"${_}_base"} += $amount } (split / /, $form->{"taxaccounts_$i"});
 
1915       map { $amount += ($form->{"${_}_base"} * $form->{"${_}_rate"}) } split / /, $form->{"taxaccounts_$i"} if !$form->{taxincluded};
 
1917       $form->{creditremaining} -= $amount;
 
1919       # redo number formatting, because invoice parse them!
 
1920       map { $form->{"${_}_$i"} = $form->format_amount(\%myconfig, $form->{"${_}_$i"}) } qw(weight listprice sellprice lastcost rop);
 
1923     $form->{"id_$i"} = $parts_id;
 
1925     # Get the actual price factor (not just the ID) for the marge calculation.
 
1926     $form->get_lists('price_factors' => 'ALL_PRICE_FACTORS');
 
1927     foreach my $pfac (@{ $form->{ALL_PRICE_FACTORS} }) {
 
1928       next if ($pfac->{id} != $newform{price_factor_id});
 
1929       $form->{"marge_price_factor_$i"} = $pfac->{factor};
 
1932     delete $form->{ALL_PRICE_FACTORS};
 
1934     delete $form->{action};
 
1936     # restore original callback
 
1937     $callback = $form->unescape($form->{callback});
 
1938     $form->{callback} = $form->unescape($form->{old_callback});
 
1939     delete $form->{old_callback};
 
1941     $form->{makemodel_rows}--;
 
1943     # put callback together
 
1944     foreach my $key (keys %$form) {
 
1946       # do single escape for Apache 2.0
 
1947       my $value = $form->escape($form->{$key}, 1);
 
1948       $callback .= qq|&$key=$value|;
 
1950     $form->{callback} = $callback;
 
1956   $lxdebug->leave_sub();
 
1960   $lxdebug->enter_sub();
 
1962   $auth->assert('part_service_assembly_edit');
 
1964   # saving the history
 
1965   if(!exists $form->{addition}) {
 
1966     $form->{snumbers} = qq|partnumber_| . $form->{partnumber};
 
1967     $form->{addition} = "SAVED AS NEW";
 
1968     $form->save_history;
 
1970   # /saving the history
 
1972   if ($form->{"original_partnumber"} &&
 
1973       ($form->{"partnumber"} eq $form->{"original_partnumber"})) {
 
1974     $form->{partnumber} = "";
 
1977   $lxdebug->leave_sub();
 
1981   $lxdebug->enter_sub();
 
1983   $auth->assert('part_service_assembly_edit');
 
1985   # saving the history
 
1986   if(!exists $form->{addition}) {
 
1987     $form->{snumbers} = qq|partnumber_| . $form->{partnumber};
 
1988     $form->{addition} = "DELETED";
 
1989     $form->save_history;
 
1991   # /saving the history
 
1992   my $rc = IC->delete(\%myconfig, \%$form);
 
1995   $form->redirect($locale->text('Item deleted!')) if ($rc > 0);
 
1996   $form->error($locale->text('Cannot delete item!'));
 
1998   $lxdebug->leave_sub();
 
2002   $lxdebug->enter_sub();
 
2004   $auth->assert('part_service_assembly_edit');
 
2009     pricegroup    => $form->{"pricegroup_$_"},
 
2010     pricegroup_id => $form->{"pricegroup_id_$_"},
 
2011     price         => $form->{"price_$_"},
 
2014   print $form->parse_html_template('ic/price_row', { PRICES => \@PRICES });
 
2016   $lxdebug->leave_sub();
 
2019 sub parts_language_selection {
 
2020   $lxdebug->enter_sub();
 
2022   $auth->assert('part_service_assembly_edit');
 
2024   my $languages = IC->retrieve_languages(\%myconfig, $form);
 
2026   if ($form->{language_values} ne "") {
 
2027     foreach my $item (split(/---\+\+\+---/, $form->{language_values})) {
 
2028       my ($language_id, $translation, $longdescription) = split(/--\+\+--/, $item);
 
2030       foreach my $language (@{ $languages }) {
 
2031         next unless ($language->{id} == $language_id);
 
2033         $language->{translation}     = $translation;
 
2034         $language->{longdescription} = $longdescription;
 
2036         $language->{translation_area}     = ($language->{translation_rows} = $form->numtextrows($language->{translation}, 40)) > 1;
 
2037         $language->{longdescription_rows} = max 4, $form->numtextrows($language->{longdescription}, 40);
 
2044   my @header_sort = qw(name longdescription);
 
2045   my %header_title = ( "name" => $locale->text("Name"),
 
2046                        "longdescription" => $locale->text("Long Description"),
 
2050     map(+{ "column_title" => $header_title{$_},
 
2055   $form->{"title"} = $locale->text("Language Values");
 
2057   print $form->parse_html_template("ic/parts_language_selection", { "HEADER"    => \@header,
 
2058                                                                     "LANGUAGES" => $languages, });
 
2060   $lxdebug->leave_sub();
 
2063 sub ajax_autocomplete {
 
2064   $main::lxdebug->enter_sub();
 
2066   my $form     = $main::form;
 
2067   my %myconfig = %main::myconfig;
 
2069   $form->{column}          = 'description'     unless $form->{column} =~ /^partnumber|description$/;
 
2070   $form->{$form->{column}} = $form->{q}           || '';
 
2071   $form->{limit}           = ($form->{limit} * 1) || 10;
 
2072   $form->{searchitems}   ||= '';
 
2074   my @results = IC->all_parts(\%myconfig, $form);
 
2076   print $form->ajax_response_header(),
 
2077         $form->parse_html_template('ic/ajax_autocomplete');
 
2079   $main::lxdebug->leave_sub();
 
2082 sub back_to_record {
 
2086   delete @{$::form}{qw(action action_add action_back_to_record back_sub description item notes partnumber sellprice taxaccount2 unit vc)};
 
2088   $::auth->restore_form_from_session($::form->{previousform}, clobber => 1);
 
2089   $::form->{rowcount}--;
 
2090   $::form->{action}   = 'display_form';
 
2091   $::form->{callback} = $::form->{script} . '?' . join('&', map { $::form->escape($_) . '=' . $::form->escape($::form->{$_}) } sort keys %{ $::form });
 
2095 sub continue { call_sub($form->{"nextsub"}); }
 
2098   my $action = first { $::form->{"action_${_}"} } qw(add back_to_record);
 
2099   $::form->error($::locale->text('No action defined.')) unless $action;
 
2101   $::form->{dispatched_action} = $action;