Zeiterfassung: Im Bericht Einträge als gebucht markieren können, …
authorBernd Bleßmann <bernd@kivitendo-premium.de>
Tue, 18 May 2021 11:28:43 +0000 (13:28 +0200)
committerBernd Bleßmann <bernd@kivitendo-premium.de>
Tue, 18 May 2021 12:15:13 +0000 (14:15 +0200)
… wenn man das Recht hat, alle zu ändern.

SL/Controller/TimeRecording.pm
locale/de/all
locale/en/all
templates/webpages/time_recording/report_bottom.html
templates/webpages/time_recording/report_top.html

index 628c8ae..38771d1 100644 (file)
@@ -22,6 +22,7 @@ use SL::Helper::Flash qw(flash);
 use SL::Helper::Number qw(_round_number _parse_number _round_total);
 use SL::Helper::UserPreferences::TimeRecording;
 use SL::Locale::String qw(t8);
+use SL::Presenter::Tag qw(checkbox_tag);
 use SL::ReportGenerator;
 
 use Rose::Object::MakeMethods::Generic
@@ -33,7 +34,9 @@ use Rose::Object::MakeMethods::Generic
 
 # safety
 __PACKAGE__->run_before('check_auth');
-__PACKAGE__->run_before('check_auth_edit', only => [ qw(edit save delete) ]);
+__PACKAGE__->run_before('check_auth_edit',     only => [ qw(edit save delete) ]);
+__PACKAGE__->run_before('check_auth_edit_all', only => [ qw(mark_as_booked) ]);
+
 
 my %sort_columns = (
   date         => t8('Date'),
@@ -83,6 +86,17 @@ sub action_list {
   $self->report_generator_list_objects(report => $self->{report}, objects => $objects);
 }
 
+sub action_mark_as_booked {
+  my ($self) = @_;
+
+  if (scalar @{ $::form->{ids} }) {
+    my $trs = SL::DB::Manager::TimeRecording->get_all(query => [id => $::form->{ids}]);
+    $_->update_attributes(booked => 1) for @$trs;
+  }
+
+  $self->redirect_to(safe_callback());
+}
+
 sub action_edit {
   my ($self) = @_;
 
@@ -279,15 +293,24 @@ sub check_auth_edit {
   }
 }
 
+sub check_auth_edit_all {
+  my ($self) = @_;
+
+  $::auth->assert('time_recording_edit_all');
+}
+
 sub prepare_report {
   my ($self) = @_;
 
   my $report      = SL::ReportGenerator->new(\%::myconfig, $::form);
   $self->{report} = $report;
 
-  my @columns  = qw(date start_time end_time order customer project part description staff_member duration booked);
+  my @columns  = qw(ids date start_time end_time order customer project part description staff_member duration booked);
 
   my %column_defs = (
+    ids          => { raw_header_data => checkbox_tag("", id => "check_all", checkall  => "[data-checkall=1]"),
+                      align           => 'center',
+                      raw_data        => sub { $_[0]->booked ? '' : checkbox_tag("ids[]", value => $_[0]->id, "data-checkall" => 1) }   },
     date         => { text => t8('Date'),         sub => sub { $_[0]->date_as_date },
                       obj_link => sub { $self->url_for(action => 'edit', 'id' => $_[0]->id, callback => $self->models->get_callback) }  },
     start_time   => { text => t8('Start'),        sub => sub { $_[0]->start_time_as_timestamp },
@@ -307,6 +330,11 @@ sub prepare_report {
     booked       => { text => t8('Booked'),       sub => sub { $_[0]->booked ? t8('Yes') : t8('No') } },
   );
 
+  if (!$self->can_edit_all) {
+    @columns = grep {'ids' ne $_} @columns;
+    delete $column_defs{ids};
+  }
+
   my $title        = t8('Time Recordings');
   $report->{title} = $title;    # for browser titlebar (title-tag)
 
@@ -373,6 +401,19 @@ sub setup_list_action_bar {
         submit    => [ '#filter_form', { action => 'TimeRecording/list' } ],
         accesskey => 'enter',
       ],
+      combobox => [
+        action => [
+          t8('Actions'),
+          only_if => $self->can_edit_all,
+        ],
+        action => [
+          t8('Mark as booked'),
+          submit  => [ '#form', { action => 'TimeRecording/mark_as_booked', callback => $self->models->get_callback } ],
+          checks  => [ [ 'kivi.check_if_entries_selected', '[name="ids[]"]' ] ],
+          confirm => $::locale->text('Do you really want to mark the selected entries as booked?'),
+          only_if => $self->can_edit_all,
+        ],
+      ],
       action => [
         t8('Add'),
         link => $self->url_for(action => 'edit', callback => $self->models->get_callback),
index aa57773..3d498ee 100755 (executable)
@@ -1106,6 +1106,7 @@ $self->{texts} = {
   'Do you really want to delete this draft?' => 'Möchten Sie diesen Entwurf wirklich löschen?',
   'Do you really want to delete this object?' => 'Möchten Sie dieses Objekt wirklich löschen?',
   'Do you really want to delete this record template?' => 'Möchten Sie diese Belegvorlage wirklich löschen?',
+  'Do you really want to mark the selected entries as booked?' => 'Möchten Sie die ausgewählten Einträge wirklich als gebucht markieren?',
   'Do you really want to print?' => 'Wollen Sie wirklich drucken?',
   'Do you really want to revert to this version?' => 'Möchten Sie wirklich auf diese Version zurücksetzen?',
   'Do you really want to undo the selected SEPA exports? You have to reassign the export again.' => 'Möchten Sie wirklich die ausgewählten SEPA-Exports rückgängig machen? Der Export muss anschließend neu erzeugt werden.',
@@ -1990,6 +1991,7 @@ $self->{texts} = {
   'Margepercent'                => 'Ertrag prozentual',
   'Margetotal'                  => 'Ertrag',
   'Margins'                     => 'Seitenränder',
+  'Mark as booked'              => 'Als gebucht markieren',
   'Mark as closed'              => 'Als geschlossen markieren',
   'Mark as paid'                => 'Als bezahlt markieren',
   'Mark as shop article if column missing' => 'Als Shopartikel setzen, falls Spalte nicht vorhanden',
index b8752cc..d796bdc 100644 (file)
@@ -1106,6 +1106,7 @@ $self->{texts} = {
   'Do you really want to delete this draft?' => '',
   'Do you really want to delete this object?' => '',
   'Do you really want to delete this record template?' => '',
+  'Do you really want to mark the selected entries as booked?' => '',
   'Do you really want to print?' => '',
   'Do you really want to revert to this version?' => '',
   'Do you really want to undo the selected SEPA exports? You have to reassign the export again.' => '',
@@ -1990,6 +1991,7 @@ $self->{texts} = {
   'Margepercent'                => '',
   'Margetotal'                  => '',
   'Margins'                     => '',
+  'Mark as booked'              => '',
   'Mark as closed'              => '',
   'Mark as paid'                => '',
   'Mark as shop article if column missing' => '',
index a27818a..71dddfe 100644 (file)
@@ -1,9 +1,3 @@
-[% USE HTML%]
-[%- USE T8 %]
-[%- USE L %][%- USE LxERP -%]
+[%- USE L %]
  [% L.paginate_controls(models=SELF.models) %]
- <input type="hidden" name="rowcount" value="[% HTML.escape(rowcount) %]">
- [%- FOREACH item = HIDDEN %]
- <input type="hidden" name="[% HTML.escape(item.key) %]" value="[% HTML.escape(item.value) %]">
- [%- END %]
 </form>
index 9541d80..12bb508 100644 (file)
@@ -1,2 +1,3 @@
 [%- PROCESS 'time_recording/_filter.html' filter=SELF.models.filtered.laundered %]
 <hr>
+<form method="post" action="controller.pl" id="form">