ParseFilter: Komplexe Methoden ermöglichen, die auch den Key ändern
[kivitendo-erp.git] / SL / Controller / PriceRule.pm
1 package SL::Controller::PriceRule;
2
3 use strict;
4
5 use parent qw(SL::Controller::Base);
6
7 use SL::Controller::Helper::GetModels;
8 use SL::Controller::Helper::ParseFilter;
9 use SL::Controller::Helper::ReportGenerator;
10 use SL::DB::PriceRule;
11 use SL::DB::PriceRuleItem;
12 use SL::DB::Pricegroup;
13 use SL::DB::PartsGroup;
14 use SL::DB::Business;
15 use SL::Helper::Flash;
16 use SL::ClientJS;
17 use SL::Locale::String;
18
19 use Rose::Object::MakeMethods::Generic
20 (
21  'scalar --get_set_init' => [ qw(models price_rule vc js pricegroups partsgroups businesses) ],
22 );
23
24 # __PACKAGE__->run_before('check_auth');
25 __PACKAGE__->run_before('add_javascripts');
26
27 #
28 # actions
29 #
30
31 sub action_list {
32   my ($self) = @_;
33
34   $self->make_filter_summary;
35
36   my $price_rules = $self->models->get;
37
38   $self->prepare_report;
39
40   $self->report_generator_list_objects(report => $self->{report}, objects => $price_rules, $::form->{inline} ? (layout => 0, header => 0) : ());
41 }
42
43 sub action_new {
44   my ($self) = @_;
45
46   $self->price_rule(SL::DB::PriceRule->new);
47   $self->price_rule->assign_attributes(%{ $::form->{price_rule} || {} });
48   $self->display_form;
49 }
50
51 sub action_edit {
52   my ($self) = @_;
53
54   $self->display_form;
55 }
56
57 sub action_create {
58   my ($self) = @_;
59
60   $self->price_rule($self->price_rule->clone_and_reset_deep);
61   $self->create_or_update;
62 }
63
64 sub action_update {
65   my ($self) = @_;
66   $self->create_or_update;
67 }
68
69  #TODO
70 sub action_destroy {
71   my ($self) = @_;
72
73   $self->price_rule->delete;
74   flash_later('info',  $::locale->text('The price rule has been deleted.'));
75
76   $self->redirect_to($::form->{callback} || (action => 'list'));
77 }
78
79 sub action_add_item_row {
80   my ($self, %params) = @_;
81
82   my $item = SL::DB::PriceRuleItem->new(type => $::form->{type});
83
84   my $html = $self->render('price_rule/item', { output => 0 }, item => $item);
85
86   $self
87     ->js
88     ->before('#price_rule_new_items', $html)
89     ->reinit_widgets
90     ->render($self);
91 }
92
93 sub action_price_type_help {
94   $_[0]->render('price_rule/price_type_help', { layout => 0 });
95 }
96
97 #
98 # filters
99 #
100
101 sub check_auth {
102   $::auth->assert('price_rule_edit');
103 }
104
105 #
106 # helpers
107 #
108
109 sub display_form {
110   my ($self, %params) = @_;
111   my $is_new  = !$self->price_rule->id;
112   my $title   = $self->form_title(($is_new ? 'create' : 'edit'), $self->price_rule->type);
113   $self->render('price_rule/form',
114     title => $title,
115     %params
116   );
117 }
118
119 sub form_title {
120   my ($self, $action, $type) = @_;
121
122   return {
123     edit => {
124       customer => t8('Edit sales price rule'),
125       vendor   => t8('Edit purchase price rule'),
126       ''       => t8('Edit price rule'),
127     },
128     create => {
129       customer => t8('Create a new sales price rule'),
130       vendor   => t8('Create a new purchase price rule'),
131       ''       => t8('Create a new price rule'),
132     },
133     list => {
134       customer => t8('Sales Price Rules'),
135       vendor   => t8('Purchase Price Rules'),
136       ''       => t8('Price Rules'),
137     },
138   }->{$action}{$type};
139 }
140
141 sub create_or_update {
142   my $self   = shift;
143   my $is_new = !$self->price_rule->id;
144   my $params = delete($::form->{price_rule}) || { };
145
146   delete $params->{id};
147   $self->price_rule->assign_attributes(%{ $params });
148
149   my @errors = $self->price_rule->validate;
150
151   if (@errors) {
152     flash('error', @errors);
153     $self->display_form(callback => $::form->{callback});
154     return;
155   }
156
157   $self->price_rule->save;
158
159   flash_later('info', $is_new ? $::locale->text('The price rule has been created.') : $::locale->text('The price rule has been saved.'));
160
161   $self->redirect_to($::form->{callback} || (action => 'list', 'filter.type' => $self->price_rule->type));
162 }
163
164 sub prepare_report {
165   my ($self)      = @_;
166
167   my $callback    = $self->models->get_callback;
168
169   my $report      = SL::ReportGenerator->new(\%::myconfig, $::form);
170   $self->{report} = $report;
171
172   my @columns     = qw(name type priority price reduction discount items);
173   my @sortable    = qw(name type priority price reduction discount      );
174
175   my %column_defs = (
176     name          => { obj_link => sub { $self->url_for(action => 'edit', 'price_rule.id' => $_[0]->id, callback => $callback) } },
177     priority      => { sub  => sub { $_[0]->priority_as_text } },
178     price         => { sub  => sub { $_[0]->price_as_number } },
179     reduction     => { sub  => sub { $_[0]->reduction_as_number } },
180     discount      => { sub  => sub { $_[0]->discount_as_number } },
181     obsolete      => { sub  => sub { $_[0]->obsolete_as_bool_yn } },
182     items         => { sub  => sub { $_[0]->item_summary } },
183   );
184
185   map { $column_defs{$_}->{text} ||= $::locale->text( $self->models->get_sort_spec->{$_}->{title} ) } keys %column_defs;
186
187   if ( $report->{options}{output_format} =~ /^(pdf|csv)$/i ) {
188     $self->models->disable_plugin('paginated');
189   }
190   $report->set_options(
191     std_column_visibility => 1,
192     controller_class      => 'PriceRule',
193     output_format         => 'HTML',
194     title                 => $self->form_title('list', $self->vc),
195     allow_pdf_export      => !$::form->{inline},
196     allow_csv_export      => !$::form->{inline},
197   );
198   $report->set_columns(%column_defs);
199   $report->set_column_order(@columns);
200   $report->set_export_options(qw(list filter));
201   $report->set_options_from_form;
202   $self->models->get_models_url_params(sub{ map { $_ => $::form->{$_} } qw(inline) });
203   $self->models->set_report_generator_sort_options(report => $report, sortable_columns => \@sortable);
204   $report->set_options(
205     raw_bottom_info_text  => $self->render('price_rule/report_bottom', { output => 0 }),
206     raw_top_info_text     => $self->render('price_rule/report_top', { output => 0 }),
207   );
208 }
209
210 sub make_filter_summary {
211   my ($self) = @_;
212
213   my $filter = $::form->{filter} || {};
214   my @filter_strings;
215
216   my @filters = (
217     [ $filter->{"name:substr::ilike"}, t8('Name')  ],
218     [ $filter->{"price:number"},       t8('Price') ],
219     [ $filter->{"discount:number"},    t8('Discount') ],
220   );
221
222   for (@filters) {
223     push @filter_strings, "$_->[1]: $_->[0]" if $_->[0];
224   }
225
226   if ($filter->{has_item_type}) {
227     push @filter_strings, sprintf "%s: %s", t8('Has item type'), join ', ', map {
228       SL::DB::Manager::PriceRuleItem->get_type($_)->{description}
229     } @{ $filter->{has_item_type} || [] };
230   }
231
232   $self->{filter_summary} = join ', ', @filter_strings;
233 }
234
235 sub all_price_rule_item_types {
236   SL::DB::Manager::PriceRuleItem->get_all_types($_[0]->vc || $_[0]->price_rule->type);
237 }
238
239 sub add_javascripts  {
240   $::request->{layout}->add_javascripts(qw(kivi.PriceRule.js autocomplete_customer.js autocomplete_vendor.js autocomplete_part.js));
241 }
242
243 sub init_price_rule {
244   my ($self) = @_;
245
246   my $price_rule = $::form->{price_rule}{id} ? SL::DB::PriceRule->new(id => $::form->{price_rule}{id})->load : SL::DB::PriceRule->new;
247
248   my $items = delete $::form->{price_rule}{items};
249
250   $price_rule->assign_attributes(%{ $::form->{price_rule} || {} });
251
252   my %old_items = map { $_->id => $_ } $price_rule->items;
253
254   my @items;
255   for my $raw_item (@$items) {
256     my $item = $raw_item->{id} ? $old_items{ $raw_item->{id} } || SL::DB::PriceRuleItem->new(id => $raw_item->{id})->load : SL::DB::PriceRuleItem->new;
257     $item->assign_attributes(%$raw_item);
258     push @items, $item;
259   }
260
261   $price_rule->items(@items) if @items;
262
263   $self->price_rule($price_rule);
264 }
265
266 sub init_vc {
267   $::form->{filter}{type};
268 }
269
270 sub init_js {
271   SL::ClientJS->new;
272 }
273
274 sub init_businesses {
275   SL::DB::Manager::Business->get_all;
276 }
277
278 sub init_pricegroups {
279   SL::DB::Manager::Pricegroup->get_all;
280 }
281
282 sub init_partsgroups {
283   SL::DB::Manager::PartsGroup->get_all;
284 }
285
286 sub all_price_types {
287   SL::DB::Manager::PriceRule->all_price_types;
288 }
289
290 sub init_models {
291   my ($self) = @_;
292
293   SL::Controller::Helper::GetModels->new(
294     controller => $self,
295     sorted => {
296       name     => t8('Name'),
297       type     => t8('Type'),
298       priority => t8('Priority'),
299       price    => t8('Price'),
300       discount => t8('Discount'),
301       reduction => t8('Reduced Master Data'),
302       obsolete => t8('Obsolete'),
303       items    => t8('Rule Details'),
304     },
305   );
306 }
307
308 1;