%PDF- %PDF-
Mini Shell

Mini Shell

Direktori : /proc/1857783/root/var/www/cwg/wp-content/plugins/addthis/backend/
Upload File :
Create Path :
Current File : //proc/1857783/root/var/www/cwg/wp-content/plugins/addthis/backend/AddThisPlugin.php

<?php
/**
 * +--------------------------------------------------------------------------+
 * | Copyright (c) 2008-2017 AddThis, LLC                                     |
 * +--------------------------------------------------------------------------+
 * | This program is free software; you can redistribute it and/or modify     |
 * | it under the terms of the GNU General Public License as published by     |
 * | the Free Software Foundation; either version 2 of the License, or        |
 * | (at your option) any later version.                                      |
 * |                                                                          |
 * | 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 'AddThisAmp.php';
require_once 'AddThisRegistrationFeature.php';
require_once 'AddThisGlobalOptionsFeature.php';
require_once 'AddThisFollowButtonsFeature.php';
require_once 'AddThisSharingButtonsFeature.php';
require_once 'AddThisRecommendedContentFeature.php';
require_once 'AddThisWidgetByDomClass.php';
require_once 'AddThisGlobalOptionsWidget.php';
require_once 'AddThisTool.php';

if (!class_exists('AddThisPlugin')) {
    /**
     * AddThis' root parent class for all its plugins
     *
     * @category   ParentClass
     * @package    AddThisWordPress
     * @subpackage Plugin
     * @author     AddThis <help@addthis.com>
     * @license    GNU General Public License, version 2
     * @link       http://addthis.com AddThis website
     */
    class AddThisPlugin
    {
        protected $enabledFeatures = array();
        protected $registrationObject = null;
        protected $globalOptionsObject = null;
        protected $followButtonsObject = null;
        protected $sharingButtonsObject = null;
        protected $recommendedContentObject = null;

        protected $registrationStatus = true;
        protected $globalOptionsStatus = true;
        protected $followButtonsStatus = false;
        protected $sharingButtonsStatus = false;
        protected $recommendedContentStatus = false;

        protected $baseName = null;
        protected $settingsLinkObject = 'globalOptionsObject';
        protected static $cmsName = 'WordPress';
        // adminJavaScriptAction needs to match here and in AddThisFeature
        protected $adminJavaScriptAction = 'addthis_admin_variables';

        // RegEx pattern that can be used to extract the type and ID of a widget from
        // a shortcode class
        protected $shortcodeIdPattern = '/^addthis_(.+)_(.{4})$/';

        protected $metaBoxId = 'at_widget';
        public static $metaBoxKey = '_at_widget';
        // the text in $metaBoxName must match msgid meta_box_title's English
        // text exactly or translations into other languages will fail
        protected $metaBoxName = 'AddThis Tools';
        // the text in $metaBoxDescription must match msgid
        // meta_box_description's English text exactly or translations into
        // other languages will fail
        protected $metaBoxDescription = 'Select "Off" to stop the AddThis plugin from automatically adding tools above and below this entry.';
        // the text in $metaBoxOnText must match msgid
        // meta_box_tools_enabled's English text exactly or translations into
        // other languages will fail
        protected $metaBoxOnText = 'On';
        // the text in $metaBoxOffText must match msgid
        // meta_box_tools_disabled's English text exactly or translations into
        // other languages will fail
        protected $metaBoxOffText = 'Off';

        // the order of the array below matters for JavaScript variable
        // precidence make a protected variable that matches objectName above
        // require the files with the objectName classes at the top of this file
        protected $features = array(
            'globalOptions'  => array(
                'name'         => 'Global Options',
                'objectName'   => 'AddThisGlobalOptionsFeature',
            ),
            'registration'  => array(
                'name'         => 'Registration',
                'objectName'   => 'AddThisRegistrationFeature',
            ),
            'followButtons'  => array(
                'name'         => 'Follow Buttons',
                'objectName'   => 'AddThisFollowButtonsFeature',
            ),
            'sharingButtons' => array(
                'name'         => 'Sharing Buttons',
                'objectName'   => 'AddThisSharingButtonsFeature',
            ),
            'recommendedContent' => array(
                'name'         => 'Recommended Content',
                'objectName'   => 'AddThisRecommendedContentFeature',
            ),
        );

        /**
         * The constructor.
         *
         * @param string $baseName the base name for this plugin
         *
         * @return null
         */
        public function __construct($baseName)
        {
            $this->baseName = $baseName;
        }

        /**
         * Returns the  version of this plugin
         *
         * @return string
         */
        public function getVersion()
        {
            return $this->version;
        }

        /**
         * Returns the product version of this plugin for lojson params
         *
         * @return string
         */
        public function getProductVersion()
        {
            $productVersion = $this->productPrefix . '-' . $this->getVersion();
            return $productVersion;
        }

        /**
         * Adds a a filter for the plugin_action_links area for this plugin, and
         * adds the main feature's addSettingsLinkToPlugin function as a filter
         * onto it. Essentially, this adds the settings link to the plugin
         * listing page.
         *
         * @return null
         */
        protected function addSettingsLinkToPlugin()
        {
            $callback = array($this, 'settingLinksFilter');

            $filterName = 'plugin_action_links_'.$this->baseName;

            if ($this->validateCallback($callback)) {
                add_filter($filterName, $callback);
            }
        }

        /**
         * This must be public as it's used in a callback for
         * plugin_action_links_$plugin filter
         *
         * Adds a link to the settings and registration pages for this plugin
         * the WordPress plugin listing
         *
         * @param string[] $links an array of strings of HTML anchor tags
         *
         * @return string[] the input with added links
         */
        public function settingLinksFilter($links)
        {
            $variableName = $this->settingsLinkObject;

            if (is_object($this->$variableName)) {
                $link = $this->$variableName->addSettingsLinkToPlugin();
                array_push($links, $link);
            }

            if (is_object($this->registrationObject)) {
                $link = $this->registrationObject->addSettingsLinkToPlugin();
                array_push($links, $link);
            }

            return $links;
        }

        /**
         * Send activation metrics to darkseid
         *
         * @return null
         */
        public function activate()
        {
            $pluginInfo = $this->getAddThisPluginInfo();
            $url = $this->globalOptionsObject->getDarkseidBaseUrl() . 'wordpress/'. 'activation';
            $body = array(
                'cmsName' => $pluginInfo['cms_name'],
                'cmsVersion' => $pluginInfo['cms_version'],
                'pluginName' => $pluginInfo['plugin_name'],
                'pluginVersion' => $pluginInfo['plugin_version'],
                'pluginMode' => $pluginInfo['plugin_mode'],
                'phpVersion' => $pluginInfo['php_version'],
                'homepageUrl' => $this->globalOptionsObject->getHomepageUrl(),
                'pubId' => $this->globalOptionsObject->getUsableProfileId()
            );
            $args = array(
                'headers' => array(
                    'Content-Type' => 'application/json',
                    'Accept'       => 'application/json',
                ),
                'body' => json_encode($body),
                'timeout' => 5,
            );
            $response = wp_remote_post($url, $args);
        }

        /**
         * Send deactivation metrics to darkseid
         *
         * Marks each feature enabled by this plugin as disabled -- if another
         * plugin also enables any of these feature, then it'll get enabled the
         * next time that plugin bootraps
         *
         * @return null
         */
        public function deactivate()
        {
            $pluginInfo = $this->getAddThisPluginInfo();
            $url = $this->globalOptionsObject->getDarkseidBaseUrl() . 'wordpress/'. 'deactivation';
            $body = array(
                'cmsName' => $pluginInfo['cms_name'],
                'cmsVersion' => $pluginInfo['cms_version'],
                'pluginName' => $pluginInfo['plugin_name'],
                'pluginVersion' => $pluginInfo['plugin_version'],
                'pluginMode' => $pluginInfo['plugin_mode'],
                'phpVersion' => $pluginInfo['php_version'],
                'homepageUrl' => $this->globalOptionsObject->getHomepageUrl(),
                'pubId' => $this->globalOptionsObject->getUsableProfileId()
            );
            $args = array(
                'headers' => array(
                    'Content-Type' => 'application/json',
                    'Accept'       => 'application/json',
                ),
                'body' => json_encode($body),
                'timeout' => 5,
            );
            $response = wp_remote_post($url, $args);

            $gooConfigs = $this->globalOptionsObject->getConfigs();

            foreach ($this->features as $feature => $info) {
                $objectVariable = $feature . 'Object';
                $enabledVariable = $feature . 'Status';
                $enabledByPlugin = $this->$enabledVariable;

                if ($enabledByPlugin && $feature != 'globalOptions') {
                    $enabledField = $this->$objectVariable->globalEnabledField;
                    $jsonField = $this->$objectVariable->globalLayersJsonField;
                    $globalOptionsConfigs[$enabledField] = false;
                    $globalOptionsConfigs[$jsonField] = '';
                }
            }

            $this->globalOptionsObject->saveConfigs($globalOptionsConfigs);
        }

        /**
         * This must be public as it's called from bootstrap.php
         *
         * This bootstraps this plugin into WordPress, including adding our
         * JavaScript and CSS onto relevant pages and calling the bootstrap
         * function for every enabled feature
         *
         * @return null
         */
        public function bootstrap()
        {
            $goo = new $this->features['globalOptions']['objectName']();
            $this->globalOptionsObject = $goo;

            $reg = new $this->features['registration']['objectName']($goo);
            $this->registrationObject = $reg;

            $reg->bootstrap();
            $goo->bootstrap();

            $gooConfigs = $goo->getConfigs();
            $saveGooConfigs = false;

            foreach ($this->features as $feature => $info) {
                $objectVariable = $feature . 'Object';
                $objectName = $info['objectName'];
                $enabledVariable = $feature . 'Status';

                if ($feature != 'globalOptions' && $feature != 'registration') {
                    $featureObject = new $objectName($goo);
                    $this->$objectVariable = $featureObject;

                    // Does this plugin enable this feature? If so, bootstrap
                    if ($this->$enabledVariable) {
                        $this->enabledFeatures[] = $feature;
                        $this->$objectVariable->bootstrap();

                        // Does the plugin know it's enabled? If not, fix it.
                        $pluginKnowsItIsEnabled = $this->$objectVariable->isEnabled();
                        if (!$pluginKnowsItIsEnabled) {
                            $field = $this->$objectVariable->globalEnabledField;
                            $gooConfigs[$field] = true;
                            $saveGooConfigs = true;
                        }
                    } else {
                        $this->$objectVariable->getConfigs();
                    }
                }
            }

            if (!isset($gooConfigs['addthis_plugin_controls'])) {
                if ($goo->isMinimalPlugin()) {
                    // the minimal plugin is only functional in AddThis mode
                    // start it off there
                    $gooConfigs['addthis_plugin_controls'] = 'AddThis';
                } else {
                    $gooConfigs['addthis_plugin_controls'] = 'WordPress';
                }

                $saveGooConfigs = true;
            }

            if ($saveGooConfigs) {
                $goo->saveConfigs($gooConfigs);
            }

            // Add a link to the main settings page & the registration page
            $this->addSettingsLinkToPlugin();

            // For adding option for show/hide AddThis tools on admin post add/edit page.
            add_action('admin_init', array($this, 'addMetaBox'));

            add_filter(
                'language_attributes',
                array($this, 'htmlNameSpacesAttributes')
            );

            if (is_admin()) {
                // load our language files -- we only use them on admin pages
                add_action(
                    'plugins_loaded',
                    array($this, 'loadTextDomain')
                );

                // addthis_share, addthis_config, addthis_layers and
                // addthis_plugin_info for all public pages
                add_action(
                    'wp_ajax_'.$this->globalOptionsObject->publicJavaScriptAction,
                    array($this, 'printJavaScriptForGlobalVariables')
                );
                add_action(
                    'wp_ajax_nopriv_'.$this->globalOptionsObject->publicJavaScriptAction,
                    array($this, 'printJavaScriptForGlobalVariables')
                );

                // make JavaScript file to ui relevant variables
                add_action(
                    'wp_ajax_'.$this->adminJavaScriptAction,
                    array($this, 'printJavascriptForAdminUi')
                );
            } else {
                // add scripts at "wp" action to ensure that AMP endpoint is
                // properly detected
                add_action('wp', array($this, 'addScripts'));
            }

            add_action('widgets_init', array($this, 'registerWidgets'));

            $this->addShortCodes();

            register_activation_hook(
                $this->baseName,
                array($this, 'activate')
            );

            register_deactivation_hook(
                $this->baseName,
                array($this, 'deactivate')
            );
        }

        /**
         * Registers the tool widget and script widgets, if they don't already
         * exist.
         *
         * @return null
         */
        public function registerWidgets()
        {
            $widgetClassName = 'AddThisWidgetByDomClass';
            if (!$this->existsWidget($widgetClassName)) {
                register_widget($widgetClassName);
            }

            $widgetClassName = 'AddThisGlobalOptionsWidget';
            if (!$this->existsWidget($widgetClassName)) {
                register_widget($widgetClassName);
            }
        }

        /**
         * Determines if a widget with the passed class name has already been
         * registered with WordPress
         *
         * @param string $widgetClassName the name of the class for a WordPress
         * widget
         *
         * @return boolean true is the widget has already been registered, false
         * if it has not
         */
        public static function existsWidget($widgetClassName)
        {
            if (empty($GLOBALS['wp_widget_factory'])) {
                return false;
            }

            $widgets = array_keys($GLOBALS['wp_widget_factory']->widgets);

            $exists = in_array($widgetClassName, $widgets);
            return $exists;
        }

        /**
         * This must be public as it's used for a callback for the
         * plugins_loaded action
         *
         * Loads our language files needed in PHP for this plugin
         *
         * @return null
         */
        public function loadTextDomain()
        {
            $path = '/' . $this->globalOptionsObject->getPluginFolder() . '/frontend/build/l10n';
            load_plugin_textdomain(AddThisFeature::$l10n_domain, false, $path);
        }

        /**
         * Performs any AMP initialization necessary for tools
         * configured in anonymous mode
         *
         * @return null
         */
        protected function initAnonymousAmpTools() {
            AddThisAmp::initConfiguredTools(
                $this->sharingButtonsObject->getConfigs(),
                $this->globalOptionsObject
            );
        }

        /**
         * This must be public as it's used for a callback for the
         * wp action
         *
         * Setup out scripts to enqueue (if async is off), adds an action that
         * echos our script onto page if async is turned on, and adds an action
         * to enqueue our styles late. For all public pages.
         *
         * @return null
         */
        public function addScripts()
        {
            $gooConfigs = $this->globalOptionsObject->getConfigs();

            // Based on AMP mode, queue anonymous tool init or queue widget scripts.
            // It is not necessary to print widget scripts in AMP mode as
            // they will be scrubbed by the HTML filter
            if (!$gooConfigs['amp_disable'] && AddThisAmp::inAmpMode()) {
                if ($this->globalOptionsObject->inAnonymousMode()) {
                    $this->initAnonymousAmpTools();
                }
            } else {
                if (!empty($gooConfigs['script_location'])
                    && $gooConfigs['script_location'] == 'footer'
                ) {
                    $pageScriptHook = 'wp_footer';
                } else {
                    $pageScriptHook = 'wp_head';
                }

                if (empty($gooConfigs['enqueue_local_settings']) ||
                    empty($gooConfigs['enqueue_client'])
                ) {
                    add_action(
                        $pageScriptHook,
                        array($this, 'printAddThisWidgetScript'),
                        19
                    );
                }

                if (!empty($gooConfigs['enqueue_local_settings']) ||
                    !empty($gooConfigs['enqueue_client'])
                ) {
                    add_action('wp_enqueue_scripts', array($this, 'enqueueScripts'));
                }
            }

            // we want this to get enqueued after the theme css, thus the 18
            add_action('wp_enqueue_scripts', array($this, 'enqueueStyles'), 18);
        }


        /**
         * This must be public as it's used for a callback in the
         * wp_enqueue_scripts action
         *
         * Enqueues our scripts the correct way when we don't have to do hacky
         * async loading stuff
         *
         * @return null
         */
        public function enqueueScripts()
        {
            $gooConfigs = $this->globalOptionsObject->getConfigs();

            if (!empty($gooConfigs['script_location'])
                && $gooConfigs['script_location'] == 'footer'
            ) {
                $enqueueInFooter = true;
            } else {
                $enqueueInFooter = false;
            }

            if (!empty($gooConfigs['enqueue_local_settings'])) {
                $localSettingsUrl = admin_url('admin-ajax.php')
                    . '?action='.$this->globalOptionsObject->publicJavaScriptAction;
                wp_enqueue_script(
                    'addthis_global_options',
                    $localSettingsUrl,
                    array(),
                    false,
                    $enqueueInFooter
                );
            }

            if (!empty($gooConfigs['enqueue_client'])) {
                $clientUrl = $this->globalOptionsObject->getAddThisWidgetJavaScriptUrl();

                $dependencies = array();
                if (!empty($gooConfigs['enqueue_local_settings'])) {
                    $dependencies[] = 'addthis_global_options';
                }

                wp_enqueue_script(
                    'addthis_widget',
                    $clientUrl,
                    $dependencies,
                    false,
                    $enqueueInFooter
                );
            }
        }

        /**
         * This must be public as it's used for a callback in the
         * wp_enqueue_scripts action
         *
         * Enqueues our styles super late, in the hopes that nothing else will
         * enquque after us, and we win! This is done so that our CSS can win
         * out over template CSS more often. For all public pages.
         *
         * @return null
         */
        public function enqueueStyles()
        {
            $gooConfigs = $this->globalOptionsObject->getConfigs();

            if (!empty($gooConfigs['script_location'])
                && $gooConfigs['script_location'] == 'footer'
            ) {
                $enqueueInFooter = true;
            } else {
                $enqueueInFooter = false;
            }

            $cssRoot = $this->globalOptionsObject->getSettingsUiBaseUrl() . 'build/';
            wp_enqueue_style(
                'addthis_all_pages',
                $cssRoot . 'addthis_wordpress_public.min.css',
                array(),
                false
            );
        }

        /**
         * This must be public as it's used for a callback in the filter
         * language_attributes
         *
         * Checks for the Facebook Graph namespace and the AddThis namespace,
         * and adds it if it doesn't already exist in the input string.
         *
         * @param string $input the string of attributes to be added the pages
         * HTML tag
         *
         * @return string the string of attributes to be added to the pages HTML
         * tag with Facebook and AddThis names spaces added if thye wheren't
         * there already
         */
        public function htmlNameSpacesAttributes($input)
        {
            $gooConfigs = $this->globalOptionsObject->getConfigs();
            if (empty($gooConfigs['xmlns_attrs'])) {
                return $input;
            }

            $output = trim($input);

            $attrs = array(
                array(
                    'attr' => 'xmlns:fb',
                    'value' => 'https://www.facebook.com/2008/fbml',
                ),
                array(
                    'attr' => 'xmlns:addthis',
                    'value' => 'https://www.addthis.com/help/api-spec',
                ),
            );

            foreach ($attrs as $info) {
                if (strpos($input, $info['attr']) === false) {
                    $output .= ' ' . $info['attr']
                        . '="' . $info['value'] . '"';
                }
            }

            $output .= ' ';
            return $output;
        }

        /**
         * Prints the JavaScript that should be used to set addthis_share,
         * addthis_config, addthis_plugin_info, as well as the params to use
         * when loading addthis.layers
         *
         * @return null
         */
        public function printJavaScriptForGlobalVariables()
        {
            if (is_feed()) {
                return null;
            }

            header('Content-type: application/x-javascript');

            $javaScript = $this->getJavascriptForGlobalVariables();

            echo $javaScript;
            die();
        }

        /**
         * Returns a string of JavaScript that should be used to set
         * addthis_share, addthis_config, addthis_plugin_info, as well as the
         * params to use when loading addthis.layers
         *
         * @return string
         */
        public function getJavascriptForGlobalVariables()
        {
            $share = array();
            $config = array();
            $layers = array();
            $layersTools = array();
            $pluginInfo = $this->getAddThisPluginInfo();

            $gooSettings = $this->globalOptionsObject->getConfigs();
            $updateGlobalOptionsSettings = false;

            reset($this->features);
            foreach ($this->features as $feature => $info) {
                $objectVariable = $feature . 'Object';
                $featureObject = $this->$objectVariable;

                if ($featureObject->isEnabled()) {
                    $featureShare = $featureObject->getAddThisShare();
                    $share = array_replace_recursive($share, $featureShare);

                    $featureConfig = $featureObject->getAddThisConfig();
                    $config = array_replace_recursive($config, $featureConfig);

                    if ($this->globalOptionsObject->inAnonymousMode()) {
                        $featureLayers = $featureObject->getAddThisLayers();
                        $layers = array_replace_recursive($layers, $featureLayers);
                        $featureLayersTools = $featureObject->getAddThisLayersTools();
                        $layersTools = array_merge($layersTools, $featureLayersTools);

                        // save a layers json string for the sharing buttons plugin
                        if (!empty($featureObject->globalLayersJsonField)) {
                            $new = json_encode((object)$featureLayers);
                            $field = $featureObject->globalLayersJsonField;

                            if (isset($gooSettings[$field])) {
                                $old = $gooSettings[$field];
                            }

                            if ((empty($old) && !empty($featureLayers))
                                || (!empty($old) && $old != $new)
                            ) {
                                $gooSettings[$field] = $new;
                                $updateGlobalOptionsSettings = true;
                            }
                        }
                    }
                }
            }

            // save a layers json string for the sharing buttons plugin
            if ($updateGlobalOptionsSettings) {
                $this->globalOptionsObject->saveConfigs($gooSettings);
            }

            if (!empty($gooSettings['addthis_share_json'])) {
                $shareFromSettings = json_decode($gooSettings['addthis_share_json'], true);
                $share = array_replace_recursive($share, $shareFromSettings);
            }
            $share = apply_filters('addthis_share_array', $share);
            $shareJson = json_encode((object)$share);
            $shareJson = apply_filters('addthis_share_json', $shareJson, $share);

            if (!empty($gooSettings['addthis_config_json'])) {
                $configFromSettings = json_decode($gooSettings['addthis_config_json'], true);
                $config = array_replace_recursive($config, $configFromSettings);
            }
            $config = apply_filters('addthis_config_array', $config);
            $configJson = json_encode((object)$config);
            $configJson = apply_filters('addthis_config_json', $configJson, $config);

            if ($this->globalOptionsObject->inAnonymousMode()) {
                if (!empty($gooSettings['addthis_layers_json'])) {
                    $layersFromSettings = json_decode($gooSettings['addthis_layers_json'], true);
                    $layers = array_replace_recursive($layers, $layersFromSettings);
                }
                $layers = apply_filters('addthis_layers_array', $layers);
                $layersJson = json_encode((object)$layers);
                $layersJson = apply_filters('addthis_layers_json', $layersJson, $layers);

                $layersToolsJson = json_encode((array)$layersTools);
                $layersToolsAlreadyDefined = '';
                foreach ($layersTools as $toolLayersConfig) {
                    $toolLayersConfig = json_encode((object)$toolLayersConfig);
                    $layersToolsAlreadyDefined .= 'window.addthis_layers_tools.push('.$toolLayersConfig.'); ';
                }
            }

            $pluginInfoJson = json_encode((object)$pluginInfo);

            $javaScript  = 'if (window.addthis_product === undefined) { ';
            $javaScript .=   'window.addthis_product = "' . $this->productPrefix . '"; ';
            $javaScript .= '} ';

            $javaScript .= 'if (window.wp_product_version === undefined) { ';
            $javaScript .=   'window.wp_product_version = "' . $this->getProductVersion() . '"; ';
            $javaScript .= '} ';

            $javaScript .= 'if (window.addthis_share === undefined) { ';
            $javaScript .=   'window.addthis_share = ' . $shareJson . '; ';
            $javaScript .= '} ';

            $javaScript .= 'if (window.addthis_config === undefined) { ';
            $javaScript .=   'window.addthis_config = ' . $configJson . '; ';
            $javaScript .= '} ';

            if ($this->globalOptionsObject->inAnonymousMode()) {
                $javaScript .= 'if (window.addthis_layers === undefined) { ';
                $javaScript .=   'window.addthis_layers = ' . $layersJson . '; ';
                $javaScript .= '} ';

                $javaScript .= 'if (window.addthis_layers_tools === undefined) { ';
                $javaScript .= 'window.addthis_layers_tools = ' . $layersToolsJson . '; ';
                $javaScript .= '} else { ';
                $javaScript .=   $layersToolsAlreadyDefined . ' ';
                $javaScript .= '} ';
            }

            $javaScript .= 'if (window.addthis_plugin_info === undefined) { ';
            $javaScript .=   'window.addthis_plugin_info = ' . $pluginInfoJson . '; ';
            $javaScript .= '} ';

            $javaScript .= $this->getJavaScriptToLoadTools();

            $javaScript = apply_filters('addthis_config_javascript', $javaScript);
            return $javaScript;
        }

        /**
         * Returns a string of JavaScript that can be used to bootstrap tools
         *
         * @return string JavaScript for bootstraping AddThis tools
         */
        public function getJavaScriptToLoadTools()
        {
            $gooSettings = $this->globalOptionsObject->getConfigs();
            if (empty($gooSettings['ajax_support'])) {
                $javaScript = '
                    (function() {
                      var first_load_interval_id = setInterval(function () {
                        if (typeof window.addthis !== \'undefined\') {
                          window.clearInterval(first_load_interval_id);
                          if (typeof window.addthis_layers !== \'undefined\' && Object.getOwnPropertyNames(window.addthis_layers).length > 0) {
                            window.addthis.layers(window.addthis_layers);
                          }
                          if (Array.isArray(window.addthis_layers_tools)) {
                            for (i = 0; i < window.addthis_layers_tools.length; i++) {
                              window.addthis.layers(window.addthis_layers_tools[i]);
                            }
                          }
                        }
                     },1000)
                    }());
                ';
            } else {
                $javaScript = '
                    (function() {
                      var new_tools_timeout = false;

                      var refresh_tools = function() {
                        new_tools_timeout = false;
                        addthis.layers.refresh();
                      };

                      var first_load_check = function () {
                        if (typeof window.addthis !== \'undefined\') {
                          window.clearInterval(first_load_interval_id);
                          if (typeof window.addthis_layers !== \'undefined\' && Object.getOwnPropertyNames(window.addthis_layers).length > 0) {
                            window.addthis.layers(window.addthis_layers);
                          }
                          if (Array.isArray(window.addthis_layers_tools)) {
                            for (i = 0; i < window.addthis_layers_tools.length; i++) {
                              window.addthis.layers(window.addthis_layers_tools[i]);
                            }
                          }

                          window.atnt = function() {
                            if (new_tools_timeout !== false) {
                              window.clearTimeout(new_tools_timeout);
                            }
                            new_tools_timeout = window.setTimeout(refresh_tools, 15);
                          };
                        }
                      };

                      var first_load_interval_id = window.setInterval(first_load_check, 1000);
                    }());
                ';
            }

            return $javaScript;
        }

        /**
         * Creates plugin specific settings for the JavaScript variable
         * addthis_plugin_info
         *
         * @return array an associative array
         */
        public function getAddThisPluginInfo()
        {
            $pluginInfo = array();

            $pluginInfo = array();
            $pluginInfo['info_status'] = 'enabled';
            $pluginInfo['cms_name'] = $this->getCmsName();
            $pluginInfo['plugin_name'] = $this->name;
            $pluginInfo['plugin_version'] = $this->getVersion();
            if ($this->globalOptionsObject->inAnonymousMode()) {
                $pluginInfo['plugin_mode'] = $this->getCmsName();
            } else {
                $pluginInfo['plugin_mode'] = 'AddThis';
            }
            $pluginInfo['anonymous_profile_id'] = $this->globalOptionsObject->getAnonymousProfileId();

            // post specific stuff that requires wp_query
            global $wp_query;
            if (isset($wp_query)) {
                $pluginInfo['page_info']['template'] = AddThisTool::currentTemplateType();
                if (isset($wp_query->query_vars['post_type'])) {
                    $pluginInfo['page_info']['post_type'] = $wp_query->query_vars['post_type'];
                }
            }

            // post specific meta box selection
            global $post;
            if (is_object($post)
                && ($post instanceof WP_Post)
                && !empty($post->ID)
            ) {
                $pluginInfo['sharing_enabled_on_post_via_metabox'] = self::metaBoxDisablesTools($post);
            }

            return $pluginInfo;
        }

        /**
         * Prints the JavaScript that should be used to include
         * addthis_widget.js on the page.
         *
         * @return null
         */
        public function printAddThisWidgetScript()
        {
            if (is_feed()) {
                return null;
            }

            $configs = $this->globalOptionsObject->getConfigs();
            $script = '';

            if (empty($configs['enqueue_local_settings'])) {
                $script .= '<script data-cfasync="false" type="text/javascript">'
                    . $this->getJavascriptForGlobalVariables()
                    . '</script>';
            }

            if (empty($configs['enqueue_client'])) {
                $clientUrl = $this->globalOptionsObject->getAddThisWidgetJavaScriptUrl();
                $script .= ' <script data-cfasync="false" type="text/javascript" '
                    . 'src="' . $clientUrl . '"';
                if (!empty($configs['addthis_asynchronous_loading'])) {
                    $script .= ' async="async"';
                }
                $script .= '></script>';
            }

            echo $script;
        }

        /**
         * The profile ID that should be used when loading addthis_widget.js
         *
         * @return string a profile ID
         */
        public function getUsableProfileId()
        {
            $profileId = $this->globalOptionsObject ->getUsableProfileId();
            return $profileId;
        }

        /**
         * Validates that the WordPress callback passed is valid.
         *
         * @param string|array $callback The name of a global function as a
         * string, or an array with the first item being an object, and the
         * second being a string naming a function in that object.
         *
         * @return boolean true for valid, false otherwise
         */
        public static function validateCallback($callback)
        {
            if (empty($callback[0])) {
                return false;
            }

            if (empty($callback[1])) {
                // check if string
                $methodName = $callback;
                if (!is_string($methodName)) {
                    return false;
                }

                if (!function_exists($methodName)) {
                    return false;
                }

                return true;
            }

            $object = $callback[0];
            $methodName = $callback[1];

            if (!is_object($object)) {
                return false;
            } elseif (!method_exists($object, $methodName)) {
                return false;
            }

            return true;
        }

        /**
         * The name of the content management system this plugin runs within
         *
         * @return string
         */
        private static function getCmsName()
        {
            return self::$cmsName;
        }

        /**
         * The full version number of the CMS (ie. 4.2.2, not 4.2)
         *
         * @return string
         */
        private static function getCmsVersion()
        {
            $version = get_bloginfo('version');
            return $version;
        }

        /**
         * The major and minor version of the CMS (ie. 4.2, not 4.2.2)
         *
         * @return string
         */
        private static function getCmsMinorVersion()
        {
            $stringVersion = substr(self::getCmsVersion(), 0, 3);
            $version = (float)$stringVersion;
            return $version;
        }


        /**
         * This must be public as it's used in a callback for add_action
         *
         * Prints JavaScript to make site specific variables available to the UI
         *
         * @return null
         */
        public function printJavascriptForAdminUi()
        {
            $this->globalOptionsObject->checkForEditPermissions(true);

            header('Content-type: application/x-javascript');

            $current_user = wp_get_current_user();

            $config = array();

            $ui = array(
                'defaults'    => array(
                    'rss'       => get_bloginfo('rss2_url'),
                    'email'     => $current_user->user_email,
                    'domain'    => $this->globalOptionsObject->getSiteDomain(),
                ),
                'urls'        => array(
                    'home'     => home_url(),
                    'admin'    => admin_url(),
                    'ajax'     => admin_url('admin-ajax.php'),
                    'widgets'  => admin_url('widgets.php'),
                    'ui'       => $this->globalOptionsObject->getSettingsUiBaseUrl(),
                    'settings' => $this->globalOptionsObject->getSettingsPageUrl(),
                ),
                'plugin'      => array(
                    'slug'     => $this->pluginSlug,
                    'pco'      => $this->productPrefix,
                    'version'  => $this->getVersion(),
                ),
                'siteName'    => get_bloginfo('name'),
                'language'    => get_bloginfo('language'),
                'locale'      => get_locale(),
                'permissions' => array(
                    'unfiltered_html' => current_user_can('unfiltered_html')
                ),
            );

            $pluginInfo = $this->getAddThisPluginInfo();

            $configJson = json_encode((object)$config);
            $uiJson = json_encode((object)$ui);
            $pluginInfoJson = json_encode((object)$pluginInfo);

            $javaScript = '
                window.addthis_config = ' . $configJson . ';
                window.addthis_ui = ' . $uiJson . ';
                window.addthis_plugin_info = ' . $pluginInfoJson . ';
            ';

            echo $javaScript;
            die();
        }


        /**
         * This must be public as it's used in a callback for
         * admin_init
         *
         * Checks if our meta box exists already
         *
         * @return null
         */
        public function addMetaBox()
        {
            $configs = $this->globalOptionsObject->getConfigs();
            if ($configs['addthis_per_post_enabled']) {
                $args = array(
                   '_builtin' => false,
                );
                $postTypes = get_post_types($args, 'names');
                $postTypes[] = 'post';
                $postTypes[] = 'page';

                foreach ($postTypes as $postType) {
                    add_meta_box(
                        $this->metaBoxId,
                        __($this->metaBoxName, AddThisFeature::$l10n_domain),
                        array($this, 'printMetaBoxHtml'),
                        $postType,
                        'advanced',
                        'high'
                    );
                }

                add_action('save_post', array($this, 'saveMetaBoxOption'));
            }
        }

        /**
         * This must be public as it's used in a callback for
         * save_post
         *
         * Validates and saves selected option for meta box
         *
         * @param string|int $postId The ID for the post
         *
         * @return null
         */
        public function saveMetaBoxOption($postId)
        {
            global $post;

            if (!isset($post, $_POST['_at_widget'])) {
                return;
            }

            if ($_POST['_at_widget'] == 1) {
                update_post_meta($postId, '_at_widget', '1');
            } else {
                update_post_meta($postId, '_at_widget', '0');
            }
        }

        /**
         * This must be public as it's used in AddThisFeature
         *
         * Checks if AddThis tools are disabled on this post via the meta box
         *
         * @param object $post a WordPress post object
         *
         * @return boolean true for enabled, false for disabled
         */
        public static function metaBoxDisablesTools($post)
        {
            $postId = $post->ID;
            $metaBoxFlag = get_post_meta($postId, self::$metaBoxKey, true);
            if ($metaBoxFlag == '0') {
                $enabled = false;
            } else {
                $enabled = true;
            }
            return !$enabled;
        }

        /**
         * This must be public as it's used in a callback for
         * add_meta_box
         *
         * Echos out the HTML for the AddThis meta box
         *
         * @param object $post a WordPress post object
         *
         * @return null
         */
        public function printMetaBoxHtml($post)
        {
            $offChecked = '';
            $onChecked = '';
            $checkedValue = 'checked="checked"';
            if (self::metaBoxDisablesTools($post)) {
                $offChecked = $checkedValue;
            } else {
                $onChecked = $checkedValue;
            }

            $descriptionText = esc_html__($this->metaBoxDescription, AddThisFeature::$l10n_domain);
            $onText = esc_html__($this->metaBoxOnText, AddThisFeature::$l10n_domain);
            $offText = esc_html__($this->metaBoxOffText, AddThisFeature::$l10n_domain);

            $html = '<p>' . $descriptionText . '</p>
                <label for="'.self::$metaBoxKey.'_on">
                    <input
                        type="radio"
                        id="'.self::$metaBoxKey.'_on"
                        name="'.self::$metaBoxKey.'"
                        value="1"
                        ' . $onChecked . '
                    />
                    <span class="addthis-checkbox-label">' . $onText . '</span>
                </label>
                <label for="'.self::$metaBoxKey.'_off">
                    <input
                        type="radio"
                        id="'.self::$metaBoxKey.'_off"
                        name="'.self::$metaBoxKey.'"
                        value="0"
                        ' . $offChecked . '
                    />
                    <span class="addthis-checkbox-label">' . $offText . '</span>
                </label>
            ';

            echo $html;
        }

        /**
         * The function shortcode_exists only works in WordPress 3.6.0+. We
         * support 3.0.0+... making a function to fall back on the ugly hacky
         * internal way of checking for this in older WordPress instances
         *
         * @param string $tag the shortcode
         *
         * @return boolean true if the shortcode exists, false if it does not
         */
        public function shortcodeExists($tag)
        {
            if (function_exists('shortcode_exists')) {
                return shortcode_exists($tag);
            }

            global $shortcode_tags;
            return isset($shortcode_tags[$tag]);
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML to use to replace a short tag for this tool. Includes
         * tags to identify its from a short code.
         *
         * @param string $cssClass a CSS class for indentifying an AddThis tool
         *
         * @return string this should be valid html
         */
        public function getInlineCodeForShortCode($cssClass)
        {
            $html = '';
            $gooSettings = $this->globalOptionsObject->getConfigs();

            if (AddThisAmp::inAmpMode()) {
                if ($gooSettings['amp_disable']) {
                    return '';
                }

                $widgetType = 'shin';
                $width = $gooSettings['amp_inline_share_width'];
                $height = $gooSettings['amp_inline_share_height'];

                if ($this->globalOptionsObject->inRegisteredMode()) {
                    $profileId = $this->globalOptionsObject->getUsableProfileId();
                    $widgetId = null;

                    if (!empty($cssClass)) {
                        preg_match($this->shortcodeIdPattern, $cssClass, $matches);
                        if (count($matches) >= 3) {
                            $widgetId = $matches[2];
                        }
                    }

                    $html = AddThisAmp::getAmpHtml($profileId, $widgetId, $widgetType, $cssClass, $width, $height);
                } else if (!empty($cssClass)) {
                    $html = AddThisAmp::getAmpHtmlByClass($cssClass, $widgetType, $width, $height);
                }
            } else {
                $html  = '<!-- Created with a shortcode from an AddThis plugin -->';

                if (!empty($cssClass)) {
                    $html .= '<div class="'.$cssClass.' addthis_tool"></div>';
                } else {
                    $html .= '<!-- No CSS class provided. Nothing to do here.-->';
                }

                if (!empty($gooSettings['ajax_support'])) {
                    $html .= '<script>if (typeof window.atnt !== \'undefined\') { window.atnt(); }</script>';
                }

                $html .= '<!-- End of short code snippet -->';
            }

            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Custom Follow buttons
         * before tool consolidation. This can't ever be deleted because then
         * people's shortCodes (which we can't migrate) will start showing up
         * as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeCustomFollow()
        {
            $cssClass = 'addthis_custom_follow';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Horizontal Follow buttons
         * before tool consolidation. This can't ever be deleted because then
         * people's shortCodes (which we can't migrate) will start showing up
         * as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeHorizontalFollow()
        {
            $cssClass = 'addthis_horizontal_follow_toolbox';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Vertical Follow buttons
         * before tool consolidation. This can't ever be deleted because then
         * people's shortCodes (which we can't migrate) will start showing up
         * as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeVerticalFollow()
        {
            $cssClass = 'addthis_vertical_follow_toolbox';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Horizontal Recommended
         * Content before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeHorizontalRecommenededContent()
        {
            $cssClass = 'addthis_recommended_horizontal';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Vertical Recommended
         * Content before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeVerticalRecommenededContent()
        {
            $cssClass = 'addthis_recommended_vertical';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Custom Share buttons
         * before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeCustomShare()
        {
            $cssClass = 'addthis_custom_sharing_buttons';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Jumbo Share buttons
         * before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeJumboShare()
        {
            $cssClass = 'addthis_jumbo_sharing_buttons';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Original Share buttons
         * before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeOriginalShare()
        {
            $cssClass = 'addthis_original_sharing_buttons';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for Responsive Share buttons
         * before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeResponsiveShare()
        {
            $cssClass = 'addthis_responsive_sharing_buttons';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with an old CSS class used for (Square) Share buttons
         * before tool consolidation. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @return string this should be valid html
         */
        public function historicShortCodeSquareShare()
        {
            $cssClass = 'addthis_sharing_buttons';
            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * This must be public as it's used in a callback for add_shortcode
         *
         * Returns HTML with the passed CSS class. This can't ever be deleted
         * because then people's shortCodes (which we can't migrate) will start
         * showing up as code on their site.
         *
         * @param array $atts associative array of attributes on a widget.
         *   This function looks for the 'tool' property
         *
         * @return string this should be valid html
         */
        public function shortCodeByDomClass($atts)
        {
            if (isset($atts['tool'])) {
                $cssClass = $atts['tool'];
            } else {
                $cssClass = false;
            }

            $html = $this->getInlineCodeForShortCode($cssClass);
            return $html;
        }

        /**
         * Adds WordPress short codes.
         *
         * @return null
         */
        public function addShortCodes()
        {
            $globlaOptionsTool = new AddThisGlobalOptionsTool();

            $shortCodes = array(
                // 'shortCode' => array(objectWithMethod, 'methodName')
                'addthis_script' => array($globlaOptionsTool, 'getInlineCodeForShortCode'),
                'addthis' => array($this, 'shortCodeByDomClass'),
                /*
                 * Historic shortcodes. These can't ever be deleted because then
                 * people's short codes inside posts/pages (which we can't
                 * migrate) will start showing up as code on their site. Yipee!
                 */
                'addthis_custom_follow_buttons'          => array(
                    $this, 'historicShortCodeCustomFollow'
                ),
                'addthis_horizontal_follow_buttons'      => array(
                    $this, 'historicShortCodeHorizontalFollow'
                ),
                'addthis_vertical_follow_buttons'        => array(
                    $this, 'historicShortCodeVerticalFollow'
                ),
                'addthis_horizontal_recommended_content' => array(
                    $this, 'historicShortCodeHorizontalRecommenededContent'
                ),
                'addthis_vertical_recommended_content'   => array(
                    $this, 'historicShortCodeVerticalRecommenededContent'
                ),
                'addthis_custom_sharing_buttons'         => array(
                    $this, 'historicShortCodeCustomShare'
                ),
                'addthis_jumbo_sharing_buttons'          => array(
                    $this, 'historicShortCodeJumboShare'
                ),
                'addthis_original_sharing_buttons'       => array(
                    $this, 'historicShortCodeOriginalShare'
                ),
                'addthis_responsive_sharing_buttons'     => array(
                    $this, 'historicShortCodeResponsiveShare'
                ),
                'addthis_sharing_buttons'                => array(
                    $this, 'historicShortCodeSquareShare'
                ),
            );

            foreach ($shortCodes as $shortCode => $callback) {
                if ($this->shortcodeExists($shortCode)) {
                    continue;
                }

                add_shortcode($shortCode, $callback);
            }
        }
    }
}

Zerion Mini Shell 1.0