X-Git-Url: http://wagnertech.de/git?a=blobdiff_plain;f=SL%2FDB%2FHelper%2FTransNumberGenerator.pm;h=d790b34847ef68a01d9622590d557c61b28272ea;hb=79ae46a435c65fa188981f6b5c61e0e15d32fb23;hp=3f58aba56c23d445a7048b0abf92be2326175a02;hpb=930e5ecb0d071e20e9fc57e6050859a0a7a5a98d;p=kivitendo-erp.git diff --git a/SL/DB/Helper/TransNumberGenerator.pm b/SL/DB/Helper/TransNumberGenerator.pm index 3f58aba56..d790b3484 100644 --- a/SL/DB/Helper/TransNumberGenerator.pm +++ b/SL/DB/Helper/TransNumberGenerator.pm @@ -9,24 +9,32 @@ use Carp; use List::Util qw(max); use SL::DB::Default; +use SL::PrefixedNumber; -my $oe_scoping = sub { +sub oe_scoping { SL::DB::Manager::Order->type_filter($_[0]); -}; +} -my $do_scoping = sub { +sub do_scoping { SL::DB::Manager::DeliveryOrder->type_filter($_[0]); -}; - -my %specs = ( ar => { number_column => 'invnumber', fill_holes_in_range => 1 }, - sales_quotation => { number_column => 'quonumber', number_range_column => 'sqnumber', scoping => $oe_scoping, }, - sales_order => { number_column => 'ordnumber', number_range_column => 'sonumber', scoping => $oe_scoping, }, - request_quotation => { number_column => 'quonumber', number_range_column => 'rfqnumber', scoping => $oe_scoping, }, - purchase_order => { number_column => 'ordnumber', number_range_column => 'ponumber', scoping => $oe_scoping, }, - sales_delivery_order => { number_column => 'donumber', number_range_column => 'sdonumber', scoping => $do_scoping, fill_holes_in_range => 1 }, - purchase_delivery_order => { number_column => 'donumber', number_range_column => 'pdonumber', scoping => $do_scoping, fill_holes_in_range => 1 }, - customer => { number_column => 'customernumber', number_range_column => 'customernumber', }, - vendor => { number_column => 'vendornumber', number_range_column => 'vendornumber', }, +} + +sub parts_scoping { + SL::DB::Manager::Part->type_filter($_[0]); +} + +my %specs = ( ar => { number_column => 'invnumber', fill_holes_in_range => 1 }, + sales_quotation => { number_column => 'quonumber', number_range_column => 'sqnumber', scoping => \&oe_scoping, }, + sales_order => { number_column => 'ordnumber', number_range_column => 'sonumber', scoping => \&oe_scoping, }, + request_quotation => { number_column => 'quonumber', number_range_column => 'rfqnumber', scoping => \&oe_scoping, }, + purchase_order => { number_column => 'ordnumber', number_range_column => 'ponumber', scoping => \&oe_scoping, }, + sales_delivery_order => { number_column => 'donumber', number_range_column => 'sdonumber', scoping => \&do_scoping, fill_holes_in_range => 1 }, + purchase_delivery_order => { number_column => 'donumber', number_range_column => 'pdonumber', scoping => \&do_scoping, fill_holes_in_range => 1 }, + customer => { number_column => 'customernumber', number_range_column => 'customernumber', }, + vendor => { number_column => 'vendornumber', number_range_column => 'vendornumber', }, + part => { number_column => 'partnumber', number_range_column => 'articlenumber', scoping => \&parts_scoping }, + service => { number_column => 'partnumber', number_range_column => 'servicenumber', scoping => \&parts_scoping }, + assembly => { number_column => 'partnumber', number_range_column => 'assemblynumber', scoping => \&parts_scoping }, ); sub get_next_trans_number { @@ -43,34 +51,23 @@ sub get_next_trans_number { return $number if $self->id && $number; - my $re = '^(.*?)(\d+)$'; - my %conditions = $scoping_conditions ? ( query => [ $scoping_conditions->($spec_type) ] ) : (); - my @numbers = map { $_->$number_column } @{ $self->_get_manager_class->get_all(%conditions) }; - my %numbers_in_use = map { ( $_ => 1 ) } @numbers; - @numbers = grep { $_ } map { my @matches = m/$re/; @matches ? $matches[-1] * 1 : undef } @numbers; - - my $defaults = SL::DB::Default->get; - my $number_range = $defaults->$number_range_column; - my @matches = $number_range =~ m/$re/; - my $prefix = (2 != scalar(@matches)) ? '' : $matches[ 0]; - my $ref_number = !@matches ? '1' : $matches[-1]; - my $min_places = length($ref_number); - - my $new_number = $fill_holes_in_range ? $ref_number : max($ref_number, @numbers); - my $new_number_full = undef; - - while (1) { - $new_number = $new_number + 1; - my $new_number_s = $new_number; - $new_number_s =~ s/\.\d+//g; - $new_number_full = $prefix . ('0' x max($min_places - length($new_number_s), 0)) . $new_number_s; - last if !$numbers_in_use{$new_number_full}; - } - - $defaults->update_attributes($number_range_column => $new_number_full) if $params{update_defaults}; - $self->$number_column($new_number_full) if $params{update_record}; - - return $new_number_full; + my %conditions = $scoping_conditions ? ( query => [ $scoping_conditions->($spec_type) ] ) : (); + my @numbers = map { $_->$number_column } @{ $self->_get_manager_class->get_all(%conditions) }; + my %numbers_in_use = map { ( $_ => 1 ) } @numbers; + + my $defaults = SL::DB::Default->get; + $number_range_column = 'articlenumber' if $number_range_column eq 'assemblynumber' and length($defaults->$number_range_column) < 1; + my $sequence = SL::PrefixedNumber->new(number => $defaults->$number_range_column); + + $sequence->set_to_max(@numbers) if !$fill_holes_in_range; + + my $new_number = $sequence->get_next; + $new_number = $sequence->get_next while $numbers_in_use{$new_number}; + + $defaults->update_attributes($number_range_column => $new_number) if $params{update_defaults}; + $self->$number_column($new_number) if $params{update_record}; + + return $new_number; } sub create_trans_number { @@ -149,7 +146,7 @@ and return a value. If it fails then it is due to exceptions. =item C -Calls and returns with the parameters +Calls and returns L with the parameters C and C. C<%params> is passed to it as well.