2009-02-12 00:10:11 +01:00
< ? php
/**
* WordPress Theme Install Administration API
*
* @ package WordPress
* @ subpackage Administration
*/
2009-02-18 22:55:00 +01:00
$themes_allowedtags = array ( 'a' => array ( 'href' => array (), 'title' => array (), 'target' => array ()),
2009-02-27 22:22:49 +01:00
'abbr' => array ( 'title' => array ()), 'acronym' => array ( 'title' => array ()),
'code' => array (), 'pre' => array (), 'em' => array (), 'strong' => array (),
'div' => array (), 'p' => array (), 'ul' => array (), 'ol' => array (), 'li' => array (),
'h1' => array (), 'h2' => array (), 'h3' => array (), 'h4' => array (), 'h5' => array (), 'h6' => array (),
'img' => array ( 'src' => array (), 'class' => array (), 'alt' => array ())
);
$theme_field_defaults = array ( 'description' => true , 'sections' => false , 'tested' => true , 'requires' => true ,
'rating' => true , 'downloaded' => true , 'downloadlink' => true , 'last_updated' => true , 'homepage' => true ,
'tags' => true , 'num_ratings' => true
);
2009-03-18 03:43:45 +01:00
2009-02-18 22:55:00 +01:00
2009-02-12 00:10:11 +01:00
/**
* Retrieve theme installer pages from WordPress Themes API .
*
* It is possible for a theme to override the Themes API result with three
* filters . Assume this is for themes , which can extend on the Theme Info to
* offer more choices . This is very powerful and must be used with care , when
* overridding the filters .
*
* The first filter , 'themes_api_args' , is for the args and gives the action as
* the second parameter . The hook for 'themes_api_args' must ensure that an
* object is returned .
*
* The second filter , 'themes_api' , is the result that would be returned .
*
* @ since 2.8 . 0
*
* @ param string $action
* @ param array | object $args Optional . Arguments to serialize for the Theme Info API .
* @ return mixed
*/
function themes_api ( $action , $args = null ) {
2009-02-27 00:35:38 +01:00
if ( is_array ( $args ) )
2009-02-12 00:10:11 +01:00
$args = ( object ) $args ;
if ( ! isset ( $args -> per_page ) )
$args -> per_page = 24 ;
$args = apply_filters ( 'themes_api_args' , $args , $action ); //NOTE: Ensure that an object is returned via this filter.
$res = apply_filters ( 'themes_api' , false , $action , $args ); //NOTE: Allows a theme to completely override the builtin WordPress.org API.
if ( ! $res ) {
$request = wp_remote_post ( 'http://api.wordpress.org/themes/info/1.0/' , array ( 'body' => array ( 'action' => $action , 'request' => serialize ( $args ))) );
if ( is_wp_error ( $request ) ) {
$res = new WP_Error ( 'themes_api_failed' , __ ( 'An Unexpected HTTP Error occured during the API request.</p> <p><a href="?" onclick="document.location.reload(); return false;">Try again</a>' ), $request -> get_error_message () );
} else {
$res = unserialize ( $request [ 'body' ]);
if ( ! $res )
2009-02-27 00:35:38 +01:00
$res = new WP_Error ( 'themes_api_failed' , __ ( 'An unknown error occured' ), $request [ 'body' ]);
2009-02-12 00:10:11 +01:00
}
}
2009-02-12 19:31:16 +01:00
//var_dump(array($args, $res));
2009-02-12 00:10:11 +01:00
return apply_filters ( 'themes_api_result' , $res , $action , $args );
}
/**
2009-04-08 01:58:35 +02:00
* Retrieve list of WordPress theme features ( aka theme tags )
2009-02-12 00:10:11 +01:00
*
* @ since 2.8 . 0
*
2010-09-24 17:28:28 +02:00
* @ deprecated since 3.1 . 0 Use get_theme_feature_list () instead .
*
2009-02-12 00:10:11 +01:00
* @ return array
*/
2009-04-07 11:44:41 +02:00
function install_themes_feature_list ( ) {
2009-04-07 23:44:23 +02:00
if ( ! $cache = get_transient ( 'wporg_theme_feature_list' ) )
set_transient ( 'wporg_theme_feature_list' , array ( ), 10800 );
2009-02-12 00:10:11 +01:00
2009-04-07 23:44:23 +02:00
if ( $cache )
return $cache ;
2009-02-12 00:10:11 +01:00
2009-04-07 11:44:41 +02:00
$feature_list = themes_api ( 'feature_list' , array ( ) );
if ( is_wp_error ( $feature_list ) )
return $features ;
2009-02-12 00:10:11 +01:00
2009-04-07 23:44:23 +02:00
set_transient ( 'wporg_theme_feature_list' , $feature_list , 10800 );
2009-02-12 00:10:11 +01:00
2009-04-07 11:44:41 +02:00
return $feature_list ;
2009-02-12 00:10:11 +01:00
}
2009-04-07 11:44:41 +02:00
/**
* Display search form for searching themes .
*
* @ since 2.8 . 0
*/
function install_theme_search_form () {
$type = isset ( $_REQUEST [ 'type' ] ) ? stripslashes ( $_REQUEST [ 'type' ] ) : '' ;
$term = isset ( $_REQUEST [ 's' ] ) ? stripslashes ( $_REQUEST [ 's' ] ) : '' ;
2009-02-12 00:10:11 +01:00
?>
2009-03-20 01:14:57 +01:00
< p class = " install-help " >< ? php _e ( 'Search for themes by keyword, author, or tag.' ) ?> </p>
2009-02-12 00:10:11 +01:00
2010-08-11 23:54:51 +02:00
< form id = " search-themes " method = " get " action = " " >
< input type = " hidden " name = " tab " value = " search " />
2009-04-07 11:44:41 +02:00
< select name = " type " id = " typeselector " >
< option value = " term " < ? php selected ( 'term' , $type ) ?> ><?php _e('Term'); ?></option>
< option value = " author " < ? php selected ( 'author' , $type ) ?> ><?php _e('Author'); ?></option>
2010-05-14 23:46:25 +02:00
< option value = " tag " < ? php selected ( 'tag' , $type ) ?> ><?php _ex('Tag', 'Theme Installer'); ?></option>
2009-04-07 11:44:41 +02:00
</ select >
2009-05-05 21:43:53 +02:00
< input type = " text " name = " s " size = " 30 " value = " <?php echo esc_attr( $term ) ?> " />
< input type = " submit " name = " search " value = " <?php esc_attr_e('Search'); ?> " class = " button " />
2009-04-07 11:44:41 +02:00
</ form >
< ? php
2009-02-12 00:10:11 +01:00
}
/**
2009-04-07 11:44:41 +02:00
* Display tags filter for themes .
2009-02-12 00:10:11 +01:00
*
* @ since 2.8 . 0
*/
2009-04-07 11:44:41 +02:00
function install_themes_dashboard () {
install_theme_search_form ();
?>
< h4 >< ? php _e ( 'Feature Filter' ) ?> </h4>
< form method = " post " action = " <?php echo admin_url( 'theme-install.php?tab=search' ); ?> " >
< p class = " install-help " >< ? php _e ( 'Find a theme based on specific features' ) ?> </p>
2009-02-27 00:35:38 +01:00
< ? php
2010-09-24 17:28:28 +02:00
$feature_list = get_theme_feature_list ( );
2009-04-07 11:44:41 +02:00
echo '<div class="feature-filter">' ;
foreach ( ( array ) $feature_list as $feature_name => $features ) {
2009-05-18 17:11:07 +02:00
$feature_name = esc_html ( $feature_name );
2009-04-07 19:02:25 +02:00
echo '<div class="feature-name">' . $feature_name . '</div>' ;
2009-04-07 11:44:41 +02:00
echo '<ol style="float: left; width: 725px;" class="feature-group">' ;
2010-09-24 17:28:28 +02:00
foreach ( $features as $feature => $feature_name ) {
2009-05-18 17:11:07 +02:00
$feature_name = esc_html ( $feature_name );
2009-05-05 21:43:53 +02:00
$feature = esc_attr ( $feature );
2009-04-07 11:44:41 +02:00
?>
< li >
2009-04-09 15:16:14 +02:00
< input type = " checkbox " name = " features[<?php echo $feature ; ?>] " id = " feature-id-<?php echo $feature ; ?> " value = " <?php echo $feature ; ?> " />
2009-04-07 19:02:25 +02:00
< label for = " feature-id-<?php echo $feature ; ?> " >< ? php echo $feature_name ; ?> </label>
2009-04-07 11:44:41 +02:00
</ li >
< ? php } ?>
</ ol >
< br class = " clear " />
< ? php
} ?>
</ div >
< br class = " clear " />
2009-05-31 03:11:07 +02:00
< p >< input type = " submit " name = " search " value = " <?php esc_attr_e('Find Themes'); ?> " class = " button " /></ p >
2009-04-07 11:44:41 +02:00
</ form >
< ? php
2009-02-12 00:10:11 +01:00
}
2010-08-11 23:54:51 +02:00
add_action ( 'install_themes_dashboard' , 'install_themes_dashboard' );
2009-02-12 00:10:11 +01:00
2009-02-26 00:43:30 +01:00
function install_themes_upload ( $page = 1 ) {
2009-04-09 15:16:14 +02:00
?>
2009-02-27 00:35:38 +01:00
< h4 >< ? php _e ( 'Install a theme in .zip format' ) ?> </h4>
2009-03-20 01:14:57 +01:00
< p class = " install-help " >< ? php _e ( 'If you have a theme in a .zip format, you may install it by uploading it here.' ) ?> </p>
2009-04-19 21:36:28 +02:00
< form method = " post " enctype = " multipart/form-data " action = " <?php echo admin_url('update.php?action=upload-theme') ?> " >
2009-04-09 15:16:14 +02:00
< ? php wp_nonce_field ( 'theme-upload' ) ?>
< input type = " file " name = " themezip " />
2009-04-19 21:36:28 +02:00
< input type = " submit "
2009-05-05 21:43:53 +02:00
class = " button " value = " <?php esc_attr_e('Install Now') ?> " />
2009-04-09 15:16:14 +02:00
</ form >
2009-04-19 21:36:28 +02:00
< ? php
2009-02-27 00:35:38 +01:00
}
2010-08-11 23:54:51 +02:00
add_action ( 'install_themes_upload' , 'install_themes_upload' , 10 , 1 );
2009-02-27 00:35:38 +01:00
function display_theme ( $theme , $actions = null , $show_details = true ) {
global $themes_allowedtags ;
2009-02-27 22:24:19 +01:00
if ( empty ( $theme ) )
return ;
2009-02-27 00:35:38 +01:00
$name = wp_kses ( $theme -> name , $themes_allowedtags );
$desc = wp_kses ( $theme -> description , $themes_allowedtags );
//if ( strlen($desc) > 30 )
2009-04-09 15:16:14 +02:00
// $desc = substr($desc, 0, 15) . '<span class="dots">...</span><span>' . substr($desc, -15) . '</span>';
2009-02-27 00:35:38 +01:00
$preview_link = $theme -> preview_url . '?TB_iframe=true&width=600&height=400' ;
2009-03-20 01:14:57 +01:00
if ( ! is_array ( $actions ) ) {
2009-02-27 00:35:38 +01:00
$actions = array ();
$actions [] = '<a href="' . admin_url ( 'theme-install.php?tab=theme-information&theme=' . $theme -> slug .
2010-05-23 14:04:24 +02:00
'&TB_iframe=true&tbWidth=500&tbHeight=385' ) . '" class="thickbox thickbox-preview onclick" title="' . esc_attr ( sprintf ( __ ( 'Install “%s”' ), $name )) . '">' . __ ( 'Install' ) . '</a>' ;
2009-05-05 21:43:53 +02:00
$actions [] = '<a href="' . $preview_link . '" class="thickbox thickbox-preview onclick previewlink" title="' . esc_attr ( sprintf ( __ ( 'Preview “%s”' ), $name )) . '">' . __ ( 'Preview' ) . '</a>' ;
2009-02-27 00:35:38 +01:00
$actions = apply_filters ( 'theme_install_action_links' , $actions , $theme );
}
$actions = implode ( ' | ' , $actions );
?>
< a class = 'thickbox thickbox-preview screenshot'
2009-05-26 06:38:09 +02:00
href = '<?php echo esc_url($preview_link); ?>'
2009-05-05 21:43:53 +02:00
title = '<?php echo esc_attr(sprintf(__(' Preview & #8220;%s”'), $name)); ?>'>
2009-05-18 18:00:33 +02:00
< img src = '<?php echo esc_url($theme->screenshot_url); ?>' width = '150' />
2009-02-27 00:35:38 +01:00
</ a >
< h3 >< ? php echo $name ?> </h3>
< span class = 'action-links' >< ? php echo $actions ?> </span>
< p >< ? php echo $desc ?> </p>
< ? php if ( $show_details ) { ?>
< a href = " #theme_detail " class = " theme-detail hide-if-no-js " tabindex = '4' >< ? php _e ( 'Details' ) ?> </a>
2009-03-15 12:04:34 +01:00
< div class = " themedetaildiv hide-if-js " >
2009-02-27 00:35:38 +01:00
< p >< strong >< ? php _e ( 'Version:' ) ?> </strong> <?php echo wp_kses($theme->version, $themes_allowedtags) ?></p>
< p >< strong >< ? php _e ( 'Author:' ) ?> </strong> <?php echo wp_kses($theme->author, $themes_allowedtags) ?></p>
2009-03-18 03:43:45 +01:00
< ? php if ( ! empty ( $theme -> last_updated ) ) : ?>
2009-02-27 22:22:49 +01:00
< p >< strong >< ? php _e ( 'Last Updated:' ) ?> </strong> <span title="<?php echo $theme->last_updated ?>"><?php printf( __('%s ago'), human_time_diff(strtotime($theme->last_updated)) ) ?></span></p>
2009-03-18 03:43:45 +01:00
< ? php endif ; if ( ! empty ( $theme -> requires ) ) : ?>
< p >< strong >< ? php _e ( 'Requires WordPress Version:' ) ?> </strong> <?php printf(__('%s or higher'), $theme->requires) ?></p>
< ? php endif ; if ( ! empty ( $theme -> tested ) ) : ?>
< p >< strong >< ? php _e ( 'Compatible up to:' ) ?> </strong> <?php echo $theme->tested ?></p>
2009-02-27 22:22:49 +01:00
< ? php endif ; if ( ! empty ( $theme -> downloaded ) ) : ?>
< p >< strong >< ? php _e ( 'Downloaded:' ) ?> </strong> <?php printf(_n('%s time', '%s times', $theme->downloaded), number_format_i18n($theme->downloaded)) ?></p>
< ? php endif ; ?>
< div class = " star-holder " title = " <?php printf(_n('(based on %s rating)', '(based on %s ratings)', $theme->num_ratings ), number_format_i18n( $theme->num_ratings )) ?> " >
2009-05-05 21:43:53 +02:00
< div class = " star star-rating " style = " width: <?php echo esc_attr( $theme->rating ) ?>px " ></ div >
2009-02-27 22:22:49 +01:00
< div class = " star star5 " >< img src = " <?php echo admin_url('images/star.gif'); ?> " alt = " <?php _e('5 stars') ?> " /></ div >
< div class = " star star4 " >< img src = " <?php echo admin_url('images/star.gif'); ?> " alt = " <?php _e('4 stars') ?> " /></ div >
< div class = " star star3 " >< img src = " <?php echo admin_url('images/star.gif'); ?> " alt = " <?php _e('3 stars') ?> " /></ div >
< div class = " star star2 " >< img src = " <?php echo admin_url('images/star.gif'); ?> " alt = " <?php _e('2 stars') ?> " /></ div >
< div class = " star star1 " >< img src = " <?php echo admin_url('images/star.gif'); ?> " alt = " <?php _e('1 star') ?> " /></ div >
</ div >
2009-02-27 00:35:38 +01:00
</ div >
< ? php }
/*
object ( stdClass )[ 59 ]
public 'name' => string 'Magazine Basic' ( length = 14 )
public 'slug' => string 'magazine-basic' ( length = 14 )
public 'version' => string '1.1' ( length = 3 )
public 'author' => string 'tinkerpriest' ( length = 12 )
public 'preview_url' => string 'http://wp-themes.com/?magazine-basic' ( length = 36 )
public 'screenshot_url' => string 'http://wp-themes.com/wp-content/themes/magazine-basic/screenshot.png' ( length = 68 )
public 'rating' => float 80
public 'num_ratings' => int 1
public 'homepage' => string 'http://wordpress.org/extend/themes/magazine-basic' ( length = 49 )
public 'description' => string 'A basic magazine style layout with a fully customizable layout through a backend interface. Designed by <a href="http://bavotasan.com">c.bavota</a> of <a href="http://tinkerpriestmedia.com">Tinker Priest Media</a>.' ( length = 214 )
public 'download_link' => string 'http://wordpress.org/extend/themes/download/magazine-basic.1.1.zip' ( length = 66 )
*/
2009-02-12 23:12:57 +01:00
}
2009-02-12 00:10:11 +01:00
/**
* Display theme content based on theme list .
*
* @ since 2.8 . 0
*/
2010-08-11 23:54:51 +02:00
function display_themes () {
2010-08-22 13:22:46 +02:00
global $wp_list_table ;
2009-04-19 21:36:28 +02:00
2010-08-22 13:22:46 +02:00
$wp_list_table -> display ();
2009-02-12 00:10:11 +01:00
}
2010-08-11 23:54:51 +02:00
add_action ( 'install_themes_search' , 'display_themes' );
add_action ( 'install_themes_featured' , 'display_themes' );
add_action ( 'install_themes_new' , 'display_themes' );
add_action ( 'install_themes_updated' , 'display_themes' );
2009-02-12 00:10:11 +01:00
/**
* Display theme information in dialog box form .
*
* @ since 2.8 . 0
*/
function install_theme_information () {
//TODO: This function needs a LOT of UI work :)
2009-03-20 01:14:57 +01:00
global $tab , $themes_allowedtags ;
2009-02-12 00:10:11 +01:00
$api = themes_api ( 'theme_information' , array ( 'slug' => stripslashes ( $_REQUEST [ 'theme' ] ) ));
if ( is_wp_error ( $api ) )
wp_die ( $api );
2009-02-27 00:35:38 +01:00
// Sanitize HTML
2009-02-12 00:10:11 +01:00
foreach ( ( array ) $api -> sections as $section_name => $content )
$api -> sections [ $section_name ] = wp_kses ( $content , $themes_allowedtags );
2010-03-19 09:03:52 +01:00
foreach ( array ( 'version' , 'author' , 'requires' , 'tested' , 'homepage' , 'downloaded' , 'slug' ) as $key ) {
if ( isset ( $api -> $key ) )
$api -> $key = wp_kses ( $api -> $key , $themes_allowedtags );
}
2009-02-12 00:10:11 +01:00
iframe_header ( __ ( 'Theme Install' ) );
2009-02-27 00:35:38 +01:00
2009-03-20 01:14:57 +01:00
if ( empty ( $api -> download_link ) ) {
echo '<div id="message" class="error"><p>' . __ ( '<strong>Error:</strong> This theme is currently not available. Please try again later.' ) . '</p></div>' ;
iframe_footer ();
exit ;
}
2009-04-13 18:24:37 +02:00
if ( ! empty ( $api -> tested ) && version_compare ( $GLOBALS [ 'wp_version' ], $api -> tested , '>' ) )
2009-02-27 00:35:38 +01:00
echo '<div class="updated"><p>' . __ ( '<strong>Warning:</strong> This theme has <strong>not been tested</strong> with your current version of WordPress.' ) . '</p></div>' ;
2009-04-13 18:24:37 +02:00
else if ( ! empty ( $api -> requires ) && version_compare ( $GLOBALS [ 'wp_version' ], $api -> requires , '<' ) )
2009-02-27 00:35:38 +01:00
echo '<div class="updated"><p>' . __ ( '<strong>Warning:</strong> This theme has not been marked as <strong>compatible</strong> with your version of WordPress.' ) . '</p></div>' ;
// Default to a "new" theme
$type = 'install' ;
// Check to see if this theme is known to be installed, and has an update awaiting it.
2010-01-08 21:49:55 +01:00
$update_themes = get_site_transient ( 'update_themes' );
2009-04-13 18:24:37 +02:00
if ( is_object ( $update_themes ) && isset ( $update_themes -> response ) ) {
foreach ( ( array ) $update_themes -> response as $theme_slug => $theme_info ) {
if ( $theme_slug === $api -> slug ) {
2009-03-20 01:14:57 +01:00
$type = 'update_available' ;
2009-04-13 18:24:37 +02:00
$update_file = $theme_slug ;
2009-03-20 01:14:57 +01:00
break ;
}
}
2009-02-27 00:35:38 +01:00
}
2009-03-20 01:14:57 +01:00
$themes = get_themes ();
foreach ( $themes as $this_theme ) {
if ( is_array ( $this_theme ) && $this_theme [ 'Stylesheet' ] == $api -> slug ) {
if ( $this_theme [ 'Version' ] == $api -> version ) {
$type = 'latest_installed' ;
} elseif ( $this_theme [ 'Version' ] > $api -> version ) {
$type = 'newer_installed' ;
$newer_version = $this_theme [ 'Version' ];
}
break ;
}
}
?>
< div class = 'available-theme' >
2009-05-18 18:00:33 +02:00
< img src = '<?php echo esc_url($api->screenshot_url) ?>' width = '300' class = " theme-preview-img " />
2009-03-20 01:14:57 +01:00
< h3 >< ? php echo $api -> name ; ?> </h3>
< p >< ? php printf ( __ ( 'by %s' ), $api -> author ); ?> </p>
< p >< ? php printf ( __ ( 'Version: %s' ), $api -> version ); ?> </p>
< ? php
$buttons = '<a class="button" id="cancel" href="#" onclick="tb_close();return false;">' . __ ( 'Cancel' ) . '</a> ' ;
2009-02-27 00:35:38 +01:00
switch ( $type ) {
2009-03-20 01:14:57 +01:00
default :
case 'install' :
if ( current_user_can ( 'install_themes' ) ) :
2009-04-19 21:36:28 +02:00
$buttons .= '<a class="button-primary" id="install" href="' . wp_nonce_url ( admin_url ( 'update.php?action=install-theme&theme=' . $api -> slug ), 'install-theme_' . $api -> slug ) . '" target="_parent">' . __ ( 'Install Now' ) . '</a>' ;
2009-02-27 00:35:38 +01:00
endif ;
break ;
case 'update_available' :
if ( current_user_can ( 'update_themes' ) ) :
2009-03-20 01:14:57 +01:00
$buttons .= '<a class="button-primary" id="install" href="' . wp_nonce_url ( admin_url ( 'update.php?action=upgrade-theme&theme=' . $update_file ), 'upgrade-theme_' . $update_file ) . '" target="_parent">' . __ ( 'Install Update Now' ) . '</a>' ;
2009-02-27 00:35:38 +01:00
endif ;
break ;
case 'newer_installed' :
if ( current_user_can ( 'install_themes' ) || current_user_can ( 'update_themes' ) ) :
2009-03-20 01:14:57 +01:00
?> <p><?php printf(__('Newer version (%s) is installed.'), $newer_version); ?></p><?php
2009-02-27 00:35:38 +01:00
endif ;
break ;
case 'latest_installed' :
if ( current_user_can ( 'install_themes' ) || current_user_can ( 'update_themes' ) ) :
2009-03-20 01:14:57 +01:00
?> <p><?php _e('This version is already installed.'); ?></p><?php
2009-02-27 00:35:38 +01:00
endif ;
break ;
2009-03-20 01:14:57 +01:00
} ?>
< br class = " clear " />
2009-02-27 00:35:38 +01:00
</ div >
2009-03-20 01:14:57 +01:00
< p class = " action-button " >
< ? php echo $buttons ; ?>
< br class = " clear " />
</ p >
< ? php
2009-02-12 00:10:11 +01:00
iframe_footer ();
exit ;
}
2010-08-11 23:54:51 +02:00
add_action ( 'install_themes_pre_theme-information' , 'install_theme_information' );