<?php /** * Infinite Scroll Presets Interface * * Stores theme-specific presets for CSS Selectors to aid with setup. Pulls community presets from CSV * stored in the plugin's SVN repo. * * The csv from the repo is cached for 24 hours as a site-transient (available to all sites on a network install) * * Custom presets (beyond the SVN CSV) are stored as a site option (also available to all sites on network) * * On a single site install, settings available to all admins. * On a network install, settings available only to super-admins (but site-admins can load those presets) * * If a user hasn't chosen CSS selectors for there theme and a preset exists, the plugin will * default to the preset (thus in many cases, no need to adjust any settings or know this exists). * * Hierarchy of presets: 1) User specified, 2) (admin specified) custom preset, 3) community specified preset * * @subpackage Presets * @package Infinite_Scroll */ require_once(ABSPATH . "/wp-admin/includes/theme.php"); class Infinite_Scroll_Presets { private $parent; // public $preset_url = 'http://plugins.svn.wordpress.org/infinite-scroll/branches/PresetDB/presets.csv'; public $preset_url = 'https://raw.github.com/benbalter/Infinite-Scroll/presetDB/presets.csv'; public $custom_preset_key = 'infinite_scroll_presets'; public $ttl = 86000; //TTL of transient cache in seconds, 1 day = 86400 = 60*60*24 public $keys = array( 'theme', 'contentSelector', 'navSelector', 'itemSelector', 'nextSelector' ); /** * Register hooks with WordPress API * * @param object $parent (reference) the parent class */ function __construct( &$parent ) { $this->parent = &$parent; add_action( 'admin_init', array( &$this, 'set_presets' ) ); add_action( 'wp_ajax_infinite-scroll-edit-preset', array( &$this, 'process_ajax_edit' ) ); add_action( 'wp_ajax_infinite-scroll-delete-preset', array( &$this, 'process_ajax_delete' ) ); add_filter( $this->parent->prefix . 'presets', array( &$this, 'merge_custom_presets' ) ); add_filter( $this->parent->prefix . 'options', array( &$this, 'default_to_presets'), 9 ); add_action( $this->parent->prefix . 'refresh_cache', array( &$this, 'get_presets' ) ); } /** * Allow for class overloading * @param string $preset the theme slug to retrieve * @return array|bool the presets or false on failure */ function __get( $preset ) { return $this->get_preset( $preset ); } function getThemes($args) { if (function_exists("wp_get_themes")) { return wp_get_themes($args); } else { return get_themes(); } } function getTheme($theme) { if (function_exists("wp_get_theme")) { return wp_get_theme($theme); } else { return get_theme($theme); } } /** * Pulls preset array from cache, or retrieves and parses * @return array an array of preset objects * @todo Consider using TLC Transients in case cron isn't working */ function get_presets() { //check cache if ( $cache = get_transient( $this->parent->prefix . 'presets' ) ) return apply_filters( $this->parent->prefix . 'presets', $cache ); $data = wp_remote_get( $this->preset_url ); if ( is_wp_error( $data ) ) return array(); $data = wp_remote_retrieve_body( $data ); //parse CSV string into array $presets = $this->parse_csv( $data ); //sort by key alpha ascending asort( $presets ); set_transient( $this->parent->prefix . 'presets', $presets, $this->ttl ); return apply_filters( $this->parent->prefix . 'presets', $presets ); } /** * Return a theme's preset object * @param string $theme the slug of theme to retrieve * @return object the preset object */ function get_preset( $theme = null ) { if ( $theme == null ) $theme = get_stylesheet(); $presets = $this->get_presets(); //direct match found, return if ( array_key_exists( $theme, $presets ) ) return $presets[ $theme ]; //no direct match found, permahps this is a child theme? //theme isn't installed, no way to know if it's a child, so skip if ( !$this->theme_installed( $theme ) ) return false; //WP version 3.4+, use the new wp_get_themes function if ( function_exists( 'wp_get_theme' ) ) { $theme = $this->getTheme($theme); //not a theme or not a child if ( is_wp_error( $theme ) || !is_object( $theme->parent() ) ) return false; return $this->get_preset( $theme->parent()->stylesheet ); } //pre 3.4 back compat.. //get theme by slug $name = $this->get_name( $theme ); $themes = $this->getThemes(array()); $child = $themes[ $name ]; //not a child theme if ( !isset( $child['Template'] ) || empty( $child['Template'] ) || $child['Template'] == $child['Stylesheet'] ) return false; //pull up parent data to get its name $parent = $themes[$name]['Template']; $parent = get_theme_data( get_theme_root( $child['Template'] ) . '/' . $child['Template'] . '/style.css' ); $preset = $this->get_preset( $parent['Stylesheet'] ); //no parent preset if ( !$preset ) return false; //rename the theme of the parent preset object for consistent return $preset->theme = $theme; $preset->parentPreset = $parent['Stylesheet']; return $preset; } /** * On plugin activation register with WP_Cron API to asynchronously refresh cache every 24 hours * This will also asynchronously prime the cache on activation */ function schedule() { wp_schedule_event( time(), 'daily', $this->parent->prefix . 'refresh_cache' ); } /** * Clear chron schedule on deactivation */ function unschedule() { wp_clear_scheduled_hook( $this->parent->prefix . 'refresh_cache' ); } /** * Conditionally prompts users on options page to use the default selectors * @uses get_preset */ function preset_prompt() { $preset = $this->get_preset( ); if ( !$preset ) return; unset( $preset->theme ); unset( $preset->parentPreset ); //if they are already using the preset, don't prompt $using_default = true; foreach ( $preset as $key => $value ) { if ( $this->parent->options->$key != $value ) $using_default = false; } if ( $using_default ) return; require dirname( $this->parent->file ) . '/templates/preset-prompt.php'; } /** * Reset selectors to default */ function set_presets() { if ( !isset( $_GET['set_presets'] ) ) return; if ( !current_user_can( 'manage_options' ) ) return; check_admin_referer( 'infinite-scroll-presets', 'nonce' ); //don't delete options if we don't have a preset $preset = $this->get_preset( ); if ( !$preset ) return; foreach ( $this->keys as $key ) $this->parent->options->$key = null; wp_redirect( admin_url( 'options-general.php?page=infinite_scroll_options&settings-updated=true' ) ); exit(); } /** * Handles AJAX edits from the manage presets form */ function process_ajax_edit() { if ( !current_user_can( 'manage_options' ) ) wp_die( -1 ); if ( is_multisite() && !is_super_admin() ) wp_die( -1 ); $data = new stdClass; foreach ( $this->keys as $key ) $data->$key = addslashes( trim( $_POST[ $key . '_column-' . $key ] ) ); $this->set_custom_preset( $data->theme, $data ); wp_die( 1 ); } /** * Handles AJAX requests to delete presets from the manage presets form */ function process_ajax_delete() { if ( !current_user_can( 'manage_options' ) ) wp_die( -1 ); if ( is_multisite() && !is_super_admin() ) wp_die( -1 ); if ( !isset( $_GET['theme'] ) ) wp_die( -1 ); $this->delete_custom_preset( $_GET['theme'] ); } /** * Retreive global custom presets * @return array the custom preset array */ function get_custom_presets( ) { $presets = get_site_option( $this->custom_preset_key, array(), true ); return apply_filters( $this->parent->prefix . 'custom_presets', $presets ); } /** * Update global custom presets * @param array $presets the presets (all) * @return bool success/fail */ function set_custom_presets( $presets ) { return update_site_option( $this->custom_preset_key, $presets ); } /** * Store a theme's global presets * @param string $theme the theme name * @param array $preset the presets * @return bool success/fail */ function set_custom_preset( $theme, $preset ) { $presets = $this->get_custom_presets(); $presets[ $theme ] = $preset; return $this->set_custom_presets( $presets ); } /** * Removes a custom preset from the database * @param string $theme the theme to remove * @return bool success/fail */ function delete_custom_preset( $theme ) { $presets = $this->get_custom_presets(); unset( $presets[ $theme ] ); return $this->set_custom_presets( $presets ); } /** * Allow custom presets to merge/override community presets * @param unknown $presets * @return unknown */ function merge_custom_presets( $presets ) { // 2nd array overrides keys that overlap with first array $presets = array_merge( $presets, $this->get_custom_presets() ); //sort by key alpha ascending asort( $presets ); return $presets; } /** * If a selector is not set, try to grab a preset to save the user trouble * @param array $options the options array * @return array the defaulted options array */ function default_to_presets( $options ) { //we don't have a preset, no need to go any further if ( !( $preset = $this->get_preset( ) ) ) return $options; foreach ( $this->keys as $key ) { if ( empty( $options[$key] ) ) $options[$key] = $preset->$key; } return $options; } /** * Converts legacy csv.php format * Removes first two lines and last line * @param string $data the contents of the CSV (usually via wp_remote_get) * @param string the equivalent standard CSV */ function parse_legacy_csv( $data ) { if ( is_string( $data ) ) $data = explode( "\n", $data ); //remove first two lines $data = array_slice( $data, 2 ); //remove the last line array_pop( $data ); $presets = $this->parse_csv( $data ); $output = array(); //convert Theme Name to stylesheet and stuff into output array foreach( $presets as $theme ) { $theme->theme = $this->get_stylesheet( $theme->theme ); $output[ $theme->theme ] = $theme; } return $output; } /** * Parse CSV into array of preset objects * @param string|array the CSV data, either as a string or as an array of lines * @return array array of preset objects */ function parse_csv( $data ) { if ( is_string( $data ) ) $data = explode( "\n", $data ); //php 5.3+ if ( function_exists( 'str_getcsv' ) ) { foreach ( $data as &$line ) $line = str_getcsv( $line ); //php 5.2 // fgetcsv needs a file handle, // so write the string to a temp file before parsing } else { $fh = tmpfile(); fwrite( $fh, implode( "\n", $data ) ); fseek( $fh, 0 ); $data = array(); while( $line = fgetcsv( $fh ) ) $data[] = $line; fclose( $fh ); } $presets = array(); //build preset objects and stuff into keyed array foreach ( $data as &$line ) { $lineObj = new stdClass; foreach ( $this->keys as $id => $key ) $lineObj->$key = $line[ $id ]; $presets[ $lineObj->theme ] = $lineObj; } return $presets; } /** * Return object representing current theme's selectors * @return object the same as would be returned from get_preset() */ function current_selectors() { $theme = new stdClass(); foreach ( $this->keys as $key ) $theme->$key = $this->parent->options->$key; $theme->theme = get_stylesheet(); return $theme; } /** * Export CSS Selectors as CSV * @param bool $all (optional) whether to include community selectors in output * @return string CSV of selectors */ function export( $all = false ) { $presets = array(); //if the current theme is not a known preset or they want all if ( !$this->get_preset( ) || $all ) $presets[ get_stylesheet() ] = $this->current_selectors(); //user has access to global custom presets if ( is_multisite() && is_super_admin() ) { if ( $custom = $this->get_custom_presets() ); $presets = array_merge( $presets, $custom ); } //include community presets, if asked if ( $all ) $presets = array_merge( $this->get_presets(), $presets ); asort( $presets ); //workaround because fputcsv needs a file handle by default $fh = tmpfile(); $length = 0; foreach ( $presets as &$preset ) $length += fputcsv( $fh, (array) $preset ); if ( $length == 0 ) return false; fseek( $fh, 0 ); $csv = fread( $fh, $length ); fclose( $fh ); return $csv; } /** * Migrates legacy csv.php files to 2.5's custom presets format * @uses parse_legacy_csv */ function migrate() { //no preset file to migrate if ( !file_exists( dirname( $this->parent->file ) . '/PresetDB.csv.php' ) ) return; $data = file_get_contents( dirname( $this->parent->file ) . '/PresetDB.csv.php' ); $presets = $this->parse_legacy_csv( $data ); //this wiill override any existing presets, // but is okay as is only being fired when no presets exist $this->set_custom_presets( $presets ); return $presets; } /** * Determines whether a given theme is installed * @param string|object $theme either the theme slug or the preset object * @return bool true if insalled, otherwise false */ function theme_installed( $theme ) { // get theme name if $theme is an preset object if ( is_object( $theme ) ) { $theme = $theme->theme; } //3.4+ if ( function_exists( 'wp_get_theme' ) ) { return wp_get_theme( $theme )->exists(); } else { //pre 3.4 $themes = get_themes(); $name = $this->get_name( $theme ); return array_key_exists( $name, $themes ); } } /** * Given a theme name, returns the coresponding theme stylesheet * * Used for converting legacy CSVs which were name based to new CSVs which are stylesheet based * since 3.4 returns themes keyed to stylesheets, not names as it did pre-3.4 * * @param string $theme the theme name * @return string the stylesheet */ function get_stylesheet( $name ) { //pre 3.4 if ( !function_exists( 'wp_get_themes' ) ) { if ( $theme = get_theme( $name ) ) return $theme->stylesheet; //3.4+ } else { //we can't use wp_filter_list_object with WP_Theme objects, so filter manually foreach ( $this->getThemes(null) as $theme ) if ( $theme->name = $name ) return $theme->stylesheet; } return false; } /** * Given a theme stylesheet, return the coresponding theme name * * Used to normalize data between 3.3 and 3.4 where keying of themes switched from name to stylesheet * * @param string $stylesheet the theme stylesheet (slug) * @return string the theme name */ function get_name( $stylesheet ) { //3.4+ if ( function_exists( 'wp_get_theme' ) ) { if ( $theme = wp_get_theme( $stylesheet ) ) return $theme->name; //pre 3.4 } else { foreach ( get_themes() as $theme ) if ( $theme->stylesheet == $stylesheet ) return $theme->name; } //theme isn't installed, use the WP.org API to grab the name rather than risk losing data on upgrade $api = themes_api( 'theme_information', array( 'slug' => $stylesheet, 'fields' => array( 'sections' => false, 'tags' => false ) ) ); if ( is_wp_error( $api ) ) return false; return $api->name; } } if (!class_exists('WP_List_Table')) { require_once ABSPATH . 'wp-admin/includes/class-wp-list-table.php'; } /** * List table for manage custom presets page */ class Infinite_Scroll_Presets_Table extends WP_List_Table { /** * Register with Parent */ function __construct( $args = array() ) { parent::__construct( array( 'singular' => 'preset', 'plural' => 'presets', 'ajax' => true, ) ); } /** * Default column callback * @param object $item the item to display * @param string $column_name the column name * @return string the HTML to display */ function column_default( $item, $column_name ) { return $item->$column_name; } /** * Callback to display the theme column * @param object $item the preset object * @return string the HTML to display */ function column_theme( $item ) { global $infinite_scroll; $s = '<strong><a href="#" class="theme-name">' . $item->theme . '</a></strong>'; $s .= '<div class="edit edit-link" style="visibility:hidden;"><a href="#">' . __( 'Edit', 'infinite-scroll' ) . '</a> | <span class="delete"><a href="#">' . __( 'Delete', 'infinite-scroll' ) . '</a></span></div>'; $s .= '<div class="save save-link" style="display:none; padding-top:5px;"><a href="#" class="button-primary">' . __( 'Save', 'infinite-scroll' ) . '</a> <a href="#" class="cancel">' . __( 'Cancel', 'infinite-scroll' ) . '</a> <img class="loader" style="display:none;" src="'. admin_url( '/images/loading.gif' ) .'" /></div>'; return $s; } /** * Callaback to return list of columns to display * @return array the columns to display */ function get_columns() { return array( 'theme' => 'Theme', 'contentSelector' => 'Content Selector', 'navSelector' => 'Navigation Selector', 'nextSelector' => 'Next Selector', 'itemSelector' => 'Item Selector', ); } /** * Grab data and filter prior to passing to table class */ function prepare_items() { global $infinite_scroll; $per_page = 25; $columns = $this->get_columns(); $hidden = array(); $sortable = array(); $this->_column_headers = array($columns, $hidden, $sortable); $data = $infinite_scroll->presets->get_presets(); //only display installed themes $data = array_filter( $data, array( &$infinite_scroll->presets, 'theme_installed' ) ); //merge in themes $themes = $infinite_scroll->presets->getThemes(null); foreach ( $themes as $theme => $theme_data ) { if ( array_key_exists( $theme, $data) ) continue; //check for parent theme's preset, if any if ( $preset = $infinite_scroll->presets->get_preset( $theme ) ) { $data[ $theme ] = $preset; continue; } $themeObj = new stdClass; foreach ( $infinite_scroll->presets->keys as $key ) $themeObj->$key = null; $themeObj->theme = $theme; $data[ $theme ] = $themeObj; } asort( $data ); $current_page = $this->get_pagenum(); $total_items = count($data); $data = array_slice($data, (($current_page-1)*$per_page), $per_page); $this->items = $data; $this->set_pagination_args( array( 'total_items' => $total_items, //WE have to calculate the total number of items 'per_page' => $per_page, //WE have to determine how many items to show on a page 'total_pages' => ceil($total_items/$per_page) //WE have to calculate the total number of pages ) ); } }