1339 lines
56 KiB
PHP
1339 lines
56 KiB
PHP
<?php
|
|
/*
|
|
Plugin Name: Crayon Syntax Highlighter
|
|
Plugin URI: https://github.com/aramk/crayon-syntax-highlighter
|
|
Description: Supports multiple languages, themes, highlighting from a URL, local file or post text.
|
|
Version: 2.8.6
|
|
Author: Aram Kocharyan
|
|
Author URI: http://aramk.com/
|
|
Text Domain: crayon-syntax-highlighter
|
|
Domain Path: /trans/
|
|
License: GPL2
|
|
Copyright 2013 Aram Kocharyan (email : akarmenia@gmail.com)
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License, version 2, as
|
|
published by the Free Software Foundation.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
require_once('global.php');
|
|
require_once(CRAYON_HIGHLIGHTER_PHP);
|
|
if (CRAYON_TAG_EDITOR) {
|
|
require_once(CRAYON_TAG_EDITOR_PHP);
|
|
}
|
|
if (CRAYON_THEME_EDITOR) {
|
|
require_once(CRAYON_THEME_EDITOR_PHP);
|
|
}
|
|
require_once('crayon_settings_wp.class.php');
|
|
|
|
crayon_set_info(array(
|
|
'Version' => '_2.7.2_beta',
|
|
'Date' => '25th April, 2015',
|
|
'AuthorName' => 'Aram Kocharyan',
|
|
'PluginURI' => 'https://github.com/aramk/crayon-syntax-highlighter',
|
|
));
|
|
|
|
/* The plugin class that manages all other classes and integrates Crayon with WP */
|
|
|
|
class CrayonWP {
|
|
// Properties and Constants ===============================================
|
|
|
|
// Associative array, keys are post IDs as strings and values are number of crayons parsed as ints
|
|
private static $post_queue = array();
|
|
// Ditto for comments
|
|
private static $comment_queue = array();
|
|
private static $post_captures = array();
|
|
private static $comment_captures = array();
|
|
// Whether we are displaying an excerpt
|
|
private static $is_excerpt = FALSE;
|
|
// Whether we have added styles and scripts
|
|
private static $enqueued = FALSE;
|
|
// Whether we have already printed the wp head
|
|
private static $wp_head = FALSE;
|
|
// Used to keep Crayon IDs
|
|
private static $next_id = 0;
|
|
// String to store the regex for capturing tags
|
|
private static $alias_regex = '';
|
|
private static $tags_regex = '';
|
|
private static $tags_regex_legacy = '';
|
|
private static $tag_regexes = array();
|
|
// Defined constants used in bitwise flags
|
|
private static $tag_types = array(
|
|
CrayonSettings::CAPTURE_MINI_TAG,
|
|
CrayonSettings::CAPTURE_PRE,
|
|
CrayonSettings::INLINE_TAG,
|
|
CrayonSettings::PLAIN_TAG,
|
|
CrayonSettings::BACKQUOTE);
|
|
private static $tag_bits = array();
|
|
// Used to find legacy tags
|
|
private static $legacy_flags = NULL;
|
|
|
|
// Used to detect the shortcode
|
|
private static $allowed_atts = array('url' => NULL, 'lang' => NULL, 'title' => NULL, 'mark' => NULL, 'range' => NULL, 'inline' => NULL);
|
|
const REGEX_CLOSED = '(?:\[\s*crayon(?:-(\w+))?\b([^\]]*)/\s*\])'; // [crayon atts="" /]
|
|
const REGEX_TAG = '(?:\[\s*crayon(?:-(\w+))?\b([^\]]*)\](.*?)\[\s*/\s*crayon\s*\])'; // [crayon atts=""] ... [/crayon]
|
|
const REGEX_INLINE_CLASS = '\bcrayon-inline\b';
|
|
|
|
const REGEX_CLOSED_NO_CAPTURE = '(?:\[\s*crayon\b[^\]]*/\])';
|
|
const REGEX_TAG_NO_CAPTURE = '(?:\[\s*crayon\b[^\]]*\].*?\[/crayon\])';
|
|
|
|
const REGEX_QUICK_CAPTURE = '(?:\[\s*crayon[^\]]*\].*?\[\s*/\s*crayon\s*\])|(?:\[\s*crayon[^\]]*/\s*\])';
|
|
|
|
const REGEX_BETWEEN_PARAGRAPH = '<p[^<]*>(?:[^<]*<(?!/?p(\s+[^>]*)?>)[^>]+(\s+[^>]*)?>)*[^<]*((?:\[\s*crayon[^\]]*\].*?\[\s*/\s*crayon\s*\])|(?:\[\s*crayon[^\]]*/\s*\]))(?:[^<]*<(?!/?p(\s+[^>]*)?>)[^>]+(\s+[^>]*)?>)*[^<]*</p[^<]*>';
|
|
const REGEX_BETWEEN_PARAGRAPH_SIMPLE = '(<p(?:\s+[^>]*)?>)(.*?)(</p(?:\s+[^>]*)?>)';
|
|
|
|
// For [crayon-id/]
|
|
const REGEX_BR_BEFORE = '#<\s*br\s*/?\s*>\s*(\[\s*crayon-\w+\])#msi';
|
|
const REGEX_BR_AFTER = '#(\[\s*crayon-\w+\])\s*<\s*br\s*/?\s*>#msi';
|
|
|
|
const REGEX_ID = '#(?<!\$)\[\s*crayon#mi';
|
|
//const REGEX_WITH_ID = '#(\[\s*crayon-\w+)\b([^\]]*["\'])(\s*/?\s*\])#mi';
|
|
const REGEX_WITH_ID = '#\[\s*(crayon-\w+)\b[^\]]*\]#mi';
|
|
|
|
const MODE_NORMAL = 0, MODE_JUST_CODE = 1, MODE_PLAIN_CODE = 2;
|
|
|
|
// Public Methods =========================================================
|
|
|
|
public static function post_captures() {
|
|
return self::$post_queue;
|
|
}
|
|
|
|
// Methods ================================================================
|
|
|
|
private function __construct() {
|
|
}
|
|
|
|
public static function regex() {
|
|
return '#(?<!\$)(?:' . self::REGEX_CLOSED . '|' . self::REGEX_TAG . ')(?!\$)#msi';
|
|
}
|
|
|
|
public static function regex_with_id($id) {
|
|
return '#\[\s*(crayon-' . $id . ')\b[^\]]*\]#mi';
|
|
}
|
|
|
|
public static function regex_no_capture() {
|
|
return '#(?<!\$)(?:' . self::REGEX_CLOSED_NO_CAPTURE . '|' . self::REGEX_TAG_NO_CAPTURE . ')(?!\$)#msi';
|
|
}
|
|
|
|
/**
|
|
* Adds the actual Crayon instance.
|
|
* $mode can be: 0 = return crayon content, 1 = return only code, 2 = return only plain code
|
|
*/
|
|
public static function shortcode($atts, $content = NULL, $id = NULL) {
|
|
CrayonLog::debug('shortcode');
|
|
|
|
// Load attributes from shortcode
|
|
$filtered_atts = shortcode_atts(self::$allowed_atts, $atts);
|
|
|
|
// Clean attributes
|
|
$keys = array_keys($filtered_atts);
|
|
for ($i = 0; $i < count($keys); $i++) {
|
|
$key = $keys[$i];
|
|
$value = $filtered_atts[$key];
|
|
if ($value !== NULL) {
|
|
$filtered_atts[$key] = trim(strip_tags($value));
|
|
}
|
|
}
|
|
|
|
// Contains all other attributes not found in allowed, used to override global settings
|
|
$extra_attr = array();
|
|
if (!empty($atts)) {
|
|
$extra_attr = array_diff_key($atts, self::$allowed_atts);
|
|
$extra_attr = CrayonSettings::smart_settings($extra_attr);
|
|
}
|
|
$url = $lang = $title = $mark = $range = $inline = '';
|
|
extract($filtered_atts);
|
|
|
|
$crayon = self::instance($extra_attr, $id);
|
|
|
|
// Set URL
|
|
$crayon->url($url);
|
|
$crayon->code($content);
|
|
// Set attributes, should be set after URL to allow language auto detection
|
|
$crayon->language($lang);
|
|
$crayon->title($title);
|
|
$crayon->marked($mark);
|
|
$crayon->range($range);
|
|
|
|
$crayon->is_inline($inline);
|
|
|
|
// Determine if we should highlight
|
|
$highlight = array_key_exists('highlight', $atts) ? CrayonUtil::str_to_bool($atts['highlight'], FALSE) : TRUE;
|
|
$crayon->is_highlighted($highlight);
|
|
return $crayon;
|
|
}
|
|
|
|
/* Returns Crayon instance */
|
|
public static function instance($extra_attr = array(), $id = NULL) {
|
|
CrayonLog::debug('instance');
|
|
|
|
// Create Crayon
|
|
$crayon = new CrayonHighlighter();
|
|
|
|
/* Load settings and merge shortcode attributes which will override any existing.
|
|
* Stores the other shortcode attributes as settings in the crayon. */
|
|
if (!empty($extra_attr)) {
|
|
$crayon->settings($extra_attr);
|
|
}
|
|
if (!empty($id)) {
|
|
$crayon->id($id);
|
|
}
|
|
|
|
return $crayon;
|
|
}
|
|
|
|
/* For manually highlighting code, useful for other PHP contexts */
|
|
public static function highlight($code, $add_tags = FALSE, $output_text = FALSE) {
|
|
$captures = CrayonWP::capture_crayons(0, $code);
|
|
$the_captures = $captures['capture'];
|
|
if (count($the_captures) == 0 && $add_tags) {
|
|
// Nothing captured, so wrap in a pre and try again
|
|
$code = '<pre>' . $code . '</pre>';
|
|
$captures = CrayonWP::capture_crayons(0, $code);
|
|
$the_captures = $captures['capture'];
|
|
}
|
|
$the_content = $captures['content'];
|
|
$the_content = CrayonUtil::strip_tags_blacklist($the_content, array('script'));
|
|
$the_content = CrayonUtil::strip_event_attributes($the_content);
|
|
foreach ($the_captures as $id => $capture) {
|
|
$atts = $capture['atts'];
|
|
$no_enqueue = array(
|
|
CrayonSettings::ENQUEUE_THEMES => FALSE,
|
|
CrayonSettings::ENQUEUE_FONTS => FALSE);
|
|
$atts = array_merge($atts, $no_enqueue);
|
|
$code = $capture['code'];
|
|
$crayon = CrayonWP::shortcode($atts, $code, $id);
|
|
$crayon_formatted = $crayon->output(TRUE, FALSE);
|
|
$the_content = CrayonUtil::preg_replace_escape_back(self::regex_with_id($id), $crayon_formatted, $the_content, 1, $count);
|
|
}
|
|
|
|
if ($output_text) {
|
|
header('Content-Type: text/plain');
|
|
} else {
|
|
header('Content-Type: text/html');
|
|
}
|
|
return $the_content;
|
|
}
|
|
|
|
public static function ajax_highlight() {
|
|
$code = isset($_POST['code']) ? $_POST['code'] : null;
|
|
if (!$code) {
|
|
$code = isset($_GET['code']) ? $_GET['code'] : null;
|
|
}
|
|
if ($code) {
|
|
echo self::highlight($code, FALSE, TRUE);
|
|
} else {
|
|
echo "No code specified.";
|
|
}
|
|
exit();
|
|
}
|
|
|
|
/* Uses the main query */
|
|
public static function wp() {
|
|
CrayonLog::debug('wp (global)');
|
|
global $wp_the_query;
|
|
if (isset($wp_the_query->posts)) {
|
|
$posts = $wp_the_query->posts;
|
|
self::the_posts($posts);
|
|
}
|
|
}
|
|
|
|
// TODO put args into an array
|
|
public static function capture_crayons($wp_id, $wp_content, $extra_settings = array(), $args = array()) {
|
|
extract($args);
|
|
CrayonUtil::set_var($callback, NULL);
|
|
CrayonUtil::set_var($callback_extra_args, NULL);
|
|
CrayonUtil::set_var($ignore, TRUE);
|
|
CrayonUtil::set_var($preserve_atts, FALSE);
|
|
CrayonUtil::set_var($flags, NULL);
|
|
CrayonUtil::set_var($skip_setting_check, FALSE);
|
|
CrayonUtil::set_var($just_check, FALSE);
|
|
|
|
// Will contain captured crayons and altered $wp_content
|
|
$capture = array('capture' => array(), 'content' => $wp_content, 'has_captured' => FALSE);
|
|
|
|
// Do not apply Crayon for posts older than a certain date.
|
|
$disable_date = trim(CrayonGlobalSettings::val(CrayonSettings::DISABLE_DATE));
|
|
if ($disable_date && get_post_time('U', true, $wp_id) <= strtotime($disable_date)) {
|
|
return $capture;
|
|
}
|
|
|
|
// Flags for which Crayons to convert
|
|
$in_flag = self::in_flag($flags);
|
|
|
|
CrayonLog::debug('capture for id ' . $wp_id . ' len ' . strlen($wp_content));
|
|
|
|
// Convert <pre> tags to crayon tags, if needed
|
|
if ((CrayonGlobalSettings::val(CrayonSettings::CAPTURE_PRE) || $skip_setting_check) && $in_flag[CrayonSettings::CAPTURE_PRE]) {
|
|
// XXX This will fail if <pre></pre> is used inside another <pre></pre>
|
|
$wp_content = preg_replace_callback('#(?<!\$)<\s*pre(?=(?:([^>]*)\bclass\s*=\s*(["\'])(.*?)\2([^>]*))?)([^>]*)>(.*?)<\s*/\s*pre\s*>#msi', 'CrayonWP::pre_tag', $wp_content);
|
|
}
|
|
|
|
// Convert mini [php][/php] tags to crayon tags, if needed
|
|
if ((CrayonGlobalSettings::val(CrayonSettings::CAPTURE_MINI_TAG) || $skip_setting_check) && $in_flag[CrayonSettings::CAPTURE_MINI_TAG]) {
|
|
$wp_content = preg_replace('#(?<!\$)\[\s*(' . self::$alias_regex . ')\b([^\]]*)\](.*?)\[\s*/\s*(?:\1)\s*\](?!\$)#msi', '[crayon lang="\1" \2]\3[/crayon]', $wp_content);
|
|
$wp_content = preg_replace('#(?<!\$)\[\s*(' . self::$alias_regex . ')\b([^\]]*)/\s*\](?!\$)#msi', '[crayon lang="\1" \2 /]', $wp_content);
|
|
}
|
|
|
|
// Convert <code> to inline tags
|
|
if (CrayonGlobalSettings::val(CrayonSettings::CODE_TAG_CAPTURE)) {
|
|
$inline = CrayonGlobalSettings::val(CrayonSettings::CODE_TAG_CAPTURE_TYPE) === 0;
|
|
$inline_setting = $inline ? 'inline="true"' : '';
|
|
$wp_content = preg_replace('#<(\s*code\b)([^>]*)>(.*?)</\1[^>]*>#msi', '[crayon ' . $inline_setting . ' \2]\3[/crayon]', $wp_content);
|
|
}
|
|
|
|
if ((CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG) || $skip_setting_check) && $in_flag[CrayonSettings::INLINE_TAG]) {
|
|
if (CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG_CAPTURE)) {
|
|
// Convert inline {php}{/php} tags to crayon tags, if needed
|
|
$wp_content = preg_replace('#(?<!\$)\{\s*(' . self::$alias_regex . ')\b([^\}]*)\}(.*?)\{/(?:\1)\}(?!\$)#msi', '[crayon lang="\1" inline="true" \2]\3[/crayon]', $wp_content);
|
|
}
|
|
// Convert <span class="crayon-inline"> tags to inline crayon tags
|
|
$wp_content = preg_replace_callback('#(?<!\$)<\s*span([^>]*)\bclass\s*=\s*(["\'])(.*?)\2([^>]*)>(.*?)<\s*/\s*span\s*>#msi', 'CrayonWP::span_tag', $wp_content);
|
|
}
|
|
|
|
// Convert [plain] tags into <pre><code></code></pre>, if needed
|
|
if ((CrayonGlobalSettings::val(CrayonSettings::PLAIN_TAG) || $skip_setting_check) && $in_flag[CrayonSettings::PLAIN_TAG]) {
|
|
$wp_content = preg_replace_callback('#(?<!\$)\[\s*plain\s*\](.*?)\[\s*/\s*plain\s*\]#msi', 'CrayonFormatter::plain_code', $wp_content);
|
|
}
|
|
|
|
// Add IDs to the Crayons
|
|
CrayonLog::debug('capture adding id ' . $wp_id . ' , now has len ' . strlen($wp_content));
|
|
$wp_content = preg_replace_callback(self::REGEX_ID, 'CrayonWP::add_crayon_id', $wp_content);
|
|
|
|
CrayonLog::debug('capture added id ' . $wp_id . ' : ' . strlen($wp_content));
|
|
|
|
// Only include if a post exists with Crayon tag
|
|
preg_match_all(self::regex(), $wp_content, $matches);
|
|
$capture['has_captured'] = count($matches[0]) != 0;
|
|
if ($just_check) {
|
|
// Backticks are matched after other tags, so they need to be captured here.
|
|
$result = self::replace_backquotes($wp_content);
|
|
$wp_content = $result['content'];
|
|
$capture['has_captured'] = $capture['has_captured'] || $result['changed'];
|
|
$capture['content'] = $wp_content;
|
|
return $capture;
|
|
}
|
|
|
|
CrayonLog::debug('capture ignore for id ' . $wp_id . ' : ' . strlen($capture['content']) . ' vs ' . strlen($wp_content));
|
|
|
|
if ($capture['has_captured']) {
|
|
|
|
// Crayons found! Load settings first to ensure global settings loaded
|
|
CrayonSettingsWP::load_settings();
|
|
|
|
CrayonLog::debug('CAPTURED FOR ID ' . $wp_id);
|
|
|
|
$full_matches = $matches[0];
|
|
$closed_ids = $matches[1];
|
|
$closed_atts = $matches[2];
|
|
$open_ids = $matches[3];
|
|
$open_atts = $matches[4];
|
|
$contents = $matches[5];
|
|
|
|
// Make sure we enqueue the styles/scripts
|
|
$enqueue = TRUE;
|
|
|
|
for ($i = 0; $i < count($full_matches); $i++) {
|
|
// Get attributes
|
|
if (!empty($closed_atts[$i])) {
|
|
$atts = $closed_atts[$i];
|
|
} else if (!empty($open_atts[$i])) {
|
|
$atts = $open_atts[$i];
|
|
} else {
|
|
$atts = '';
|
|
}
|
|
|
|
// Capture attributes
|
|
preg_match_all('#([^="\'\s]+)[\t ]*=[\t ]*("|\')(.*?)\2#', $atts, $att_matches);
|
|
// Add extra attributes
|
|
$atts_array = $extra_settings;
|
|
if (count($att_matches[0]) != 0) {
|
|
for ($j = 0; $j < count($att_matches[1]); $j++) {
|
|
$atts_array[trim(strtolower($att_matches[1][$j]))] = trim($att_matches[3][$j]);
|
|
}
|
|
}
|
|
|
|
if (isset($atts_array[CrayonSettings::IGNORE]) && $atts_array[CrayonSettings::IGNORE]) {
|
|
// TODO(aramk) Revert to the original content.
|
|
continue;
|
|
}
|
|
|
|
// Capture theme
|
|
$theme_id = array_key_exists(CrayonSettings::THEME, $atts_array) ? $atts_array[CrayonSettings::THEME] : '';
|
|
$theme = CrayonResources::themes()->get($theme_id);
|
|
// If theme not found, use fallbacks
|
|
if (!$theme) {
|
|
// Given theme is invalid, try global setting
|
|
$theme_id = CrayonGlobalSettings::val(CrayonSettings::THEME);
|
|
$theme = CrayonResources::themes()->get($theme_id);
|
|
if (!$theme) {
|
|
// Global setting is invalid, fall back to default
|
|
$theme = CrayonResources::themes()->get_default();
|
|
$theme_id = CrayonThemes::DEFAULT_THEME;
|
|
}
|
|
}
|
|
// If theme is now valid, change the array
|
|
if ($theme) {
|
|
if (!$preserve_atts || isset($atts_array[CrayonSettings::THEME])) {
|
|
$atts_array[CrayonSettings::THEME] = $theme_id;
|
|
}
|
|
$theme->used(TRUE);
|
|
}
|
|
|
|
// Capture font
|
|
$font_id = array_key_exists(CrayonSettings::FONT, $atts_array) ? $atts_array[CrayonSettings::FONT] : '';
|
|
$font = CrayonResources::fonts()->get($font_id);
|
|
// If font not found, use fallbacks
|
|
if (!$font) {
|
|
// Given font is invalid, try global setting
|
|
$font_id = CrayonGlobalSettings::val(CrayonSettings::FONT);
|
|
$font = CrayonResources::fonts()->get($font_id);
|
|
if (!$font) {
|
|
// Global setting is invalid, fall back to default
|
|
$font = CrayonResources::fonts()->get_default();
|
|
$font_id = CrayonFonts::DEFAULT_FONT;
|
|
}
|
|
}
|
|
|
|
// If font is now valid, change the array
|
|
if ($font /* != NULL && $font_id != CrayonFonts::DEFAULT_FONT*/) {
|
|
if (!$preserve_atts || isset($atts_array[CrayonSettings::FONT])) {
|
|
$atts_array[CrayonSettings::FONT] = $font_id;
|
|
}
|
|
$font->used(TRUE);
|
|
}
|
|
|
|
// Add array of atts and content to post queue with key as post ID
|
|
// XXX If at this point no ID is added we have failed!
|
|
$id = !empty($open_ids[$i]) ? $open_ids[$i] : $closed_ids[$i];
|
|
//if ($ignore) {
|
|
$code = self::crayon_remove_ignore($contents[$i]);
|
|
//}
|
|
$c = array('post_id' => $wp_id, 'atts' => $atts_array, 'code' => $code);
|
|
$capture['capture'][$id] = $c;
|
|
CrayonLog::debug('capture finished for post id ' . $wp_id . ' crayon-id ' . $id . ' atts: ' . count($atts_array) . ' code: ' . strlen($code));
|
|
$is_inline = isset($atts_array['inline']) && CrayonUtil::str_to_bool($atts_array['inline'], FALSE) ? '-i' : '';
|
|
if ($callback === NULL) {
|
|
$wp_content = str_replace($full_matches[$i], '[crayon-' . $id . $is_inline . '/]', $wp_content);
|
|
} else {
|
|
$wp_content = call_user_func($callback, $c, $full_matches[$i], $id, $is_inline, $wp_content, $callback_extra_args);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if ($ignore) {
|
|
// We need to escape ignored Crayons, since they won't be captured
|
|
// XXX Do this after replacing the Crayon with the shorter ID tag, otherwise $full_matches will be different from $wp_content
|
|
$wp_content = self::crayon_remove_ignore($wp_content);
|
|
}
|
|
|
|
$result = self::replace_backquotes($wp_content);
|
|
$wp_content = $result['content'];
|
|
|
|
$capture['content'] = $wp_content;
|
|
return $capture;
|
|
}
|
|
|
|
public static function replace_backquotes($wp_content) {
|
|
// Convert `` backquote tags into <code></code>, if needed
|
|
// XXX Some code may contain `` so must do it after all Crayons are captured
|
|
$result = array();
|
|
$prev_count = strlen($wp_content);
|
|
if (CrayonGlobalSettings::val(CrayonSettings::BACKQUOTE)) {
|
|
$wp_content = preg_replace('#(?<!\\\\)`([^`]*)`#msi', '<code>$1</code>', $wp_content);
|
|
}
|
|
$result['changed'] = $prev_count !== strlen($wp_content);
|
|
$result['content'] = $wp_content;
|
|
return $result;
|
|
}
|
|
|
|
/* Search for Crayons in posts and queue them for creation */
|
|
public static function the_posts($posts) {
|
|
CrayonLog::debug('the_posts');
|
|
|
|
// Whether to enqueue syles/scripts
|
|
CrayonSettingsWP::load_settings(TRUE); // We will eventually need more than the settings
|
|
|
|
self::init_tags_regex();
|
|
$crayon_posts = CrayonSettingsWP::load_posts(); // Loads posts containing crayons
|
|
|
|
// Search for shortcode in posts
|
|
foreach ($posts as $post) {
|
|
$wp_id = $post->ID;
|
|
$is_page = $post->post_type == 'page';
|
|
if (!in_array($wp_id, $crayon_posts)) {
|
|
// If we get query for a page, then that page might have a template and load more posts containing Crayons
|
|
// By this state, we would be unable to enqueue anything (header already written).
|
|
if (CrayonGlobalSettings::val(CrayonSettings::SAFE_ENQUEUE) && $is_page) {
|
|
CrayonGlobalSettings::set(CrayonSettings::ENQUEUE_THEMES, false);
|
|
CrayonGlobalSettings::set(CrayonSettings::ENQUEUE_FONTS, false);
|
|
}
|
|
// Only include crayon posts
|
|
continue;
|
|
}
|
|
|
|
$id_str = strval($wp_id);
|
|
|
|
if (wp_is_post_revision($wp_id)) {
|
|
// Ignore post revisions, use the parent, which has the updated post content
|
|
continue;
|
|
}
|
|
|
|
if (isset(self::$post_captures[$id_str])) {
|
|
// Don't capture twice
|
|
// XXX post->post_content is reset each loop, replace content
|
|
// Doing this might cause content changed by other plugins between the last loop
|
|
// to fail, so be cautious
|
|
$post->post_content = self::$post_captures[$id_str];
|
|
continue;
|
|
}
|
|
// Capture post Crayons
|
|
$captures = self::capture_crayons(intval($post->ID), $post->post_content);
|
|
|
|
// XXX Careful not to undo changes by other plugins
|
|
// XXX Must replace to remove $ for ignored Crayons
|
|
$post->post_content = $captures['content'];
|
|
self::$post_captures[$id_str] = $captures['content'];
|
|
if ($captures['has_captured'] === TRUE) {
|
|
self::$post_queue[$id_str] = array();
|
|
foreach ($captures['capture'] as $capture_id => $capture_content) {
|
|
self::$post_queue[$id_str][$capture_id] = $capture_content;
|
|
}
|
|
}
|
|
|
|
// Search for shortcode in comments
|
|
if (CrayonGlobalSettings::val(CrayonSettings::COMMENTS)) {
|
|
$comments = get_comments(array('post_id' => $post->ID));
|
|
foreach ($comments as $comment) {
|
|
$id_str = strval($comment->comment_ID);
|
|
if (isset(self::$comment_queue[$id_str])) {
|
|
// Don't capture twice
|
|
continue;
|
|
}
|
|
// Capture comment Crayons, decode their contents if decode not specified
|
|
$content = apply_filters('get_comment_text', $comment->comment_content, $comment);
|
|
$captures = self::capture_crayons($comment->comment_ID, $content, array(CrayonSettings::DECODE => TRUE));
|
|
self::$comment_captures[$id_str] = $captures['content'];
|
|
if ($captures['has_captured'] === TRUE) {
|
|
self::$comment_queue[$id_str] = array();
|
|
foreach ($captures['capture'] as $capture_id => $capture_content) {
|
|
self::$comment_queue[$id_str][$capture_id] = $capture_content;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return $posts;
|
|
}
|
|
|
|
private static function add_crayon_id($content) {
|
|
$uid = $content[0] . '-' . str_replace('.', '', uniqid('', true));
|
|
CrayonLog::debug('add_crayon_id ' . $uid);
|
|
return $uid;
|
|
}
|
|
|
|
private static function get_crayon_id() {
|
|
return self::$next_id++;
|
|
}
|
|
|
|
public static function enqueue_resources() {
|
|
if (!self::$enqueued) {
|
|
|
|
CrayonLog::debug('enqueue');
|
|
global $CRAYON_VERSION;
|
|
CrayonSettingsWP::load_settings(TRUE);
|
|
if (CRAYON_MINIFY) {
|
|
wp_enqueue_style('crayon', plugins_url(CRAYON_STYLE_MIN, __FILE__), array(), $CRAYON_VERSION);
|
|
wp_enqueue_script('crayon_js', plugins_url(CRAYON_JS_MIN, __FILE__), array('jquery'), $CRAYON_VERSION, CrayonGlobalSettings::val(CrayonSettings::DELAY_LOAD_JS));
|
|
} else {
|
|
wp_enqueue_style('crayon_style', plugins_url(CRAYON_STYLE, __FILE__), array(), $CRAYON_VERSION);
|
|
wp_enqueue_style('crayon_global_style', plugins_url(CRAYON_STYLE_GLOBAL, __FILE__), array(), $CRAYON_VERSION);
|
|
wp_enqueue_script('crayon_util_js', plugins_url(CRAYON_JS_UTIL, __FILE__), array('jquery'), $CRAYON_VERSION);
|
|
CrayonSettingsWP::other_scripts();
|
|
}
|
|
CrayonSettingsWP::init_js_settings();
|
|
self::$enqueued = TRUE;
|
|
}
|
|
}
|
|
|
|
private static function init_tags_regex($force = FALSE, $flags = NULL, &$tags_regex = NULL) {
|
|
CrayonSettingsWP::load_settings();
|
|
self::init_tag_bits();
|
|
|
|
// Default output
|
|
if ($tags_regex === NULL) {
|
|
$tags_regex = & self::$tags_regex;
|
|
}
|
|
|
|
if ($force || $tags_regex === "") {
|
|
// Check which tags are in $flags. If it's NULL, then all flags are true.
|
|
$in_flag = self::in_flag($flags);
|
|
|
|
if (($in_flag[CrayonSettings::CAPTURE_MINI_TAG] && (CrayonGlobalSettings::val(CrayonSettings::CAPTURE_MINI_TAG)) || $force) ||
|
|
($in_flag[CrayonSettings::INLINE_TAG] && (CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG) && CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG_CAPTURE)) || $force)
|
|
) {
|
|
$aliases = CrayonResources::langs()->ids_and_aliases();
|
|
self::$alias_regex = '';
|
|
for ($i = 0; $i < count($aliases); $i++) {
|
|
$alias = $aliases[$i];
|
|
$alias_regex = CrayonUtil::esc_hash(CrayonUtil::esc_regex($alias));
|
|
if ($i != count($aliases) - 1) {
|
|
$alias_regex .= '|';
|
|
}
|
|
self::$alias_regex .= $alias_regex;
|
|
}
|
|
}
|
|
|
|
// Add other tags
|
|
$tags_regex = '#(?<!\$)(?:(\s*\[\s*crayon\b)';
|
|
|
|
// TODO this is duplicated in capture_crayons()
|
|
$tag_regexes = array(
|
|
CrayonSettings::CAPTURE_MINI_TAG => '(\[\s*(' . self::$alias_regex . ')\b)',
|
|
CrayonSettings::CAPTURE_PRE => '(<\s*pre\b)',
|
|
CrayonSettings::INLINE_TAG => '(' . self::REGEX_INLINE_CLASS . ')' . '|(\{\s*(' . self::$alias_regex . ')\b([^\}]*)\})',
|
|
CrayonSettings::PLAIN_TAG => '(\s*\[\s*plain\b)',
|
|
CrayonSettings::BACKQUOTE => '(`[^`]*`)'
|
|
);
|
|
|
|
foreach ($tag_regexes as $tag => $regex) {
|
|
if ($in_flag[$tag] && (CrayonGlobalSettings::val($tag) || $force)) {
|
|
$tags_regex .= '|' . $regex;
|
|
}
|
|
}
|
|
$tags_regex .= ')#msi';
|
|
}
|
|
|
|
}
|
|
|
|
private static function init_tag_bits() {
|
|
if (count(self::$tag_bits) == 0) {
|
|
$values = array();
|
|
for ($i = 0; $i < count(self::$tag_types); $i++) {
|
|
$j = pow(2, $i);
|
|
self::$tag_bits[self::$tag_types[$i]] = $j;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static function tag_bit($tag) {
|
|
self::init_tag_bits();
|
|
if (isset(self::$tag_bits[$tag])) {
|
|
return self::$tag_bits[$tag];
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static function in_flag($flags) {
|
|
$in_flag = array();
|
|
foreach (self::$tag_types as $tag) {
|
|
$in_flag[$tag] = $flags === NULL || ($flags & self::tag_bit($tag)) > 0;
|
|
}
|
|
return $in_flag;
|
|
}
|
|
|
|
private static function init_legacy_tag_bits() {
|
|
if (self::$legacy_flags === NULL) {
|
|
self::$legacy_flags = self::tag_bit(CrayonSettings::CAPTURE_MINI_TAG) |
|
|
self::tag_bit(CrayonSettings::INLINE_TAG) |
|
|
self::tag_bit(CrayonSettings::PLAIN_TAG);
|
|
}
|
|
if (self::$tags_regex_legacy === "") {
|
|
self::init_tags_regex(TRUE, self::$legacy_flags, self::$tags_regex_legacy);
|
|
}
|
|
}
|
|
|
|
// Add Crayon into the_content
|
|
public static function the_content($the_content) {
|
|
CrayonLog::debug('the_content');
|
|
|
|
// Some themes make redundant queries and don't need extra work...
|
|
if (strlen($the_content) == 0) {
|
|
CrayonLog::debug('the_content blank');
|
|
return $the_content;
|
|
}
|
|
|
|
global $post;
|
|
|
|
// Go through queued posts and find crayons
|
|
$post_id = strval($post->ID);
|
|
|
|
if (self::$is_excerpt) {
|
|
CrayonLog::debug('excerpt');
|
|
if (CrayonGlobalSettings::val(CrayonSettings::EXCERPT_STRIP)) {
|
|
CrayonLog::debug('excerpt strip');
|
|
// Remove Crayon from content if we are displaying an excerpt
|
|
$the_content = preg_replace(self::REGEX_WITH_ID, '', $the_content);
|
|
}
|
|
// Otherwise Crayon remains with ID and replaced later
|
|
return $the_content;
|
|
}
|
|
|
|
// Find if this post has Crayons
|
|
if (array_key_exists($post_id, self::$post_queue)) {
|
|
self::enqueue_resources();
|
|
|
|
// XXX We want the plain post content, no formatting
|
|
$the_content_original = $the_content;
|
|
|
|
// Replacing may cause <p> tags to become disjoint with a <div> inside them, close and reopen them if needed
|
|
$the_content = preg_replace_callback('#' . self::REGEX_BETWEEN_PARAGRAPH_SIMPLE . '#msi', 'CrayonWP::add_paragraphs', $the_content);
|
|
// Loop through Crayons
|
|
$post_in_queue = self::$post_queue[$post_id];
|
|
|
|
foreach ($post_in_queue as $id => $v) {
|
|
$atts = $v['atts'];
|
|
$content = $v['code']; // The code we replace post content with
|
|
$crayon = self::shortcode($atts, $content, $id);
|
|
if (is_feed()) {
|
|
// Convert the plain code to entities and put in a <pre></pre> tag
|
|
$crayon_formatted = CrayonFormatter::plain_code($crayon->code(), $crayon->setting_val(CrayonSettings::DECODE));
|
|
} else {
|
|
// Apply shortcode to the content
|
|
$crayon_formatted = $crayon->output(TRUE, FALSE);
|
|
}
|
|
// Replace the code with the Crayon
|
|
CrayonLog::debug('the_content: id ' . $post_id . ' has UID ' . $id . ' : ' . intval(stripos($the_content, $id) !== FALSE));
|
|
$the_content = CrayonUtil::preg_replace_escape_back(self::regex_with_id($id), $crayon_formatted, $the_content, 1, $count);
|
|
CrayonLog::debug('the_content: REPLACED for id ' . $post_id . ' from len ' . strlen($the_content_original) . ' to ' . strlen($the_content));
|
|
}
|
|
}
|
|
|
|
return $the_content;
|
|
}
|
|
|
|
public static function pre_comment_text($text) {
|
|
global $comment;
|
|
$comment_id = strval($comment->comment_ID);
|
|
if (array_key_exists($comment_id, self::$comment_captures)) {
|
|
// Replace with IDs now that we need to
|
|
$text = self::$comment_captures[$comment_id];
|
|
}
|
|
return $text;
|
|
}
|
|
|
|
public static function comment_text($text) {
|
|
global $comment;
|
|
$comment_id = strval($comment->comment_ID);
|
|
// Find if this post has Crayons
|
|
if (array_key_exists($comment_id, self::$comment_queue)) {
|
|
// XXX We want the plain post content, no formatting
|
|
$the_content_original = $text;
|
|
// Loop through Crayons
|
|
$post_in_queue = self::$comment_queue[$comment_id];
|
|
|
|
foreach ($post_in_queue as $id => $v) {
|
|
$atts = $v['atts'];
|
|
$content = $v['code']; // The code we replace post content with
|
|
$crayon = self::shortcode($atts, $content, $id);
|
|
$crayon_formatted = $crayon->output(TRUE, FALSE);
|
|
// Replacing may cause <p> tags to become disjoint with a <div> inside them, close and reopen them if needed
|
|
if (!$crayon->is_inline()) {
|
|
$text = preg_replace_callback('#' . self::REGEX_BETWEEN_PARAGRAPH_SIMPLE . '#msi', 'CrayonWP::add_paragraphs', $text);
|
|
}
|
|
// Replace the code with the Crayon
|
|
$text = CrayonUtil::preg_replace_escape_back(self::regex_with_id($id), $crayon_formatted, $text, 1, $text);
|
|
}
|
|
}
|
|
return $text;
|
|
}
|
|
|
|
public static function add_paragraphs($capture) {
|
|
if (count($capture) != 4) {
|
|
CrayonLog::debug('add_paragraphs: 0');
|
|
return $capture[0];
|
|
}
|
|
$capture[2] = preg_replace('#(?:<\s*br\s*/\s*>\s*)?(\[\s*crayon-\w+/\])(?:<\s*br\s*/\s*>\s*)?#msi', '</p>$1<p>', $capture[2]);
|
|
// If [crayon appears right after <p> then we will generate <p></p>, remove all these
|
|
$paras = $capture[1] . $capture[2] . $capture[3];
|
|
return $paras;
|
|
}
|
|
|
|
// Remove Crayons from the_excerpt
|
|
public static function the_excerpt($the_excerpt) {
|
|
CrayonLog::debug('excerpt');
|
|
global $post;
|
|
if (!empty($post->post_excerpt)) {
|
|
// Use custom excerpt if defined
|
|
$the_excerpt = wpautop($post->post_excerpt);
|
|
} else {
|
|
// Pass wp_trim_excerpt('') to gen from content (and remove [crayons])
|
|
$the_excerpt = wpautop(wp_trim_excerpt(''));
|
|
}
|
|
// XXX Returning "" may cause it to default to full contents...
|
|
return $the_excerpt . ' ';
|
|
}
|
|
|
|
// Used to capture pre and span tags which have settings in class attribute
|
|
public static function class_tag($matches) {
|
|
// If class exists, atts is not captured
|
|
$pre_class = $matches[1];
|
|
$quotes = $matches[2];
|
|
$class = $matches[3];
|
|
$post_class = $matches[4];
|
|
$atts = $matches[5];
|
|
$content = $matches[6];
|
|
|
|
// If we find a crayon=false in the attributes, or a crayon[:_]false in the class, then we should not capture
|
|
$ignore_regex_atts = '#crayon\s*=\s*(["\'])\s*(false|no|0)\s*\1#msi';
|
|
$ignore_regex_class = '#crayon\s*[:_]\s*(false|no|0)#msi';
|
|
if (preg_match($ignore_regex_atts, $atts) !== 0 ||
|
|
preg_match($ignore_regex_class, $class) !== 0
|
|
) {
|
|
return $matches[0];
|
|
}
|
|
|
|
if (!empty($class)) {
|
|
if (preg_match('#\bignore\s*:\s*true#', $class)) {
|
|
// Prevent any changes if ignoring the tag.
|
|
return $matches[0];
|
|
}
|
|
// crayon-inline is turned into inline="1"
|
|
$class = preg_replace('#' . self::REGEX_INLINE_CLASS . '#mi', 'inline="1"', $class);
|
|
// "setting[:_]value" style settings in the class attribute
|
|
$class = preg_replace('#\b([A-Za-z-]+)[_:](\S+)#msi', '$1=' . $quotes . '$2' . $quotes, $class);
|
|
}
|
|
|
|
// data-url is turned into url=""
|
|
if (!empty($post_class)) {
|
|
$post_class = preg_replace('#\bdata-url\s*=#mi', 'url=', $post_class);
|
|
}
|
|
if (!empty($pre_class)) {
|
|
$pre_class = preg_replace('#\bdata-url\s*=#mi', 'url=', $pre_class);
|
|
}
|
|
|
|
if (!empty($class)) {
|
|
return "[crayon $pre_class $class $post_class]{$content}[/crayon]";
|
|
} else {
|
|
return "[crayon $atts]{$content}[/crayon]";
|
|
}
|
|
}
|
|
|
|
// Capture span tag and extract settings from the class attribute, if present.
|
|
public static function span_tag($matches) {
|
|
// Only use <span> tags with crayon-inline class
|
|
if (preg_match('#' . self::REGEX_INLINE_CLASS . '#mi', $matches[3])) {
|
|
// no $atts
|
|
$matches[6] = $matches[5];
|
|
$matches[5] = '';
|
|
return self::class_tag($matches);
|
|
} else {
|
|
// Don't turn regular <span>s into Crayons
|
|
return $matches[0];
|
|
}
|
|
}
|
|
|
|
// Capture pre tag and extract settings from the class attribute, if present.
|
|
public static function pre_tag($matches) {
|
|
return self::class_tag($matches);
|
|
}
|
|
|
|
/**
|
|
* Check if the $ notation has been used to ignore [crayon] tags within posts and remove all matches
|
|
* Can also remove if used without $ as a regular crayon
|
|
*
|
|
* @deprecated
|
|
*/
|
|
public static function crayon_remove_ignore($the_content, $ignore_flag = '$') {
|
|
if ($ignore_flag == FALSE) {
|
|
$ignore_flag = '';
|
|
}
|
|
$ignore_flag_regex = preg_quote($ignore_flag);
|
|
|
|
$the_content = preg_replace('#' . $ignore_flag_regex . '(\s*\[\s*crayon)#msi', '$1', $the_content);
|
|
$the_content = preg_replace('#(crayon\s*\])\s*\$#msi', '$1', $the_content);
|
|
|
|
if (CrayonGlobalSettings::val(CrayonSettings::CAPTURE_PRE)) {
|
|
$the_content = str_ireplace(array($ignore_flag . '<pre', 'pre>' . $ignore_flag), array('<pre', 'pre>'), $the_content);
|
|
// Remove any <code> tags wrapping around the whole code, since we won't needed them
|
|
// XXX This causes <code> tags to be stripped in the post content! Disabled now.
|
|
// $the_content = preg_replace('#(^\s*<\s*code[^>]*>)|(<\s*/\s*code[^>]*>\s*$)#msi', '', $the_content);
|
|
}
|
|
if (CrayonGlobalSettings::val(CrayonSettings::PLAIN_TAG)) {
|
|
$the_content = str_ireplace(array($ignore_flag . '[plain', 'plain]' . $ignore_flag), array('[plain', 'plain]'), $the_content);
|
|
}
|
|
if (CrayonGlobalSettings::val(CrayonSettings::CAPTURE_MINI_TAG) ||
|
|
(CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG && CrayonGlobalSettings::val(CrayonSettings::INLINE_TAG_CAPTURE)))
|
|
) {
|
|
self::init_tags_regex();
|
|
// $the_content = preg_replace('#'.$ignore_flag_regex.'\s*([\[\{])\s*('. self::$alias_regex .')#', '$1$2', $the_content);
|
|
// $the_content = preg_replace('#('. self::$alias_regex .')\s*([\]\}])\s*'.$ignore_flag_regex.'#', '$1$2', $the_content);
|
|
$the_content = preg_replace('#' . $ignore_flag_regex . '(\s*[\[\{]\s*(' . self::$alias_regex . ')[^\]]*[\]\}])#', '$1', $the_content);
|
|
}
|
|
if (CrayonGlobalSettings::val(CrayonSettings::BACKQUOTE)) {
|
|
$the_content = str_ireplace('\\`', '`', $the_content);
|
|
}
|
|
return $the_content;
|
|
}
|
|
|
|
public static function wp_head() {
|
|
CrayonLog::debug('head');
|
|
|
|
self::$wp_head = TRUE;
|
|
if (!self::$enqueued) {
|
|
CrayonLog::debug('head: missed enqueue');
|
|
// We have missed our chance to check before enqueuing. Use setting to either load always or only in the_post
|
|
CrayonSettingsWP::load_settings(TRUE); // Ensure settings are loaded
|
|
// If we need the tag editor loaded at all times, we must enqueue at all times
|
|
if (!CrayonGlobalSettings::val(CrayonSettings::EFFICIENT_ENQUEUE) || CrayonGlobalSettings::val(CrayonSettings::TAG_EDITOR_FRONT)) {
|
|
CrayonLog::debug('head: force enqueue');
|
|
// Efficient enqueuing disabled, always load despite enqueuing or not in the_post
|
|
self::enqueue_resources();
|
|
}
|
|
}
|
|
// Enqueue Theme CSS
|
|
if (CrayonGlobalSettings::val(CrayonSettings::ENQUEUE_THEMES)) {
|
|
self::crayon_theme_css();
|
|
}
|
|
// Enqueue Font CSS
|
|
if (CrayonGlobalSettings::val(CrayonSettings::ENQUEUE_FONTS)) {
|
|
self::crayon_font_css();
|
|
}
|
|
}
|
|
|
|
public static function save_post($update_id, $post) {
|
|
self::refresh_post($post);
|
|
}
|
|
|
|
public static function filter_post_data($data, $postarr) {
|
|
// Remove the selected CSS that may be present from the tag editor.
|
|
CrayonTagEditorWP::init_settings();
|
|
$css_selected = CrayonTagEditorWP::$settings['css_selected'];
|
|
$data['post_content'] = preg_replace("#(class\s*=\s*(\\\\[\"'])[^\"']*)$css_selected([^\"']*\\2)#msi", '$1$3', $data['post_content']);
|
|
return $data;
|
|
}
|
|
|
|
public static function refresh_post($post, $refresh_legacy = TRUE, $save = TRUE) {
|
|
$postID = $post->ID;
|
|
if (wp_is_post_revision($postID)) {
|
|
// Ignore revisions
|
|
return;
|
|
}
|
|
if (CrayonWP::scan_post($post)) {
|
|
CrayonSettingsWP::add_post($postID, $save);
|
|
if ($refresh_legacy) {
|
|
if (self::scan_legacy_post($post)) {
|
|
CrayonSettingsWP::add_legacy_post($postID, $save);
|
|
} else {
|
|
CrayonSettingsWP::remove_legacy_post($postID, $save);
|
|
}
|
|
}
|
|
} else {
|
|
CrayonSettingsWP::remove_post($postID, $save);
|
|
CrayonSettingsWP::remove_legacy_post($postID, $save);
|
|
}
|
|
}
|
|
|
|
public static function refresh_posts() {
|
|
CrayonSettingsWP::remove_posts();
|
|
CrayonSettingsWP::remove_legacy_posts();
|
|
foreach (CrayonWP::get_posts() as $post) {
|
|
self::refresh_post($post, TRUE, FALSE);
|
|
}
|
|
CrayonSettingsWP::save_posts();
|
|
CrayonSettingsWP::save_legacy_posts();
|
|
|
|
}
|
|
|
|
public static function save_comment($id, $is_spam = NULL, $comment = NULL) {
|
|
self::init_tags_regex();
|
|
if ($comment === NULL) {
|
|
$comment = get_comment($id);
|
|
}
|
|
$content = $comment->comment_content;
|
|
$post_id = $comment->comment_post_ID;
|
|
$found = preg_match(self::$tags_regex, $content);
|
|
if ($found) {
|
|
CrayonSettingsWP::add_post($post_id);
|
|
}
|
|
return $found;
|
|
}
|
|
|
|
public static function crayon_theme_css() {
|
|
global $CRAYON_VERSION;
|
|
CrayonSettingsWP::load_settings();
|
|
$css = CrayonResources::themes()->get_used_css();
|
|
foreach ($css as $theme => $url) {
|
|
wp_enqueue_style('crayon-theme-' . $theme, $url, array(), $CRAYON_VERSION);
|
|
}
|
|
}
|
|
|
|
public static function crayon_font_css() {
|
|
global $CRAYON_VERSION;
|
|
CrayonSettingsWP::load_settings();
|
|
$css = CrayonResources::fonts()->get_used_css();
|
|
foreach ($css as $font_id => $url) {
|
|
wp_enqueue_style('crayon-font-' . $font_id, $url, array(), $CRAYON_VERSION);
|
|
}
|
|
}
|
|
|
|
public static function init($request) {
|
|
CrayonLog::debug('init');
|
|
crayon_load_plugin_textdomain();
|
|
}
|
|
|
|
public static function init_ajax() {
|
|
add_action('wp_ajax_crayon-tag-editor', 'CrayonTagEditorWP::content');
|
|
add_action('wp_ajax_nopriv_crayon-tag-editor', 'CrayonTagEditorWP::content');
|
|
add_action('wp_ajax_crayon-highlight', 'CrayonWP::ajax_highlight');
|
|
add_action('wp_ajax_nopriv_crayon-highlight', 'CrayonWP::ajax_highlight');
|
|
if (current_user_can('manage_options')) {
|
|
add_action('wp_ajax_crayon-ajax', 'CrayonWP::ajax');
|
|
add_action('wp_ajax_crayon-theme-editor', 'CrayonThemeEditorWP::content');
|
|
add_action('wp_ajax_crayon-theme-editor-save', 'CrayonThemeEditorWP::save');
|
|
add_action('wp_ajax_crayon-theme-editor-delete', 'CrayonThemeEditorWP::delete');
|
|
add_action('wp_ajax_crayon-theme-editor-duplicate', 'CrayonThemeEditorWP::duplicate');
|
|
add_action('wp_ajax_crayon-theme-editor-submit', 'CrayonThemeEditorWP::submit');
|
|
add_action('wp_ajax_crayon-show-posts', 'CrayonSettingsWP::show_posts');
|
|
add_action('wp_ajax_crayon-show-langs', 'CrayonSettingsWP::show_langs');
|
|
add_action('wp_ajax_crayon-show-preview', 'CrayonSettingsWP::show_preview');
|
|
}
|
|
}
|
|
|
|
public static function ajax() {
|
|
$allowed = array(CrayonSettings::HIDE_HELP);
|
|
foreach ($allowed as $allow) {
|
|
if (array_key_exists($allow, $_GET)) {
|
|
CrayonGlobalSettings::set($allow, $_GET[$allow]);
|
|
CrayonSettingsWP::save_settings();
|
|
}
|
|
}
|
|
}
|
|
|
|
public static function get_posts() {
|
|
$query = new WP_Query(array('post_type' => 'any', 'suppress_filters' => TRUE, 'posts_per_page' => '-1'));
|
|
if (isset($query->posts)) {
|
|
return $query->posts;
|
|
} else {
|
|
return array();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return an array of post IDs where crayons occur.
|
|
* Comments are ignored by default.
|
|
*/
|
|
public static function scan_posts($check_comments = FALSE) {
|
|
$crayon_posts = array();
|
|
foreach (self::get_posts() as $post) {
|
|
if (self::scan_post($post)) {
|
|
$crayon_posts[] = $post->ID;
|
|
}
|
|
}
|
|
return $crayon_posts;
|
|
}
|
|
|
|
public static function scan_legacy_posts($init_regex = TRUE, $check_comments = FALSE) {
|
|
if ($init_regex) {
|
|
// We can skip this if needed
|
|
self::init_tags_regex();
|
|
}
|
|
$crayon_posts = array();
|
|
foreach (self::get_posts() as $post) {
|
|
if (self::scan_legacy_post($post)) { // TODO this part is different
|
|
$crayon_posts[] = $post->ID;
|
|
}
|
|
}
|
|
return $crayon_posts;
|
|
}
|
|
|
|
/**
|
|
* Returns TRUE if a given post contains a Crayon tag
|
|
*/
|
|
public static function scan_post($post, $scan_comments = TRUE, $flags = NULL) {
|
|
if ($flags === NULL) {
|
|
self::init_tags_regex(TRUE);
|
|
}
|
|
|
|
$id = $post->ID;
|
|
|
|
$args = array(
|
|
'ignore' => FALSE,
|
|
'flags' => $flags,
|
|
'skip_setting_check' => TRUE,
|
|
'just_check' => TRUE
|
|
);
|
|
$captures = self::capture_crayons($id, $post->post_content, array(), $args);
|
|
|
|
if ($captures['has_captured']) {
|
|
return TRUE;
|
|
} else if ($scan_comments) {
|
|
CrayonSettingsWP::load_settings(TRUE);
|
|
if (CrayonGlobalSettings::val(CrayonSettings::COMMENTS)) {
|
|
$comments = get_comments(array('post_id' => $id));
|
|
foreach ($comments as $comment) {
|
|
if (self::scan_comment($comment, $flags)) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
public static function scan_legacy_post($post, $scan_comments = TRUE) {
|
|
self::init_legacy_tag_bits();
|
|
return self::scan_post($post, $scan_comments, self::$legacy_flags);
|
|
}
|
|
|
|
/**
|
|
* Returns TRUE if the comment contains a Crayon tag
|
|
*/
|
|
public static function scan_comment($comment, $flags = NULL) {
|
|
if ($flags === NULL) {
|
|
self::init_tags_regex();
|
|
}
|
|
$args = array(
|
|
'ignore' => FALSE,
|
|
'flags' => $flags,
|
|
'skip_setting_check' => TRUE,
|
|
'just_check' => TRUE
|
|
);
|
|
$content = apply_filters('get_comment_text', $comment->comment_content, $comment);
|
|
$captures = self::capture_crayons($comment->comment_ID, $content, array(), $args);
|
|
return $captures['has_captured'];
|
|
}
|
|
|
|
public static function install() {
|
|
self::refresh_posts();
|
|
self::update();
|
|
}
|
|
|
|
public static function uninstall() {
|
|
|
|
}
|
|
|
|
public static function update() {
|
|
global $CRAYON_VERSION;
|
|
CrayonSettingsWP::load_settings(TRUE);
|
|
$settings = CrayonSettingsWP::get_settings();
|
|
if ($settings === NULL || !isset($settings[CrayonSettings::VERSION])) {
|
|
return;
|
|
}
|
|
|
|
$version = $settings[CrayonSettings::VERSION];
|
|
|
|
// Only upgrade if the version differs
|
|
if ($version != $CRAYON_VERSION) {
|
|
$defaults = CrayonSettings::get_defaults_array();
|
|
$touched = FALSE;
|
|
|
|
// Upgrade database and settings
|
|
|
|
if (CrayonUtil::version_compare($version, '1.7.21') < 0) {
|
|
$settings[CrayonSettings::SCROLL] = $defaults[CrayonSettings::SCROLL];
|
|
$touched = TRUE;
|
|
}
|
|
|
|
if (CrayonUtil::version_compare($version, '1.7.23') < 0 && $settings[CrayonSettings::FONT] == 'theme-font') {
|
|
$settings[CrayonSettings::FONT] = $defaults[CrayonSettings::FONT];
|
|
$touched = TRUE;
|
|
}
|
|
|
|
if (CrayonUtil::version_compare($version, '1.14') < 0) {
|
|
CrayonLog::syslog("Updated to v1.14: Font size enabled");
|
|
$settings[CrayonSettings::FONT_SIZE_ENABLE] = TRUE;
|
|
}
|
|
|
|
if (CrayonUtil::version_compare($version, '1.17') < 0) {
|
|
$settings[CrayonSettings::HIDE_HELP] = FALSE;
|
|
}
|
|
|
|
// Save new version
|
|
$settings[CrayonSettings::VERSION] = $CRAYON_VERSION;
|
|
CrayonSettingsWP::save_settings($settings);
|
|
CrayonLog::syslog("Updated from $version to $CRAYON_VERSION");
|
|
|
|
// Refresh to show new settings
|
|
header('Location: ' . CrayonUtil::current_url());
|
|
exit();
|
|
}
|
|
}
|
|
|
|
public static function basename() {
|
|
return plugin_basename(__FILE__);
|
|
}
|
|
|
|
public static function pre_excerpt($e) {
|
|
CrayonLog::debug('pre_excerpt');
|
|
self::$is_excerpt = TRUE;
|
|
return $e;
|
|
}
|
|
|
|
public static function post_excerpt($e) {
|
|
CrayonLog::debug('post_excerpt');
|
|
self::$is_excerpt = FALSE;
|
|
$e = self::the_content($e);
|
|
return $e;
|
|
}
|
|
|
|
public static function post_get_excerpt($e) {
|
|
CrayonLog::debug('post_get_excerpt');
|
|
self::$is_excerpt = FALSE;
|
|
return $e;
|
|
}
|
|
|
|
/**
|
|
* Converts Crayon tags found in WP to <pre> form.
|
|
* XXX: This will alter blog content, so backup before calling.
|
|
* XXX: Do NOT call this while updating posts or comments, it may cause an infinite loop or fail.
|
|
* @param $encode Whether to detect missing "decode" attribute and encode html entities in the code.
|
|
*/
|
|
public static function convert_tags($encode = FALSE) {
|
|
$crayon_posts = CrayonSettingsWP::load_legacy_posts();
|
|
if ($crayon_posts === NULL) {
|
|
return;
|
|
}
|
|
|
|
self::init_legacy_tag_bits();
|
|
$args = array(
|
|
'callback' => 'CrayonWP::capture_replace_pre',
|
|
'callback_extra_args' => array('encode' => $encode),
|
|
'ignore' => FALSE,
|
|
'preserve_atts' => TRUE,
|
|
'flags' => self::$legacy_flags,
|
|
'skip_setting_check' => TRUE
|
|
);
|
|
|
|
foreach ($crayon_posts as $postID) {
|
|
$post = get_post($postID);
|
|
$post_content = $post->post_content;
|
|
$post_captures = self::capture_crayons($postID, $post_content, array(), $args);
|
|
|
|
if ($post_captures['has_captured'] === TRUE) {
|
|
$post_obj = array();
|
|
$post_obj['ID'] = $postID;
|
|
$post_obj['post_content'] = addslashes($post_captures['content']);
|
|
wp_update_post($post_obj);
|
|
CrayonLog::syslog("Converted Crayons in post ID $postID to pre tags", 'CONVERT');
|
|
}
|
|
|
|
if (CrayonGlobalSettings::val(CrayonSettings::COMMENTS)) {
|
|
$comments = get_comments(array('post_id' => $postID));
|
|
foreach ($comments as $comment) {
|
|
$commentID = $comment->comment_ID;
|
|
$comment_captures = self::capture_crayons($commentID, $comment->comment_content, array(CrayonSettings::DECODE => TRUE), $args);
|
|
|
|
if ($comment_captures['has_captured'] === TRUE) {
|
|
$comment_obj = array();
|
|
$comment_obj['comment_ID'] = $commentID;
|
|
$comment_obj['comment_content'] = $comment_captures['content'];
|
|
wp_update_comment($comment_obj);
|
|
CrayonLog::syslog("Converted Crayons in post ID $postID, comment ID $commentID to pre tags", 'CONVERT');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
self::refresh_posts();
|
|
}
|
|
|
|
// Used as capture_crayons callback
|
|
public static function capture_replace_pre($capture, $original, $id, $is_inline, $wp_content, $args = array()) {
|
|
$code = $capture['code'];
|
|
$oldAtts = $capture['atts'];
|
|
$newAtts = array();
|
|
$encode = isset($args['encode']) ? $args['encode'] : FALSE;
|
|
if (!isset($oldAtts[CrayonSettings::DECODE]) && $encode) {
|
|
// Encode the content, since no decode information exists.
|
|
$code = CrayonUtil::htmlentities($code);
|
|
}
|
|
// We always set decode=1 irrespectively - so at this point the code is assumed to be encoded
|
|
$oldAtts[CrayonSettings::DECODE] = TRUE;
|
|
$newAtts['class'] = CrayonUtil::html_attributes($oldAtts, CrayonGlobalSettings::val_str(CrayonSettings::ATTR_SEP), '');
|
|
return str_replace($original, CrayonUtil::html_element('pre', $code, $newAtts), $wp_content);
|
|
}
|
|
|
|
// Add TinyMCE to comments
|
|
public static function tinymce_comment_enable($args) {
|
|
if (function_exists('wp_editor')) {
|
|
ob_start();
|
|
wp_editor('', 'comment', array('tinymce'));
|
|
$args['comment_field'] = ob_get_clean();
|
|
}
|
|
return $args;
|
|
}
|
|
|
|
public static function allowed_tags() {
|
|
global $allowedtags;
|
|
$tags = array('pre', 'span', 'code');
|
|
foreach ($tags as $tag) {
|
|
$current_atts = isset($allowedtags[$tag]) ? $allowedtags[$tag] : array();
|
|
// TODO data-url isn't recognised by WP
|
|
$new_atts = array('class' => TRUE, 'title' => TRUE, 'data-url' => TRUE);
|
|
$allowedtags[$tag] = array_merge($current_atts, $new_atts);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// Only if WP is loaded
|
|
if (defined('ABSPATH')) {
|
|
if (!is_admin()) {
|
|
// Filters and Actions
|
|
|
|
add_filter('init', 'CrayonWP::init');
|
|
|
|
CrayonSettingsWP::load_settings(TRUE);
|
|
if (CrayonGlobalSettings::val(CrayonSettings::MAIN_QUERY)) {
|
|
add_action('wp', 'CrayonWP::wp', 100);
|
|
} else {
|
|
add_filter('the_posts', 'CrayonWP::the_posts', 100);
|
|
}
|
|
|
|
// XXX Some themes like to play with the content, make sure we replace after they're done
|
|
add_filter('the_content', 'CrayonWP::the_content', 100);
|
|
|
|
// Highlight bbPress content
|
|
add_filter('bbp_get_reply_content', 'CrayonWP::highlight', 100);
|
|
add_filter('bbp_get_topic_content', 'CrayonWP::highlight', 100);
|
|
add_filter('bbp_get_forum_content', 'CrayonWP::highlight', 100);
|
|
add_filter('bbp_get_topic_excerpt', 'CrayonWP::highlight', 100);
|
|
|
|
// Allow tags
|
|
add_action('init', 'CrayonWP::allowed_tags', 11);
|
|
|
|
if (CrayonGlobalSettings::val(CrayonSettings::COMMENTS)) {
|
|
/* XXX This is called first to match Crayons, then higher priority replaces after other filters.
|
|
Prevents Crayon from being formatted by the filters, and also keeps original comment formatting. */
|
|
add_filter('comment_text', 'CrayonWP::pre_comment_text', 1);
|
|
add_filter('comment_text', 'CrayonWP::comment_text', 100);
|
|
}
|
|
|
|
// This ensures Crayons are not formatted by WP filters. Other plugins should specify priorities between 1 and 100.
|
|
add_filter('get_the_excerpt', 'CrayonWP::pre_excerpt', 1);
|
|
add_filter('get_the_excerpt', 'CrayonWP::post_get_excerpt', 100);
|
|
add_filter('the_excerpt', 'CrayonWP::post_excerpt', 100);
|
|
|
|
add_action('template_redirect', 'CrayonWP::wp_head', 0);
|
|
|
|
if (CrayonGlobalSettings::val(CrayonSettings::TAG_EDITOR_FRONT)) {
|
|
add_filter('comment_form_defaults', 'CrayonWP::tinymce_comment_enable');
|
|
}
|
|
} else {
|
|
// Update between versions
|
|
CrayonWP::update();
|
|
// For marking a post as containing a Crayon
|
|
add_action('update_post', 'CrayonWP::save_post', 10, 2);
|
|
add_action('save_post', 'CrayonWP::save_post', 10, 2);
|
|
add_filter('wp_insert_post_data', 'CrayonWP::filter_post_data', '99', 2);
|
|
}
|
|
register_activation_hook(__FILE__, 'CrayonWP::install');
|
|
register_deactivation_hook(__FILE__, 'CrayonWP::uninstall');
|
|
if (CrayonGlobalSettings::val(CrayonSettings::COMMENTS)) {
|
|
add_action('comment_post', 'CrayonWP::save_comment', 10, 2);
|
|
add_action('edit_comment', 'CrayonWP::save_comment', 10, 2);
|
|
}
|
|
add_filter('init', 'CrayonWP::init_ajax');
|
|
}
|