epic-ts
[kivitendo-erp.git] / SL / Controller / PriceSource.pm
1 package SL::Controller::PriceSource;
2
3 use strict;
4
5 use parent qw(SL::Controller::Base);
6
7 use List::MoreUtils qw(any uniq apply);
8 use SL::Locale::String qw(t8);
9 use SL::PriceSource;
10
11 use Rose::Object::MakeMethods::Generic
12 (
13  scalar => [ qw(record_item) ],
14  'scalar --get_set_init' => [ qw(record) ],
15 );
16
17 __PACKAGE__->run_before('check_auth');
18
19 #
20 # actions
21 #
22
23 sub action_price_popup {
24   my ($self) = @_;
25
26   my $record_item = _make_record_item($::form->{row});
27   my $old_unit;
28   if (($old_unit = $record_item->{__additional_form_attributes}{unit_old}) && $old_unit ne $record_item->unit) {
29     # reset unit changes. the way these interact on update breaks stuff
30     $record_item->unit_obj(SL::DB::Manager::Unit->find_by(name => $old_unit));
31     $self->js->val("select[name='unit_$::form->{row}']", $old_unit);
32   }
33
34   $self->render_price_dialog($record_item);
35 }
36
37 sub render_price_dialog {
38   my ($self, $record_item) = @_;
39
40   my $price_source = SL::PriceSource->new(record_item => $record_item, record => $self->record);
41
42   $self->js
43     ->run(
44       'kivi.io.price_chooser_dialog',
45       t8('Available Prices'),
46       $self->render('oe/price_sources_dialog', { output => 0 }, price_source => $price_source)
47     )
48     ->reinit_widgets;
49
50 #   if (@errors) {
51 #     $self->js->text('#dialog_flash_error_content', join ' ', @errors);
52 #     $self->js->show('#dialog_flash_error');
53 #   }
54
55   $self->js->render;
56 }
57
58
59 #
60 # internal stuff
61 #
62
63 sub check_auth {
64   if ($::form->{vc} eq 'customer') {
65     $::auth->assert('sales_edit_prices');
66   } elsif ($::form->{vc} eq 'vendor') {
67     $::auth->assert('purchase_edit_prices');
68   } else {
69     $::auth->assert('no_such_right');
70   }
71 }
72
73 sub init_record {
74   _make_record();
75 }
76
77 sub _make_record_item {
78   my ($row) = @_;
79
80   my $class = {
81     sales_order             => 'OrderItem',
82     purchase_order          => 'OrderItem',
83     sales_quotation         => 'OrderItem',
84     request_quotation       => 'OrderItem',
85     invoice                 => 'InvoiceItem',
86     purchase_invoice        => 'InvoiceItem',
87     credit_note             => 'InvoiceItem',
88     purchase_delivery_order => 'DeliveryOrderItem',
89     sales_delivery_order    => 'DeliveryOrderItem',
90   }->{$::form->{type}};
91
92   return unless $class;
93
94   $class = 'SL::DB::' . $class;
95
96   my %translated_methods = (
97     'SL::DB::OrderItem' => {
98       id                      => 'parts_id',
99       orderitems_id           => 'id',
100     },
101     'SL::DB::DeliveryOrderItem' => {
102       id                      => 'parts_id',
103       delivery_order_items_id => 'id',
104     },
105     'SL::DB::InvoiceItem' => {
106       id                      => 'parts_id',
107       invoice_id => 'id',
108     },
109   );
110
111   eval "require $class";
112
113   my $obj = $::form->{"orderitems_id_$row"}
114           ? $class->meta->convention_manager->auto_manager_class_name->find_by(id => $::form->{"orderitems_id_$row"})
115           : $class->new;
116
117   for my $key (grep { /_$row$/ } keys %$::form) {
118     my $method = $key;
119     $method =~ s/_$row$//;
120     $method = $translated_methods{$class}{$method} // $method;
121     my $value = $::form->{$key};
122     if ($obj->meta->column($method)) {
123       if ($obj->meta->column($method)->isa('Rose::DB::Object::Metadata::Column::Date')) {
124         $obj->${\"$method\_as_date"}($value);
125       } elsif ((ref $obj->meta->column($method)) =~ /^Rose::DB::Object::Metadata::Column::(?:Numeric|Float|DoublePrecsion)$/) {
126         $obj->${\"$method\_as_number"}($value);
127       } elsif ((ref $obj->meta->column($method)) =~ /^Rose::DB::Object::Metadata::Column::Boolean$/) {
128         $obj->$method(!!$value);
129       } else {
130         $obj->$method($value);
131       }
132     } else {
133       $obj->{__additional_form_attributes}{$method} = $value;
134     }
135   }
136
137   if ($::form->{"id_$row"}) {
138     $obj->part(SL::DB::Part->load_cached($::form->{"id_$row"}));
139   }
140
141   return $obj;
142 }
143
144 sub _make_record {
145   my ($with_items) = @_;
146
147   my $class = {
148     sales_order             => 'Order',
149     purchase_order          => 'Order',
150     sales_quotation         => 'Order',
151     request_quotation       => 'Order',
152     purchase_invoice        => 'PurchaseInvoice',
153     purchase_delivery_order => 'DeliveryOrder',
154     sales_delivery_order    => 'DeliveryOrder',
155   }->{$::form->{type}};
156
157   if ($::form->{type} =~ /invoice|credit_note/) {
158     $class = $::form->{vc} eq 'customer' ? 'Invoice'
159            : $::form->{vc} eq 'vendor'   ? 'PurchaseInvoice'
160            : do { die 'unknown invoice type' };
161   }
162
163   return unless $class;
164
165   $class = 'SL::DB::' . $class;
166
167   eval "require $class";
168
169   my $obj = $::form->{id}
170           ? $class->meta->convention_manager->auto_manager_class_name->find_by(id => $::form->{id})
171           : $class->new;
172
173   for my $method (keys %$::form) {
174     next unless $obj->can($method);
175     next unless $obj->meta->column($method);
176
177     if ($obj->meta->column($method)->isa('Rose::DB::Object::Metadata::Column::Date')) {
178       $obj->${\"$method\_as_date"}($::form->{$method});
179     } elsif ((ref $obj->meta->column($method)) =~ /^Rose::DB::Object::Metadata::Column::(?:Numeric|Float|DoublePrecsion)$/) {
180       $obj->${\"$method\_as\_number"}($::form->{$method});
181     } elsif ((ref $obj->meta->column($method)) =~ /^Rose::DB::Object::Metadata::Column::Boolean$/) {
182       $obj->$method(!!$::form->{$method});
183     } else {
184       $obj->$method($::form->{$method});
185     }
186   }
187
188   if ($with_items) {
189     my @items;
190     for my $i (1 .. $::form->{rowcount}) {
191       next unless $::form->{"id_$i"};
192       push @items, _make_record_item($i)
193     }
194
195     $obj->items(@items) if @items;
196   }
197   $obj->is_sales(!!$obj->customer_id) if $class eq 'SL::DB::DeliveryOrder';
198
199   return $obj;
200 }
201
202 1;