1 package SL::DB::Manager::Part;
5 use SL::DB::Helper::Manager;
6 use SL::DB::Helper::Sorted;
7 use SL::DB::Helper::Paginated;
8 use SL::DB::Helper::Filtered;
9 use base qw(SL::DB::Helper::Manager);
13 use SL::MoreCommon qw(listify);
15 sub object_class { 'SL::DB::Part' }
17 __PACKAGE__->make_manager_methods;
18 __PACKAGE__->add_filter_specs(
20 my ($key, $value, $prefix) = @_;
21 return __PACKAGE__->type_filter($value, $prefix);
24 my ($key, $value, $prefix) = @_;
25 return or => [ map { $prefix . $_ => $value } qw(partnumber description ean) ]
27 all_with_makemodel => sub {
28 my ($key, $value, $prefix) = @_;
29 return or => [ map { $prefix . $_ => $value } qw(partnumber description ean makemodels.model) ],
30 $prefix . 'makemodels';
32 all_with_customer_partnumber => sub {
33 my ($key, $value, $prefix) = @_;
34 return or => [ map { $prefix . $_ => $value } qw(partnumber description ean customerprices.customer_partnumber) ],
35 $prefix . 'customerprices';
40 my ($class, $type, $prefix) = @_;
42 return () unless $type;
46 # this is to make selections like part_type => { part => 1, service => 1 } work
47 if ('HASH' eq ref $type) {
48 $type = [ grep { $type->{$_} } keys %$type ];
51 my @types = grep { $_ } listify($type);
54 for my $type (@types) {
55 if ($type =~ m/^part/) {
56 push @filter, ($prefix . part_type => 'part');
57 } elsif ($type =~ m/^service/) {
58 push @filter, ($prefix . part_type => 'service');
59 } elsif ($type =~ m/^assembly/) {
60 push @filter, ($prefix . part_type => 'assembly');
61 } elsif ($type =~ m/^assortment/) {
62 push @filter, ($prefix . part_type => 'assortment');
66 return @filter > 2 ? (or => \@filter) : @filter;
73 return () unless @part_ids;
75 my $placeholders = join ',', ('?') x @part_ids;
77 SELECT oi.parts_id, SUM(oi.base_qty) AS qty
79 LEFT JOIN oe ON (oi.trans_id = oe.id)
80 WHERE (oi.parts_id IN ($placeholders))
81 AND oe.record_type = 'purchase_order'
82 AND (NOT COALESCE(oe.closed, FALSE))
83 AND (NOT COALESCE(oe.delivered, FALSE))
87 my %qty_by_id = map { $_->{parts_id} => $_->{qty} * 1 } @{ selectall_hashref_query($::form, $class->object_class->init_db->dbh, $query, @part_ids) };
88 map { $qty_by_id{$_} ||= 0 } @part_ids;
93 sub get_open_ordered_qty {
96 return () unless $part_id;
101 SELECT parts_id, sum(oi.qty) as sum
103 LEFT OUTER JOIN oe o ON (oi.trans_id = o.id)
106 AND (o.record_type = 'purchase_order')
107 AND (NOT COALESCE(o.closed, FALSE))
108 AND (NOT COALESCE(o.delivered, FALSE))
109 AND (COALESCE(o.vendor_id, 0) <> 0)
113 open_orderitems_ids AS (
114 SELECT oi.id, parts_id
116 LEFT OUTER JOIN oe o ON (oi.trans_id = o.id)
119 AND (o.record_type = 'purchase_order')
120 AND (NOT COALESCE(o.closed, FALSE))
121 AND (NOT COALESCE(o.delivered, FALSE))
122 AND (o.vendor_id is not null)
126 SELECT parts_id, sum(qty) AS sum
127 FROM delivery_order_items
129 SELECT to_id from record_links
131 from_id IN ( SELECT id FROM open_orderitems_ids)
132 AND from_table = 'orderitems'
133 AND to_table = 'delivery_order_items'
138 open_ordered_qty AS (
141 oq.sum AS ordered_sum,
142 COALESCE(dq.sum,0.00) AS sum,
143 sum(COALESCE(oq.sum,0.00) - COALESCE(dq.sum,0.00)) AS open_qty
145 LEFT JOIN delivered_qty dq ON dq.parts_id = oq.parts_id
146 GROUP BY oq.parts_id, oq.sum, dq.sum
149 SELECT open_qty FROM open_ordered_qty
153 my ($open_qty) = selectfirst_array_query(
154 $::form, $class->object_class->init_db->dbh,
155 $query, $part_id, $part_id, $part_id
164 default => [ 'partnumber', 1 ],
181 SL::DB::Manager::Part - RDBO manager for the C<parts> table
187 =item C<get_ordered_qty @part_ids>
189 For each of the given part IDs the ordered quantity is
190 calculated. This is done by summing over all open purchase orders.
192 Returns a hash with the part IDs being the keys and the ordered
193 quantities being the values.
195 =item C<type_filter @types>
197 Constructs a partial filter for matching any of the article types
198 given with C<@types>. The returned partial filter is suitable for a
201 Each type can be either 'C<part>', 'C<service>' or 'C<assembly>'
202 (their plurals are recognized as well). If multiple types are given
203 then they're combined with C<OR>.
213 Sven Schöling E<lt>s.schoeling@linet-services.deE<gt>,
214 Moritz Bunkus E<lt>m.bunkus@linet-services.deE<gt>