View source
<?php
namespace Drupal\typogrify\Plugin\Filter;
use Drupal\typogrify\SmartyPants;
use Drupal\typogrify\Typogrify;
use Drupal\typogrify\UnicodeConversion;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Language\LanguageInterface;
use Drupal\filter\FilterProcessResult;
use Drupal\filter\Plugin\FilterBase;
class TypogrifyFilter extends FilterBase {
const TYPOGRIFY_VERSION = '1.0';
protected static $arraySettingsKeys = [
'ligatures',
'arrows',
'fractions',
'quotes',
];
protected static function settingsSerialize(array &$settings) {
foreach (static::$arraySettingsKeys as $key) {
if (isset($settings[$key]) && is_array($settings[$key])) {
$settings[$key] = serialize(array_filter($settings[$key]));
}
}
}
protected static function settingsUnserialize(array &$settings) {
foreach (static::$arraySettingsKeys as $key) {
if (isset($settings[$key]) && is_string($settings[$key])) {
$settings[$key] = unserialize($settings[$key]);
}
}
}
public function settingsForm(array $form, FormStateInterface $form_state) {
$settings = $this->settings;
static::settingsUnserialize($settings);
$form['help'] = [
'#type' => 'markup',
'#value' => '<p>' . $this
->t('Enable the following typographic refinements:') . '</p>',
];
$form['smartypants_enabled'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Use typographers quotation marks and dashes (<a href="@smarty-pants-url">SmartyPants</a>)', [
'@smarty-pants-url' => 'http://daringfireball.net/projects/smartypants',
]),
'#default_value' => $settings['smartypants_enabled'],
];
$form['smartypants_hyphens'] = [
'#type' => 'select',
'#title' => $this
->t('Dash replacement settings for SmartyPants'),
'#default_value' => $settings['smartypants_hyphens'],
'#options' => [
1 => $this
->t('“--” for em-dashes; no en-dash support'),
3 => $this
->t('“--” for em-dashes; “---” for en-dashes'),
2 => $this
->t('“---” for em-dashes; “--” for en-dashes'),
],
];
$form['space_hyphens'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Replace stand-alone dashes (normal dashes between whitespace) em-dashes.'),
'#description' => $this
->t('" - " will turn into " — ".'),
'#default_value' => $settings['space_hyphens'],
];
$form['widont_enabled'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Remove widows'),
'#default_value' => $settings['widont_enabled'],
];
$form['hyphenate_shy'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Replace <code>=</code> with <code>&shy;</code>'),
'#description' => $this
->t('Words may be broken at the hyphenation points marked by “=”.'),
'#default_value' => $settings['hyphenate_shy'],
];
$form['space_to_nbsp'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Replace normal spaces with non-breaking spaces before "double punctuation marks" <code>!?:;</code>.'),
'#description' => $this
->t('This is especially useful for french.'),
'#default_value' => $settings['space_to_nbsp'],
];
$form['wrap_caps'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Wrap caps'),
'#default_value' => $settings['wrap_caps'],
];
$form['wrap_ampersand'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Wrap ampersands'),
'#default_value' => $settings['wrap_ampersand'],
];
$form['wrap_abbr'] = [
'#type' => 'select',
'#title' => $this
->t('Thin space in abbreviations'),
'#description' => $this
->t('Wraps abbreviations with <code>@wrapper-code</code> and inserts space after the dots.', [
'@wrapper-code' => '<span class="abbr">…</span>',
]),
'#default_value' => $settings['wrap_abbr'],
'#options' => [
0 => $this
->t('Do nothing'),
4 => $this
->t('Insert no space'),
1 => $this
->t('“U+202F“ Narrow no-break space'),
2 => $this
->t('“U+2009“ Thin space'),
3 => $this
->t('span with margin-left: 0.167em'),
],
];
$form['wrap_numbers'] = [
'#type' => 'select',
'#title' => $this
->t('Digit grouping in numbers'),
'#description' => $this
->t('Wraps numbers with <code>@wrapper-code</code> and inserts thin space for digit grouping.', [
'@wrapper-code' => '<span class="number">…</span>',
]),
'#default_value' => $settings['wrap_numbers'],
'#options' => [
0 => $this
->t('Do nothing'),
1 => $this
->t('“U+202F“ Narrow no-break space'),
2 => $this
->t('“U+2009“ Thin space'),
3 => $this
->t('span with margin-left: 0.167em'),
4 => $this
->t('just wrap numbers'),
],
];
$form['wrap_initial_quotes'] = [
'#type' => 'checkbox',
'#title' => $this
->t('Wrap quotation marks'),
'#default_value' => $settings['wrap_initial_quotes'],
];
$ligature_options = [];
foreach (UnicodeConversion::map('ligature') as $ascii => $unicode) {
$ligature_options[$ascii] = $this
->t('Convert <code>@ascii</code> to <code>@unicode</code>', [
'@ascii' => $ascii,
'@unicode' => $unicode,
]);
}
$form['ligatures'] = [
'#type' => 'checkboxes',
'#title' => $this
->t('Ligatures'),
'#options' => $ligature_options,
'#default_value' => $settings['ligatures'],
];
$arrow_options = [];
foreach (UnicodeConversion::map('arrow') as $ascii => $unicode) {
$arrow_options[$ascii] = $this
->t('Convert <code>@ascii</code> to <code>@unicode</code>', [
'@ascii' => $this
->unquote($ascii),
'@unicode' => $unicode,
]);
}
$form['arrows'] = [
'#type' => 'checkboxes',
'#title' => $this
->t('Arrows'),
'#options' => $arrow_options,
'#default_value' => $settings['arrows'],
];
$fraction_options = [];
foreach (UnicodeConversion::map('fraction') as $ascii => $unicode) {
$fraction_options[$ascii] = $this
->t('Convert <code>@ascii</code> to <code>@unicode</code>', [
'@ascii' => $ascii,
'@unicode' => $unicode,
]);
}
$form['fractions'] = [
'#type' => 'checkboxes',
'#title' => $this
->t('Fractions'),
'#options' => $fraction_options,
'#default_value' => $settings['fractions'],
];
$quotes_options = [];
foreach (UnicodeConversion::map('quotes') as $quotes => $unicode) {
$quotes_options[$quotes] = $this
->t('Convert <code>@ascii</code> to <code>@unicode</code>', [
'@ascii' => $this
->unquote($quotes),
'@unicode' => $unicode,
]);
}
$form['quotes'] = [
'#type' => 'checkboxes',
'#title' => $this
->t('Quotes'),
'#options' => $quotes_options,
'#default_value' => $settings['quotes'],
];
$version_strings = [];
$version_strings[] = $this
->t('SmartyPants PHP version: <a href=":smarty-pants-url">@version</a>', [
':smarty-pants-url' => 'http://www.michelf.com/projects/php-smartypants/',
'@version' => SmartyPants::SMARTYPANTS_PHP_VERSION,
]);
$version_strings[] = $this
->t('PHP Typogrify Version: <a href=":php-typogrify-url">@version</a>', [
':php-typogrify-url' => 'http://blog.hamstu.com/',
'@version' => static::TYPOGRIFY_VERSION,
]);
$form['info']['typogrify_status'] = [
'#theme' => 'item_list',
'#items' => $version_strings,
'#title' => $this
->t('Versions'),
];
return $form;
}
public function setConfiguration(array $configuration) {
static::settingsSerialize($configuration['settings']);
parent::setConfiguration($configuration);
}
public function process($text, $langcode) {
$settings = $this->settings;
static::settingsUnserialize($settings);
$characters_to_convert = [];
$ctx = [];
if ($langcode == LanguageInterface::LANGCODE_NOT_SPECIFIED) {
$language = \Drupal::languageManager()
->getCurrentLanguage();
$ctx['langcode'] = $language
->getId();
}
else {
$ctx['langcode'] = $langcode;
}
foreach (UnicodeConversion::map('ligature') as $ascii => $unicode) {
if (isset($settings['ligatures'][$ascii]) && $settings['ligatures'][$ascii]) {
$characters_to_convert[] = $ascii;
}
}
if ($settings['wrap_caps']) {
$text = Typogrify::caps($text);
}
foreach (UnicodeConversion::map('arrow') as $ascii => $unicode) {
$htmle = $this
->unquote($ascii);
if (isset($settings['arrows'][$ascii]) && $settings['arrows'][$ascii] || isset($settings['arrows'][$htmle]) && $settings['arrows'][$htmle]) {
$characters_to_convert[] = $ascii;
}
}
foreach (UnicodeConversion::map('fraction') as $ascii => $unicode) {
if (isset($settings['fractions'][$ascii]) && $settings['fractions'][$ascii]) {
$characters_to_convert[] = $ascii;
}
}
foreach (UnicodeConversion::map('quotes') as $ascii => $unicode) {
if (isset($settings['quotes'][$ascii]) && $settings['quotes'][$ascii]) {
$characters_to_convert[] = $ascii;
}
}
if (count($characters_to_convert) > 0) {
$text = UnicodeConversion::convertCharacters($text, $characters_to_convert);
}
if ($settings['wrap_ampersand']) {
$text = SmartyPants::smartAmpersand($text);
}
if ($settings['smartypants_enabled']) {
$text = SmartyPants::process($text, $settings['smartypants_hyphens'], $ctx);
}
if ($settings['wrap_abbr'] > 0) {
$text = SmartyPants::smartAbbreviation($text, $settings['wrap_abbr']);
}
if ($settings['wrap_numbers'] > 0) {
$text = SmartyPants::smartNumbers($text, $settings['wrap_numbers']);
}
if ($settings['wrap_initial_quotes']) {
$text = Typogrify::initialQuotes($text);
}
if ($settings['hyphenate_shy']) {
$text = SmartyPants::hyphenate($text);
}
if ($settings['widont_enabled']) {
$text = Typogrify::widont($text);
}
if (isset($settings['space_to_nbsp']) && $settings['space_to_nbsp']) {
$text = SmartyPants::spaceToNbsp($text);
}
if (isset($settings['space_hyphens']) && $settings['space_hyphens']) {
$text = SmartyPants::spaceHyphens($text);
}
$result = new FilterProcessResult($text);
$result
->setAttachments([
'library' => [
'typogrify/typogrify',
],
]);
return $result;
}
public function tips($long = FALSE) {
$settings = $this->settings;
static::settingsUnserialize($settings);
if ($long) {
$output = $this
->t('Typogrify.module brings the typographic refinements of Typogrify to Drupal.');
$output .= '<ul>';
if ($settings['wrap_ampersand']) {
$output .= '<li>' . $this
->t('Wraps ampersands (the “&” character) with <code>@wrapper-code</code>.', [
'@wrapper-code' => '<span class="amp">&</span>',
]) . '</li>';
}
if ($settings['widont_enabled']) {
$output .= '<li>' . $this
->t("Prevents single words from wrapping onto their own line using Shaun Inman's Widont technique.") . '</li>';
}
if ($settings['wrap_initial_quotes']) {
$output .= '<li>' . $this
->t("Converts straight quotation marks to typographer's quotation marks, using SmartyPants.");
$output .= '</li><li>' . $this
->t('Wraps initial quotation marks with <code>@wrapper-code-quote</code> or <code>@wrapper-code-dquote</code>.', [
'@wrapper-code-quote' => '<span class="quo"></span>;',
'@wrapper-code-dquote' => '<span class="dquo"></span>;',
]) . '</li>';
}
$output .= $this
->t('<li>Converts multiple hyphens to en dashes and em dashes (according to your preferences), using SmartyPants.</li>');
if ($settings['hyphenate_shy']) {
$output .= '<li>' . $this
->t('Words may be broken at the hyphenation points marked by “=”.') . '</li>';
}
if ($settings['wrap_abbr']) {
$output .= '<li>' . $this
->t('Wraps abbreviations as “e.g.” to <code>@wrapper-code</code> and adds a thin space (1/6 em) after the dots.</li>', [
'@wrapper-code' => '<span class="abbr">e.g.</span>',
]) . '</li>';
}
if ($settings['wrap_numbers']) {
$output .= '<li>' . $this
->t('Wraps large numbers > 1 000 with <code>@wrapper-code</code> and inserts thin space for digit grouping.', [
'@wrapper-code' => '<span class="number">…</span>',
]) . '</li>';
}
if ($settings['wrap_caps']) {
$output .= '<li>' . $this
->t('Wraps multiple capital letters with <code>@wrapper-code</code>.', [
'@wrapper-code' => '<span class="caps">CAPS</span>',
]) . '</li>';
}
$output .= '<li>' . $this
->t('Adds a css style sheet that uses the <span> tags to substitute a showy ampersand in headlines, switch caps to small caps, and hang initial quotation marks.') . '</li>';
foreach (UnicodeConversion::map('quotes') as $ascii => $unicode) {
if (!empty($settings['quotes'][$ascii])) {
$ascii_to_unicode = $this
->t('Converts <code>@ascii</code> to @unicode', [
'@ascii' => $ascii,
'@unicode' => $unicode,
]);
$output .= "<li>{$ascii_to_unicode}</li>\n";
}
}
$output .= '</ul>';
}
else {
$output = $this
->t('Typographic refinements will be added.');
}
return $output;
}
private function unquote($text) {
$text = str_replace([
'<',
'>',
], [
'<',
'>',
], $text);
return $text;
}
}