Skip to content

Instantly share code, notes, and snippets.

@jonathonbyrdziak
Last active September 29, 2022 22:00
Show Gist options
  • Star 74 You must be signed in to star a gist
  • Fork 42 You must be signed in to fork a gist
  • Save jonathonbyrdziak/1229641 to your computer and use it in GitHub Desktop.
Save jonathonbyrdziak/1229641 to your computer and use it in GitHub Desktop.
EMPTY WIDGET Plugin code to create a single widget in wordpress.
<?php
/**
* Duplicate this file as many times as you would like, just be sure to change the
* Empty_Widget class name to a custom name of your choice. Have fun! redrokk.com
*
* Plugin Name: Empty Widget
* Description: Single Widget Class handles all of the widget responsibility, all that you need to do is create the html. Just use Find/Replace on the Contact_RedRokk_Widget keyword to rebrand this class for your needs.
* Author: RedRokk Interactive Media
* Version: 1.0.0
* Author URI: http://www.redrokk.com
*/
/**
* Actions and Filters
*
* Register any and all actions here. Nothing should actually be called
* directly, the entire system will be based on these actions and hooks.
*/
add_action( 'widgets_init', create_function( '', 'register_widget("Empty_Widget");' ) );
/**
* This is the class that you'll be working with. Duplicate this class as many times as you want. Make sure
* to include an add_action call to each class, like the line above.
*
* @author byrd
*
*/
class Empty_Widget extends Empty_Widget_Abstract
{
/**
* Widget settings
*
* Simply use the following field examples to create the WordPress Widget options that
* will display to administrators. These options can then be found in the $params
* variable within the widget method.
*
*
*/
protected $widget = array(
// you can give it a name here, otherwise it will default
// to the classes name. BTW, you should change the class
// name each time you create a new widget. Just use find
// and replace!
'name' => false,
// this description will display within the administrative widgets area
// when a user is deciding which widget to use.
'description' => 'Single Widget Class handles all of the widget responsibility, all that you need to do is create the html and change the description. RedRokk',
// determines whether or not to use the sidebar _before and _after html
'do_wrapper' => true,
// determines whether or not to display the widgets title on the frontend
'do_title' => false,
// string : if you set a filename here, it will be loaded as the view
// when using a file the following array will be given to the file :
// array('widget'=>array(),'params'=>array(),'sidebar'=>array(),
// alternatively, you can return an html string here that will be used
'view' => false,
// If you desire to change the size of the widget administrative options
// area
'width' => 350,
'height' => 350,
// Shortcode button row
'buttonrow' => 4,
// The image to use as a representation of your widget.
// Whatever you place here will be used as the img src
// so we have opted to use a basencoded image.
'thumbnail' => '',
/* The field options that you have available to you. Please
* contribute additional field options if you create any.
*
*/
'fields' => array(
// You should always offer a widget title
array(
'name' => 'Title',
'desc' => '',
'id' => 'title',
'type' => 'text',
'default' => 'Widget Title'
),
array(
'name' => 'Textarea',
'desc' => 'Enter big text here',
'id' => 'textarea_id',
'type' => 'textarea',
'default' => 'Default value 2'
),
array(
'name' => 'Radio',
'desc' => '',
'id' => 'radio_id',
'type' => 'radio',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Checkbox',
'desc' => '',
'id' => 'checkbox_id',
'type' => 'checkbox',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Select box',
'desc' => '',
'id' => 'select_id',
'type' => 'select',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Menus',
'desc' => '',
'id' => 'menu',
'type' => 'select_menu',
),
array(
'name' => 'Users',
'desc' => '',
'id' => 'Users',
'type' => 'select_users',
),
array(
'name' => 'Capabilities',
'desc' => '',
'id' => 'Capabilities',
'type' => 'select_capabilities',
),
array(
'name' => 'Roles',
'desc' => '',
'id' => 'Roles',
'type' => 'select_roles',
),
array(
'name' => 'Categories',
'desc' => '',
'id' => 'Categories',
'type' => 'select_categories',
),
)
);
/**
* Widget HTML
*
* If you want to have an all inclusive single widget file, you can do so by
* dumping your css styles with base_encoded images along with all of your
* html string, right into this method.
*
* @param array $widget
* @param array $params
* @param array $sidebar
*/
function html($widget = array(), $params = array(), $sidebar = array())
{
?>
<!-- Your widget html goes here -->
<?php //print_r($params);?>
<?php
}
}
<?php
/**
* @package WordPress
* @subpackage Empty Widget Class
* @version 0.5
*/
/**
* Protection
*
* This string of code will prevent hacks from accessing the file directly.
*/
defined('ABSPATH') or die("Cannot access pages directly.");
/**
* Initializing
*
* The directory separator is different between linux and microsoft servers.
* Thankfully php sets the DIRECTORY_SEPARATOR constant so that we know what
* to use.
*/
defined("DS") or define("DS", DIRECTORY_SEPARATOR);
/**
* This abstract class is not intended to be called directly. This class is a worker class that must
* be extended by another class. The class that you will be working with is called Empty_Widget
* and exists at the bottom of this file.
*
*/
if (!class_exists('Empty_Widget_Abstract')):
abstract class Empty_Widget_Abstract extends WP_Widget
{
/**
* Widget settings
*
* Simply use the following field examples to create the WordPress Widget options that
* will display to administrators. These options can then be found in the $params
* variable within the widget method.
*
*
*/
protected $widget = array(
'name' => false,
'description' => 'Single Widget Class handles all of the widget responsibility, all that you need to do is create the html and change the description. RedRokk',
'do_wrapper' => true,
'do_title' => false,
'view' => false,
'width' => 350,
'height' => 350,
'buttonrow' => 4,
'thumbnail' => '',
'fields' => array(
array(
'name' => 'Title',
'desc' => '',
'id' => 'title',
'type' => 'text',
'default' => 'Widget Title'
),
array(
'name' => 'Textarea',
'desc' => 'Enter big text here',
'id' => 'textarea_id',
'type' => 'textarea',
'default' => 'Default value 2'
),
array(
'name' => 'Radio',
'desc' => '',
'id' => 'radio_id',
'type' => 'radio',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Checkbox',
'desc' => '',
'id' => 'checkbox_id',
'type' => 'checkbox',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Select box',
'desc' => '',
'id' => 'select_id',
'type' => 'select',
'options' => array(
'KEY1' => 'Value 1',
'KEY2' => 'Value 2',
'KEY3' => 'Value 3'
)
),
array(
'name' => 'Menus',
'desc' => '',
'id' => 'menu',
'type' => 'select_menu',
),
array(
'name' => 'Users',
'desc' => '',
'id' => 'Users',
'type' => 'select_users',
),
array(
'name' => 'Capabilities',
'desc' => '',
'id' => 'Capabilities',
'type' => 'select_capabilities',
),
array(
'name' => 'Roles',
'desc' => '',
'id' => 'Roles',
'type' => 'select_roles',
),
array(
'name' => 'Categories',
'desc' => '',
'id' => 'Categories',
'type' => 'select_categories',
),
array(
'name' => 'Posts',
'desc' => '',
'id' => 'posts',
'type' => 'select_posts',
'args' => array(
'post_type' => 'posts'
)
),
)
);
/**
* Widget HTML
*
* If you want to have an all inclusive single widget file, you can do so by
* dumping your css styles with base_encoded images along with all of your
* html string, right into this method.
*
* @param array $widget
* @param array $params
* @param array $sidebar
*/
abstract function html($widget = array(), $params = array(), $sidebar = array());
/* *****************************************
* NO NEED TO MODIFY ANYTHING BELOW THIS POINT
*/
/**
* Constructor
*
* Registers the widget details with the parent class, based off of the options
* that were defined within the widget property. This method does not need to be
* changed.
*/
function __construct()
{
//Initializing
$this->_file = basename(__file__);
$this->_class = $this->get_called_class();
$this->widget['name'] = isset($this->widget['name']) && $this->widget['name']
? $this->widget['name']
: ucwords(str_replace('_',' ',sanitize_title(get_class($this))));
// widget actual processes
parent::WP_Widget(
$id = $this->_class,
$name = $this->widget['name'],
$widget_options = array(
'description' => $this->widget['description']
),
$control_options = array(
'width' => isset($this->widget['width']) && $this->widget['width']?$this->widget['width']:250,
'height' => isset($this->widget['height']) && $this->widget['height']?$this->widget['height']:200,
)
);
// expanding the potential
add_action( 'init', array(&$this, 'set_wysiwyg_button'));
add_action( "after_plugin_row_$this->_file", array(&$this, 'get_plugin_support'));
add_filter( "{$this->_class}_params", array(&$this, 'set_defaults') );
add_filter( "plugin_action_links_$this->_file", array(&$this, 'set_plugin_links'));
add_shortcode( $this->_class, array(&$this, 'shortcode') );
}
/**
* Method returns the called class
*
*/
function get_called_class()
{
if (function_exists('get_called_class')) {
return get_called_class();
}
$called_class = false;
$objects = array();
$traces = debug_backtrace();
foreach ($traces as $trace)
{
if (isset($trace['object'])) {
if (is_object($trace['object'])) {
$objects[] = $trace['object'];
}
}
}
if (count($objects)) {
$called_class = get_class($objects[0]);
}
return $called_class;
}
/**
* Widget View
*
* This method determines what view method is being used and gives that view
* method the proper parameters to operate. This method does not need to be
* changed.
*
* @param array $sidebar
* @param array $params
*/
function widget($sidebar = array(), $params = array())
{
//initializing variables
$sidebar = wp_parse_args($sidebar, array(
'before_widget' => '',
'before_title' => '',
'after_title' => '',
'after_widget' => '',
));
$this->widget['number'] = $this->number;
$params = apply_filters( "{$this->_class}_params", $params, $this );
$sidebar = apply_filters( "{$this->_class}_sidebar", $sidebar, $this );
$params['title'] = isset($params['title']) && $params['title']
? $params['title']
: '';
$params['title'] = trim(apply_filters( "{$this->_class}_title", $params['title'] ));
$do_wrapper = (isset($this->widget['do_wrapper']) && $this->widget['do_wrapper']);
$do_title = (isset($this->widget['do_title']) && $this->widget['do_title']);
if ( $do_wrapper )
echo $sidebar['before_widget'];
if ( $do_title && $params['title'] )
echo $sidebar['before_title'] . $params['title'] . $sidebar['after_title'];
//loading a file that is isolated from other variables
if (file_exists($this->widget['view']))
$this->getViewFile($widget, $params, $sidebar);
elseif ($this->widget['view'])
echo $this->widget['view'];
else $this->html($this->widget, $params, $sidebar);
if ( $do_wrapper )
echo $sidebar['after_widget'];
}
/**
* Compress the content and return it
*
* @param $buffer
*/
function compress($buffer)
{
/* remove comments */
//$buffer = preg_replace('!/\*[^*]*\*+([^/][^*]*\*+)*/!', '', $buffer);
$buffer = preg_replace("/((?:\/\*(?:[^*]|(?:\*+[^*\/]))*\*+\/)|(?:\/\/.*))/", "", $buffer);
/* remove tabs, spaces, newlines, etc. */
$buffer = str_replace(array("\r\n", "\r", "\n", "\t", ' ', ' ', ' '), '', $buffer);
return $buffer;
}
/**
*
* @param array $atts
* @param string $content
*/
function shortcode( $atts = array(), $content = '' )
{
$fields = array();
foreach((array)$this->widget['fields'] as $field)
{
$meta = $atts[$field['id']];
$fields[$field['id']] = $meta ?$meta :$field['default'];
}
ob_start();
$this->widget(array(), $fields);
$html = ob_get_clean();
return $this->compress($html);
}
/**
* Administration Form
*
* This method is called from within the wp-admin/widgets area when this
* widget is placed into a sidebar. The resulting is a widget options form
* that allows the administration to modify how the widget operates.
*
* You do not need to adjust this method what-so-ever, it will parse the array
* parameters given to it from the protected widget property of this class.
*
* @param array $instance
* @param boolean $shortcode
* @return boolean
*/
function form( $instance = array(), $shortcode = false )
{
// initializing
$fields = $this->widget['fields'];
$defaults = array(
'name' => '',
'desc' => '',
'id' => '',
'type' => 'text',
'options' => array(),
'default' => '',
'value' => '',
'class' => '',
'multiple' => '',
'args' => array(
'hide_empty' => 0,
'name' => 'element_name',
'hierarchical' => true
),
);
// reasons to fail
if (!empty($fields))
{
do_action("{$this->_class}_before");
foreach ($fields as $field)
{
// initializing the individual field
$field = wp_parse_args($field, $defaults);
$field['args'] = wp_parse_args($field['args'], $defaults['args']);
extract($field);
$field['args']['name'] = $element_name = $id;
// grabbing the meta value
if (array_key_exists($id, $instance))
@$meta = esc_attr($instance[$id]);
else
$meta = $default;
if (!$shortcode)
{
$field['args']['name'] = $element_name = $this->get_field_name($id);
$id = $this->get_field_id($id);
}
switch ($type) : default: ?>
<?php case 'text': ?>
<p>
<label for="<?php echo $id; ?>">
<?php echo $name; ?> :
</label>
<input id="<?php echo $id; ?>" name="<?php echo $element_name; ?>"
value="<?php echo $meta; ?>" type="<?php echo $type; ?>"
class="text large-text <?php echo $class; ?>" />
<br/>
<span class="description"><?php echo $desc; ?></span>
</p>
<?php break; ?>
<?php case 'textarea': ?>
<p>
<label for="<?php echo $id; ?>">
<?php echo $name; ?> :
</label>
<textarea cols="60" rows="4" style="width:97%"
id="<?php echo $id; ?>" name="<?php echo $element_name; ?>"
class="large-text <?php echo $class; ?>"
><?php echo trim($meta); ?></textarea>
<br/>
<span class="description"><?php echo $desc; ?></span>
</p>
<?php break; ?>
<?php case 'select_capabilities': ?>
<?php $options = $type=='select_capabilities' ?$this->get_options_capabilities() :$options; ?>
<?php case 'select_roles': ?>
<?php $options = $type=='select_roles' ?$this->get_options_roles() :$options; ?>
<?php case 'select_menu': ?>
<?php $options = $type=='select_menu' ?$this->get_options_menus() :$options; ?>
<?php case 'select_posts': ?>
<?php $options = $type=='select_posts' ?$this->get_options_posts( $args ) :$options; ?>
<?php case 'select_users': ?>
<?php $options = $type=='select_users' ?$this->get_options_users() :$options; ?>
<?php case 'select_categories': ?>
<?php case 'select': ?>
<p>
<label for="<?php echo $id; ?>">
<?php echo $name; ?> :
</label>
<?php if ($type == 'select_categories'): ?>
<?php wp_dropdown_categories( array( 'id' => $id, 'name' => $element_name, 'class' => $class, 'selected' => $meta )); ?>
<?php else: ?>
<select <?php echo $multiple ?"MULTIPLE SIZE='$multiple'" :''; ?>
id="<?php echo $id; ?>" name="<?php echo $element_name; ?>"
class="<?php echo $class; ?>">
<?php foreach ((array)$options as $_value => $_name): ?>
<?php $_value = !is_int($_value)?$_value:$_name; ?>
<option
value="<?php echo $_value; ?>"
<?php echo $meta == $_value?' selected="selected"' :''; ?>
><?php echo $_name; ?>
</option>
<?php endforeach; ?>
</select>
<?php endif; ?>
<br/>
<span class="description"><?php echo $desc; ?></span>
</p>
<?php break; ?>
<?php case 'radio': ?>
<p>
<?php echo $name; ?> :
</p>
<p>
<?php foreach ((array)$options as $_value => $_name): ?>
<input name="<?php echo $element_name; ?>" id="<?php echo $element_name . $_value; ?>"
value="<?php echo $_value; ?>" type="<?php echo $type; ?>"
<?php echo $meta == $_value?'checked="checked"' :''; ?>
class="<?php echo $class; ?>" />
<label class="<?php echo $element_name; ?>" for="<?php echo $element_name . $_value; ?>">
<?php echo $_name; ?>
</label>
<?php endforeach; ?>
<br/>
<span class="description"><?php echo $desc; ?></span>
</p>
<?php break; ?>
<?php case 'checkbox': ?>
<p>
<?php echo $name; ?> :
</p>
<p>
<!-- first hidden input forces this item to be submitted
via javascript, when it is not checked -->
<input type="hidden" name="<?php echo $element_name; ?>" value="" />
<?php foreach ((array)$options as $_value => $_name): ?>
<input value="<?php echo $_value; ?>" type="<?php echo $type; ?>"
name="<?php echo $element_name; ?>" id="<?php echo $element_name . $_value; ?>"
<?php echo $meta == $_value? 'checked="checked"' :''; ?>
class="<?php echo $class; ?>" />
<label class="<?php echo $element_name; ?>" for="<?php echo $element_name . $_value; ?>">
<?php echo $_name; ?>
</label>
<?php endforeach; ?>
<br/>
<span class="description"><?php echo $desc; ?></span>
</p>
<?php break; ?>
<?php case 'title': ?>
<h3 style="border: 1px solid #ddd;
padding: 10px;
background: #eee;
border-radius: 2px;
color: #666;
margin: 0;"><?php echo $name; ?></h3>
<?php break; ?>
<?php case 'hidden': ?>
<input
id="<?php echo $id; ?>" name="<?php echo $element_name; ?>"
value="<?php echo $meta; ?>" type="<?php echo $type; ?>"
style="visibility:hidden;" />
<?php break; ?>
<?php case 'custom': ?>
<?php echo $default; ?>
<?php break; ?>
<?php endswitch;
}
do_action("{$this->_class}_after");
}
return true;
}
/**
* Returns an options list of menus
*/
function get_options_posts( $args = array() )
{
// initializing
$options = array();
$posts = get_posts($args);
foreach((array)$posts as $post) {
$options[$post->slug] = $post->name;
}
return $options;
}
/**
* Returns an options list of menus
*/
function get_options_menus()
{
// initializing
$options = array();
$menus = get_terms('nav_menu');
foreach($menus as $menu) {
$options[$menu->slug] = $menu->name;
}
return $options;
}
/**
* Returns an options list of users
*/
function get_options_users()
{
// initializing
global $wpdb;
$options = array();
$query = $wpdb->prepare("SELECT $wpdb->users.ID, $wpdb->users.display_name FROM $wpdb->users");
$results = $wpdb->get_results( $query );
foreach ((array)$results as $result)
{
$options[$result->ID] = $result->display_name;
}
return $options;
}
/**
* Returns an options list of capabilities
*/
function get_options_capabilities()
{
// initializing
global $wpdb;
$options = array();
$roles = get_option($wpdb->prefix . 'user_roles');
foreach ((array)$roles as $role)
{
if(!isset($role['capabilities'])) continue;
foreach ((array)$role['capabilities'] as $cap => $v)
{
$options[$role['name']."::$cap"] = $role['name']."::$cap";
}
}
return $options;
}
/**
* Returns an options list of roles
*/
function get_options_roles()
{
// initializing
global $wpdb;
$options = array();
$roles = get_option($wpdb->prefix . 'user_roles');
foreach ((array)$roles as $role)
{
$options[] = $role['name'];
}
return $options;
}
/**
* Method adds a WYSIWYG button to the editor
*
*/
function set_wysiwyg_button()
{
// bail early and bail often
if ((!current_user_can('edit_posts')
&& !current_user_can('edit_pages'))
|| get_user_option('rich_editing') != 'true')
return;
// display the javascript
$this->view_preview();
$this->view_javascript();
$this->view_config();
// hooking it up
add_filter( 'mce_external_plugins', array(&$this, 'set_wysiwyg_js'));
add_filter( "mce_buttons_{$this->widget['buttonrow']}", array(&$this, 'set_wysiwyg_button_callback'));
}
/**
* Method displays the administrative preview for this given widget
*
*/
function view_preview()
{
// fail early, fail often
if (!array_key_exists("{$this->_class}_Preview", $_GET))
return;
// sending the headers
header("Content-type: text/html; charset=UTF-8");
header("Pragma: no-cache");
header("Cache-Control: no-cache");
header('Expires: ' . gmdate('D, d M Y H:i:s', strtotime('-1 years')) . ' GMT');
// the actual PREVIEW code
$this->widget(array(), $_GET);
die();
}
/**
* Method displays the Administrative configurations for this shortcode
*
*/
function view_config()
{
// fail early, fail often
if (!array_key_exists("{$this->_class}_Config", $_GET))
return;
// sending the headers
header("Content-type: text/html; charset=UTF-8");
header("Pragma: no-cache");
header("Cache-Control: no-cache");
header('Expires: ' . gmdate('D, d M Y H:i:s', strtotime('-1 years')) . ' GMT');
// the actual HTML code
?>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<title><?php echo $this->widget['name']; ?></title>
<link rel="stylesheet" href="<?php echo site_url('wp-includes/js/tinymce/themes/advanced/skins/wp_theme/dialog.css'); ?>">
<script type="text/javascript" src="<?php echo site_url('/wp-includes/js/tinymce/tiny_mce_popup.js'); ?>"></script>
<script type="text/javascript" src="<?php echo site_url('/wp-includes/js/tinymce/utils/mctabs.js'); ?>"></script>
</head>
<body onload="tinyMCEPopup.resizeToInnerSize();" role="application" dir="ltr" class="forceColors">
<form onsubmit="tinyMCEPopup.restoreSelection();tinyMCEPopup.editor.<?php echo $this->_class; ?>_Command(this);tinyMCEPopup.close();return false;"
id="popup_form" action="#">
<div class="tabs" role="tablist" tabindex="-1">
<ul>
<li id="general_tab" class="current" aria-controls="general_panel" role="tab" tabindex="0">
<span><a href="javascript:mcTabs.displayTab('general_tab','general_panel');"
onmousedown="return false;" tabindex="-1">General</a></span>
</li>
<li id="preview_tab" aria-controls="preview_panel" role="tab" tabindex="-1">
<span><a href="javascript:tinyMCEPopup.editor.<?php echo $this->_class; ?>_Preview(document.getElementById('popup_form'),document.getElementById('preview_inner_div'));mcTabs.displayTab('preview_tab','preview_panel');"
onmousedown="return false;" tabindex="-1">Preview</a></span>
</li>
<!-- IF YOU WANT TO IMPLEMENT A HELP TAB
<li id="help_tab" aria-controls="help_panel" role="tab" tabindex="-1">
<span><a href="javascript:mcTabs.displayTab('help_tab','help_panel');"
onmousedown="return false;" tabindex="-1">Help</a></span>
</li> -->
</ul>
</div>
<div class="panel_wrapper">
<div id="general_panel" style="height:auto;" class="panel current">
<fieldset>
<legend>Widget Options</legend>
<?php $this->form(array(),true); ?>
</fieldset>
<div style="clear:both;width:100%;"></div>
</div>
<div id="preview_panel" style="height:auto;" class="panel">
<fieldset>
<legend>Widget Preview</legend>
<div id="preview_inner_div"></div>
</fieldset>
<div style="clear:both;width:100%;"></div>
</div>
<div id="help_panel" style="height:auto;" class="panel">
<fieldset>
<legend>Help</legend>
</fieldset>
<div style="clear:both;width:100%;"></div>
</div>
</div>
<div class="mceActionPanel">
<input type="submit" id="insert" name="insert" value="Insert">
<input type="button" id="cancel" name="cancel" value="Cancel" onclick="tinyMCEPopup.close();">
</div>
</form>
</body>
</html>
<?php
die();
}
/**
* Method displays the javascript for the shortcode
*
*/
function view_javascript()
{
// fail early, fail often
if (!array_key_exists("{$this->_class}_Javascript", $_GET))
return;
// sending the headers
$expires = 60*60*24*14;
header("Content-type: text/javascript");
header("Pragma: public");
header("Cache-Control: maxage=".$expires);
header('Expires: ' . gmdate('D, d M Y H:i:s', time()+$expires) . ' GMT');
// the actual JS code
?>
(function() {
tinymce.create('tinymce.plugins.<?php echo $this->_class; ?>', {
mceTout : 0,
init : function(editor, site_url)
{
// Create the WYSIWYG button
editor.addButton('<?php echo $this->_class; ?>', {
title : '<?php echo $this->widget['name']; ?>',
image : '<?php echo $this->widget['thumbnail']; ?>',
onclick : function() {
editor.windowManager.open({
file : site_url + '/?<?php echo $this->_class; ?>_Config=true',
width : (<?php echo $this->widget['width']; ?> + 30) + editor.getLang('example.delta_width', 0),
height : (<?php echo $this->widget['height']; ?> + 35) + editor.getLang('example.delta_height', 0),
inline : 1
}, {
plugin_url : site_url,
some_custom_arg : '' // Custom argument
});
}
});
// The command to add the shortcode
editor.<?php echo $this->_class; ?>_Command = function(form) {
tinyMCEPopup.restoreSelection();
var insert = '[<?php echo $this->_class ?> ';
jQuery.each( jQuery(form).serializeArray(), function(k, f){
insert += f.name + '="'+ f.value +'" ';
});
insert += ']';
editor.execCommand('mceInsertContent', false, insert);
};
// Preview the shortcode/widget
editor.<?php echo $this->_class; ?>_Preview = function(form, wrapper) {
jQuery.get(site_url + '/?<?php echo $this->_class; ?>_Preview=true',
jQuery(form).serializeArray(),
function(data){
jQuery(wrapper).html(data);
}
);
};
},
createControl : function(n, cm) {
return null;
},
getInfo : function() {
return {
longname : "<?php echo $this->widget['name']; ?> Shortcode",
author : 'RedRokk',
authorurl : 'http://redrokk.com',
infourl : 'http://redrokk.com',
version : "1.0.0"
};
}
});
tinymce.PluginManager.add('<?php echo $this->_class; ?>',
tinymce.plugins.<?php echo $this->_class; ?>);
})();
<?php
die();
}
/**
* String to image
*
* @var string
*/
var $screen_icon = "%3D%3D";
/**
* Adds links to the plugins action list
*
* @param array $actions
*/
function set_plugin_links( $actions )
{
// Update the database for show options
$key = "RedRokk_Plugin_Info";
if (array_key_exists($key, $_GET)) {
update_user_option(get_current_user_id(), $key, $_GET[$key], true);
}
// set the link for show options
if (get_user_option($key, get_current_user_id())) {
$actions['info'] = '<a href="'
. site_url("/wp-admin/plugins.php?$key=0")
. '" alt="Display the widget information on this page">Show Info</a>';
}
else {
$actions['info'] = '<a href="'
. site_url("/wp-admin/plugins.php?$key=true")
. '" alt="Hide the widget information on this page">Hide Info</a>';
}
// update the database for show link
$key = "{$this->_class}_Link";
if (array_key_exists($key, $_GET)) {
update_user_option(get_current_user_id(), $key, $_GET[$key], true);
}
// set the link for show link
if (get_user_option($key, get_current_user_id())) {
$actions['publiclink'] = '<a href="'
. site_url("/wp-admin/plugins.php?$key=0")
. '" alt="Display the link to redrokk below the widget">Show Link</a>';
}
else {
$actions['publiclink'] = '<a href="'
. site_url("/wp-admin/plugins.php?$key=true")
. '" alt="Hide the link to redrokk below the widget">Hide Link</a>';
}
return $actions;
}
/**
* This is a tiny contribution to our cause, I thank you for leaving this intact.
* Please add your own support blurb, in addition to ours as you see fit.
*
* @return string
*/
function get_plugin_support()
{
// initializing
global $redrokk_display_once;
if (!isset($redrokk_display_once) && !get_user_option("RedRokk_Plugin_Info", get_current_user_id()))
{
$redrokk_display_once = true;
$title = "Red Rokk Interactive &trade;";
$descriptionencode = urlencode($description);
$description = "The WordPress Total Widget Control Plugin is an amazing plugin!";
$descriptionencode = urlencode($description);
$url = 'http://redrokk.com';
$urlencode = urlencode($url);
ob_start();
?>
<tr>
<td><hr style="margin: 0 0 10px;border: none;border-bottom: 1px dashed #CCC;width:100%;clear:both"/>
</td>
<td>
<hr style="margin: 0 0 10px;border: none;border-bottom: 1px dashed #CCC;width:100%;clear:both"/>
<a href="<?php echo $url; ?>" style="text-decoration:none;">
<img src="<?php echo $this->screen_icon; ?>"
height="41px" width="194px"
style="margin:15px 20px 30px 0;position:relative;float:left;" />
</a>
</td>
<td>
<hr style="margin: 0 0 10px;border: none;border-bottom: 1px dashed #CCC;width:100%;clear:both"/>
<div style="position:relative;float:right;width:200px;">
<iframe src="http://www.facebook.com/plugins/like.php?href=<?php echo $urlencode ?>&amp;layout=box_count&amp;show_faces=false&amp;width=50&amp;action=like&amp;colorscheme=light&amp;height=65"
scrolling="no"
frameborder="0"
style="border:none; overflow:hidden; width:50px; height:65px;margin-bottom: -5px;"
allowTransparency="true" ></iframe>
<a href="http://twitter.com/share" class="twitter-share-button"
data-url="<?php echo $url ?>"
data-text="<?php echo $description ?>"
data-count="vertical"
data-via="jonathonbyrd">
Tweet</a>
<script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script>
<a href="http://digg.com/submit?url=<?php echo $urlencode ?>&bodytext=Total%20Widget%20Control%20allows%20you%20to%20customize%20any%20and%20every%20page%20of%20your%20WordPress%20website."
class="DiggThisButton DiggMedium">
<img src="http://developers.diggstatic.com/sites/all/themes/about/img/digg-btn.jpg"
alt="Digg <?php echo $title; ?>"
title="Digg <?php echo $title; ?>" />
<?php echo $title; ?></a>
<script type="text/javascript">
(function() {
var s = document.createElement('SCRIPT'), s1 = document.getElementsByTagName('SCRIPT')[0];
s.type = 'text/javascript';
s.async = true;
s.src = 'http://widgets.digg.com/buttons.js';
s1.parentNode.insertBefore(s, s1);
})();
</script>
</div>
<p>
<a href="<?php echo $url; ?>" style="text-decoration:none;">
<h4 style="margin:20px 0 0px;"><?php echo $title; ?></h4>
</a>
<?php echo $title; ?> is a software development firm that accepts
projects of all sizes. We contribute back to the community as much
as possible by providing amazing plugins with exceptional API's for
developers.
</p>
</td>
</tr>
<?php
echo ob_get_clean();
}
}
/**
* Method adds our button to the WYSIWYG array of buttons
*
* @params array $buttons
*/
function set_wysiwyg_button_callback( $buttons )
{
array_push($buttons, $this->_class);
return $buttons;
}
/**
* Method adds the required Javascript to the WYSIWYG
*
* @params array $scripts
*/
function set_wysiwyg_js( $scripts = array() )
{
$scripts[$this->_class] = site_url("/?{$this->_class}_Javascript=true");
return $scripts;
}
/**
* Get the View file
*
* Isolates the view file from the other variables and loads the view file,
* giving it the three parameters that are needed. This method does not
* need to be changed.
*
* @param array $widget
* @param array $params
* @param array $sidebar
*/
function getViewFile($widget, $params, $sidebar) {
require $this->widget['view'];
}
/**
*
* @param $params
*/
function set_defaults( $params = array() )
{
// initializing
$defaults = array();
foreach ((array)$this->widget['fields'] as $key => $field)
{
if (!isset($field['default'])) continue;
$defaults[$field['id']] = $field['default'];
}
$params = wp_parse_args($params, $defaults);
return $params;
}
/**
* Update the Administrative parameters
*
* This function will merge any posted paramters with that of the saved
* parameters. This ensures that the widget options never get lost. This
* method does not need to be changed.
*
* @param array $new_instance
* @param array $old_instance
* @return array
*/
function update($new_instance, $old_instance)
{
// processes widget options to be saved
$instance = wp_parse_args($new_instance, $old_instance);
return $instance;
}
}
endif;
@jonathonbyrdziak
Copy link
Author

@CHANGELOG 5/29/2012 : Adjusted the default value handling with wp-admin configurations

The system was using an old form field creation method. I've since updated my other plugins and now this class to use a new style form field creation method. This new method is quite a bit cleaner and handles the default field values in a much more stable manner. This new method will also allow us to expand on the default form functionality with ease.

@humanshell
Copy link

The new form() method looks far more powerful. I'll update my plugin to use the new method tonight. I'll let you know how it goes.

@jonathonbyrdziak
Copy link
Author

@CHANGELOG 5/29/2012.2 : Added Shortcode WYSIWYG button support

The Empty_Widget class now turns your widget into a shortcode with a supporting WYSIWYG custom button. Upon clicking the button you will see the same widget options available to you for sidebars. You also have the option to view a preview of what the widget will look like when creating the shortcode. Finally, upon hitting submit within the buttons popup, the shortcode string will be created and inserted into your post or page.

@jonathonbyrdziak
Copy link
Author

@CHANGELOG 5/30/2012 : Added additional form fields

The additional fields are select boxes of the default wordpress data, such as: capabilities, categories, roles, users n more!

@kabadabra
Copy link

Hi - this plugin looks epic!

I added the widget, but clicking save doesnt save the values, returns back to default. Using the latest WordPress version. Any ideas?

@jonathonbyrdziak
Copy link
Author

Sorry, I've been making lots of changes. I'm working on it right now and should post an update within the hour.

@kabadabra
Copy link

Ahh hehe thanks man! Awesome work - just post a notice when updated, working on a project as we speak and your widget is SUPER USEFUL and easy to understand and manage!!

@jonathonbyrdziak
Copy link
Author

Alright! Should be good to go bro. I tested it in WP 3.3.2, worked great.

@kabadabra
Copy link

Oh yeah! Works GREAT! Thanks so much man....perfect!!!

@kabadabra
Copy link

Just something weird...

I use the function "dynamic_sidebar( 'Home Pane' )" to get the contents of my Widget's HTML on my Home Page, but it only gets the default values set in the Widget's php file, not the values from WordPress Widget itself...not sure why, any advice?

@jonathonbyrdziak
Copy link
Author

Thanks for bug testin this with me today :) I just had to toggle lines 336 and 337.

 $field['args']['name'] = $element_name = $this->get_field_name($id);

is now above

 $id = $this->get_field_id($id);

@kabadabra
Copy link

Hehe I'm trying :) it's now doing that no saving thing again....

The toggle makes the value show up correctly, but in the widget settings it doesnt return the value just keeps the default. However it does update the fields...

@jonathonbyrdziak
Copy link
Author

Fixed!

            // grabbing the meta value
            if (array_key_exists($id, $instance))
                @$meta = attribute_escape($instance[$id]);
            else
                $meta = $default;

Had to move this above the if (!$shortcode) command

@kabadabra
Copy link

Nice one bro! Working great!! Thank you very much for looking into this today, and the additions you implemented are epic! bows down

@humanshell
Copy link

Johnathan,

The checkbox part of the updated form() method has been working perfectly for me, thanks. I just wanted to point out that according to the Codex, attribute_escape() (line 316 of the abstract class' form method) has been deprecated since 2.8.0. That should probably be changed to esc_attr()?

Thanks again for releasing this excellent code.

@jonathonbyrdziak
Copy link
Author

jonathonbyrdziak commented Jun 28, 2012 via email

@humanshell
Copy link

Jonathan,

Just added a textarea to my plugin and thought you should know that the line break on line 349 of the abstract class' form method was inserting all that whitespace into the textarea when clicking save in the admin. I moved that line up so it doesn't break right in the "echo $meta" output. That fixed the issue. Not sure if anyone else is experiencing this issue, but I thought you'd want to know incase you wanted to update that section.

@jonathonbyrdziak
Copy link
Author

jonathonbyrdziak commented Jul 8, 2012 via email

@humanshell
Copy link

You're welcome, looking forward to the updates. Don't forget about changing attribute_escape() to esc_attr() on line 316. :-)

@mrwweb
Copy link

mrwweb commented Sep 27, 2012

(Someday, I'll figure out github better to submit a real patch. Hopefully this is useful.)

Possible bug: If there are multiple checkboxes/radioes in a single field, clicking any label toggles the first checkbox or radio button because of the id/for attributes.

I fixed that with the following:
` $_name): ?>
<input name="" id=""
value="" type=""

class="" />

<label class="<?php echo $element_name; ?>" for="<?php echo $id; ?>">
    <?php echo $_name; ?>
</label>
`

For some reason, using only <?php echo $_value; ?> as the id/for attribute didn't work. Also, I moved the <br /> tag inside the foreach so each checkbox/radio button gets its own line.

I love this thing. Thanks!

@mrwweb
Copy link

mrwweb commented Sep 27, 2012

Should've previewed that. Sorry. I can't get code to paste. Weird. Anyway...snippet is here: http://pastebin.com/wTxPhvTj

@mrwweb
Copy link

mrwweb commented Sep 27, 2012

(Sorry if this got submitted. I posted this and then it disappeared. Also, someday I'll learn github, but right now my priority is submitting this patch.)

When using multiple checkboxes or radio buttons in a single option, clicking on any label toggles the first box/button. This is because the id/for values of the input/label aren't unique. I changed that, and now they work. I also moved the linebreak into the foreach loop so that each box/button gets its own line.

Code here: http://pastebin.com/wTxPhvTj

@jonathonbyrdziak
Copy link
Author

Thanks buddy, I don't think I ever received an email that you commented, but noticed it now and made the adjustment.

@foxpcteam
Copy link

Hello, when ui try to activate this plugin i get error:
Fatal error: Class 'Empty_Widget_Abstract' not found in /plugins/123/CustomWidgetFile.php on line 28

why i get this error?

thanks for help.

@osirismqz
Copy link

I'm using your code on some widgets and found a little problem using the "select_categories" field type.

On line 422:

<?php wp_dropdown_categories($args); ?>

You forgot to actually pass arguments to wp_dropdown_categories().

This is what I'm using and is working fine:

<?php wp_dropdown_categories( array( 'id' => $id, 'name' => $element_name, 'class' => $class, 'selected' => $meta ) ); ?>

Hope it helps :)

@jonathonbyrdziak
Copy link
Author

Nice @ozzysong I'll get that added

@alessandrotesoro
Copy link

I'm using your class and i've noticed that it doesn't allow widgets to be localized/translated. Anytime i try using __() to localize a string (widget title/fields names/desc/ etc), it returns an unexpected "(" error. Any idea how to fix it?

Update 21st November
This also seem to cause issues with WPML when trying to translate widgets fields/titles they are not being recognized.

@AndPicc
Copy link

AndPicc commented Dec 24, 2014

Hello!
First of all thanx for this code, it's very well made and I'm learning a lot by working with it.
I made a few experiments in the last year, and everything was working fine.

then I've noticed that on versions after wordpress 4.0 the tinymce popup is not working properly.

when you click to submit, it loads the homepage into the popup, and it doesn't close it, nor it adds the actual shortcode into the post content..

I've tried to search around and I found out that actually there have been some changes in tinymce popups, but I've not been able to figure out how to fix your code..

could you have a look at this issue please?

cheers!
and Merry Christmas!
Andrea

@alexanderbiscajin
Copy link

Thanks for an amazing tutorial on how to create widgets. WordPress give a best drag and drop design option through widgets. This components allow users to add design and structure control elements through WordPress themes. Create Widget Plugin

@Adzetko
Copy link

Adzetko commented Nov 20, 2020

Careful! In widget.class.phpn at line 176, it shows

parent::WP_Widget(

which is deprecated. However, the fix is simple, simply replace WP_Widget( by __construct(.

This piece of code was so useful and makes the plugin developement so much streamlined, thanks alot ♥

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment