Projekt

Allgemein

Profil

« Zurück | Weiter » 

Revision 43a5f43b

Von Hans Peter Schlaepfer vor fast 6 Jahren hinzugefügt

  • ID 43a5f43b3abc776bf21c1b13cf19290c82fcf4fa
  • Vorgänger 5af1f24c
  • Nachfolger 37a322ba

Neues kivitendo Design Aenderungen in templates/webpages/csv_import/..

Unterschiede anzeigen:

templates/webpages/csv_import/_deferred_results.html
1
[%- USE L %]
2
[%- USE T8 %]
3
[%- USE HTML %]
1
[% USE L %]
2
[% USE T8 %]
3
[% USE HTML %]
4 4

  
5 5
<h2>[% 'Import Status' | $T8 %]</h2>
6 6

  
templates/webpages/csv_import/_form_banktransactions.html
1 1
[% USE LxERP %]
2 2
[% USE L %]
3 3
<tr>
4
 <th align="right">[%- LxERP.t8("Existing bank transactions") %]:</th>
5
 <td colspan="10">
4
 <th>[% LxERP.t8("Existing bank transactions") %]:</th>
5
 <td>
6 6
  [% opts = [ [ 'skip', LxERP.t8('Skip entry') ] , [ 'insert_new', LxERP.t8('Insert new') ] ] %]
7
  [% L.select_tag('settings.update_policy', opts, default = SELF.profile.get('update_policy'), style = 'width: 300px') %]
8
 </td>
7
  [% L.select_tag('settings.update_policy', opts, default = SELF.profile.get('update_policy'), class='wi-normal') %]
8
</td>
9 9
</tr>
templates/webpages/csv_import/_form_customers_vendors.html
2 2
[% USE L %]
3 3

  
4 4
<tr>
5
 <th align="right">[%- LxERP.t8('Target table') %]:</th>
6
 <td colspan="10">
5
 <th>[%- LxERP.t8('Target table') %]:</th>
6
 <td>
7 7
  [% opts = [ [ 'customer', LxERP.t8('Customers') ], [ 'vendor', LxERP.t8('Vendors') ] ] %]
8 8
  [% L.select_tag('settings.table', opts, default = SELF.profile.get('table'), style = 'width: 300px') %]
9 9
 </td>
10 10
</tr>
11 11

  
12 12
<tr>
13
 <th align="right">[%- LxERP.t8('Existing customers/vendors with same customer/vendor number') %]:</th>
14
 <td colspan="10">
13
 <th>[%- LxERP.t8('Existing customers/vendors with same customer/vendor number') %]:</th>
14
 <td>
15 15
  [% opts = [ [ 'update_existing', LxERP.t8('Update properties of existing entries') ], [ 'insert_new', LxERP.t8('Insert with new customer/vendor number') ], [ 'skip', LxERP.t8('Skip entry') ] ] %]
16 16
  [% L.select_tag('settings.update_policy', opts, default = SELF.profile.get('update_policy'), style = 'width: 300px') %]
17 17
 </td>
18 18
</tr>
19 19

  
20 20
<tr>
21
 <th align="right" valign="top">[%- LxERP.t8('Default taxzone') %]:</th>
22
 <td colspan="10" valign="top">
21
 <th>[%- LxERP.t8('Default taxzone') %]:</th>
22
 <td>
23 23
  [% L.select_tag('settings.default_taxzone', SELF.all_taxzones, default = SELF.profile.get('default_taxzone'), title_key = 'description', style => 'width: 300px') %]
24 24
 </td>
25 25
</tr>
templates/webpages/csv_import/_mapping_item.html
1
[%- USE HTML %]
2
[%- USE LxERP %]
3
[%- USE L %]
4
[%- USE T8 %]
5
 <tr class=listrow>
6
    <td><a class='remove_line interact cursor-pointer'>✘</a></td>
7
[%- IF item.from %]
8
  <td>[% L.hidden_tag('mappings[+].from', item.from) %][% item.from | html %]</td>
9
[%- ELSE %]
10
  <td>[% L.input_tag('mappings[+].from', '') %]</td>
11
[%- END %]
1
[% USE HTML %]
2
[% USE LxERP %]
3
[% USE L %]
4
[% USE T8 %]
5

  
6
<tr>
7
  <td><a class='remove_line interact cursor-pointer'>✘</a></td>
8
  [% IF item.from %]
9
    <td>[% L.hidden_tag('mappings[+].from', item.from) %][% item.from | html %]</td>
10
  [% ELSE %]
11
    <td>[% L.input_tag('mappings[+].from', '') %]</td>
12
  [% END %]
12 13
  <td>[% L.select_tag('mappings[].to', SELF.displayable_columns, value_key='name', title_key='name', default=item.to) %]</td>
13
 </tr>
14
</tr>
14 15

  
templates/webpages/csv_import/form.html
1
[%- USE HTML %]
2
[%- USE LxERP %]
3
[%- USE L %]
4
[%- USE T8 %]
5
 <h1>[% FORM.title %]</h1>
1
[% USE HTML %]
2
[% USE LxERP %]
3
[% USE L %]
4
[% USE T8 %]
6 5

  
7
 [%- INCLUDE 'common/flash.html' %]
6
<h1>[% FORM.title %]</h1>
8 7

  
9
 <form method="post" action="controller.pl" enctype="multipart/form-data" id="form">
10
  [% L.hidden_tag('form_sent', '1') %]
11
  [% L.hidden_tag('action', 'CsvImport/dispatch') %]
12
  [% L.hidden_tag('profile.type', SELF.profile.type) %]
13
  [% L.hidden_tag('tmp_profile_id', SELF.profile.id) %]
8
[% INCLUDE 'common/flash.html' %]
14 9

  
15
 [%- IF SELF.profile.get('dont_edit_profile') %]
10
<form method="post" action="controller.pl" enctype="multipart/form-data" id="form">
11
[% L.hidden_tag('form_sent', '1') %]
12
[% L.hidden_tag('action', 'CsvImport/dispatch') %]
13
[% L.hidden_tag('profile.type', SELF.profile.type) %]
14
[% L.hidden_tag('tmp_profile_id', SELF.profile.id) %]
15

  
16
[% IF SELF.profile.get('dont_edit_profile') %]
16 17
  [% L.hidden_tag('force_profile', 1) %]
17 18
  [% L.hidden_tag('profile.id', SELF.profile.id) %]
18
 [%- ELSE %][%# IF SELF.profile.get('dont_edit_profile') %]
19
  <h2>[%- LxERP.t8('Import profiles') %]</h2>
20 19

  
21
  <table>
22
   [%- IF SELF.profile.id %]
20
[% ELSE %]
21

  
22
<div class="wrapper">
23

  
24
<table class="tbl-horizontal">
25
  <caption>[% LxERP.t8('Import profiles') %]</caption>
26
  <colgroup><col class="wi-mediumsmall"><col class="wi-wide"></colgroup>
27
  <tbody>
28
    [% IF SELF.profile.id %]
23 29
    <tr>
24
     <th align="right">[%- LxERP.t8('Current profile') %]:</th>
25
     <td>[%- HTML.escape(SELF.profile.name) %]</td>
30
      <th>[% LxERP.t8('Current profile') %]:</th>
31
      <td><h3>[% HTML.escape(SELF.profile.name) %]</h3></td>
26 32
    </tr>
27
   [%- END %][%# IF SELF.profile.id %]
28

  
29
   [%- IF SELF.all_profiles.size %]
33
    [% END %]
34
    [% IF SELF.all_profiles.size %]
35
    <tr>
36
      <th>[% LxERP.t8('Existing profiles') %]:</th>
37
      <td>
38
        [% L.select_tag('profile.id', SELF.all_profiles, title_key = 'name', default = SELF.profile.id, style = 'width: 300px') %] <br>
39
        [% L.submit_tag('action_new', LxERP.t8('Load profile')) %]
40
        [% L.submit_tag('action_destroy', LxERP.t8('Delete profile'), confirm => LxERP.t8('Do you really want to delete this object?')) %]
41
      </td>
42
    </tr>
43
    [% END %]
30 44
    <tr>
31
     <th align="right">[%- LxERP.t8('Existing profiles') %]:</th>
32
     <td>
33
      [% L.select_tag('profile.id', SELF.all_profiles, title_key = 'name', default = SELF.profile.id, style = 'width: 300px') %]
34
     </td>
35
     <td>
36
      [% L.submit_tag('action_new', LxERP.t8('Load profile')) %]
37
      [% L.submit_tag('action_destroy', LxERP.t8('Delete profile'), confirm => LxERP.t8('Do you really want to delete this object?')) %]
38
     </td>
45
      <th>[% LxERP.t8('Save settings as') %]:</th>
46
      <td>
47
        [% L.input_tag('profile.name', '', style => 'width: 300px') %]<br>
48
        [% L.checkbox_tag('profile.is_default', label => LxERP.t8('Make default profile')) %]<br>
49
        <div class="buttons below">[% L.submit_tag('action_save', LxERP.t8('Save profile'), class='below') %]</div>
50
      </td>
39 51
    </tr>
40
   [%- END %][%# IF SELF.all_profiles.size %]
52
  </tbody>
53
</table>
41 54

  
42
   <tr>
43
    <th align="right" valign="top">[%- LxERP.t8('Save settings as') %]:</th>
44
    <td valign="top">
45
     [% L.input_tag('profile.name', '', style => 'width: 300px') %]
46
     <br>
47
     [% L.checkbox_tag('profile.is_default', label => LxERP.t8('Make default profile')) %]
48
    </td>
49
    <td valign="top">[% L.submit_tag('action_save', LxERP.t8('Save profile')) %]</td>
50
   </tr>
51
  </table>
55
</div><!-- /.wrapper -->
52 56

  
53
  <hr>
57
<div class="wrapper">
54 58

  
55
  <h2>[%- LxERP.t8('Help on column names') %]</h2>
59
<div class="col wi-verywide">
56 60

  
57
  <div class="help_toggle">
58
   <a href="#" onClick="javascript:$('.help_toggle').toggle()">[% LxERP.t8("Show help text") %]</a>
59
  </div>
61
<h3 class="caption">[% LxERP.t8('Settings') %][% IF SELF.profile.id %] [% 'for Profile' | $T8 %]«[% HTML.escape(SELF.profile.name) %][% END %]</h3>
60 62

  
61
  <div class="help_toggle" style="display:none">
62
   <p><a href="#" onClick="javascript:$('.help_toggle').toggle()">[% LxERP.t8("Hide help text") %]</a></p>
63

  
64
   [%- IF SELF.worker.is_multiplexed %]
65
     <table>
66
       <tr class="listheading">
67
         [%- FOREACH p = SELF.worker.profile %]
68
           <th>[%- p.row_ident %]</th>
69
         [%- END %][%# FOREACH SELF.worker.profile %]
70
       </tr>
71
       <tr style="vertical-align:top">
72
         [%- FOREACH p = SELF.worker.profile %]
73
           [% SET ri = p.row_ident %]
74
         <td>
75
           <table>
76
             <tr class="listheading">
77
               <th>[%- LxERP.t8('Column name') %]</th>
78
               <th>[%- LxERP.t8('Meaning') %]</th>
79
             </tr>
80

  
81
             [%- FOREACH row = SELF.displayable_columns.$ri %]
82
             <tr class="listrow[% loop.count % 2 %]">
83
               <td>[%- HTML.escape(row.name) %]</td>
84
               <td>[%- HTML.escape(row.description) %]</td>
85
             </tr>
86
             [%- END %][%# FOREACH SELF.displayable_columns.$ri %]
87
           </table>
88
         </td>
89
         [%- END %][%# FOREACH SELF.worker.profile %]
90
       </tr>
91
     </table>
92
   [%- ELSE %][%# IF SELF.worker.is_multiplexed %]
93
     <table>
94
       <tr class="listheading">
95
         <th>[%- LxERP.t8('Column name') %]</th>
96
         <th>[%- LxERP.t8('Meaning') %]</th>
97
       </tr>
98

  
99
       [%- FOREACH row = SELF.displayable_columns %]
100
       <tr class="listrow[% loop.count % 2 %]">
101
         <td>[%- HTML.escape(row.name) %]</td>
102
         <td>[%- HTML.escape(row.description) %]</td>
103
       </tr>
104
       [%- END %][%# FOREACH SELF.displayable_columns %]
105
     </table>
106
   [%- END %][%# SELF.worker.is_multiplexed %]
107

  
108
[%- IF SELF.type == 'contacts' %]
109
   <p>
110
    [%- LxERP.t8("You can update existing contacts by providing the 'cp_id' column with their database IDs. Otherwise: ") %]
111
    [%- LxERP.t8('At least one of the columns #1, customer, customernumber, customer_gln, vendor, vendornumber, vendor_gln (depending on the target table) is required for matching the entry to an existing customer or vendor.', 'cp_cv_id') %]
112
   </p>
113

  
114
[%- ELSIF SELF.type == 'addresses' %]
115
   <p>
116
    [%- LxERP.t8('At least one of the columns #1, customer, customernumber, customer_gln, vendor, vendornumber, vendor_gln (depending on the target table) is required for matching the entry to an existing customer or vendor.', 'trans_id') %]
117
   </p>
118

  
119
[%- ELSIF SELF.type == 'parts' %]
120
   <p>
121
    [1]:
122
    [% LxERP.t8('The three columns "make_X", "model_X" and "lastcost_X" with the same number "X" are used to import vendor part numbers and vendor prices.') %]
123
    [% LxERP.t8('The column triplets can occur multiple times with different numbers "X" each time (e.g. "make_1", "model_1", "lastcost_1", "make_2", "model_2", "lastcost_2", "make_3", "model_3", "lastcost_3" etc).') %]
124
    [% LxERP.t8('The items are imported accoring do their number "X" regardless of the column order inside the file.') %]
125
    [% LxERP.t8('The column "make_X" can contain either a vendor\'s database ID, a vendor number or a vendor\'s name.') %]
126
   </p>
63
[% BLOCK panel_1 %]
64
<table class="tbl-horizontal">
65
  <colgroup><col class="wi-mediumsmall"><col class="wi-wide"></colgroup>
66
  <tbody>
67
    <tr>
68
      <th>[% LxERP.t8('Number Format') %]:</th>
69
      <td>[% L.select_tag('settings.numberformat', ['1.000,00', '1000,00', '1,000.00', '1000.00', "1'000.00"], default = SELF.profile.get('numberformat'), class='wi-normal') %]</td>
70
    </tr>
71
    <tr>
72
      <th>[% LxERP.t8('Charset') %]:</th>
73
      <td>[% L.select_tag('settings.charset', SELF.all_charsets, default = SELF.profile.get('charset'), class='wi-normal') %]</td>
74
    </tr>
75
    <tr>
76
      <th>[% LxERP.t8('Separator') %]:</th>
77
      <td>
78
      [% SET custom_sep_char = SELF.sep_char %]
79
      [% FOREACH entry = SELF.all_sep_chars %]
80
        [% IF SELF.sep_char == entry.first %] [% SET custom_sep_char = '' %] [% END %]
81
        [% L.radio_button_tag('sep_char', value => entry.first, label => entry.last, checked => SELF.sep_char == entry.first) %]<br>
82
      [% END %]
83
      [% L.radio_button_tag('sep_char', value => 'custom', checked => custom_sep_char != '') %]
84
      [% L.input_tag('custom_sep_char', custom_sep_char, size => 3, maxlength => 1) %]
85
      </td>
86
    </tr>
87
    <tr>
88
      <th>[% LxERP.t8('Quote character') %]:</th>
89
      <td>
90
        [% SET custom_quote_char = SELF.quote_char %]
91
        [% FOREACH entry = SELF.all_quote_chars %]
92
          [% IF SELF.quote_char == entry.first %] [% SET custom_quote_char = '' %] [% END %]
93
          [% L.radio_button_tag('quote_char', value => entry.first, label => entry.last, checked => SELF.quote_char == entry.first) %]<br>
94
        [% END %]
95
        [% L.radio_button_tag('quote_char', value => 'custom', checked => custom_quote_char != '') %]
96
        [% L.input_tag('custom_quote_char', custom_quote_char, size => 3, maxlength => 1) %]
97
      </td>
98
    </tr>
99
    <tr>
100
      <th>[% LxERP.t8('Escape character') %]:</th>
101
      <td>
102
        [% SET custom_escape_char = SELF.escape_char %]
103
        [% FOREACH entry = SELF.all_escape_chars %]
104
          [% IF SELF.escape_char == entry.first %] [% SET custom_escape_char = '' %] [% END %]
105
          [% L.radio_button_tag('escape_char', value => entry.first, label => entry.last, checked => SELF.escape_char == entry.first) %]<br>
106
        [% END %]
107
        [% L.radio_button_tag('escape_char', value => 'custom', checked => custom_escape_char != '') %]
108
        [% L.input_tag('custom_escape_char', custom_escape_char, size => 3, maxlength => 1) %]
109
      </td>
110
    </tr>
111
    [% duplicate_fields = SELF.worker.get_duplicate_check_fields() %]
112
    [% IF ( duplicate_fields.size ) %]
113
      <tr>
114
        <th>[% LxERP.t8('Check for duplicates') %]:</th>
115
        <td>
116
        [% FOREACH key = duplicate_fields.keys %]
117
          [% IF (SELF.profile.get('duplicates_'_ key) || (duplicate_fields.$key.default && !FORM.form_sent)) %] [% SET chckd=' checked' %]
118
          [% ELSE %] [% SET chckd='' %]
119
          [% END %]
120
          <input type="checkbox" name="settings.duplicates_[% key | html %]" id="settings.duplicates_[% key | html %]" value="1"[% chckd %]>
121
          <label for="settings.duplicates_[% key | html %]">[% duplicate_fields.$key.label | html %]</label><br>
122
        [% END %]
123
        </td>
124
      </tr>
125
      <tr>
126
        <th></th>
127
        <td>
128
          [% opts = [ [ 'no_check',  LxERP.t8('Do not check for duplicates') ],
129
                      [ 'check_csv', LxERP.t8('Discard duplicate entries in CSV file') ],
130
                      [ 'check_db',  LxERP.t8('Discard entries with duplicates in database or CSV file') ] ] %]
131
          [% L.select_tag('settings.duplicates', opts, default = SELF.profile.get('duplicates'), style = 'width: 300px') %]
132
        </td>
133
      </tr>
134
    [% END %]
135
    [% IF SELF.type == 'parts' %]
136
      [% INCLUDE 'csv_import/_form_parts.html' %]
137
    [% ELSIF SELF.type == 'customers_vendors' %]
138
      [% INCLUDE 'csv_import/_form_customers_vendors.html' %]
139
    [% ELSIF SELF.type == 'contacts' %]
140
      [% INCLUDE 'csv_import/_form_contacts.html' %]
141
    [% ELSIF SELF.type == 'inventories' %]
142
      [% INCLUDE 'csv_import/_form_inventories.html' %]
143
    [% ELSIF SELF.type == 'orders' %]
144
      [% INCLUDE 'csv_import/_form_orders.html' %]
145
    [% ELSIF SELF.type == 'ar_transactions' %]
146
      [% INCLUDE 'csv_import/_form_artransactions.html' %]
147
    [% ELSIF SELF.type == 'bank_transactions' %]
148
      [% INCLUDE 'csv_import/_form_banktransactions.html' %]
149
    [% END %]
150
    <tr>
151
      <th>[% LxERP.t8('Preview Mode') %]:</th>
152
      <td>
153
        [% L.radio_button_tag('settings.full_preview', value=2, checked=SELF.profile.get('full_preview')==2, label=LxERP.t8('Full Preview')) %]
154
        [% L.radio_button_tag('settings.full_preview', value=1, checked=SELF.profile.get('full_preview')==1, label=LxERP.t8('Only Warnings and Errors')) %]
155
        [% L.radio_button_tag('settings.full_preview', value=0, checked=!SELF.profile.get('full_preview'),   label=LxERP.t8('First 20 Lines')) %]
156
      </td>
157
    </tr>
158
    <tr>
159
      <th>[% LxERP.t8('Import file') %]:</th>
160
      <td>[% L.input_tag('file', '', type => 'file', accept => '*') %]</td>
161
    </tr>
162
    [% IF SELF.file.exists %]
163
      <tr>
164
        <th>[% LxERP.t8('Existing file on server') %]:</th>
165
        <td>[% LxERP.t8('Uploaded on #1, size #2 kB', SELF.file.displayable_mtime, LxERP.format_amount(SELF.file.size / 1024, 2)) %]</td>
166
      </tr>
167
    [% END %]
168
  </tbody>
169
</table>
170
[% END # panel_1 %]
171

  
172
[%
173
  INCLUDE 'common/toggle_panel.html'
174
  block_name    = 'panel_1'
175
  toggle_class  = 'toggle_panel_1'
176
  button_closed = LxERP.t8("Show settings")
177
  button_open   = LxERP.t8("Hide settings")
178
%]
179

  
180
</div><!-- /.col -->
181
<div class="col wi-verywide">
182

  
183
[% UNLESS SELF.worker.is_multiplexed %]
184

  
185
<h3>[% 'Mappings (csv_import)' | $T8 %]</h3>
186

  
187
[% BLOCK panel_2 %]
188
<p>[% 'These mappings can be used to map heading from non standard csv files to known columns. These will also be saved in profiles, so you can save profiles for every source of formats.' | $T8 %]</p>
189
<table id="csv_import_mappings" class="tbl-list">
190
  <thead>
191
    <tr>
192
      <th>[% 'Action' | $T8 %]</th>
193
      <th>[% 'Text in CSV File' | $T8 %]</th>
194
      <th>[% 'Known Column' | $T8 %]</th>
195
    </tr>
196
  </thead>
197
  <tbody>
198
    <tr id="mapping_empty" style="display:none">
199
      <td colspan="3">[% 'There is nothing here yet (csv_import)' | $T8 %]</td>
200
    </tr>
201
    [% FOREACH row = SELF.mappings %]
202
      [% PROCESS 'csv_import/_mapping_item.html', item=row IF row.from %]
203
    [% END %]
204
    [% PROCESS 'csv_import/_mapping_item.html', item={} %]
205
  </tbody>
206
</table>
207
<div class="button">
208
  <input type="button" id="add_empty_mapping_line" value="[% 'Add empty line (csv_import)' | $T8 %]">
209
  <input type="button" id="add_mapping_from_upload" value="[% 'Add headers from last uploaded file (csv_import)' | $T8 %]">
210
</div>
211
[% END # panel_2 %]
212

  
213
[%
214
  #INCLUDE 'common/toggle_panel.html'
215
  block_name    = 'panel_2'
216
  toggle_class  = 'toggle_panel_2'
217
  button_closed = LxERP.t8("Show mappings (csv_import)")
218
  button_open   = LxERP.t8("Hide mappings (csv_import)s")
219
%]
220

  
221
[% END # UNLESS SELF.worker.is_multiplexed %]
222

  
223
[% IF SELF.deferred %]
224
  <div id="results">
225
    [% PROCESS 'csv_import/_deferred_results.html' %]
226
  </div>
227
[% END %]
228

  
229

  
230
<h3 class="caption">[% LxERP.t8('Help on column names') %]</h3>
231

  
232
[% BLOCK panel_3 %]
233
  [% IF SELF.worker.is_multiplexed %]
234
    <table>
235
      <tbody>
236
        <tr>
237
          [% FOREACH p = SELF.worker.profile %]
238
            <th>[% p.row_ident %]</th>
239
          [% END %]
240
        </tr>
241
        <tr>
242
          [% FOREACH p = SELF.worker.profile %]
243
            [% SET ri = p.row_ident %]
244
            <td>
245
              <table class="tbl-list">
246
                <thead>
247
                  <tr>
248
                    <th>[% LxERP.t8('Column name') %]</th>
249
                    <th>[% LxERP.t8('Meaning') %]</th>
250
                  </tr>
251
                </thead>
252
                <tbody>
253
                  [% FOREACH row = SELF.displayable_columns.$ri %]
254
                    <tr>
255
                      <td>[% HTML.escape(row.name) %]</td>
256
                      <td>[% HTML.escape(row.description) %]</td>
257
                    </tr>
258
                  [% END %]
259
                </tbody>
260
              </table>
261
            </td>
262
          [% END %]
263
        </tr>
264
      </tbody>
265
    </table>
266
  [% ELSE %]
267
    <table class="tbl-list">
268
      <thead>
269
        <tr>
270
          <th>[% LxERP.t8('Column name') %]</th>
271
          <th>[% LxERP.t8('Meaning') %]</th>
272
        </tr>
273
      </thead>
274
      <tbody>
275
        [% FOREACH row = SELF.displayable_columns %]
276
        <tr>
277
          <td>[% HTML.escape(row.name) %]</td>
278
          <td>[% HTML.escape(row.description) %]</td>
279
        </tr>
280
        [% END %]
281
      </tbody>
282
    </table>
283
  [% END %]
284
  [% IF SELF.type == 'contacts' %]
285
    <p>
286
      [% LxERP.t8("You can update existing contacts by providing the 'cp_id' column with their database IDs. Otherwise: ") %]
287
      [% LxERP.t8('At least one of the columns #1, customer, customernumber, customer_gln, vendor, vendornumber, vendor_gln (depending on the target table) is required for matching the entry to an existing customer or vendor.', 'cp_cv_id') %]
288
    </p>
289
  [% ELSIF SELF.type == 'addresses' %]
290
    <p>[% LxERP.t8('At least one of the columns #1, customer, customernumber, customer_gln, vendor, vendornumber, vendor_gln (depending on the target table) is required for matching the entry to an existing customer or vendor.', 'trans_id') %]</p>
291
  [% ELSIF SELF.type == 'parts' %]
292
    <p>
293
      [1]:
294
      [% LxERP.t8('The three columns "make_X", "model_X" and "lastcost_X" with the same number "X" are used to import vendor part numbers and vendor prices.') %]
295
      [% LxERP.t8('The column triplets can occur multiple times with different numbers "X" each time (e.g. "make_1", "model_1", "lastcost_1", "make_2", "model_2", "lastcost_2", "make_3", "model_3", "lastcost_3" etc).') %]
296
      [% LxERP.t8('The items are imported accoring do their number "X" regardless of the column order inside the file.') %]
297
      [% LxERP.t8('The column "make_X" can contain either a vendor\'s database ID, a vendor number or a vendor\'s name.') %]
298
    </p>
299
    <p>
300
      [2]:
301
      [% LxERP.t8('Onhand only sets the quantity in master data, not in inventory. This is only a legacy info field and will be overwritten as soon as a inventory transfer happens.') %]
302
    </p>
303
    <p>
304
      [3]:
305
      [% LxERP.t8("If the article type is set to 'mixed' then a column called 'part_type' or called 'pclass' must be present.") %]
306
      [% LxERP.t8("Type can be either 'part', 'service' or 'assembly'.") %]
307
      [% LxERP.t8("If column 'pclass' is present the article type is then irrelevant or used as default ") %]
308
      [% LxERP.t8("The 'pclass' column has the same abbreviation like a part export. The first letter is for the type Part,Assembly or Service, the second(and third) for Part Classification") %]
309
    </p>
310
  [% ELSIF SELF.type == 'inventories' %]
311
    <p>[% LxERP.t8('One of the columns "qty" or "target_qty" must be given. If "target_qty" is given, the quantity to transfer for each transfer will be calculate, so that the quantity for this part, warehouse and bin will result in the given "target_qty" after each transfer.') %]</p>
312
  [% ELSIF SELF.type == 'orders' OR SELF.type == 'ar_transactions' %]
313
    <p>
314
      [1]:
315
      [% LxERP.t8('The column "datatype" must be present and must be at the same position / column in each data set. The values must be the row names (see settings) for order and item data respectively.') %]
316
    </p>
127 317
    <p>
128
    [2]:
129
    [% LxERP.t8('Onhand only sets the quantity in master data, not in inventory. This is only a legacy info field and will be overwritten as soon as a inventory transfer happens.') %]
130
   </p>
131
   <p>
132
    [3]:
133
    [% LxERP.t8("If the article type is set to 'mixed' then a column called 'part_type' or called 'pclass' must be present.") %]
134
    [% LxERP.t8("Type can be either 'part', 'service' or 'assembly'.") %]
135
    [%- LxERP.t8("If column 'pclass' is present the article type is then irrelevant or used as default ") %]
136
    [% LxERP.t8("The 'pclass' column has the same abbreviation like a part export. The first letter is for the type Part,Assembly or Service, the second(and third) for Part Classification") %]
137
   </p>
138

  
139
[%- ELSIF SELF.type == 'inventories' %]
140
   <p>
141
    [%- LxERP.t8('One of the columns "qty" or "target_qty" must be given. If "target_qty" is given, the quantity to transfer for each transfer will be calculate, so that the quantity for this part, warehouse and bin will result in the given "target_qty" after each transfer.') %]
142
   </p>
143

  
144
[%- ELSIF SELF.type == 'orders' OR SELF.type == 'ar_transactions' %]
145
   <p>
146
    [1]:
147
    [% LxERP.t8('The column "datatype" must be present and must be at the same position / column in each data set. The values must be the row names (see settings) for order and item data respectively.') %]
148
   </p>
149
   <p>
150
    [2]:
151
    [%- LxERP.t8('Amount and net amount are calculated by kivitendo. "verify_amount" and "verify_netamount" can be used for sanity checks.') %]<br>
152
    [%- LxERP.t8('If amounts differ more than "Maximal amount difference" (see settings), this item is marked as invalid.') %]<br>
153
   </p>
154
[%- END %][%# IF SELF.type == … %]
155

  
156
   <p>
157
    [%- L.submit_tag('action_download_sample', LxERP.t8('Download sample file')) %]
158
   </p>
318
      [2]:
319
      [% LxERP.t8('Amount and net amount are calculated by kivitendo. "verify_amount" and "verify_netamount" can be used for sanity checks.') %]<br>
320
      [% LxERP.t8('If amounts differ more than "Maximal amount difference" (see settings), this item is marked as invalid.') %]<br>
321
    </p>
322
  [% END %]
323
  <p>[% L.submit_tag('action_download_sample', LxERP.t8('Download sample file')) %]</p>
324
[% END # panel_3 %]
159 325

  
160
  </div>
326
[%
327
  #INCLUDE 'common/toggle_panel.html'
328
  block_name    = 'panel_3'
329
  toggle_class  = 'toggle_panel_3'
330
  button_closed = LxERP.t8("Show help text")
331
  button_open   = LxERP.t8("Hide help text")
332
%]
161 333

  
162
  <hr>
163 334

  
164
  <h2>[%- LxERP.t8('Settings') %]</h2>
335
</div><!-- /.col -->
165 336

  
166
  <div class="settings_toggle"[% UNLESS SELF.deferred || SELF.import_status %] style="display:none"[% END %]>
167
   <a href="#" onClick="javascript:$('.settings_toggle').toggle()">[% LxERP.t8("Show settings") %]</a>
168
  </div>
337
[% END %]<!-- # /ELSE SELF.profile.get('dont_edit_profile') -->
169 338

  
170
  <div class="settings_toggle"[% IF SELF.deferred || SELF.import_status %] style="display:none"[% END %]>
171
   <p><a href="#" onClick="javascript:$('.settings_toggle').toggle()">[% LxERP.t8("Hide settings") %]</a></p>
172

  
173
  <table>
174
   <tr>
175
    <th align="right">[%- LxERP.t8('Number Format') %]:</th>
176
    <td colspan="10">
177
     [% L.select_tag('settings.numberformat', ['1.000,00', '1000,00', '1,000.00', '1000.00', "1'000.00"], default = SELF.profile.get('numberformat'), style = 'width: 300px') %]
178
    </td>
179
   </tr>
180

  
181
   <tr>
182
    <th align="right">[%- LxERP.t8('Charset') %]:</th>
183
    <td colspan="10">[% L.select_tag('settings.charset', SELF.all_charsets, default = SELF.profile.get('charset'), style = 'width: 300px') %]</td>
184
   </tr>
185

  
186
   <tr>
187
    <th align="right">[%- LxERP.t8('Separator') %]:</th>
188
    [% SET custom_sep_char = SELF.sep_char %]
189
    [% FOREACH entry = SELF.all_sep_chars %]
190
     <td>
191
      [% IF SELF.sep_char == entry.first %] [% SET custom_sep_char = '' %] [%- END %]
192
      [% L.radio_button_tag('sep_char', value => entry.first, label => entry.last, checked => SELF.sep_char == entry.first) %]
193
     </td>
194
    [%- END %][%# FOREACH SELF.all_sep_chars %]
195

  
196
    <td>
197
     [% L.radio_button_tag('sep_char', value => 'custom', checked => custom_sep_char != '') %]
198
     [% L.input_tag('custom_sep_char', custom_sep_char, size => 3, maxlength => 1) %]
199
    </td>
200
   </tr>
201

  
202
   <tr>
203
    <th align="right">[%- LxERP.t8('Quote character') %]:</th>
204
    [% SET custom_quote_char = SELF.quote_char %]
205
    [% FOREACH entry = SELF.all_quote_chars %]
206
     <td>
207
      [% IF SELF.quote_char == entry.first %] [% SET custom_quote_char = '' %] [%- END %]
208
      [% L.radio_button_tag('quote_char', value => entry.first, label => entry.last, checked => SELF.quote_char == entry.first) %]
209
     </td>
210
    [%- END %][%# FOREACH SELF.all_quote_chars %]
211

  
212
    <td>
213
     [% L.radio_button_tag('quote_char', value => 'custom', checked => custom_quote_char != '') %]
214
     [% L.input_tag('custom_quote_char', custom_quote_char, size => 3, maxlength => 1) %]
215
    </td>
216
   </tr>
217

  
218
   <tr>
219
    <th align="right">[%- LxERP.t8('Escape character') %]:</th>
220
    [% SET custom_escape_char = SELF.escape_char %]
221
    [% FOREACH entry = SELF.all_escape_chars %]
222
     <td>
223
      [% IF SELF.escape_char == entry.first %] [% SET custom_escape_char = '' %] [%- END %]
224
      [% L.radio_button_tag('escape_char', value => entry.first, label => entry.last, checked => SELF.escape_char == entry.first) %]
225
     </td>
226
    [%- END %][%# FOREACH SELF.all_escape_chars %]
227

  
228
    <td>
229
     [% L.radio_button_tag('escape_char', value => 'custom', checked => custom_escape_char != '') %]
230
     [% L.input_tag('custom_escape_char', custom_escape_char, size => 3, maxlength => 1) %]
231
    </td>
232
   </tr>
233

  
234
   [% duplicate_fields = SELF.worker.get_duplicate_check_fields() %]
235
   [% IF ( duplicate_fields.size ) %]
236
     <tr>
237
       <th align="right">[%- LxERP.t8('Check for duplicates') %]:</th>
238

  
239
       <td colspan=10>
240
         [% FOREACH key = duplicate_fields.keys %]
241
           <input type="checkbox" name="settings.duplicates_[% key | html %]" id="settings.duplicates_[% key | html %]" value="1"[% IF ( SELF.profile.get('duplicates_'_ key) || (duplicate_fields.$key.default && !FORM.form_sent ) ) %] checked="checked"[% END %]>
242
           <label for="settings.duplicates_[% key | html %]">[% duplicate_fields.$key.label | html %]</label>
243
         [% END %][%# FOREACH duplicate_fields.keys %]
244
       </td>
245
     </tr>
246

  
247
     <tr>
248
       <th align="right"></th>
249

  
250
       <td colspan=10>
251
         [% opts = [ [ 'no_check',  LxERP.t8('Do not check for duplicates') ],
252
                     [ 'check_csv', LxERP.t8('Discard duplicate entries in CSV file') ],
253
                     [ 'check_db',  LxERP.t8('Discard entries with duplicates in database or CSV file') ] ] %]
254
         [% L.select_tag('settings.duplicates', opts, default = SELF.profile.get('duplicates'), style = 'width: 300px') %]
255
       </td>
256
     </tr>
257
   [% END %][%# IF duplicate_fields.size %]
258

  
259
[%- IF SELF.type == 'parts' %]
260
 [%- INCLUDE 'csv_import/_form_parts.html' %]
261
[%- ELSIF SELF.type == 'customers_vendors' %]
262
 [%- INCLUDE 'csv_import/_form_customers_vendors.html' %]
263
[%- ELSIF SELF.type == 'contacts' %]
264
 [%- INCLUDE 'csv_import/_form_contacts.html' %]
265
[%- ELSIF SELF.type == 'inventories' %]
266
 [%- INCLUDE 'csv_import/_form_inventories.html' %]
267
[%- ELSIF SELF.type == 'orders' %]
268
 [%- INCLUDE 'csv_import/_form_orders.html' %]
269
[%- ELSIF SELF.type == 'ar_transactions' %]
270
 [%- INCLUDE 'csv_import/_form_artransactions.html' %]
271
[%- ELSIF SELF.type == 'bank_transactions' %]
272
 [%- INCLUDE 'csv_import/_form_banktransactions.html' %]
273
[%- END %]
274

  
275
   <tr>
276
    <th align="right">[%- LxERP.t8('Preview Mode') %]:</th>
277
    <td colspan="10">
278
      [% L.radio_button_tag('settings.full_preview', value=2, checked=SELF.profile.get('full_preview')==2, label=LxERP.t8('Full Preview')) %]
279
      [% L.radio_button_tag('settings.full_preview', value=1, checked=SELF.profile.get('full_preview')==1, label=LxERP.t8('Only Warnings and Errors')) %]
280
      [% L.radio_button_tag('settings.full_preview', value=0, checked=!SELF.profile.get('full_preview'),   label=LxERP.t8('First 20 Lines')) %]
281
    </td>
282
   </tr>
283

  
284
   <tr>
285
    <th align="right">[%- LxERP.t8('Import file') %]:</th>
286
    <td colspan="10">[% L.input_tag('file', '', type => 'file', accept => '*') %]</td>
287
   </tr>
288

  
289
   [%- IF SELF.file.exists %]
290
    <tr>
291
     <th align="right">[%- LxERP.t8('Existing file on server') %]:</th>
292
     <td colspan="10">[%- LxERP.t8('Uploaded on #1, size #2 kB', SELF.file.displayable_mtime, LxERP.format_amount(SELF.file.size / 1024, 2)) %]</td>
293
    </tr>
294
   [%- END %][%# IF SELF.file.exists %]
295 339

  
296
  </table>
297 340

  
298
  </div>
299
  <hr>
341
</form>
300 342

  
301
[%- UNLESS SELF.worker.is_multiplexed %]
302
  <h2>[% 'Mappings (csv_import)' | $T8 %]</h2>
343
<div class="col wi-wide">
303 344

  
304
  <div class="mappings_toggle"[% UNLESS SELF.deferred || SELF.import_status %] style="display:none"[% END %]>
305
   <a href="#" onClick="javascript:$('.mappings_toggle').toggle()">[% LxERP.t8("Show mappings (csv_import)") %]</a>
306
  </div>
307
  <div class="mappings_toggle"[% IF SELF.deferred || SELF.import_status %] style="display:none"[% END %]>
308
   <p><a href="#" onClick="javascript:$('.mappings_toggle').toggle()">[% LxERP.t8("Hide mappings (csv_import)") %]</a></p>
309

  
310
    <p>[% 'These mappings can be used to map heading from non standard csv files to known columns. These will also be saved in profiles, so you can save profiles for every source of formats.' | $T8 %]</p>
311

  
312
  <table id="csv_import_mappings">
313
   <tr class=listheading>
314
    <th></th>
315
    <th>[% 'Text in CSV File' | $T8 %]</th>
316
    <th>[% 'Known Column' | $T8 %]</th>
317
   </tr>
318
   <tr id='mapping_empty' style='display:none'>
319
    <td colspan=3>[% 'There is nothing here yet (csv_import)' | $T8 %]</td>
320
   </tr>
321
[%- FOREACH row = SELF.mappings %]
322
   [% PROCESS 'csv_import/_mapping_item.html', item=row IF row.from %]
323
[%- END %][%# FOREACH SELF.mappings %]
324
   [% PROCESS 'csv_import/_mapping_item.html', item={} %]
325
  </table>
326

  
327
  <input type=button id='add_empty_mapping_line' value='[% 'Add empty line (csv_import)' | $T8 %]'>
328
  <input type=button id='add_mapping_from_upload' value='[% 'Add headers from last uploaded file (csv_import)' | $T8 %]'>
329 345

  
330
  </div>
331
  <hr>
332
[%- END %][%# UNLESS SELF.worker.is_multiplexed %]
333
[%- END %][%# IF SELF.profile.get('dont_edit_profile') %]
334
 </form>
335 346

  
336
 <div id='results'>
337
 [%- IF SELF.deferred %]
338
   [%- PROCESS 'csv_import/_deferred_results.html' %]
339
 [%- END %][%# IF SELF.deferred %]
340
 </div>
347

  
348
</div><!-- /.col -->
349

  
350
</div><!-- /.wrapper -->
341 351

  
342 352

  
343
 <script type="text/javascript">
353
<script type="text/javascript">
344 354
  <!--
345 355
    $(document).ready(function() {
346 356
      $('#action_save').click(function() {
......
358 368
      $('#csv_import_mappings').on('click', '.remove_line', function(){ $(this).closest('tr').remove(); if (1==$('#csv_import_mappings tr:visible').length) $('#mapping_empty').show() });
359 369
    });
360 370
    -->
361
 </script>
371
</script>
templates/webpages/csv_import/report.html
2 2
[% USE LxERP %]
3 3
[% USE L %]
4 4
 <h2>
5
 [%- IF SELF.report.test_mode %]
6
  [%- LxERP.t8('Import preview') %]
7
 [%- ELSE %]
8
  [%- LxERP.t8('Import result') %]
9
 [%- END %]
5
 [% IF SELF.report.test_mode %]
6
  [% LxERP.t8('Import preview') %]
7
 [% ELSE %]
8
  [% LxERP.t8('Import result') %]
9
 [% END %]
10 10
</h2>
11 11

  
12
[%- IF SELF.report.test_mode %]
12
[% IF SELF.report.test_mode %]
13 13
 <p>
14 14
  [% LxERP.t8("The following is only a preview.") %]
15 15
  [% LxERP.t8("No entries have been imported yet.") %]
16 16
 </p>
17
[%- END %]
17
[% END %]
18 18

  
19
[%- PROCESS 'common/paginate.html' pages=SELF.pages, base_url = SELF.base_url %]
19
[% PROCESS 'common/paginate.html' pages=SELF.pages, base_url = SELF.base_url %]
20 20
 <table>
21 21

  
22
[%- SET max_col = 0 %]
23
[%- FOREACH rownum = SELF.display_rows %]
24
  [%- SET max_col = max_col > SELF.report_rows.${rownum}.size ? max_col : SELF.report_rows.${rownum}.size  %]
25
  [%- LAST IF rownum >= SELF.report_numheaders %]
26
[%- END %]
22
[% SET max_col = 0 %]
23
[% FOREACH rownum = SELF.display_rows %]
24
  [% SET max_col = max_col > SELF.report_rows.${rownum}.size ? max_col : SELF.report_rows.${rownum}.size  %]
25
  [% LAST IF rownum >= SELF.report_numheaders %]
26
[% END %]
27 27

  
28
[%- FOREACH rownum = SELF.display_rows %]
29
 [%- SET to_pad = max_col - SELF.report_rows.${rownum}.size %]
30
 [%- IF rownum < SELF.report_numheaders %]
31
  <tr class="listheading">
32
  [%- FOREACH value = SELF.report_rows.${rownum} %]
28
[% FOREACH rownum = SELF.display_rows %]
29
 [% SET to_pad = max_col - SELF.report_rows.${rownum}.size %]
30
 [% IF rownum < SELF.report_numheaders %]
31
  <tr>
32
  [% FOREACH value = SELF.report_rows.${rownum} %]
33 33
   <th>[% value | html %]</th>
34
  [%- END %]
35
   [%- IF to_pad -%]<th style="text-align:center" colspan="[%- to_pad -%]">-</th>[%- END -%]
36
   <th>[%- LxERP.t8('Notes') %]</th>
34
  [% END %]
35
   [% IF to_pad %]<th style="text-align:center" colspan="[% to_pad %]">-</th>[% END %]
36
   <th>[% LxERP.t8('Notes') %]</th>
37 37
  </tr>
38
 [%- ELSE %]
38
 [% ELSE %]
39 39
  [% csv_import_report_errors = SELF.report_status.${rownum}.errors %]
40 40
  <tr class="listrow[% IF csv_import_report_errors && csv_import_report_errors.size %]_error[% END %][% (loop.count + SELF.report_numheaders) % 2 %]">
41
  [%- FOREACH value = SELF.report_rows.${rownum} %]
42
   <td>[%- value | html  %]</td>
43
  [%- END %]
44
   [%- IF to_pad -%]<td align="center" colspan="[%- to_pad -%]">-</td>[%- END -%]
41
  [% FOREACH value = SELF.report_rows.${rownum} %]
42
   <td>[% value | html  %]</td>
43
  [% END %]
44
   [% IF to_pad %]<td colspan="[% to_pad %]">-</td>[% END %]
45 45
   <td>
46
    [%- FOREACH error = csv_import_report_errors %][%- error | html %][% UNLESS loop.last %]<br>[%- END %][%- END %]
47
    [%- FOREACH info  = SELF.report_status.${rownum}.information %][% IF rownum >= SELF.report_numheaders || csv_import_report_errors.size %]<br>[%- END %][%- info | html %][%- END %]
46
    [% FOREACH error = csv_import_report_errors %][% error | html %][% UNLESS loop.last %]<br>[% END %][% END %]
47
    [% FOREACH info  = SELF.report_status.${rownum}.information %][% IF rownum >= SELF.report_numheaders || csv_import_report_errors.size %]<br>[% END %][% info | html %][% END %]
48 48
   </td>
49 49
  </tr>
50
 [%- END %]
51
[%- END %]
50
 [% END %]
51
[% END %]
52 52

  
53 53
 </table>
54 54

  
55
[%- PROCESS 'common/paginate.html' pages=SELF.pages, base_url = SELF.base_url %]
55
[% PROCESS 'common/paginate.html' pages=SELF.pages, base_url = SELF.base_url %]
56 56

  
57 57
<script type='text/javascript'>
58 58
  $(function(){
59
    [%- IF SELF.reporterror %]
59
    [% IF SELF.reporterror %]
60 60
      kivi.clear_flash('info',0);
61 61
      kivi.display_flash('error','[% SELF.reporterror %]',0);
62
    [%- ELSIF SELF.report.test_mode %]
62
    [% ELSIF SELF.report.test_mode %]
63 63
      kivi.ActionBar.Action('#action_import').enable();
64
    [%- END %]
64
    [% END %]
65 65
  });
66 66
</script>

Auch abrufbar als: Unified diff