Projekt

Allgemein

Profil

« Zurück | Weiter » 

Revision 5fdc44cb

Von Moritz Bunkus vor mehr als 17 Jahren hinzugefügt

  • ID 5fdc44cb545488ac82b3dd851b40dfea4d35d779
  • Vorgänger 4d20c181
  • Nachfolger 6bd1a382

Die Funktion "Zahlung buchen" bei Ausgangsrechnungen komplett umgeschrieben. Sie verlässt sich nun nicht mehr auf die aktuellen Daten in $form, um die alten Einträge in acc_trans zu löschen, sondern lädt den vorherigen Stand aus der Datenbank, entfernt darauf basierend die Einträge in acc_trans und lässt IS->post_transaction() selber die Zahlungen eintragen.

Unterschiede anzeigen:

SL/IS.pm
34 34

  
35 35
package IS;
36 36

  
37
use Data::Dumper;
38 37
use SL::AM;
39 38
use SL::Common;
40 39
use SL::DBUtils;
40
use SL::MoreCommon;
41 41

  
42 42
sub invoice_details {
43 43
  $main::lxdebug->enter_sub();
......
481 481
sub post_invoice {
482 482
  $main::lxdebug->enter_sub();
483 483

  
484
  my ($self, $myconfig, $form) = @_;
484
  my ($self, $myconfig, $form, $provided_dbh, $payments_only) = @_;
485 485

  
486 486
  # connect to database, turn off autocommit
487
  my $dbh = $form->dbconnect_noauto($myconfig);
487
  my $dbh = $provided_dbh ? $provided_dbh : $form->dbconnect_noauto($myconfig);
488 488

  
489
  my ($query, $sth, $null, $project_id, $deliverydate, @values);
489
  my ($query, $sth, $null, $project_id, @values);
490 490
  my $exchangerate = 0;
491 491

  
492 492
  if (!$form->{employee_id}) {
......
497 497

  
498 498
  my $all_units = AM->retrieve_units($myconfig, $form);
499 499

  
500
  if ($form->{id}) {
501

  
502
    &reverse_invoice($dbh, $form);
500
  if (!$payments_only) {
501
    if ($form->{id}) {
502
      &reverse_invoice($dbh, $form);
503 503

  
504
  } else {
505
    $query = qq|SELECT nextval('glid')|;
506
    ($form->{"id"}) = selectrow_query($form, $dbh, $query);
504
    } else {
505
      $query = qq|SELECT nextval('glid')|;
506
      ($form->{"id"}) = selectrow_query($form, $dbh, $query);
507 507

  
508
    $query = qq|INSERT INTO ar (id, invnumber) VALUES (?, ?)|;
509
    do_query($form, $dbh, $query, $form->{"id"}, $form->{"id"});
508
      $query = qq|INSERT INTO ar (id, invnumber) VALUES (?, ?)|;
509
      do_query($form, $dbh, $query, $form->{"id"}, $form->{"id"});
510 510

  
511
    if (!$form->{invnumber}) {
512
      $form->{invnumber} =
513
        $form->update_defaults($myconfig, $form->{type} eq "credit_note" ?
514
                               "cnnumber" : "invnumber", $dbh);
511
      if (!$form->{invnumber}) {
512
        $form->{invnumber} =
513
          $form->update_defaults($myconfig, $form->{type} eq "credit_note" ?
514
                                 "cnnumber" : "invnumber", $dbh);
515
      }
515 516
    }
516 517
  }
517 518

  
518 519
  my ($netamount, $invoicediff) = (0, 0);
519 520
  my ($amount, $linetotal, $lastincomeaccno);
520 521

  
521
  if ($form->{currency} eq $form->{defaultcurrency}) {
522
  my ($currencies)    = selectfirst_array_query($form, $dbh, qq|SELECT curr FROM defaults|);
523
  my $defaultcurrency = (split m/:/, $currencies)[0];
524

  
525
  if ($form->{currency} eq $defaultcurrency) {
522 526
    $form->{exchangerate} = 1;
523 527
  } else {
524 528
    $exchangerate =
......
638 642
        $form->round_amount($form->{"sellprice_$i"} * $form->{exchangerate},
639 643
                            $decimalplaces);
640 644

  
645
      next if $payments_only;
646

  
641 647
      if ($form->{"inventory_accno_$i"} || $form->{"assembly_$i"}) {
642 648

  
643 649
        # adjust parts onhand quantity
......
767 773
  $form->{amount}{ $form->{id} }{ $form->{AR} } *= -1;
768 774

  
769 775
  # update exchangerate
770
  if (($form->{currency} ne $form->{defaultcurrency}) && !$exchangerate) {
776
  if (($form->{currency} ne $defaultcurrency) && !$exchangerate) {
771 777
    $form->update_exchangerate($dbh, $form->{currency}, $form->{invdate},
772 778
                               $form->{exchangerate}, 0);
773 779
  }
......
777 783
  foreach my $trans_id (keys %{ $form->{amount} }) {
778 784
    foreach my $accno (keys %{ $form->{amount}{$trans_id} }) {
779 785
      next unless ($form->{expense_inventory} =~ /$accno/);
780
      if (
781
          ($form->{amount}{$trans_id}{$accno} =
782
           $form->round_amount($form->{amount}{$trans_id}{$accno}, 2)
783
          ) != 0
784
        ) {
786

  
787
      $form->{amount}{$trans_id}{$accno} = $form->round_amount($form->{amount}{$trans_id}{$accno}, 2);
788

  
789
      if (!$payments_only && ($form->{amount}{$trans_id}{$accno} != 0)) {
785 790
        $query =
786 791
          qq|INSERT INTO acc_trans (trans_id, chart_id, amount, transdate, taxkey, project_id)
787 792
             VALUES (?, (SELECT id FROM chart WHERE accno = ?), ?, ?,
......
793 798
    }
794 799

  
795 800
    foreach my $accno (keys %{ $form->{amount}{$trans_id} }) {
796
      if (
797
          ($form->{amount}{$trans_id}{$accno} =
798
           $form->round_amount($form->{amount}{$trans_id}{$accno}, 2)
799
          ) != 0
800
        ) {
801
      $form->{amount}{$trans_id}{$accno} = $form->round_amount($form->{amount}{$trans_id}{$accno}, 2);
802

  
803
      if (!$payments_only && ($form->{amount}{$trans_id}{$accno} != 0)) {
801 804
        $query =
802 805
          qq|INSERT INTO acc_trans (trans_id, chart_id, amount, transdate, taxkey, project_id)
803 806
             VALUES (?, (SELECT id FROM chart WHERE accno = ?), ?, ?,
......
830 833

  
831 834
      $exchangerate = 0;
832 835

  
833
      if ($form->{currency} eq $form->{defaultcurrency}) {
836
      if ($form->{currency} eq $defaultcurrency) {
834 837
        $form->{"exchangerate_$i"} = 1;
835 838
      } else {
836 839
        $exchangerate =
......
884 887
      $diff = 0;
885 888

  
886 889
      # update exchange rate
887
      if (($form->{currency} ne $form->{defaultcurrency}) && !$exchangerate) {
890
      if (($form->{currency} ne $defaultcurrency) && !$exchangerate) {
888 891
        $form->update_exchangerate($dbh, $form->{currency},
889 892
                                   $form->{"datepaid_$i"},
890 893
                                   $form->{"exchangerate_$i"}, 0);
......
892 895
    }
893 896
  }
894 897

  
898
  if ($payments_only) {
899
    $query = qq|UPDATE ar SET paid = ?, datepaid = ? WHERE id = ?|;
900
    do_query($form, $dbh, $query,  $form->{paid}, $form->{paid} ? conv_date($form->{datepaid}) : undef, conv_i($form->{id}));
901

  
902
    if (!$provided_dbh) {
903
      $dbh->commit();
904
      $dbh->disconnect();
905
    }
906

  
907
    $main::lxdebug->leave_sub();
908
    return;
909
  }
910

  
895 911
  # record exchange rate differences and gains/losses
896 912
  foreach my $accno (keys %{ $form->{fx} }) {
897 913
    foreach my $transdate (keys %{ $form->{fx}{$accno} }) {
......
913 929

  
914 930
  $amount = $netamount + $tax;
915 931

  
916
  # set values which could be empty to 0
917
  my $datepaid = conv_date($form->{paid});
918
  my $duedate  = conv_date($form->{duedate});
919
  $deliverydate = conv_date($form->{deliverydate});
920

  
921 932
  # fill in subject if there is none
922 933
  $form->{subject} = qq|$form->{label} $form->{invnumber}|
923 934
    unless $form->{subject};
......
1023 1034

  
1024 1035
  Common::webdav_folder($form) if ($main::webdav);
1025 1036

  
1026
  my $rc = $dbh->commit;
1027
  $dbh->disconnect;
1037
  my $rc = 1;
1038
  if (!$provided_dbh) {
1039
    $dbh->commit();
1040
    $dbh->disconnect();
1041
  }
1028 1042

  
1029 1043
  $main::lxdebug->leave_sub();
1030 1044

  
1031 1045
  return $rc;
1032 1046
}
1033 1047

  
1034
sub post_payment {
1035
  $main::lxdebug->enter_sub() and my ($self, $myconfig, $form, $locale) = @_;
1048
sub _delete_payments {
1049
  $main::lxdebug->enter_sub();
1036 1050

  
1037
  # connect to database, turn off autocommit
1038
  my $dbh = $form->dbconnect_noauto($myconfig);
1051
  my ($self, $form, $dbh) = @_;
1039 1052

  
1040
  $form->{datepaid} = $form->{invdate};
1053
  my @delete_oids;
1041 1054

  
1042
  # total payments, don't move we need it here
1043
  for my $i ( 1 .. $form->{paidaccounts} ) {
1044
    $form->{"paid_$i"}  = $form->parse_amount($myconfig, $form->{"paid_$i"});
1045
    $form->{"paid_$i"} *= -1                     if $form->{type} eq "credit_note";
1046
    $form->{"paid"}    += $form->{"paid_$i"};
1047
    $form->{"datepaid"} = $form->{"datepaid_$i"} if $form->{"datepaid_$i"};
1055
  # Delete old payment entries from acc_trans.
1056
  my $query =
1057
    qq|SELECT oid
1058
       FROM acc_trans
1059
       WHERE (trans_id = ?) AND fx_transaction
1060

  
1061
       UNION
1062

  
1063
       SELECT at.oid
1064
       FROM acc_trans at
1065
       LEFT JOIN chart c ON (at.chart_id = c.id)
1066
       WHERE (trans_id = ?) AND (c.link LIKE '%AR_paid%')|;
1067
  push @delete_oids, selectall_array_query($form, $dbh, $query, conv_i($form->{id}), conv_i($form->{id}));
1068

  
1069
  $query =
1070
    qq|SELECT at.oid
1071
       FROM acc_trans at
1072
       LEFT JOIN chart c ON (at.chart_id = c.id)
1073
       WHERE (trans_id = ?)
1074
         AND ((c.link = 'AR') OR (c.link LIKE '%:AR') OR (c.link LIKE 'AR:%'))
1075
       ORDER BY at.oid
1076
       OFFSET 1|;
1077
  push @delete_oids, selectall_array_query($form, $dbh, $query, conv_i($form->{id}));
1078

  
1079
  if (@delete_oids) {
1080
    $query = qq|DELETE FROM acc_trans WHERE oid IN (| . join(", ", @delete_oids) . qq|)|;
1081
    do_query($form, $dbh, $query);
1048 1082
  }
1049 1083

  
1050
  $form->{exchangerate} = $form->get_exchangerate($dbh, $form->{currency}, $form->{invdate}, "buy");
1084
  $main::lxdebug->leave_sub();
1085
}
1051 1086

  
1052
  # record payments and offsetting AR
1053
  for my $i (1 .. $form->{paidaccounts}) {
1054
    if ($form->{"paid_$i"}) {
1087
sub post_payment {
1088
  $main::lxdebug->enter_sub();
1055 1089

  
1056
      my ($accno) = split /--/, $form->{"AR_paid_$i"};
1057
      $form->{"datepaid_$i"} = $form->{invdate} unless ($form->{"datepaid_$i"});
1058
      $form->{datepaid} = $form->{"datepaid_$i"};
1090
  my ($self, $myconfig, $form, $locale) = @_;
1059 1091

  
1060
      $exchangerate = 0;
1061
      if (($form->{currency} eq $form->{defaultcurrency}) || ($form->{defaultcurrency} eq "")) {
1062
        $form->{"exchangerate_$i"} = 1;
1063
      } else {
1064
        $exchangerate = $form->check_exchangerate($myconfig, $form->{currency}, $form->{"datepaid_$i"}, 'buy');
1065
        $form->{"exchangerate_$i"} = ($exchangerate) ? $exchangerate : $form->parse_amount($myconfig, $form->{"exchangerate_$i"});
1066
      }
1092
  # connect to database, turn off autocommit
1093
  my $dbh = $form->dbconnect_noauto($myconfig);
1067 1094

  
1068
      # record AR
1069
      $amount = $form->round_amount($form->{"paid_$i"} * $form->{"exchangerate"}, 2);
1095
  my (%payments, $old_form, $row, $item, $query, %keep_vars);
1070 1096

  
1071
      $query =
1072
        qq|DELETE FROM acc_trans
1073
           WHERE (trans_id = ?)
1074
             AND (chart_id = (SELECT id FROM chart WHERE accno = ?))
1075
             AND (amount = ?) AND (transdate = ?)|;
1076
      do_query($form, $dbh, $query, $form->{id}, $form->{AR}, $amount, conv_date($form->{"datepaid_$i"}));
1077
      $query =
1078
        qq|INSERT INTO acc_trans (trans_id, chart_id, amount, transdate, project_id, taxkey)
1079
           VALUES (?, (SELECT id FROM chart WHERE accno = ?), ?, ?, ?,
1080
                   (SELECT taxkey_id FROM chart WHERE accno = ?))|;
1081
      do_query($form, $dbh, $query, $form->{id}, $form->{AR}, $amount, $form->{"datepaid_$i"}, conv_i($form->{"globalproject_id"}), $accno);
1097
  $old_form = save_form();
1082 1098

  
1083
      # record payment
1084
      $form->{"paid_$i"} *= -1;
1099
  # Delete all entries in acc_trans from prior payments.
1100
  $self->_delete_payments($form, $dbh);
1085 1101

  
1086
      $query =
1087
        qq|DELETE FROM acc_trans
1088
           WHERE (trans_id = ?)
1089
             AND (chart_id = (SELECT id FROM chart WHERE accno = ?))
1090
             AND (amount = ?) AND (transdate = ?) AND (source = ?) AND (memo = ?)|;
1091
      do_query($form, $dbh, $query, $form->{id}, $accno, $form->{"paid_$i"}, $form->{"datepaid_$i"}, $form->{"source_$i"}, $form->{"memo_$i"});
1102
  my @after_dp;
1103
  push @after_dp, selectall_hashref_query($form, $dbh, qq|SELECT * FROM acc_trans WHERE trans_id = ? ORDER BY oid|, $form->{id});
1092 1104

  
1093
      $query =
1094
        qq|INSERT INTO acc_trans (trans_id, chart_id, amount, transdate, source, memo, project_id, taxkey)
1095
           VALUES (?, (SELECT id FROM chart WHERE accno = ?), ?, ?, ?, ?, ?,
1096
                   (SELECT taxkey_id FROM chart WHERE accno = ?))|;
1097
      do_query($form, $dbh, $query, $form->{id}, $accno, $form->{"paid_$i"}, $form->{"datepaid_$i"}, $form->{"source_$i"}, $form->{"memo_$i"},
1098
               conv_i($form->{"globalproject_id"}), $accno);
1105
  # Save the new payments the user made before cleaning up $form.
1106
  map { $payments{$_} = $form->{$_} } grep m/^datepaid_\d+$|^memo_\d+$|^source_\d+$|^exchangerate_\d+$|^paid_\d+$|^AR_paid_\d+$|^paidaccounts$/, keys %{ $form };
1099 1107

  
1100
      # gain/loss
1101
      $amount = $form->{"paid_$i"} * $form->{exchangerate} - $form->{"paid_$i"} * $form->{"exchangerate_$i"};
1102
      $form->{fx}{ $form->{($amount > 0 ? 'fxgain_accno' : 'fxloss_accno')} }{ $form->{"datepaid_$i"} } += $amount;
1108
  # Clean up $form so that old content won't tamper the results.
1109
  %keep_vars = map { $_, 1 } qw(login password id);
1110
  map { delete $form->{$_} unless $keep_vars{$_} } keys %{ $form };
1103 1111

  
1104
      $diff = 0;
1112
  # Retrieve the invoice from the database.
1113
  $self->retrieve_invoice($myconfig, $form);
1105 1114

  
1106
      # update exchange rate
1107
      if (($form->{currency} ne $form->{defaultcurrency}) && !$exchangerate) {
1108
        $form->update_exchangerate($dbh, $form->{currency}, $form->{"datepaid_$i"}, $form->{"exchangerate_$i"}, 0);
1109
      }
1115
  # Set up the content of $form in the way that IR::post_invoice() expects.
1116
  $form->{exchangerate} = $form->format_amount($myconfig, $form->{exchangerate});
1110 1117

  
1111
    }
1118
  for $row (1 .. scalar @{ $form->{invoice_details} }) {
1119
    $item = $form->{invoice_details}->[$row - 1];
1120

  
1121
    map { $item->{$_} = $form->format_amount($myconfig, $item->{$_}) } qw(qty sellprice discount);
1122

  
1123
    map { $form->{"${_}_${row}"} = $item->{$_} } keys %{ $item };
1112 1124
  }
1113 1125

  
1114
  # record exchange rate differences and gains/losses
1115
  foreach my $accno (keys %{ $form->{fx} }) {
1116
    foreach my $transdate (keys %{ $form->{fx}{$accno} }) {
1126
  $form->{rowcount} = scalar @{ $form->{invoice_details} };
1117 1127

  
1118
      if ($form->{fx}{$accno}{$transdate} = $form->round_amount($form->{fx}{$accno}{$transdate}, 2)) { # '=' is no typo, it's an assignment
1119
        $query =
1120
          qq|DELETE FROM acc_trans
1121
             WHERE (trans_id = ?)
1122
               AND (chart_id = (SELECT c.id FROM chart c WHERE c.accno = ?))
1123
               AND (amount = ?) AND (transdate = ?) AND (cleared = ?) AND (fx_transaction = ?)|;
1124
        do_query($form, $dbh, $query, $form->{id}, $accno, $form->{fx}{$accno}{$transdate}, $transdate, 0, 1);
1125
        $query =
1126
          qq|INSERT INTO acc_trans (trans_id, chart_id, amount, transdate, cleared, fx_transaction, project_id, taxkey)
1127
             VALUES (?, (SELECT id FROM chart WHERE accno = ?), ?, ?, ?, ?, ?,
1128
                     (SELECT taxkey_id FROM chart WHERE accno = ?))|;
1129
        do_query($form, $dbh, $query, $form->{id}, $accno, $form->{fx}{$accno}{$transdate}, $transdate, 0, 1, conv_i($form->{"globalproject_id"}), $accno);
1130
      }
1128
  delete @{$form}{qw(invoice_details paidaccounts storno paid)};
1131 1129

  
1132
    }
1133
  }
1130
  # Restore the payment options from the user input.
1131
  map { $form->{$_} = $payments{$_} } keys %payments;
1134 1132

  
1135
  # save AR record
1136
  delete $form->{datepaid} unless $form->{paid};
1133
  # Get the AP accno (which is normally done by Form::create_links()).
1134
  $query =
1135
    qq|SELECT c.accno
1136
       FROM acc_trans at
1137
       LEFT JOIN chart c ON (at.chart_id = c.id)
1138
       WHERE (trans_id = ?)
1139
         AND ((c.link = 'AR') OR (c.link LIKE '%:AR') OR (c.link LIKE 'AR:%'))
1140
       ORDER BY at.oid
1141
       LIMIT 1|;
1137 1142

  
1138
  my $query = qq|UPDATE ar SET paid = ?, datepaid = ? WHERE id = ?|;
1139
  do_query($form, $dbh, $query, $form->{"paid"}, conv_date($form->{"datepaid"}), conv_i($form->{"id"}));
1143
  ($form->{AR}) = selectfirst_array_query($form, $dbh, $query, conv_i($form->{id}));
1140 1144

  
1141
  my $rc = $dbh->commit;
1142
  $dbh->disconnect;
1145
  # Post the new payments.
1146
  $self->post_invoice($myconfig, $form, $dbh, 1);
1147

  
1148
  restore_form($old_form);
1143 1149

  
1144
  $main::lxdebug->leave_sub() and return $rc;
1150
  my $rc = $dbh->commit();
1151
  $dbh->disconnect();
1152

  
1153
  $main::lxdebug->leave_sub();
1154

  
1155
  return $rc;
1145 1156
}
1146 1157

  
1147 1158
sub process_assembly {

Auch abrufbar als: Unified diff