1
|
<?php
|
2
|
|
3
|
/**
|
4
|
* @file
|
5
|
* Admin settings and update page.
|
6
|
*/
|
7
|
|
8
|
/**
|
9
|
* Project has a new release available.
|
10
|
*/
|
11
|
define('L10N_UPDATE_NOT_CURRENT', 4);
|
12
|
|
13
|
/**
|
14
|
* Project is up to date.
|
15
|
*/
|
16
|
define('L10N_UPDATE_CURRENT', 5);
|
17
|
|
18
|
/**
|
19
|
* Project's status cannot be checked.
|
20
|
*/
|
21
|
define('L10N_UPDATE_NOT_CHECKED', -1);
|
22
|
|
23
|
/**
|
24
|
* No available update data was found for project.
|
25
|
*/
|
26
|
define('L10N_UPDATE_UNKNOWN', -2);
|
27
|
|
28
|
/**
|
29
|
* There was a failure fetching available update data for this project.
|
30
|
*/
|
31
|
define('L10N_UPDATE_NOT_FETCHED', -3);
|
32
|
|
33
|
// Include l10n_update API
|
34
|
module_load_include('check.inc', 'l10n_update');
|
35
|
// And project api
|
36
|
module_load_include('project.inc', 'l10n_update');
|
37
|
|
38
|
/**
|
39
|
* Page callback: Admin overview page.
|
40
|
*/
|
41
|
function l10n_update_admin_overview() {
|
42
|
// For now we get package information provided by modules.
|
43
|
$projects = l10n_update_get_projects();
|
44
|
$languages = l10n_update_language_list('name');
|
45
|
|
46
|
$build = array();
|
47
|
if ($languages) {
|
48
|
$history = l10n_update_get_history();
|
49
|
$available = l10n_update_available_releases();
|
50
|
$updates = l10n_update_build_updates($history, $available);
|
51
|
$build['project_status'] = array(
|
52
|
'#theme' => 'l10n_update_project_status',
|
53
|
'#projects' => $projects,
|
54
|
'#languages' => $languages,
|
55
|
'#history' => $history,
|
56
|
'#available' => $available,
|
57
|
'#updates' => $updates,
|
58
|
);
|
59
|
$build['admin_import_form'] = drupal_get_form('l10n_update_admin_import_form', $projects, $updates);
|
60
|
}
|
61
|
else {
|
62
|
$build['no_languages'] = array('#markup' => t('No translatable language defined. <a href="/admin/config/regional/language">Add a language</a>.'));
|
63
|
}
|
64
|
return $build;
|
65
|
}
|
66
|
|
67
|
/**
|
68
|
* Translation update form.
|
69
|
*
|
70
|
* @todo selectable packages
|
71
|
* @todo check language support in server
|
72
|
* @todo check file update dates
|
73
|
*
|
74
|
* @param $form_state
|
75
|
* Form states array.
|
76
|
* @param $projects
|
77
|
* @todo $projects are not used in the form.
|
78
|
* @param $updates
|
79
|
* Updates to be displayed in the form.
|
80
|
*/
|
81
|
function l10n_update_admin_import_form($form, $form_state, $projects, $updates) {
|
82
|
//module_load_include('inc', 'l10n_update');
|
83
|
// For now we get package information provided by modules
|
84
|
$projects = l10n_update_get_projects();
|
85
|
$languages = l10n_update_language_list('name');
|
86
|
|
87
|
// Absence of projects is an error and only occurs if the database table
|
88
|
// was truncated. In this case we rebuild the project data.
|
89
|
if (!$projects) {
|
90
|
l10n_update_build_projects();
|
91
|
$projects = l10n_update_get_projects();
|
92
|
}
|
93
|
|
94
|
if ($projects && $languages) {
|
95
|
$form['updates'] = array(
|
96
|
'#type' => 'value',
|
97
|
'#value' => $updates,
|
98
|
);
|
99
|
|
100
|
if (count($languages) > 1) {
|
101
|
$form['lang'] = array(
|
102
|
'#type' => 'fieldset',
|
103
|
'#title' => t('Languages'),
|
104
|
'#collapsible' => TRUE,
|
105
|
'#collapsed' => FALSE ,
|
106
|
'#description' => t('Select one or more languages to download and update. If you select none, all of them will be updated.'),
|
107
|
);
|
108
|
$form['lang']['languages'] = array(
|
109
|
'#type' => 'checkboxes',
|
110
|
'#options' => $languages,
|
111
|
);
|
112
|
}
|
113
|
|
114
|
if ($updates) {
|
115
|
$form['actions']['download'] = array(
|
116
|
'#type' => 'submit',
|
117
|
'#value' => t('Update translations'),
|
118
|
);
|
119
|
}
|
120
|
}
|
121
|
$form['actions']['refresh'] = array(
|
122
|
'#type' => 'submit',
|
123
|
'#value' => t('Refresh information'),
|
124
|
);
|
125
|
return $form;
|
126
|
}
|
127
|
|
128
|
/**
|
129
|
* Submit handler for Update form.
|
130
|
*
|
131
|
* Handles both submit buttons to update translations and to update the
|
132
|
* form information.
|
133
|
*/
|
134
|
function l10n_update_admin_import_form_submit($form, $form_state) {
|
135
|
$op = isset($form_state['values']['op']) ? $form_state['values']['op'] : '';
|
136
|
$projects = l10n_update_get_projects();
|
137
|
|
138
|
if ($op == t('Update translations')) {
|
139
|
$languages = isset($form_state['values']['languages']) ? array_filter($form_state['values']['languages']) : NULL;
|
140
|
$updates = $form_state['values']['updates'];
|
141
|
$mode = variable_get('l10n_update_import_mode', LOCALE_IMPORT_KEEP);
|
142
|
|
143
|
if ($projects && $updates) {
|
144
|
module_load_include('batch.inc', 'l10n_update');
|
145
|
// Filter out updates in other languages. If no languages, all of them will be updated
|
146
|
$updates = _l10n_update_prepare_updates($updates, NULL, $languages);
|
147
|
$batch = l10n_update_batch_multiple($updates, $mode);
|
148
|
batch_set($batch);
|
149
|
}
|
150
|
}
|
151
|
elseif ($op == t('Refresh information')) {
|
152
|
// Get current version of projects.
|
153
|
l10n_update_build_projects();
|
154
|
|
155
|
// Get available translation updates and update file history.
|
156
|
if ($available = l10n_update_available_releases(TRUE)) {
|
157
|
l10n_update_flag_history($available);
|
158
|
drupal_set_message(t('Fetched information about available updates from the server'));
|
159
|
}
|
160
|
else {
|
161
|
drupal_set_message(t('Failed to fetch information about available updates from the server.'), 'error');
|
162
|
}
|
163
|
}
|
164
|
}
|
165
|
|
166
|
/**
|
167
|
* Page callback: Settings form.
|
168
|
*/
|
169
|
function l10n_update_admin_settings_form($form, &$form_state) {
|
170
|
$form['l10n_update_check_mode'] = array(
|
171
|
'#type' => 'radios',
|
172
|
'#title' => t('Update source'),
|
173
|
'#default_value' => variable_get('l10n_update_check_mode', L10N_UPDATE_CHECK_ALL),
|
174
|
'#options' => _l10n_update_admin_check_options(),
|
175
|
);
|
176
|
$form['l10n_update_import_mode'] = array(
|
177
|
'#type' => 'radios',
|
178
|
'#title' => t('Update mode'),
|
179
|
'#default_value' => variable_get('l10n_update_import_mode', LOCALE_IMPORT_KEEP),
|
180
|
'#options' => _l10n_update_admin_import_options(),
|
181
|
);
|
182
|
$form['l10n_update_check_frequency'] = array(
|
183
|
'#type' => 'radios',
|
184
|
'#title' => t('Check for updates'),
|
185
|
'#default_value' => variable_get('l10n_update_check_frequency', 0),
|
186
|
'#options' => array(
|
187
|
0 => t('Never (manually)'),
|
188
|
1 => t('Daily'),
|
189
|
7 => t('Weekly'),
|
190
|
),
|
191
|
'#description' => t('Select how frequently you want to automatically check for updated translations for installed modules and themes.'),
|
192
|
);
|
193
|
$form['l10n_update_check_disabled'] = array(
|
194
|
'#type' => 'checkbox',
|
195
|
'#title' => t('Check for updates of disabled modules and themes'),
|
196
|
'#default_value' => variable_get('l10n_update_check_disabled', 0),
|
197
|
'#description' => t('Note that this comes with a performance penalty, so it is not recommended.'),
|
198
|
);
|
199
|
$form['l10n_update_download_store'] = array(
|
200
|
'#title' => t('Store downloaded files'),
|
201
|
'#type' => 'textfield',
|
202
|
'#default_value' => variable_get('l10n_update_download_store', ''),
|
203
|
'#description' => t('A path relative to the Drupal installation directory where translation files will be stored, e.g. sites/all/translations. Saved translation files can be reused by other installations. If left empty the downloaded translation will not be saved.'),
|
204
|
);
|
205
|
|
206
|
$form = system_settings_form($form);
|
207
|
$form['#submit'][] = 'l10n_update_admin_settings_form_submit';
|
208
|
|
209
|
return $form;
|
210
|
}
|
211
|
|
212
|
/**
|
213
|
* Additional validation handler for update settings.
|
214
|
*
|
215
|
* Check for existing files directory and creates one when required.
|
216
|
*/
|
217
|
function l10n_update_admin_settings_form_validate($form, &$form_state) {
|
218
|
$form_values = $form_state['values'];
|
219
|
if (!empty($form_values['l10n_update_download_store'])) {
|
220
|
if (!file_prepare_directory($form_values['l10n_update_download_store'], FILE_CREATE_DIRECTORY, 'l10n_update_download_store')) {
|
221
|
form_set_error('l10n_update_download_store', t('The directory %directory does not exist or is not writable.', array('%directory' => $form_values['l10n_update_download_store'])));
|
222
|
watchdog('file system', 'The directory %directory does not exist or is not writable.', array('%directory' => $form_values['l10n_update_download_store']), WATCHDOG_ERROR);
|
223
|
}
|
224
|
}
|
225
|
}
|
226
|
|
227
|
/**
|
228
|
* Additional submit handler for update settings.
|
229
|
*/
|
230
|
function l10n_update_admin_settings_form_submit($form, &$form_state) {
|
231
|
// Add .htaccess file to the translations directory.
|
232
|
l10n_update_ensure_htaccess();
|
233
|
}
|
234
|
|
235
|
/**
|
236
|
* Get array of import options.
|
237
|
*
|
238
|
* The import options of the Locale module are used but the UI text is altered
|
239
|
* to suit the Localization update cases.
|
240
|
*
|
241
|
* @return
|
242
|
* Keyed array of import options.
|
243
|
*/
|
244
|
function _l10n_update_admin_import_options() {
|
245
|
return array(
|
246
|
LOCALE_IMPORT_OVERWRITE => t('Translation updates replace existing ones, new ones are added'),
|
247
|
LOCALE_UPDATE_OVERRIDE_DEFAULT => t('Edited translations are kept, only previously imported ones are overwritten and new translations are added'),
|
248
|
LOCALE_IMPORT_KEEP => t('All existing translations are kept, only new translations are added.'),
|
249
|
);
|
250
|
}
|
251
|
|
252
|
/**
|
253
|
* Get array of check options.
|
254
|
*
|
255
|
* @return
|
256
|
* Keyed array of source download options.
|
257
|
*/
|
258
|
function _l10n_update_admin_check_options() {
|
259
|
return array(
|
260
|
L10N_UPDATE_CHECK_ALL => t('Local files and remote server.'),
|
261
|
L10N_UPDATE_CHECK_LOCAL => t('Local files only.'),
|
262
|
L10N_UPDATE_CHECK_REMOTE => t('Remote server only.'),
|
263
|
);
|
264
|
}
|
265
|
|
266
|
/**
|
267
|
* Format project update status.
|
268
|
*
|
269
|
* @param $variables
|
270
|
* An associative array containing:
|
271
|
* - projects: An array containing all enabled projects.
|
272
|
* - languages: An array of all enabled languages.
|
273
|
* - history: An array of the current translations per project.
|
274
|
* - available: An array of translation sources per project.
|
275
|
* - updates: An array of available translation updates per project.
|
276
|
* Only recommended translations are listed.
|
277
|
*
|
278
|
* @return string
|
279
|
* HTML output.
|
280
|
*/
|
281
|
function theme_l10n_update_project_status($variables) {
|
282
|
$header = $rows = array();
|
283
|
|
284
|
// Get module and theme data for the project title.
|
285
|
$projects = system_rebuild_module_data();
|
286
|
$projects += system_rebuild_theme_data();
|
287
|
|
288
|
foreach ($variables['projects'] as $name => $project) {
|
289
|
if (isset($variables['history'][$name])) {
|
290
|
if (isset($variables['updates'][$name])) {
|
291
|
$project_status = 'updatable';
|
292
|
$project_class = 'warning';
|
293
|
}
|
294
|
else {
|
295
|
$project_status = 'uptodate';
|
296
|
$project_class = 'ok';
|
297
|
}
|
298
|
}
|
299
|
elseif (isset($variables['available'][$name])) {
|
300
|
$project_status = 'available';
|
301
|
$project_class = 'warning';
|
302
|
}
|
303
|
else {
|
304
|
// Remote information not checked
|
305
|
$project_status = 'unknown';
|
306
|
$project_class = 'unknown';
|
307
|
}
|
308
|
|
309
|
// Get the project title and module version.
|
310
|
$project->title = isset($projects[$name]->info['name']) ? $projects[$name]->info['name'] : '';
|
311
|
$project->module_version = isset($projects[$name]->info['version']) ? $projects[$name]->info['version'] : $project->version;
|
312
|
|
313
|
// Project with related language states.
|
314
|
$row = theme('l10n_update_single_project_wrapper', array(
|
315
|
'project' => $project,
|
316
|
'project_status' => $project_status,
|
317
|
'languages' => $variables['languages'],
|
318
|
'available' => $variables['available'],
|
319
|
'history' => $variables['history'],
|
320
|
'updates' => $variables['updates'],
|
321
|
));
|
322
|
|
323
|
$rows[$project->project_type][] = array(
|
324
|
'data' => array(
|
325
|
array(
|
326
|
'data' => $row,
|
327
|
'class' => 'l10n-update-wrapper collapsed',
|
328
|
),
|
329
|
),
|
330
|
'class' => array($project_class),
|
331
|
);
|
332
|
}
|
333
|
|
334
|
// Build tables of update states grouped by project type. Similar to the
|
335
|
// status report by the Update module.
|
336
|
$output = '';
|
337
|
$project_types = array(
|
338
|
'core' => t('Drupal core'),
|
339
|
'module' => t('Modules'),
|
340
|
'theme' => t('Themes'),
|
341
|
'module-disabled' => t('Disabled modules'),
|
342
|
'theme-disabled' => t('Disabled themes'),
|
343
|
);
|
344
|
foreach ($project_types as $type_name => $type_label) {
|
345
|
if (!empty($rows[$type_name])) {
|
346
|
ksort($rows[$type_name]);
|
347
|
$output .= "\n<h3>" . $type_label . "</h3>\n";
|
348
|
$output .= theme('table', array('header' => $header, 'rows' => $rows[$type_name], 'attributes' => array('class' => array('update l10n-update'))));
|
349
|
}
|
350
|
}
|
351
|
|
352
|
// We use the core update module CSS to re-use the color definitions.
|
353
|
// Plus add our own css and js.
|
354
|
drupal_add_css(drupal_get_path('module', 'update') . '/update.css');
|
355
|
drupal_add_css(drupal_get_path('module', 'l10n_update') . '/css/l10n_update.admin.css');
|
356
|
drupal_add_js('misc/collapse.js');
|
357
|
drupal_add_js(drupal_get_path('module', 'l10n_update') . '/js/l10n_update.js');
|
358
|
|
359
|
return $output;
|
360
|
}
|
361
|
|
362
|
/**
|
363
|
* Format project translation state with states per language.
|
364
|
*
|
365
|
* @param $variables
|
366
|
* An associative array containing:
|
367
|
* - project: Project data object
|
368
|
* - project_status: Project status
|
369
|
* - languages: Available languages.
|
370
|
* @return string
|
371
|
* HTML output.
|
372
|
*/
|
373
|
function theme_l10n_update_single_project_wrapper($variables) {
|
374
|
$project = $variables['project'];
|
375
|
$name = $project->name;
|
376
|
$project_status = $variables['project_status'];
|
377
|
$languages = $variables['languages'];
|
378
|
$history = $variables['history'];
|
379
|
$updates = $variables['updates'];
|
380
|
$availables = $variables['available'];
|
381
|
|
382
|
// Output project title and project summary status.
|
383
|
$output = theme('l10n_update_single_project_status', array(
|
384
|
'project' => $project,
|
385
|
'server' => l10n_update_server($project->l10n_server),
|
386
|
'status' => $project_status,
|
387
|
));
|
388
|
|
389
|
// Translation status per language is displayed in a table, one language per row.
|
390
|
// For each language the current translation is listed. And optionally the
|
391
|
// most recent update.
|
392
|
$rows = array();
|
393
|
foreach ($languages as $lang => $language) {
|
394
|
// Determine current translation status and update status.
|
395
|
$installed = isset($history[$name][$lang]) ? $history[$name][$lang] : NULL;
|
396
|
$update = isset($updates[$name][$lang]) ? $updates[$name][$lang] : NULL;
|
397
|
$available = isset($availables[$name][$lang]) ? $availables[$name][$lang] : NULL;
|
398
|
if ($installed) {
|
399
|
if ($update) {
|
400
|
$status = 'updatable';
|
401
|
$class = 'messages warning';
|
402
|
}
|
403
|
else {
|
404
|
$status = 'uptodate';
|
405
|
$class = 'ok';
|
406
|
}
|
407
|
}
|
408
|
elseif ($available) {
|
409
|
$status = 'available';
|
410
|
$class = 'warning';
|
411
|
}
|
412
|
else {
|
413
|
$status = 'unknown';
|
414
|
$class = 'unknown';
|
415
|
}
|
416
|
|
417
|
// The current translation version.
|
418
|
$row = theme('l10n_update_current_release', array('language' => $language, 'release' => $installed, 'status' => $status));
|
419
|
|
420
|
// If an update is available, add it.
|
421
|
if ($update) {
|
422
|
$row .= theme('l10n_update_available_release', array('release' => $update));
|
423
|
}
|
424
|
|
425
|
$rows[] = array(
|
426
|
'data' => array($row),
|
427
|
'class' => array($class),
|
428
|
);
|
429
|
}
|
430
|
|
431
|
// Output tables with translation status per language.
|
432
|
$output .= '<div class="fieldset-wrapper">' . "\n";
|
433
|
$output .= theme('table', array('header' => array(), 'rows' => $rows));
|
434
|
$output .= "</div>\n";
|
435
|
|
436
|
return $output;
|
437
|
}
|
438
|
|
439
|
/**
|
440
|
* Format a single project translation state.
|
441
|
*
|
442
|
* @param $variables
|
443
|
* An associative array containing:
|
444
|
* - project: project data object.
|
445
|
* - server: (optional) remote server data object.
|
446
|
* - status: project summary status.
|
447
|
* @return string
|
448
|
* HTML output.
|
449
|
*/
|
450
|
function theme_l10n_update_single_project_status($variables) {
|
451
|
$project = $variables['project'];
|
452
|
$server = $variables['server'];
|
453
|
$title = $project->title ? $project->title : $project->name;
|
454
|
|
455
|
$output = '<div class="project">';
|
456
|
$output .= '<span class="project-title">' . check_plain($title) . '</span>' . ' ' . check_plain($project->module_version) ;
|
457
|
if ($server = l10n_update_server($project->l10n_server)) {
|
458
|
$output .= '<span class="project-server">' . t('(translation source: !server)', array('!server' => l($server['name'], $server['link']))) . '</span>';
|
459
|
}
|
460
|
$output .= theme('l10n_update_version_status', array('status' => $variables['status']));
|
461
|
$output .= "</div>\n";
|
462
|
|
463
|
return $output;
|
464
|
}
|
465
|
|
466
|
/**
|
467
|
* Format current translation version.
|
468
|
*
|
469
|
* @param $variables
|
470
|
* An associative array containing:
|
471
|
* - language: Language name.
|
472
|
* - release: Current file data.
|
473
|
* - status: Release status.
|
474
|
* @return string
|
475
|
* HTML output.
|
476
|
*/
|
477
|
function theme_l10n_update_current_release($variables) {
|
478
|
if (isset($variables['release'])) {
|
479
|
$date = $variables['release']->timestamp;
|
480
|
$version = $variables['release']->version;
|
481
|
$text = t('@language: @version (!date)', array('@language' => $variables['language'], '@version' => $version, '!date' => format_date($date, 'custom', 'Y-M-d')));
|
482
|
}
|
483
|
else {
|
484
|
$text = t('@language: <em>No installed translation</em>', array('@language' => $variables['language']));
|
485
|
}
|
486
|
|
487
|
$output = '<div class="language">';
|
488
|
$output .= $text;
|
489
|
$output .= theme('l10n_update_version_status', $variables);
|
490
|
$output .= "</div>\n";
|
491
|
|
492
|
return $output;
|
493
|
}
|
494
|
|
495
|
/**
|
496
|
* Format current translation version.
|
497
|
*
|
498
|
* @param object $release
|
499
|
* Update file data.
|
500
|
* @return string
|
501
|
* HTML output.
|
502
|
*/
|
503
|
function theme_l10n_update_available_release($variables) {
|
504
|
$date = $variables['release']->timestamp;
|
505
|
$version = $variables['release']->version;
|
506
|
if (!empty($variables['release']->fileurl)) {
|
507
|
// Remote file, straight link
|
508
|
$link = l(t('Download'), $variables['release']->fileurl);
|
509
|
}
|
510
|
elseif (!empty($variables['release']->uri)) {
|
511
|
// Local file, try something
|
512
|
$link = l(t('Download'), $variables['release']->uri, array('absolute' => TRUE));
|
513
|
}
|
514
|
|
515
|
$output = '<div class="version version-recommended">';
|
516
|
$output .= t('Recommended version: @version (!date)', array('@version' => $version, '!date' => format_date($date, 'custom', 'Y-M-d')));
|
517
|
$output .= '<span class="version-links">' . $link . '</span>';
|
518
|
$output .= "</div>\n";
|
519
|
return $output;
|
520
|
}
|
521
|
|
522
|
/**
|
523
|
* Format version status with icon.
|
524
|
*
|
525
|
* @param string $status
|
526
|
* Version status: 'uptodate', 'updatable', 'available', 'unknown'.
|
527
|
* @param string $type
|
528
|
* Update type: 'download', 'localfile'.
|
529
|
*
|
530
|
* @return sting
|
531
|
* HTML output.
|
532
|
*/
|
533
|
function theme_l10n_update_version_status($variables) {
|
534
|
$icon = '';
|
535
|
$msg = '';
|
536
|
|
537
|
switch ($variables['status']) {
|
538
|
case 'uptodate':
|
539
|
$icon = theme('image', array('path' => 'misc/watchdog-ok.png', 'alt' => t('ok'), 'title' => t('ok')));
|
540
|
$msg = '<span class="current">' . t('Up to date') . '</span>';
|
541
|
break;
|
542
|
case 'updatable':
|
543
|
$icon = theme('image', array('path' => 'misc/watchdog-warning.png', 'alt' => t('warning'), 'title' => t('warning')));
|
544
|
$msg = '<span class="not-current">' . t('Update available') . '</span>';
|
545
|
break;
|
546
|
case 'available':
|
547
|
$icon = theme('image', array('path' => 'misc/watchdog-warning.png', 'alt' => t('warning'), 'title' => t('warning')));
|
548
|
$msg = '<span class="not-current">' . t('Uninstalled translation available') . '</span>';
|
549
|
break;
|
550
|
case 'unknown':
|
551
|
$icon = theme('image', array('path' => 'misc/watchdog-warning.png', 'alt' => t('warning'), 'title' => t('warning')));
|
552
|
$msg = '<span class="not-supported">' . t('No available translations found') . '</span>';
|
553
|
break;
|
554
|
}
|
555
|
$output = '<div class="version-status">';
|
556
|
$output .= $msg;
|
557
|
$output .= '<span class="icon">' . $icon . '</span>';
|
558
|
$output .= "</div>\n";
|
559
|
return $output;
|
560
|
}
|