-
Notifications
You must be signed in to change notification settings - Fork 8
/
metatag.inc
777 lines (650 loc) · 23.7 KB
/
metatag.inc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
<?php
/**
* @file
* Metatag primary classes.
*/
/**
* The master interface for all tags.
*/
interface DrupalMetaTagInterface {
/**
* Constructor.
*
* @param array $info
* The information about the meta tag from metatag_get_info().
*/
function __construct(array $info, array $data = array());
public function getForm();
//function validateForm();
//function processForm();
public function getValue();
function getWeight();
function getElement();
function tidyValue($value);
function convertUrlToAbsolute($url);
function truncate($value);
function maxlength();
static function text_summary($text, $size);
}
/**
* The default meta tag class from which all others inherit.
*/
class DrupalDefaultMetaTag implements DrupalMetaTagInterface {
/**
* All of the basic information about this tag.
*
* @var array
*/
protected $info;
/**
* The values submitted for this tag.
*
* @var array
*/
protected $data = array('value' => '');
/**
* This item's weight; used for sorting the output.
*
* @var float
*/
protected $weight = 0;
/**
* Constructor.
*/
function __construct(array $info, array $data = NULL) {
$this->info = $info;
if (isset($data)) {
$this->data = $data;
}
}
/**
* Calculate the weight of this meta tag.
*
* @return int
* Weight.
*/
function getWeight() {
static $counter = 0;
// If no weight value is found, stack this meta tag at the end.
$weight = 100;
if (!empty($this->info['weight'])) {
$weight = $this->info['weight'];
}
return $weight + ($counter++ * 0.1);
}
/**
* Build the form for this meta tag.
*
* @return array
* A standard FormAPI array.
*/
public function getForm(array $options = array()) {
return array();
}
/**
* Get the string value of this meta tag.
*
* @return string
* The value of this meta tag.
*/
public function getValue(array $options = array()) {
$value = $this->tidyValue($this->data['value']);
// Translate the final output string prior to output. Use the
// 'output' i18n_string object type, and pass along the meta tag's
// options as the context so it can be handled appropriately.
$value = metatag_translate_metatag($value, $this->info['name'], $options, NULL, TRUE);
return $this->truncate($this->tidyValue($this->data['value']));
}
/**
* Get the HTML tag for this meta tag.
*
* @return array
* A render array for this meta tag.
*/
public function getElement(array $options = array()) {
$value = $this->getValue($options);
if (strlen($value) === 0) {
return array();
}
// The stack of elements that will be output.
$elements = array();
// Dynamically add each option to this setting.
$base_element = isset($this->info['element']) ? $this->info['element'] : array();
// Single item.
if (empty($this->info['multiple'])) {
$values = array($value);
}
// Multiple items.
else {
$values = array_filter(explode(',', $value));
}
// Loop over each item.
if (!empty($values)) {
foreach ($values as $ctr => $value) {
$value = trim($value);
// Some meta tags must be output as secure URLs.
if (!empty($this->info['secure'])) {
$value = str_replace('http://', 'https://', $value);
}
// Combine the base configuration for this meta tag with the value.
$element = $base_element + array(
'#theme' => 'metatag',
'#tag' => 'meta',
'#id' => 'metatag_' . $this->info['name'] . '_' . $ctr,
'#name' => $this->info['name'],
'#value' => $value,
'#weight' => $this->getWeight(),
);
// Add header information if desired.
if (!empty($this->info['header'])) {
$element['#attached']['drupal_add_http_header'][] = array($this->info['header'], $value);
}
$elements[] = array($element, $element['#id']);
}
}
if (!empty($elements)) {
return array(
'#attached' => array('drupal_add_html_head' => $elements),
);
}
}
/**
* Remove unwanted formatting from a meta tag.
*
* @param string $value
* The meta tag value to be tidied up.
*
* @return string
* The meta tag value after it has been tidied up.
*/
public function tidyValue($value) {
// This shouldn't happen, but protect against tokens returning arrays.
if (!is_string($value)) {
return '';
}
// Check for Media strings from the WYSIWYG submodule.
if (module_exists('media_wysiwyg') && strpos($value, '[[{') !== FALSE) {
// In https://www.drupal.org/node/2129273 media_wysiwyg_filter() was
// changed to require several additional arguments.
$langcode = language_default('language');
$value = media_wysiwyg_filter($value, NULL, NULL, $langcode, NULL, NULL);
}
// Specifically replace encoded spaces, because some WYSIWYG editors are
// silly. Do this before decoding the other HTML entities so that the output
// doesn't end up with a bunch of a-circumflex characters.
$value = str_replace(' ', ' ', $value);
// Decode HTML entities.
$value = decode_entities($value);
// First off, remove the <style> tag, because strip_tags() leaves the CSS
// inline.
$value = preg_replace('/<style\b[^>]*>(.*?)<\/style>/is', '', $value);
// Ditto for JavaScript.
$value = preg_replace('/<script\b[^>]*>(.*?)<\/script>/is', '', $value);
// Remove any HTML code that might have been included.
$value = strip_tags($value);
// Strip errant whitespace.
$value = str_replace(array("\r\n", "\n", "\r", "\t"), ' ', $value);
$value = str_replace(' ', ' ', $value);
$value = str_replace(' ', ' ', $value);
$value = trim($value);
return $value;
}
/**
* Make sure a given URL is absolute.
*
* @param string $url
* The URL to convert to an absolute URL.
*
* @return string
* The argument converted to an absolute URL.
*/
function convertUrlToAbsolute($url) {
// Convert paths relative to the hostname, that start with a slash, to
// ones that are relative to the Drupal root path; ignore protocol-relative
// URLs.
if (strpos($url, base_path()) === 0 && strpos($url, '//') !== 0) {
// Logic:
// * Get the length of the base_path(),
// * Get a portion of the image's path starting from the position equal
// to the base_path()'s length; this will result in a path relative
// to the Drupal installation's base directory.
$len = strlen(base_path());
$url = substr($url, $len);
}
// Pass everything else through file_create_url(). The alternative is to
// use url() but it would insert '?q=' into the path.
return file_create_url($url);
}
/**
* Shorten a string to a certain length using text_summary().
*
* @param string $value
* String to shorten.
*
* @return string
* Shortened string.
*/
function truncate($value) {
$maxlength = $this->maxlength();
if (!empty($value) && $maxlength > 0) {
$value = $this->text_summary($value, $maxlength);
}
return $value;
}
/**
* Identify the maximum length of which strings will be allowed.
*
* @return int
* Maxlenght.
*/
function maxlength() {
if (isset($this->info['maxlength'])) {
return intval(variable_get('metatag_maxlength_' . $this->info['name'], $this->info['maxlength']));
}
return 0;
}
/**
* Copied from text.module with the following changes:.
*
* Change 1: $size is required.
* Change 2: $format is removed.
* Change 3: Don't trim at the end of short sentences
* (https://www.drupal.org/node/1620104).
* Change 4: Word boundaries (https://www.drupal.org/node/1482178).
* Change 5: Trim the final string.
*/
static function text_summary($text, $size) {
// if (!isset($size)) {
// // What used to be called 'teaser' is now called 'summary', but
// // the variable 'teaser_length' is preserved for backwards compatibility.
// $size = variable_get('teaser_length', 600);
// }
// Find where the delimiter is in the body.
$delimiter = strpos($text, '<!--break-->');
// If the size is zero, and there is no delimiter,
// the entire body is the summary.
if ($size == 0 && $delimiter === FALSE) {
return $text;
}
// If a valid delimiter has been specified, use it to chop off the summary.
if ($delimiter !== FALSE) {
return substr($text, 0, $delimiter);
}
// We check for the presence of the PHP evaluator filter in the current
// format. If the body contains PHP code, we do not split it up to prevent
// parse errors.
// if (isset($format)) {
// $filters = filter_list_format($format);
// if (isset($filters['php_code']) && $filters['php_code']->status && strpos($text, '<?') !== FALSE) {
// return $text;
// }
// }
// If we have a short body, the entire body is the summary.
if (drupal_strlen($text) <= $size) {
return $text;
}
// If the delimiter has not been specified, try to split at paragraph or
// sentence boundaries.
// The summary may not be longer than maximum length specified.
// Initial slice.
$summary = truncate_utf8($text, $size);
// Store the actual length of the UTF8 string -- which might not be the same
// as $size.
$max_rpos = strlen($summary);
// How much to cut off the end of the summary so that it doesn't end in the
// middle of a paragraph, sentence, or word.
// Initialize it to maximum in order to find the minimum.
$min_rpos = $max_rpos;
// Store the reverse of the summary. We use strpos on the reversed needle
// and haystack for speed and convenience.
$reversed = strrev($summary);
// Build an array of arrays of break points grouped by preference.
$break_points = array();
// A paragraph near the end of sliced summary is most preferable.
$break_points[] = array('</p>' => 0);
// If no complete paragraph then treat line breaks as paragraphs.
// $line_breaks = array('<br />' => 6, '<br>' => 4);
// Newline only indicates a line break if line break converter
// filter is present.
// if (isset($filters['filter_autop'])) {
// $line_breaks["\n"] = 1;
// }
// $break_points[] = $line_breaks;
// If the first paragraph is too long, split at the end of a sentence.
// $break_points[] = array('. ' => 1, '! ' => 1, '? ' => 1, '。' => 0, '؟ ' => 1);
// From https://www.drupal.org/node/1482178.
// If the first sentence is too long, split at the first word break.
$word_breaks = array(' ' => 0, "\t" => 0);
$break_points[] = $word_breaks;
// Iterate over the groups of break points until a break point is found.
foreach ($break_points as $points) {
// Look for each break point, starting at the end of the summary.
foreach ($points as $point => $offset) {
// The summary is already reversed, but the break point isn't.
$rpos = strpos($reversed, strrev($point));
if ($rpos !== FALSE) {
$min_rpos = min($rpos + $offset, $min_rpos);
}
}
// If a break point was found in this group, slice and stop searching.
if ($min_rpos !== $max_rpos) {
// Don't slice with length 0. Length must be <0 to slice from RHS.
$summary = ($min_rpos === 0) ? $summary : substr($summary, 0, 0 - $min_rpos);
break;
}
}
// If the htmlcorrector filter is present, apply it to the generated
// summary.
// if (isset($filters['filter_htmlcorrector'])) {
// $summary = _filter_htmlcorrector($summary);
// }
return trim($summary);
}
}
/**
* Text-based meta tag controller.
*/
class DrupalTextMetaTag extends DrupalDefaultMetaTag {
/**
* {@inheritdoc}
*/
public function getForm(array $options = array()) {
$options += array(
'token types' => array(),
);
$form['value'] = isset($this->info['form']) ? $this->info['form'] : array();
$form['value'] += array(
'#type' => 'textfield',
'#title' => $this->info['label'],
'#description' => !empty($this->info['description']) ? $this->info['description'] : '',
'#default_value' => isset($this->data['value']) ? $this->data['value'] : '',
'#element_validate' => array('token_element_validate'),
'#token_types' => $options['token types'],
'#maxlength' => 1024,
);
// Optional handling for items that allow multiple values.
if (!empty($this->info['multiple'])) {
$form['value']['#description'] .= ' ' . t('Multiple values may be used, separated by a comma. Note: Tokens that return multiple values will be handled automatically.');
}
// Optionally limit the field to a certain length.
$maxlength = $this->maxlength();
if (!empty($maxlength)) {
$form['value']['#description'] .= ' ' . t('This will be truncated to a maximum of %max characters.', array('%max' => $maxlength));
}
// Optional handling for images.
if (!empty($this->info['image'])) {
$form['value']['#description'] .= ' ' . t('This will be able to extract the URL from an image field.');
}
// Optional handling for languages.
if (!empty($this->info['is_language'])) {
$form['value']['#description'] .= ' ' . t('This will not be displayed if it is set to the "Language neutral" (i.e. "und").');
}
// Optional support for select_or_other.
if ($form['value']['#type'] == 'select' && !empty($this->info['select_or_other']) && module_exists('select_or_other')) {
$form['value']['#type'] = 'select_or_other';
$form['value']['#other'] = t('Other (please type a value)');
$form['value']['#multiple'] = FALSE;
$form['value']['#other_unknown_defaults'] = 'other';
$form['value']['#other_delimiter'] = FALSE;
$form['value']['#theme'] = 'select_or_other';
$form['value']['#select_type'] = 'select';
$form['value']['#element_validate'] = array('select_or_other_element_validate');
}
// Support for dependencies, using Form API's #states system.
// @see metatag.api.php.
// @see https://api.drupal.org/drupal_process_states
if (!empty($this->info['dependencies'])) {
foreach ($this->info['dependencies'] as $specs) {
$form['value']['#states']['visible'][':input[name*="[' . $specs['dependency'] . '][' . $specs['attribute'] . ']"]'] = array(
$specs['condition'] => $specs['value'],
);
}
}
return $form;
}
/**
* {@inheritdoc}
*/
public function getValue(array $options = array()) {
$options += array(
'instance' => '',
'token data' => array(),
// Remove any remaining token after the string is parsed.
'clear' => TRUE,
'sanitize' => variable_get('metatag_token_sanitize', FALSE),
'raw' => FALSE,
);
// If the value wasn't set there's no point in proceeding.
if (!isset($this->data['value'])) {
return '';
}
$value = $this->data['value'];
if (empty($options['raw'])) {
// Keep a copy of the original body value from before the summary string
// is extracted, so that this doesn't break output from other modules.
$old_value = NULL;
// There can be problems extracting the [node:summary] token due to
// certain modules using custom placeholders, e.g. Media WYSIWYG. To avoid
// that problem, the string needs to be filtered using tidyValue() before
// the tokens are processed.
if (strpos($value, '[node:summary]') !== FALSE) {
// Make sure there is a node to work with.
if (isset($options['token data']['node'])) {
// Get language to use for selecting body field value.
$lang = field_language('node', $options['token data']['node'], 'body');
if (!empty($options['token data']['node']->body[$lang][0]['value'])) {
$old_value = $options['token data']['node']->body[$lang][0]['value'];
// Pre-tidy the node body for token_replace if it's not empty.
$options['token data']['node']->body[$lang][0]['value'] = $this->tidyValue($old_value);
}
}
}
// Give other modules the opportunity to use hook_metatag_pattern_alter()
// to modify defined token patterns and values before replacement.
drupal_alter('metatag_pattern', $value, $options['token data'], $this->info['name']);
$value = token_replace($value, $options['token data'], $options);
// Put back the original value, if one was retained earlier.
if (!is_null($old_value)) {
$options['token data']['node']->body[$lang][0]['value'] = $old_value;
}
}
// Special handling for language meta tags.
if (!empty($this->info['is_language'])) {
// If the meta tag value equals LANGUAGE_NONE, i.e. "und", then don't
// output it.
if (is_string($value) && $value == LANGUAGE_NONE) {
$value = '';
}
}
// Special handling for images and other URLs.
if (!empty($this->info['image']) || !empty($this->info['url'])) {
// Support multiple items, whether it's needed or not. Also remove the
// empty values and reindex the array.
$values = array_values(array_filter(explode(',', $value)));
// If this meta tag does *not* allow multiple items, only keep the first
// one.
if (empty($this->info['multiple']) && !empty($values[0])) {
$values = array($values[0]);
}
foreach ($values as $key => &$image_value) {
// Remove any unwanted whitespace around the value.
$image_value = trim($image_value);
// If this contains embedded image tags, extract the image URLs.
if (!empty($this->info['image']) && strip_tags($image_value) != $image_value) {
$matches = array();
preg_match('/src="([^"]*)"/', $image_value, $matches);
if (!empty($matches[1])) {
$image_value = $matches[1];
}
}
// Convert the URL to an absolute URL.
$image_value = $this->convertUrlToAbsolute($image_value);
// Replace spaces the URL encoded entity to avoid validation problems.
$image_value = str_replace(' ', '%20', $image_value);
}
// Combine the multiple values into a single string.
$value = implode(',', $values);
}
// Clean up the string a bit.
$value = $this->tidyValue($value);
// Optionally truncate the value.
$value = $this->truncate($value);
// Translate the final output string prior to output. Use the
// 'output' i18n_string object type, and pass along the meta tag's
// options as the context so it can be handled appropriately.
$value = metatag_translate_metatag($value, $this->info['name'], $options, NULL, TRUE);
return $value;
}
}
/**
* Link type meta tag controller.
*/
class DrupalLinkMetaTag extends DrupalTextMetaTag {
/**
* {@inheritdoc}
*/
public function getElement(array $options = array()) {
$element = isset($this->info['element']) ? $this->info['element'] : array();
$value = $this->getValue($options);
if (strlen($value) === 0) {
return array();
}
$element += array(
'#theme' => 'metatag_link_rel',
'#tag' => 'link',
'#id' => 'metatag_' . $this->info['name'],
'#name' => $this->info['name'],
'#value' => $value,
'#weight' => $this->getWeight(),
);
if (!isset($this->info['header']) || !empty($this->info['header'])) {
// Also send the generator in the HTTP header.
// @todo This does not support 'rev' or alternate link headers.
$element['#attached']['drupal_add_http_header'][] = array(
'Link', '<' . $value . '>;' . drupal_http_header_attributes(array('rel' => $element['#name'])), TRUE,
);
}
return array(
'#attached' => array('drupal_add_html_head' => array(array($element, $element['#id']))),
);
}
}
/**
* Title meta tag controller.
*
* This extends DrupalTextMetaTag as we need to alter variables in
* template_preprocess_html() rather output a normal meta tag.
*/
class DrupalTitleMetaTag extends DrupalTextMetaTag {
/**
* {@inheritdoc}
*/
public function getElement(array $options = array()) {
$element = array();
if ($value = $this->getValue($options)) {
$element['#attached']['metatag_set_preprocess_variable'][] = array(
'html',
'head_title',
decode_entities($value),
);
$element['#attached']['metatag_set_preprocess_variable'][] = array(
'html',
'head_array',
array('title' => $value),
);
}
return $element;
}
}
/**
* Multiple value meta tag controller.
*/
class DrupalListMetaTag extends DrupalDefaultMetaTag {
/**
* {@inheritdoc}
*/
function __construct(array $info, array $data = NULL) {
// Ensure that the $data['value] argument is an array.
if (empty($data['value'])) {
$data['value'] = array();
}
$data['value'] = (array) $data['value'];
parent::__construct($info, $data);
}
/**
* {@inheritdoc}
*/
public function getForm(array $options = array()) {
$form['value'] = isset($this->info['form']) ? $this->info['form'] : array();
$form['value'] += array(
'#type' => 'checkboxes',
'#title' => $this->info['label'],
'#description' => !empty($this->info['description']) ? $this->info['description'] : '',
'#default_value' => isset($this->data['value']) ? $this->data['value'] : array(),
);
return $form;
}
/**
* {@inheritdoc}
*/
public function getValue(array $options = array()) {
$values = array_keys(array_filter($this->data['value']));
sort($values);
$value = implode(', ', $values);
$value = $this->tidyValue($value);
// Translate the final output string prior to output. Use the
// 'output' i18n_string object type, and pass along the meta tag's
// options as the context so it can be handled appropriately.
$value = metatag_translate_metatag($value, $this->info['name'], $options, NULL, TRUE);
return $value;
}
}
/**
* Date interval meta tag controller.
*/
class DrupalDateIntervalMetaTag extends DrupalDefaultMetaTag {
/**
* {@inheritdoc}
*/
public function getForm(array $options = array()) {
$form['value'] = array(
'#type' => 'textfield',
'#title' => t('!title interval', array('!title' => $this->info['label'])),
'#default_value' => isset($this->data['value']) ? $this->data['value'] : '',
'#element_validate' => array('element_validate_integer_positive'),
'#maxlength' => 4,
'#description' => isset($this->info['description']) ? $this->info['description'] : '',
);
$form['period'] = array(
'#type' => 'select',
'#title' => t('!title interval type', array('!title' => $this->info['label'])),
'#default_value' => isset($this->data['period']) ? $this->data['period'] : '',
'#options' => array(
'' => t('- none -'),
'day' => t('Day(s)'),
'week' => t('Week(s)'),
'month' => t('Month(s)'),
'year' => t('Year(s)'),
),
);
return $form;
}
/**
* {@inheritdoc}
*/
public function getValue(array $options = array()) {
$value = '';
if (!empty($this->data['value'])) {
$interval = intval($this->data['value']);
if (!empty($interval) && !empty($this->data['period'])) {
$period = $this->data['period'];
$value = format_plural($interval, '@count ' . $period, '@count ' . $period . 's');
}
}
// Translate the final output string prior to output. Use the 'output'
// i18n_string object type, and pass along the meta tag's options as the
// context so it can be handled appropriately.
$value = metatag_translate_metatag($value, $this->info['name'], $options, NULL, TRUE);
return $value;
}
}