1 package SL::Controller::ClientConfig;
 
   4 use parent qw(SL::Controller::Base);
 
   6 use File::Copy::Recursive ();
 
   7 use List::Util qw(first);
 
  14 use SL::Helper::Flash;
 
  15 use SL::Locale::String qw(t8);
 
  18 __PACKAGE__->run_before('check_auth');
 
  20 use Rose::Object::MakeMethods::Generic (
 
  21   'scalar --get_set_init' => [ qw(defaults all_warehouses all_weightunits all_languages all_currencies all_templates posting_options payment_options accounting_options inventory_options profit_options accounts) ],
 
  25   my ($self, %params) = @_;
 
  27   $::form->{use_templates} = $self->defaults->templates ? 'existing' : 'new';
 
  32   my ($self, %params)      = @_;
 
  34   my $defaults             = delete($::form->{defaults}) || {};
 
  35   my $entered_currencies   = delete($::form->{currencies}) || [];
 
  36   my $original_currency_id = $self->defaults->currency_id;
 
  38   # undef several fields if an empty value has been selected.
 
  39   foreach (qw(warehouse_id bin_id warehouse_id_ignore_onhand bin_id_ignore_onhand)) {
 
  40     undef $defaults->{$_} if !$defaults->{$_};
 
  43   $self->defaults->assign_attributes(%{ $defaults });
 
  48   my (%new_currency_names);
 
  49   foreach my $existing_currency (@{ $self->all_currencies }) {
 
  50     my $new_name     = $existing_currency->name;
 
  51     my $new_currency = first { $_->{id} == $existing_currency->id } @{ $entered_currencies };
 
  52     $new_name        = $new_currency->{name} if $new_currency;
 
  55       $errors_idx{0} = t8('Currency names must not be empty.');
 
  56     } elsif ($new_currency_names{$new_name}) {
 
  57       $errors_idx{1} = t8('Currency names must be unique.');
 
  61       $new_currency_names{$new_name} = 1;
 
  62       $existing_currency->name($new_name);
 
  66   if ($::form->{new_currency} && $new_currency_names{ $::form->{new_currency} }) {
 
  67     $errors_idx{1} = t8('Currency names must be unique.');
 
  70   my @errors = map { $errors_idx{$_} } sort keys %errors_idx;
 
  73   $::form->{new_templates}        =~ s:/::g;
 
  74   $::form->{new_master_templates} =~ s:/::g;
 
  76   if (($::form->{use_templates} eq 'existing') && ($self->defaults->templates !~ m:^templates/[^/]+$:)) {
 
  77     push @errors, t8('You must select existing print templates or create a new set.');
 
  79   } elsif ($::form->{use_templates} eq 'new') {
 
  80     if (!$::form->{new_templates}) {
 
  81       push @errors, t8('You must enter a name for your new print templates.');
 
  82     } elsif (-d "templates/" . $::form->{new_templates}) {
 
  83       push @errors, t8('A directory with the name for the new print templates exists already.');
 
  84     } elsif (! -d "templates/print/" . $::form->{new_master_templates}) {
 
  85       push @errors, t8('The master templates where not found.');
 
  89   # Show form again if there were any errors. Nothing's been changed
 
  90   # yet in the database.
 
  92     flash('error', @errors);
 
  93     return $self->edit_form;
 
  96   # Save currencies. As the names must be unique we cannot simply save
 
  97   # them as they are -- the user might want to swap to names. So make
 
  98   # them unique first and assign the actual names in a second step.
 
  99   my %currency_names_by_id = map { ($_->id => $_->name) } @{ $self->all_currencies };
 
 100   $_->update_attributes(name => '__039519735__' . $_->{id})        for @{ $self->all_currencies };
 
 101   $_->update_attributes(name => $currency_names_by_id{ $_->{id} }) for @{ $self->all_currencies };
 
 103   # Create new currency if required
 
 105   if ($::form->{new_currency}) {
 
 106     $new_currency = SL::DB::Currency->new(name => $::form->{new_currency});
 
 110   # If the user wants the new currency to be the default then replace
 
 111   # the ID placeholder with the proper value. However, if no new
 
 112   # currency has been created then don't change the value at all.
 
 113   if (-1 == $self->defaults->currency_id) {
 
 114     $self->defaults->currency_id($new_currency ? $new_currency->id : $original_currency_id);
 
 117   # Create new templates if requested.
 
 118   if ($::form->{use_templates} eq 'new') {
 
 119     local $File::Copy::Recursive::SkipFlop = 1;
 
 120     File::Copy::Recursive::dircopy('templates/print/' . $::form->{new_master_templates}, 'templates/' . $::form->{new_templates});
 
 121     $self->defaults->templates('templates/' . $::form->{new_templates});
 
 124   # Finally save defaults.
 
 125   $self->defaults->save;
 
 127   flash_later('info', t8('Client Configuration saved!'));
 
 129   $self->redirect_to(action => 'edit');
 
 136 sub init_defaults        { SL::DB::Default->get                                           }
 
 137 sub init_all_warehouses  { SL::DB::Manager::Warehouse->get_all_sorted                     }
 
 138 sub init_all_languages   { SL::DB::Manager::Language->get_all_sorted                      }
 
 139 sub init_all_currencies  { SL::DB::Manager::Currency->get_all_sorted                      }
 
 140 sub init_all_weightunits { SL::DB::Manager::Unit->find_by(name => 'g')->convertible_units }
 
 141 sub init_all_templates   { +{ SL::Template->available_templates }                         }
 
 143 sub init_posting_options {
 
 144   [ { title => t8("never"),           value => 0           },
 
 145     { title => t8("every time"),      value => 1           },
 
 146     { title => t8("on the same day"), value => 2           }, ]
 
 149 sub init_payment_options {
 
 150   [ { title => t8("never"),           value => 0           },
 
 151     { title => t8("every time"),      value => 1           },
 
 152     { title => t8("on the same day"), value => 2           }, ]
 
 155 sub init_accounting_options {
 
 156   [ { title => t8("Accrual"),         value => "accrual"   },
 
 157     { title => t8("cash"),            value => "cash"      }, ]
 
 160 sub init_inventory_options {
 
 161   [ { title => t8("perpetual"),       value => "perpetual" },
 
 162     { title => t8("periodic"),        value => "periodic"  }, ]
 
 165 sub init_profit_options {
 
 166   [ { title => t8("balance"),         value => "balance"   },
 
 167     { title => t8("income"),          value => "income"    }, ]
 
 173   foreach my $chart (@{ SL::DB::Manager::Chart->get_all(where => [ link => { like => '%IC%' } ], sort_by => 'accno ASC') }) {
 
 176     foreach my $link (split m/:/, $chart->link) {
 
 177       my $key = lc($link =~ /cogs/ ? 'IC_expense' : $link =~ /sale/ ? 'IC_income' : $link);
 
 178       next if $added{$key};
 
 181       $accounts{$key} ||= [];
 
 182       push @{ $accounts{$key} }, $chart;
 
 186   $accounts{fx_gain} = SL::DB::Manager::Chart->get_all(where => [ category => 'I', charttype => 'A' ], sort_by => 'accno ASC');
 
 187   $accounts{fx_loss} = SL::DB::Manager::Chart->get_all(where => [ category => 'E', charttype => 'A' ], sort_by => 'accno ASC');
 
 188   $accounts{ar_paid} = SL::DB::Manager::Chart->get_all(where => [ link => { like => '%AR_paid%' }   ], sort_by => 'accno ASC');
 
 198   $::auth->assert('admin');
 
 208   $self->render('client_config/form', title => t8('Client Configuration'),
 
 209                 make_chart_title     => sub { $_[0]->accno . '--' . $_[0]->description },
 
 210                 make_templates_value => sub { 'templates/' . $_[0] },