From: Moritz Bunkus
Date: Thu, 3 Apr 2014 12:26:47 +0000 (+0200)
Subject: Merge branch 'erweiterung-wiederkehrender-rechnungen'
X-Git-Tag: release-3.2.0beta~476
X-Git-Url: http://wagnertech.de/gitweb/gitweb.cgi/mfinanz.git/commitdiff_plain/648c1ad77f5c9c512b89783e8c8af13b29280c63?hp=0bbc7ffddd20e42362b9d959f1e57b444da44408
Merge branch 'erweiterung-wiederkehrender-rechnungen'
---
diff --git a/SL/BackgroundJob/CreatePeriodicInvoices.pm b/SL/BackgroundJob/CreatePeriodicInvoices.pm
index fa5d64935..dc740cc30 100644
--- a/SL/BackgroundJob/CreatePeriodicInvoices.pm
+++ b/SL/BackgroundJob/CreatePeriodicInvoices.pm
@@ -5,6 +5,7 @@ use strict;
use parent qw(SL::BackgroundJob::Base);
use Config::Std;
+use DateTime::Format::Strptime;
use English qw(-no_match_vars);
use SL::DB::AuthUser;
@@ -79,21 +80,22 @@ sub _generate_time_period_variables {
$::locale->text('January'), $::locale->text('February'), $::locale->text('March'), $::locale->text('April'), $::locale->text('May'), $::locale->text('June'),
$::locale->text('July'), $::locale->text('August'), $::locale->text('September'), $::locale->text('October'), $::locale->text('November'), $::locale->text('December'));
- my $vars = { current_quarter => $period_start_date->quarter,
- previous_quarter => $period_start_date->clone->subtract(months => 3)->quarter,
- next_quarter => $period_start_date->clone->add( months => 3)->quarter,
+ my $vars = {
+ current_quarter => [ $period_start_date->clone->truncate(to => 'month'), sub { $_[0]->quarter } ],
+ previous_quarter => [ $period_start_date->clone->truncate(to => 'month')->subtract(months => 3), sub { $_[0]->quarter } ],
+ next_quarter => [ $period_start_date->clone->truncate(to => 'month')->add( months => 3), sub { $_[0]->quarter } ],
- current_month => $period_start_date->month,
- previous_month => $period_start_date->clone->subtract(months => 1)->month,
- next_month => $period_start_date->clone->add( months => 1)->month,
+ current_month => [ $period_start_date->clone->truncate(to => 'month'), sub { $_[0]->month } ],
+ previous_month => [ $period_start_date->clone->truncate(to => 'month')->subtract(months => 1), sub { $_[0]->month } ],
+ next_month => [ $period_start_date->clone->truncate(to => 'month')->add( months => 1), sub { $_[0]->month } ],
- current_year => $period_start_date->year,
- previous_year => $period_start_date->year - 1,
- next_year => $period_start_date->year + 1,
+ current_year => [ $period_start_date->clone->truncate(to => 'year'), sub { $_[0]->year } ],
+ previous_year => [ $period_start_date->clone->truncate(to => 'year')->subtract(years => 1), sub { $_[0]->year } ],
+ next_year => [ $period_start_date->clone->truncate(to => 'year')->add( years => 1), sub { $_[0]->year } ],
- period_start_date => $::locale->format_date(\%::myconfig, $period_start_date),
- period_end_date => $::locale->format_date(\%::myconfig, $period_end_date),
- };
+ period_start_date => [ $period_start_date->clone->truncate(to => 'month'), sub { $::locale->format_date(\%::myconfig, $_[0]) } ],
+ period_end_date => [ $period_end_date ->clone->truncate(to => 'month'), sub { $::locale->format_date(\%::myconfig, $_[0]) } ],
+ };
map { $vars->{"${_}_month_long"} = $month_names[ $vars->{"${_}_month"} ] } qw(current previous next);
@@ -106,8 +108,23 @@ sub _replace_vars {
my $sub = shift;
my $str = $object->$sub;
- my ($key, $value);
- $str =~ s|<\%${key}\%>|$value|g while ($key, $value) = each %{ $vars };
+ $str =~ s{ <\% ([a-z0-9_]+) ( \s+ format \s*=\s* (.*?) \s* )? \%>}{
+ my ($key, $format) = ($1, $3);
+ if (!$vars->{$key}) {
+ '';
+
+ } elsif ($format) {
+ DateTime::Format::Strptime->new(
+ pattern => $format,
+ locale => 'de_DE',
+ time_zone => 'local',
+ )->format_datetime($vars->{$key}->[0]);
+
+ } else {
+ $vars->{$1}->[1]->($vars->{$1}->[0]);
+ }
+ }eigx;
+
$object->$sub($str);
}
@@ -171,27 +188,8 @@ sub _create_periodic_invoice {
}
sub _calculate_dates {
- my $config = shift;
-
- my $cur_date = $config->start_date;
- my $start_date = $config->get_previous_invoice_date || DateTime->new(year => 1970, month => 1, day => 1);
- my $end_date = $config->end_date || DateTime->new(year => 2100, month => 1, day => 1);
- my $tomorrow = DateTime->today_local->add(days => 1);
- my $period_len = $config->get_period_length;
-
- $end_date = $tomorrow if $end_date > $tomorrow;
-
- my @dates;
-
- while (1) {
- last if $cur_date >= $end_date;
-
- push @dates, $cur_date->clone if $cur_date > $start_date;
-
- $cur_date->add(months => $period_len);
- }
-
- return @dates;
+ my ($config) = @_;
+ return $config->calculate_invoice_dates(end_date => DateTime->today_local->add(days => 1));
}
sub _send_email {
diff --git a/SL/Controller/FinancialControllingReport.pm b/SL/Controller/FinancialControllingReport.pm
index bedc7623f..f0be9ada9 100644
--- a/SL/Controller/FinancialControllingReport.pm
+++ b/SL/Controller/FinancialControllingReport.pm
@@ -24,7 +24,6 @@ my %sort_columns = (
transaction_description => t8('Transaction description'),
globalprojectnumber => t8('Project'),
globalproject_type => t8('Project Type'),
- netamount => t8('Order amount'),
);
sub action_list {
@@ -51,13 +50,13 @@ sub prepare_report {
my $report = SL::ReportGenerator->new(\%::myconfig, $::form);
$self->{report} = $report;
- my @columns = qw(customer globalprojectnumber globalproject_type ordnumber netamount delivered_amount delivered_amount_p billed_amount billed_amount_p paid_amount paid_amount_p
+ my @columns = qw(customer globalprojectnumber globalproject_type ordnumber net_amount delivered_amount delivered_amount_p billed_amount billed_amount_p paid_amount paid_amount_p
billable_amount billable_amount_p other_amount);
- my @sortable = qw(ordnumber transdate customer netamount globalprojectnumber globalproject_type);
- $self->{number_columns} = [ qw(netamount billed_amount billed_amount_p delivered_amount delivered_amount_p paid_amount paid_amount_p other_amount billable_amount billable_amount_p) ];
+ my @sortable = qw(ordnumber transdate customer globalprojectnumber globalproject_type);
+ $self->{number_columns} = [ qw(net_amount billed_amount billed_amount_p delivered_amount delivered_amount_p paid_amount paid_amount_p other_amount billable_amount billable_amount_p) ];
my %column_defs = (
- netamount => { },
+ net_amount => { text => $::locale->text('Order amount') },
billed_amount => { text => $::locale->text('Billed amount') },
billed_amount_p => { text => $::locale->text('%') },
delivered_amount => { text => $::locale->text('Delivered amount') },
@@ -120,8 +119,16 @@ sub calculate_data {
$order->{other_amount} = $billed_amount - $order->{billed_amount};
$order->{billable_amount} = $order->{delivered_amount} - $order->{billed_amount};
+ if ($order->periodic_invoices_config) {
+ my @dates = $order->periodic_invoices_config->calculate_invoice_dates(past_dates => 1, end_date => $order->periodic_invoices_config->end_date || DateTime->today_local);
+ $order->{net_amount} = $order->netamount * scalar(@dates);
+
+ } else {
+ $order->{net_amount} = $order->netamount;
+ }
+
foreach (qw(delivered billed paid billable)) {
- $order->{"${_}_amount_p"} = $order->netamount * 1 ? $order->{"${_}_amount"} * 100 / $order->netamount : undef;
+ $order->{"${_}_amount_p"} = $order->{net_amount} * 1 ? $order->{"${_}_amount"} * 100 / $order->{net_amount} : undef;
}
}
}
@@ -219,8 +226,20 @@ sub init_models {
'globalproject.active' => 1,
'globalproject.valid' => 1,
]],
+ # keine WR
+ # oder aber (WR aktiv und (kein enddatum oder enddatum noch nicht überschritten))
+ or => [
+ 'periodic_invoices_config.id' => undef,
+ # and => [
+ 'periodic_invoices_config.active' => 1,
+ # or => [
+ # 'periodic_invoices_config.end_date' => undef,
+ # 'periodic_invoices_config.end_date' => { le => DateTime->today_local },
+ # ]
+ # ]
+ ],
],
- with_objects => [ 'customer', 'globalproject', 'globalproject.project_type' ],
+ with_objects => [ 'customer', 'globalproject', 'globalproject.project_type', 'periodic_invoices_config' ],
);
}
diff --git a/SL/Controller/FinancialOverview.pm b/SL/Controller/FinancialOverview.pm
index 38d2fc61b..49c04698c 100644
--- a/SL/Controller/FinancialOverview.pm
+++ b/SL/Controller/FinancialOverview.pm
@@ -5,14 +5,17 @@ use parent qw(SL::Controller::Base);
use List::MoreUtils qw(none);
+use SL::DB::Employee;
use SL::DB::Invoice;
use SL::DB::Order;
+use SL::DB::PeriodicInvoicesConfig;
use SL::DB::PurchaseInvoice;
use SL::Controller::Helper::ReportGenerator;
use SL::Locale::String;
use Rose::Object::MakeMethods::Generic (
- scalar => [ qw(report number_columns year current_year types objects data subtotals_per_quarter) ],
+ scalar => [ qw(report number_columns year current_year objects subtotals_per_quarter salesman_id) ],
+ 'scalar --get_set_init' => [ qw(employees types data) ],
);
__PACKAGE__->run_before(sub { $::auth->assert('report'); });
@@ -20,10 +23,11 @@ __PACKAGE__->run_before(sub { $::auth->assert('report'); });
sub action_list {
my ($self) = @_;
- $self->subtotals_per_quarter($::form->{subtotals_per_quarter});
+ $self->$_($::form->{$_}) for qw(subtotals_per_quarter salesman_id);
$self->get_objects;
- $self->calculate_data;
+ $self->calculate_one_time_data;
+ $self->calculate_periodic_invoices;
$self->prepare_report;
$self->list_data;
}
@@ -64,7 +68,7 @@ sub prepare_report {
);
$self->report->set_columns(%column_defs);
$self->report->set_column_order(@columns);
- $self->report->set_export_options(qw(list year subtotals_per_quarter));
+ $self->report->set_export_options(qw(list year subtotals_per_quarter salesman_id));
$self->report->set_options_from_form;
}
@@ -77,22 +81,26 @@ sub get_objects {
my $start = DateTime->new(year => $self->year, month => 1, day => 1);
my $end = DateTime->new(year => $self->year, month => 12, day => 31);
- my @date_filter = (and => [ transdate => { ge => $start }, transdate => { le => $end } ]);
+ my @f_date = (transdate => { ge => $start }, transdate => { le => $end });
+ my @f_salesman = $self->salesman_id ? (salesman_id => $self->salesman_id) : ();
$self->objects({
- sales_quotations => SL::DB::Manager::Order->get_all( where => [ and => [ @date_filter, SL::DB::Manager::Order->type_filter('sales_quotation') ]]),
- sales_orders => SL::DB::Manager::Order->get_all( where => [ and => [ @date_filter, SL::DB::Manager::Order->type_filter('sales_order') ]]),
- requests_for_quotation => SL::DB::Manager::Order->get_all( where => [ and => [ @date_filter, SL::DB::Manager::Order->type_filter('request_quotation') ]]),
- purchase_orders => SL::DB::Manager::Order->get_all( where => [ and => [ @date_filter, SL::DB::Manager::Order->type_filter('purchase_order') ]]),
- sales_invoices => SL::DB::Manager::Invoice->get_all( where => \@date_filter),
- purchase_invoices => SL::DB::Manager::PurchaseInvoice->get_all(where => \@date_filter),
+ sales_quotations => SL::DB::Manager::Order->get_all( where => [ and => [ @f_date, @f_salesman, SL::DB::Manager::Order->type_filter('sales_quotation') ]]),
+ sales_orders => SL::DB::Manager::Order->get_all( where => [ and => [ @f_date, @f_salesman, SL::DB::Manager::Order->type_filter('sales_order') ]], with_objects => [ qw(periodic_invoices_config) ]),
+ requests_for_quotation => SL::DB::Manager::Order->get_all( where => [ and => [ @f_date, @f_salesman, SL::DB::Manager::Order->type_filter('request_quotation') ]]),
+ purchase_orders => SL::DB::Manager::Order->get_all( where => [ and => [ @f_date, @f_salesman, SL::DB::Manager::Order->type_filter('purchase_order') ]]),
+ sales_invoices => SL::DB::Manager::Invoice->get_all( where => [ and => [ @f_date, @f_salesman, ]]),
+ purchase_invoices => SL::DB::Manager::PurchaseInvoice->get_all(where => [ and => \@f_date ]),
+ periodic_invoices_cfg => SL::DB::Manager::PeriodicInvoicesConfig->get_all(where => [ active => 1 ]),
});
+
+ $self->objects->{sales_orders} = [ grep { !$_->periodic_invoices_config || !$_->periodic_invoices_config->active } @{ $self->objects->{sales_orders} } ];
}
-sub calculate_data {
- my ($self) = @_;
+sub init_types { [ qw(sales_quotations sales_orders sales_invoices requests_for_quotation purchase_orders purchase_invoices) ] }
- $self->types([ qw(sales_quotations sales_orders sales_invoices requests_for_quotation purchase_orders purchase_invoices) ]);
+sub init_data {
+ my ($self) = @_;
my %data = (
year => [ ($self->year) x 12 ],
@@ -107,18 +115,47 @@ sub calculate_data {
} @{ $self->types },
);
- foreach my $type (keys %{ $self->objects }) {
+ return \%data;
+}
+
+sub calculate_one_time_data {
+ my ($self) = @_;
+
+ foreach my $type (@{ $self->types }) {
foreach my $object (@{ $self->objects->{ $type } }) {
my $month = $object->transdate->month - 1;
- my $tdata = $data{$type};
+ my $tdata = $self->data->{$type};
$tdata->{months}->[$month] += $object->netamount;
$tdata->{quarters}->[int($month / 3)] += $object->netamount;
$tdata->{year} += $object->netamount;
}
}
+}
+
+sub calculate_periodic_invoices {
+ my ($self) = @_;
- $self->data(\%data);
+ my $start_date = DateTime->new(year => $self->year, month => 1, day => 1, time_zone => $::locale->get_local_time_zone);
+ my $end_date = DateTime->new(year => $self->year, month => 12, day => 31, time_zone => $::locale->get_local_time_zone);
+
+ $self->calculate_one_periodic_invoice(config => $_, start_date => $start_date, end_date => $end_date) for @{ $self->objects->{periodic_invoices_cfg} };
+}
+
+sub calculate_one_periodic_invoice {
+ my ($self, %params) = @_;
+
+ my @dates = $params{config}->calculate_invoice_dates(start_date => $params{start_date}, end_date => $params{end_date}, past_dates => 1);
+ my $first_date = $dates[0];
+
+ return if !$first_date;
+
+ my $net = $params{config}->order->netamount * scalar(@dates);
+ my $sord = $self->data->{sales_orders};
+
+ $sord->{months }->[ $first_date->month - 1 ] += $net;
+ $sord->{quarters}->[ $first_date->quarter - 1 ] += $net;
+ $sord->{year} += $net;
}
sub list_data {
@@ -162,4 +199,6 @@ sub list_data {
return $self->report->generate_with_headers;
}
+sub init_employees { SL::DB::Manager::Employee->get_all_sorted }
+
1;
diff --git a/SL/DB/MetaSetup/PeriodicInvoicesConfig.pm b/SL/DB/MetaSetup/PeriodicInvoicesConfig.pm
index e914a6449..eae8481d4 100644
--- a/SL/DB/MetaSetup/PeriodicInvoicesConfig.pm
+++ b/SL/DB/MetaSetup/PeriodicInvoicesConfig.pm
@@ -14,6 +14,7 @@ __PACKAGE__->meta->columns(
copies => { type => 'integer' },
end_date => { type => 'date' },
extend_automatically_by => { type => 'integer' },
+ first_billing_date => { type => 'date' },
id => { type => 'integer', not_null => 1, sequence => 'id' },
oe_id => { type => 'integer', not_null => 1 },
periodicity => { type => 'varchar', length => 10, not_null => 1 },
diff --git a/SL/DB/PeriodicInvoicesConfig.pm b/SL/DB/PeriodicInvoicesConfig.pm
index 7480c3a5f..074d4e317 100644
--- a/SL/DB/PeriodicInvoicesConfig.pm
+++ b/SL/DB/PeriodicInvoicesConfig.pm
@@ -4,6 +4,8 @@ use strict;
use SL::DB::MetaSetup::PeriodicInvoicesConfig;
+use List::Util qw(max min);
+
__PACKAGE__->meta->initialize;
# Creates get_all, get_all_count, get_all_iterator, delete_all and update_all.
@@ -58,20 +60,42 @@ sub handle_automatic_extension {
return $end_date;
}
-sub get_previous_invoice_date {
+sub get_previous_billed_period_start_date {
my $self = shift;
my $query = <dbh->selectrow_array($query, undef, $self->id);
+ my ($date) = $self->dbh->selectrow_array($query, undef, $self->id);
+
+ return undef unless $date;
+ return ref $date ? $date : $self->db->parse_date($date);
+}
+
+sub calculate_invoice_dates {
+ my ($self, %params) = @_;
+
+ my $period_len = $self->get_period_length;
+ my $cur_date = $self->first_billing_date || $self->start_date;
+ my $end_date = $self->end_date || DateTime->today_local->add(years => 10);
+ my $start_date = $params{past_dates} ? undef : $self->get_previous_billed_period_start_date;
+ $start_date = $start_date ? $start_date->add(days => 1) : $cur_date->clone;
+
+ $start_date = max($start_date, $params{start_date}) if $params{start_date};
+ $end_date = min($end_date, $params{end_date}) if $params{end_date};
+
+ my @dates;
+
+ while ($cur_date <= $end_date) {
+ push @dates, $cur_date->clone if $cur_date >= $start_date;
+
+ $cur_date->add(months => $period_len);
+ }
- return undef unless $max_transdate;
- return ref $max_transdate ? $max_transdate : $self->db->parse_date($max_transdate);
+ return @dates;
}
1;
diff --git a/SL/InstallationCheck.pm b/SL/InstallationCheck.pm
index a3aa509ac..499ef673e 100644
--- a/SL/InstallationCheck.pm
+++ b/SL/InstallationCheck.pm
@@ -21,6 +21,7 @@ BEGIN {
{ name => "Clone", url => "http://search.cpan.org/~rdf/", debian => 'libclone-perl' },
{ name => "Config::Std", url => "http://search.cpan.org/~dconway/", debian => 'libconfig-std-perl' },
{ name => "DateTime", url => "http://search.cpan.org/~drolsky/", debian => 'libdatetime-perl' },
+ { name => "DateTime::Format::Strptime", url => "http://search.cpan.org/~drolsky/", debian => 'libdatetime-format-strptime-perl' },
{ name => "DBI", version => '1.50', url => "http://search.cpan.org/~timb/", debian => 'libdbi-perl' },
{ name => "DBD::Pg", version => '1.49', url => "http://search.cpan.org/~dbdpg/", debian => 'libdbd-pg-perl' },
{ name => "Email::Address", url => "http://search.cpan.org/~rjbs/", debian => 'libemail-address-perl' },
diff --git a/SL/OE.pm b/SL/OE.pm
index 13e735a38..266945ce3 100644
--- a/SL/OE.pm
+++ b/SL/OE.pm
@@ -662,7 +662,7 @@ sub load_periodic_invoice_config {
my $config_obj = SL::DB::Manager::PeriodicInvoicesConfig->find_by(oe_id => $form->{id});
if ($config_obj) {
- my $config = { map { $_ => $config_obj->$_ } qw(active terminated periodicity start_date_as_date end_date_as_date extend_automatically_by ar_chart_id
+ my $config = { map { $_ => $config_obj->$_ } qw(active terminated periodicity start_date_as_date end_date_as_date first_billing_date_as_date extend_automatically_by ar_chart_id
print printer_id copies) };
$form->{periodic_invoices_config} = YAML::Dump($config);
}
diff --git a/bin/mozilla/oe.pl b/bin/mozilla/oe.pl
index a2d7175bc..7bb96a18e 100644
--- a/bin/mozilla/oe.pl
+++ b/bin/mozilla/oe.pl
@@ -35,6 +35,7 @@
use Carp;
use POSIX qw(strftime);
+use SL::DB::Order;
use SL::DO;
use SL::FU;
use SL::OE;
@@ -312,6 +313,18 @@ sub form_header {
# Container for template variables. Unfortunately this has to be
# visible in form_footer too, so package local level and not my here.
%TMPL_VAR = ();
+ if ($form->{id}) {
+ my $obj = SL::DB::Order->new(id => $form->{id})->load;
+ $TMPL_VAR{warn_save_active_periodic_invoice} =
+ $obj->is_type('sales_order')
+ && $obj->periodic_invoices_config
+ && $obj->periodic_invoices_config->active
+ && ( !$obj->periodic_invoices_config->end_date
+ || ($obj->periodic_invoices_config->end_date > DateTime->today_local))
+ && $obj->periodic_invoices_config->get_previous_billed_period_start_date;
+
+ $TMPL_VAR{oe_obj} = $obj;
+ }
$form->{defaultcurrency} = $form->get_default_currency(\%myconfig);
@@ -1930,13 +1943,13 @@ sub edit_periodic_invoices_config {
if ('HASH' ne ref $config) {
$config = { periodicity => 'y',
- start_date_as_date => $::form->{transdate},
+ start_date_as_date => $::form->{transdate} || $::form->current_date,
extend_automatically_by => 12,
active => 1,
};
}
- $config->{periodicity} = 'm' if none { $_ eq $config->{periodicity} } qw(m q y);
+ $config->{periodicity} = 'm' if none { $_ eq $config->{periodicity} } qw(m q b y);
$::form->get_lists(printers => "ALL_PRINTERS",
charts => { key => 'ALL_CHARTS',
@@ -1962,9 +1975,10 @@ sub save_periodic_invoices_config {
my $config = { active => $::form->{active} ? 1 : 0,
terminated => $::form->{terminated} ? 1 : 0,
- periodicity => (any { $_ eq $::form->{periodicity} } qw(m q y)) ? $::form->{periodicity} : 'm',
+ periodicity => (any { $_ eq $::form->{periodicity} } qw(m q b y)) ? $::form->{periodicity} : 'm',
start_date_as_date => $::form->{start_date_as_date},
end_date_as_date => $::form->{end_date_as_date},
+ first_billing_date_as_date => $::form->{first_billing_date_as_date},
print => $::form->{print} ? 1 : 0,
printer_id => $::form->{print} ? $::form->{printer_id} * 1 : undef,
copies => $::form->{copies} * 1 ? $::form->{copies} : 1,
diff --git a/doc/dokumentation.xml b/doc/dokumentation.xml
index aeeb56fce..5e86d1a0d 100644
--- a/doc/dokumentation.xml
+++ b/doc/dokumentation.xml
@@ -2232,6 +2232,12 @@ ln -s $(pwd)/kivitendo-task-server.service /etc/systemd/system/
Abrechnungszeitraum explizit auszuweisen. Eine Variable hat dabei die Syntax <%variablenname%>.
+
+ Sofern es sich um eine Datumsvariable handelt, kann das Ausgabeformat weiter bestimmt werden, indem an den Variablennamen
+ Formatoptionen angehängt werden. Die Syntax sieht dabei wie folgt aus: <%variablenname
+ FORMAT=Formatinformation%>. Die zur verfügung stehenden Formatinformationen werden unten genauer beschrieben.
+
+
Diese Variablen werden in den folgenden Elementen des Auftrags ersetzt:
@@ -2297,6 +2303,172 @@ ln -s $(pwd)/kivitendo-task-server.service /etc/systemd/system/
+
+
+ Die invidiuellen Formatinformationen bestehen aus Paaren von Prozentzeichen und einem Buchstaben, welche beide zusammen durch den
+ dazugehörigen Wert ersetzt werden. So wird z.B. %Y durch das viertstellige Jahr ersetzt. Alle möglichen
+ Platzhalter sind:
+
+
+
+
+
+ %a
+
+
+ Der abgekürzte Wochentagsname.
+
+
+
+
+ %A
+
+
+ Der ausgeschriebene Wochentagsname.
+
+
+
+
+ %b
+
+
+ Der abgekürzte Monatsname.
+
+
+
+
+ %B
+
+
+ Der ausgeschriebene Monatsname.
+
+
+
+
+ %C
+
+
+ Das Jahrhundert (Jahr/100) als eine zweistellige Zahl.
+
+
+
+
+ %d
+
+
+ Der Monatstag als Zahl zwischen 01 und 31.
+
+
+
+
+ %D
+
+
+ Entspricht %m/%d/%y (amerikanisches Datumsformat).
+
+
+
+
+ %e
+
+
+ Wie %d (Monatstag als Zahl zwischen 1 und 31), allerdings werden führende Nullen durch Leerzeichen ersetzt.
+
+
+
+
+ %F
+
+
+ Entspricht %Y-%m-%d (das ISO-8601-Datumsformat).
+
+
+
+
+ %j
+
+
+ Der Tag im Jahr als Zahl zwischen 001 und 366 inklusive.
+
+
+
+
+ %m
+
+
+ Der Monat als Zahl zwischen 01 und 12 inklusive.
+
+
+
+
+ %u
+
+
+ Der Wochentag als Zahl zwischen 1 und 7 inklusive, wobei die 1 dem Montag entspricht.
+
+
+
+
+ %U
+
+
+ Die Wochennummer als Zahl zwischen 00 und 53 inklusive, wobei der erste Sonntag im Jahr das Startdatum von Woche 01 ist.
+
+
+
+
+ %V
+
+
+ Die ISO-8601:1988-Wochennummer als Zahl zwischen 01 und 53 inklusive, wobei Woche 01 die erste Woche, von der mindestens vier Tage im Jahr liegen; Montag ist erster Tag der Woche.
+
+
+
+
+ %w
+
+
+ Der Wochentag als Zahl zwischen 0 und 6 inklusive, wobei die 0 dem Sonntag entspricht.
+
+
+
+
+ %W
+
+
+ Die Wochennummer als Zahl zwischen 00 und 53 inklusive, wobei der erste Montag im Jahr das Startdatum von Woche 01 ist.
+
+
+
+
+ %y
+
+
+ Das Jahr als zweistellige Zahl zwischen 00 und 99 inklusive.
+
+
+
+
+ %Y
+
+
+ Das Jahr als vierstellige Zahl.
+
+
+
+
+ %%
+
+
+ Das Prozentzeichen selber.
+
+
+
+
+
+ Anwendungsbeispiel für die Ausgabe, von welchem Monat und Jahr bis zu welchem Monat und Jahr die aktuelle Abrechnungsperiode
+ dauert: Abrechnungszeitrum: <%period_start_date FORMAT=%m/%Y%> bis <%period_end_date FORMAT=%m/%Y%>
+
diff --git a/doc/html/ch03.html b/doc/html/ch03.html
index 8b59e52b0..5e91f1a6b 100644
--- a/doc/html/ch03.html
+++ b/doc/html/ch03.html
@@ -42,6 +42,10 @@
Um die erzeugten Rechnungen individualisieren zu können, werden beim Umwandeln des Auftrags in eine Rechnung einige speziell
formatierte Variablen durch für die jeweils aktuelle Abrechnungsperiode gültigen Werte ersetzt. Damit ist es möglich, z.B. den
Abrechnungszeitraum explizit auszuweisen. Eine Variable hat dabei die Syntax <%variablenname%>.
+
+ Sofern es sich um eine Datumsvariable handelt, kann das Ausgabeformat weiter bestimmt werden, indem an den Variablennamen
+ Formatoptionen angehängt werden. Die Syntax sieht dabei wie folgt aus: <%variablenname
+ FORMAT=Formatinformation%>. Die zur verfügung stehenden Formatinformationen werden unten genauer beschrieben.
Diese Variablen werden in den folgenden Elementen des Auftrags ersetzt:
Bemerkungen
Interne Bemerkungen
Vorgangsbezeichnung
In den Beschreibungs- und Langtextfeldern aller Positionen
Die zur Verfügung stehenden Variablen sind die Folgenden:
@@ -65,7 +69,53 @@
Formatiertes Datum des ersten und letzten Tages im Abrechnungszeitraum (z.B. bei quartalsweiser Abrechnung und im ersten
Quartal von 2013 wären dies der 01.01.2013 und 31.03.2013).
-
3.1.4. Auflisten
Unter Verkauf->Berichte->Aufträge finden sich zwei neue
+
+ Die invidiuellen Formatinformationen bestehen aus Paaren von Prozentzeichen und einem Buchstaben, welche beide zusammen durch den
+ dazugehörigen Wert ersetzt werden. So wird z.B. %Y durch das viertstellige Jahr ersetzt. Alle möglichen
+ Platzhalter sind:
+
+ %a
+
Der abgekürzte Wochentagsname.
+ %A
+
Der ausgeschriebene Wochentagsname.
+ %b
+
Der abgekürzte Monatsname.
+ %B
+
Der ausgeschriebene Monatsname.
+ %C
+
Das Jahrhundert (Jahr/100) als eine zweistellige Zahl.
Wie %d (Monatstag als Zahl zwischen 1 und 31), allerdings werden führende Nullen durch Leerzeichen ersetzt.
+ %F
+
Entspricht %Y-%m-%d (das ISO-8601-Datumsformat).
+ %j
+
Der Tag im Jahr als Zahl zwischen 001 und 366 inklusive.
+ %m
+
Der Monat als Zahl zwischen 01 und 12 inklusive.
+ %u
+
Der Wochentag als Zahl zwischen 1 und 7 inklusive, wobei die 1 dem Montag entspricht.
+ %U
+
Die Wochennummer als Zahl zwischen 00 und 53 inklusive, wobei der erste Sonntag im Jahr das Startdatum von Woche 01 ist.
+ %V
+
Die ISO-8601:1988-Wochennummer als Zahl zwischen 01 und 53 inklusive, wobei Woche 01 die erste Woche, von der mindestens vier Tage im Jahr liegen; Montag ist erster Tag der Woche.
+ %w
+
Der Wochentag als Zahl zwischen 0 und 6 inklusive, wobei die 0 dem Sonntag entspricht.
+ %W
+
Die Wochennummer als Zahl zwischen 00 und 53 inklusive, wobei der erste Montag im Jahr das Startdatum von Woche 01 ist.
+ %y
+
Das Jahr als zweistellige Zahl zwischen 00 und 99 inklusive.
+ %Y
+
Das Jahr als vierstellige Zahl.
+ %%
+
Das Prozentzeichen selber.
+ Anwendungsbeispiel für die Ausgabe, von welchem Monat und Jahr bis zu welchem Monat und Jahr die aktuelle Abrechnungsperiode
+ dauert: Abrechnungszeitrum: <%period_start_date FORMAT=%m/%Y%> bis <%period_end_date FORMAT=%m/%Y%>
+
+
3.1.4. Auflisten
Unter Verkauf->Berichte->Aufträge finden sich zwei neue
Checkboxen, "Wiederkehrende Rechnungen aktiv" und "Wiederkehrende
Rechnungen inaktiv", mit denen man sich einen Ãberglick über die
wiederkehrenden Rechnungen verschaffen kann.
3.1.5. Erzeugung der eigentlichen Rechnungen
Die zeitliche und periodische Ãberprüfung, ob eine
diff --git a/doc/html/ch03s02.html b/doc/html/ch03s02.html
index 7bbd9e572..a3381b1e6 100644
--- a/doc/html/ch03s02.html
+++ b/doc/html/ch03s02.html
@@ -423,6 +423,9 @@
ordnumber_oe
Auftragsnummer des Originalauftrags, wenn die Rechnung
aus einem Sammelauftrag erstellt wurde
+ donumber_do
+
Lieferscheinnummer desjenigen Lieferscheins, aus dem die Position stammt, sofern die Rechnung aus einem oder
+ mehreren Lieferscheinen erstellt wurde
p_discount
Rabatt in Prozent
partnotes
@@ -566,7 +569,7 @@
invdate
Rechnungsdatum
invnumber
-
Rechnungsnummer
3.2.10. Variablen in anderen Vorlagen
3.2.10.1. Einführung
Die Variablen in anderen Vorlagen sind ähnlich wie in der
+
Rechnungsnummer
3.2.10. Variablen in anderen Vorlagen
3.2.10.1. Einführung
Die Variablen in anderen Vorlagen sind ähnlich wie in der
Rechnung. Allerdings heiÃen die Variablen, die mit
inv beginnen, jetzt anders. Bei den Angeboten
fangen sie mit quo für "quotation" an:
diff --git a/doc/html/ch04.html b/doc/html/ch04.html
index 55a00d7b8..72b87bbbf 100644
--- a/doc/html/ch04.html
+++ b/doc/html/ch04.html
@@ -1,6 +1,6 @@
Globale Variablen liegen in einem speziellen namespace namens
"main", der von überall erreichbar ist. Darüber hinaus sind bareword
globs global und die meisten speziellen Variablen sind...
speziell.
Daraus ergeben sich folgende Formen:
@@ -25,7 +25,7 @@
$PACKAGE::form.
local $form
Alle Ãnderungen an $form werden am Ende
- des scopes zurückgesetzt
4.1.2. Warum sind globale Variablen ein Problem?
Das erste Problem ist FCGIâ¢.
+ des scopes zurückgesetzt
4.1.2. Warum sind globale Variablen ein Problem?
Das erste Problem ist FCGIâ¢.
SQL-Ledger⢠hat fast alles im globalen
namespace abgelegt, und erwartet, dass es da auch wiederzufinden ist.
Unter FCGI⢠müssen diese Sachen aber wieder
@@ -39,7 +39,7 @@
dies hat, seit der Einführung, u.a. schon so manche langwierige
Bug-Suche verkürzt. Da globale Variablen aber implizit mit Package
angegeben werden, werden die nicht geprüft, und somit kann sich
- schnell ein Tippfehler einschleichen.
4.1.3. Kanonische globale Variablen
Um dieses Problem im Griff zu halten gibt es einige wenige
+ schnell ein Tippfehler einschleichen.
4.1.3. Kanonische globale Variablen
Um dieses Problem im Griff zu halten gibt es einige wenige
globale Variablen, die kanonisch sind, d.h. sie haben bestimmte
vorgegebenen Eigenschaften, und alles andere sollte anderweitig
umhergereicht werden.
Diese Variablen sind im Moment die folgenden neun:
@@ -62,7 +62,7 @@
$::request
Damit diese nicht erneut als Müllhalde missbraucht werden, im
Folgenden eine kurze Erläuterung der bestimmten vorgegebenen
- Eigenschaften (Konventionen):
4.1.3.1. $::form
Ist ein Objekt der Klasse
+ Eigenschaften (Konventionen):
4.1.3.1. $::form
Ist ein Objekt der Klasse
"Form"
Wird nach jedem Request gelöscht
Muss auch in Tests und Konsolenscripts vorhanden
sein.
Enthält am Anfang eines Requests die Requestparameter vom
User
Kann zwar intern über Requestgrenzen ein Datenbankhandle
@@ -110,7 +110,7 @@
push @{ $form->{TEMPLATE_ARRAYS}{number} }, $form->{"partnumber_$i"};
push @{ $form->{TEMPLATE_ARRAYS}{description} }, $form->{"description_$i"};
# ...
-}
4.1.3.2. %::myconfig
Das einzige Hash unter den globalen Variablen
Wird spätestens benötigt wenn auf die Datenbank
+}
4.1.3.2. %::myconfig
Das einzige Hash unter den globalen Variablen
Wird spätestens benötigt wenn auf die Datenbank
zugegriffen wird
Wird bei jedem Request neu erstellt.
Enthält die Userdaten des aktuellen Logins
Sollte nicht ohne Filterung irgendwo gedumpt werden oder
extern serialisiert werden, weil da auch der Datenbankzugriff
für diesen user drinsteht.
Enthält unter anderem Listenbegrenzung vclimit,
@@ -122,10 +122,10 @@
überwiegend die Daten, die sich unter Programm
-> Einstellungen befinden, bzw. die
Informationen über den Benutzer die über die
- Administrator-Schnittstelle eingegeben wurden.
4.1.3.3. $::locale
Objekt der Klasse "Locale"
Wird pro Request erstellt
Muss auch für Tests und Scripte immer verfügbar
+ Administrator-Schnittstelle eingegeben wurden.
4.1.3.3. $::locale
Objekt der Klasse "Locale"
Wird pro Request erstellt
Muss auch für Tests und Scripte immer verfügbar
sein.
Lokalisierung für den aktuellen User. Alle Ãbersetzungen,
- Zahlen- und Datumsformatierungen laufen über dieses Objekt.
4.1.3.4. $::lxdebug
Objekt der Klasse "LXDebug"
Wird global gecached
Muss immer verfügbar sein, in nahezu allen
+ Zahlen- und Datumsformatierungen laufen über dieses Objekt.
4.1.3.4. $::lxdebug
Objekt der Klasse "LXDebug"
Wird global gecached
Muss immer verfügbar sein, in nahezu allen
Funktionen
$::lxdebug stellt Debuggingfunktionen
bereit, wie "enter_sub" und
@@ -135,14 +135,14 @@
"message" und "dump" mit
denen man flott Informationen ins Log (tmp/kivitendo-debug.log)
packen kann.
Beispielsweise so:
$main::lxdebug->message(0, 'Meine Konfig:' . Dumper (%::myconfig));
-$main::lxdebug->message(0, 'Wer bin ich? Kunde oder Lieferant:' . $form->{vc});
4.1.3.5. $::auth
Objekt der Klasse "SL::Auth"
Wird global gecached
Hat eine permanente DB Verbindung zur Authdatenbank
Wird nach jedem Request resettet.
+$main::lxdebug->message(0, 'Wer bin ich? Kunde oder Lieferant:' . $form->{vc});
4.1.3.5. $::auth
Objekt der Klasse "SL::Auth"
Wird global gecached
Hat eine permanente DB Verbindung zur Authdatenbank
Wird nach jedem Request resettet.
$::auth stellt Funktionen bereit um die
Rechte des aktuellen Users abzufragen. Obwohl diese Informationen
vom aktuellen User abhängen wird das Objekt aus
Geschwindigkeitsgründen nur einmal angelegt und dann nach jedem
Request kurz resettet.
Dieses Objekt kapselt auch den gerade aktiven Mandanten. Dessen Einstellungen können über
$::auth->client abgefragt werden; Rückgabewert ist ein Hash mit den Werten aus der Tabelle
- auth.clients.
4.1.3.6. $::lx_office_conf
Objekt der Klasse
+ auth.clients.
4.1.3.6. $::lx_office_conf
Objekt der Klasse
"SL::LxOfficeConf"
Global gecached
Repräsentation der
config/kivitendo.conf[.default]-Dateien
Globale Konfiguration. Configdateien werden zum Start gelesen
und danach nicht mehr angefasst. Es ist derzeit nicht geplant, dass
@@ -152,16 +152,16 @@ $main::lxdebug->message(0, 'Wer bin ich? Kunde oder Lieferant:' . $form->{
file_name = /tmp/kivitendo-debug.log
ist der Key file im Programm als
$::lx_office_conf->{debug}{file}
erreichbar.
Warnung
Zugriff auf die Konfiguration erfolgt im Moment über
- Hashkeys, sind also nicht gegen Tippfehler abgesichert.
4.1.3.7. $::instance_conf
Objekt der Klasse
+ Hashkeys, sind also nicht gegen Tippfehler abgesichert.
4.1.3.7. $::instance_conf
Objekt der Klasse
"SL::InstanceConfiguration"
wird pro Request neu erstellt
Funktioniert wie $::lx_office_conf,
speichert aber Daten die von der Instanz abhängig sind. Eine Instanz
ist hier eine Mandantendatenbank. Beispielsweise überprüft
- ob die berüchtigte Bestandsmethode zur Anwendung kommt.
4.1.3.8. $::dispatcher
Objekt der Klasse
+ ob die berüchtigte Bestandsmethode zur Anwendung kommt.
4.1.3.8. $::dispatcher
Objekt der Klasse
"SL::Dispatcher"
wird pro Serverprozess erstellt.
enthält Informationen über die technische Verbindung zum
Server
Der dritte Punkt ist auch der einzige Grund warum das Objekt
global gespeichert wird. Wird vermutlich irgendwann in einem anderen
- Objekt untergebracht.
4.1.3.9. $::request
Hashref (evtl später Objekt)
Wird pro Request neu initialisiert.
Keine Unterstruktur garantiert.
+ Objekt untergebracht.
4.1.3.9. $::request
Hashref (evtl später Objekt)
Wird pro Request neu initialisiert.
Keine Unterstruktur garantiert.
$::request ist ein generischer Platz um
Daten "für den aktuellen Request" abzulegen. Sollte nicht für action
at a distance benutzt werden, sondern um lokales memoizing zu
@@ -174,20 +174,20 @@ file_name = /tmp/kivitendo-debug.log
ist der Key f
$::request
Muss ich von anderen Teilen des Programms lesend drauf
zugreifen? Dann $::request, aber Zugriff über
- Wrappermethode
4.1.4. Ehemalige globale Variablen
Die folgenden Variablen waren einmal im Programm, und wurden
- entfernt.
4.1.4.1. $::cgi
war nötig, weil cookie Methoden nicht als
+ Wrappermethode
4.1.4. Ehemalige globale Variablen
Die folgenden Variablen waren einmal im Programm, und wurden
+ entfernt.
4.1.4.1. $::cgi
war nötig, weil cookie Methoden nicht als
Klassenfunktionen funktionieren
Aufruf als Klasse erzeugt Dummyobjekt was im
Klassennamespace gehalten wird und über Requestgrenzen
leaked
liegt jetzt unter
$::request->{cgi}
-
4.1.4.2. $::all_units
war nötig, weil einige Funktionen in Schleifen zum Teil
+
4.1.4.2. $::all_units
war nötig, weil einige Funktionen in Schleifen zum Teil
ein paar hundert mal pro Request eine Liste der Einheiten
brauchen, und de als Parameter durch einen Riesenstack von
Funktionen geschleift werden müssten.
Liegt jetzt unter
$::request->{cache}{all_units}
Wird nur in
AM->retrieve_all_units() gesetzt oder
- gelesen.
4.1.4.3. %::called_subs
wurde benutzt um callsub deep recursions
+ gelesen.
4.1.4.3. %::called_subs
wurde benutzt um callsub deep recursions
abzufangen.
Wurde entfernt, weil callsub nur einen Bruchteil der
möglichen Rekursioenen darstellt, und da nie welche
auftreten.
komplette recursion protection wurde entfernt.
\ No newline at end of file
diff --git a/doc/html/index.html b/doc/html/index.html
index 7827645f6..8fa725c49 100644
--- a/doc/html/index.html
+++ b/doc/html/index.html
@@ -3,7 +3,7 @@
kivitendo 3.1.0: Installation, Konfiguration, Entwicklung
kivitendo 3.1.0: Installation, Konfiguration, Entwicklung