[11/40] Work on layout editor interface.
[koha.git] / labels / label-edit-layout.pl
1 #!/usr/bin/perl
2 #
3 # Copyright 2006 Katipo Communications.
4 # Parts Copyright 2009 Foundations Bible College.
5 #
6 # This file is part of Koha.
7 #       
8 # Koha is free software; you can redistribute it and/or modify it under the
9 # terms of the GNU General Public License as published by the Free Software
10 # Foundation; either version 2 of the License, or (at your option) any later
11 # version.
12 #
13 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
14 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
15 # A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
16 #
17 # You should have received a copy of the GNU General Public License along with
18 # Koha; if not, write to the Free Software Foundation, Inc., 59 Temple Place,
19 # Suite 330, Boston, MA  02111-1307 USA
20
21 use strict;
22 use warnings;
23 use Sys::Syslog qw(syslog);
24 use CGI;
25 use HTML::Template::Pro;
26 use Data::Dumper;
27 use POSIX;
28 use Text::CSV_XS;
29
30 use C4::Auth;
31 use C4::Output;
32 use C4::Context;
33 use C4::Debug;
34 use C4::Labels::Lib 1.000000 qw(get_barcode_types get_label_types get_font_types get_text_justification_types);
35 use C4::Labels::Layout 1.000000;
36
37 my $cgi = new CGI;
38 my ( $template, $loggedinuser, $cookie ) = get_template_and_user(
39     {
40         template_name   => "labels/label-edit-layout.tmpl",
41         query           => $cgi,
42         type            => "intranet",
43         authnotrequired => 0,
44         flagsrequired   => { catalogue => 1 },
45         debug           => 1,
46     }
47 );
48
49
50 my $op = $cgi->param('op') || $ARGV[0] || '';
51 my $layout_id = $cgi->param('layout_id') || $cgi->param('element_id') || $ARGV[1] || '';
52 my $layout = '';
53
54 sub _set_selected {
55     my ($type_list, $object, $data_type) = @_;
56     SET_SELECTED:
57     foreach my $type (@$type_list) {
58         if ($layout->get_attr($data_type)) {
59             if ($type->{'type'} eq $layout->get_attr($data_type)) {
60                 $type->{'selected'} = 1;
61             }
62         }
63         else {
64             $type->{'selected'} = 1;
65             last SET_SELECTED;
66         }
67     };
68     return $type_list;
69 }
70
71 sub _select_format_string {     # generate field table based on format_string
72     my $format_string = shift;
73     $format_string =~ s/(?<=,) (?![A-Z][a-z][0-9])//g;  # remove spaces between fields
74     my $table = [];
75     my $fields = [];
76     my ($row_index, $col_index, $field_index) = (0,0,0);
77     my $cols = 5;       # number of columns to wrap on
78     my $csv = Text::CSV_XS->new({ allow_whitespace => 1 });
79     my $status = $csv->parse($format_string);
80     my @text_fields = $csv->fields();
81     syslog("LOG_ERR", "labels/label-edit-layout.pl : Error parsing format_string. Parser returned: %s",$csv->error_input()) if $csv->error_input();
82     my $field_count = $#text_fields + 1;
83     POPULATE_TABLE:
84     foreach my $text_field (@text_fields) {
85         $$fields[$col_index] = {field_empty => 0, field_name => ($text_field . "_tbl"), field_label => $text_field, order => [{num => '', selected => 0}]};
86         for (my $order_i = 1; $order_i <= $field_count; $order_i++) {
87             $$fields[$col_index]{'order'}[$order_i] = {num => $order_i, selected => ($field_index == $order_i-1 ? 1 : 0)};
88         }
89         $col_index++;
90         $field_index++;
91         if ((($col_index > 0) && !($col_index % $cols)) || ($field_index == $field_count)) {    # wrap to new row
92             if (($field_index == $field_count) && ($row_index > 0)) { # in this case fill out row with empty fields
93                 while ($col_index < $cols) {
94                     $$fields[$col_index] = {field_empty => 1, field_name => '', field_label => '', order => [{num => '', selected => 0}]};
95                     $col_index++;
96                 }
97                 $$table[$row_index] = {text_fields => $fields};
98                 last POPULATE_TABLE;
99             }
100             $$table[$row_index] = {text_fields => $fields};
101             $row_index++;
102             $fields = [];
103             $col_index = 0;
104         }
105     }
106     return $table;
107 }
108
109 if ($op eq 'edit') {
110     syslog("LOG_ERR", "labels/label-edit-layout.pl : Error performing '%s': No 'layout_id' passed in.", $op) unless ($layout);
111     $layout = C4::Labels::Layout->retrieve(layout_id => $layout_id);
112
113 }
114 elsif  ($op eq 'save') {
115     my $format_string = '';
116     if ($cgi->param('layout_choice') eq 'layout_table') {       # translate the field table into a format_string
117         my @layout_table = ();
118         foreach my $cgi_param ($cgi->param()) {
119             if (($cgi_param =~ m/^(.*)_tbl$/) && ($cgi->param($cgi_param))) {
120                 my $value = $cgi->param($cgi_param);
121                 $layout_table[$value - 1] = $1;
122             }
123         }
124         @layout_table = grep {$_} @layout_table;        # this removes numerically 'skipped' fields. ie. user omits a number in sequential order
125         $format_string = join ', ', @layout_table;
126         $cgi->param('format_string', $format_string);
127     }
128     my @params = (
129                     barcode_type    => $cgi->param('barcode_type'),
130                     printing_type   => $cgi->param('printing_type'),
131                     layout_name     => $cgi->param('layout_name'),
132                     guidebox        => ($cgi->param('guidebox') ? 1 : 0),
133                     font            => $cgi->param('font'),
134                     font_size       => $cgi->param('font_size'),
135                     callnum_split   => ($cgi->param('callnum_split') ? 1 : 0),
136                     text_justify    => $cgi->param('text_justify'),
137                     format_string   => $cgi->param('format_string'),
138     );
139     if ($layout_id) {   # if a label_id was passed in, this is an update to an existing layout
140         $layout = C4::Labels::Layout->retrieve(layout_id => $layout_id);
141         $layout->set_attr(@params);
142         $layout->save();
143     }
144     else {      # if no label_id, this is a new layout so insert it
145         $layout = C4::Labels::Layout->new(@params);
146         $layout->save();
147     }
148     print $cgi->redirect("label-manage.pl?label_element=layout");
149     exit;
150 }
151 else {  # if we get here, this is a new layout
152     $layout = C4::Labels::Layout->new();
153 }
154
155 my $barcode_types = _set_selected(get_barcode_types(), $layout, 'barcode_type');
156 my $label_types = _set_selected(get_label_types(), $layout, 'printing_type');
157 my $font_types = _set_selected(get_font_types(), $layout, 'font');
158 my $text_justification_types = _set_selected(get_text_justification_types(), $layout, 'text_justify');
159 my $select_text_fields = _select_format_string($layout->get_attr('format_string'));
160
161 $template->param(
162         barcode_types   => $barcode_types,
163         label_types     => $label_types,
164         font_types      => $font_types,
165         text_justification_types    => $text_justification_types,
166         field_table     => $select_text_fields,
167         layout_id       => $layout->get_attr('layout_id') > -1 ? $layout->get_attr('layout_id') : '',
168         layout_name     => $layout->get_attr('layout_name'),
169         guidebox        => $layout->get_attr('guidebox'),
170         font_size       => $layout->get_attr('font_size'),
171         callnum_split   => $layout->get_attr('callnum_split'),
172         format_string   => $layout->get_attr('format_string'),
173 );
174 output_html_with_http_headers $cgi, $cookie, $template->output;