File Editor: Add support for more than one sub-directory level.
The theme and plugin editors now list all files in the selected theme or plugin, recursing through subdirectories as necessary. Props WraithKenny, schlessera, chsxf, MikeHansenMe, Daedalon, valendesigns, westonruter, pento. Fixes #6531. Built from https://develop.svn.wordpress.org/trunk@41806 git-svn-id: http://core.svn.wordpress.org/trunk@41640 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
parent
1a22fb3b60
commit
58db3cb54e
|
@ -120,35 +120,53 @@ function get_home_path() {
|
||||||
* The depth of the recursiveness can be controlled by the $levels param.
|
* The depth of the recursiveness can be controlled by the $levels param.
|
||||||
*
|
*
|
||||||
* @since 2.6.0
|
* @since 2.6.0
|
||||||
|
* @since 4.9.0 Added the `$exclusions` parameter.
|
||||||
*
|
*
|
||||||
* @param string $folder Optional. Full path to folder. Default empty.
|
* @param string $folder Optional. Full path to folder. Default empty.
|
||||||
* @param int $levels Optional. Levels of folders to follow, Default 100 (PHP Loop limit).
|
* @param int $levels Optional. Levels of folders to follow, Default 100 (PHP Loop limit).
|
||||||
|
* @param array $exclusions Optional. List of folders and files to skip.
|
||||||
* @return bool|array False on failure, Else array of files
|
* @return bool|array False on failure, Else array of files
|
||||||
*/
|
*/
|
||||||
function list_files( $folder = '', $levels = 100 ) {
|
function list_files( $folder = '', $levels = 100, $exclusions = array() ) {
|
||||||
if ( empty($folder) )
|
if ( empty( $folder ) ) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if ( ! $levels )
|
$folder = trailingslashit( $folder );
|
||||||
|
|
||||||
|
if ( ! $levels ) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
$files = array();
|
$files = array();
|
||||||
if ( $dir = @opendir( $folder ) ) {
|
|
||||||
|
$dir = @opendir( $folder );
|
||||||
|
if ( $dir ) {
|
||||||
while ( ( $file = readdir( $dir ) ) !== false ) {
|
while ( ( $file = readdir( $dir ) ) !== false ) {
|
||||||
if ( in_array($file, array('.', '..') ) )
|
// Skip current and parent folder links.
|
||||||
|
if ( in_array( $file, array( '.', '..' ), true ) ) {
|
||||||
continue;
|
continue;
|
||||||
if ( is_dir( $folder . '/' . $file ) ) {
|
}
|
||||||
$files2 = list_files( $folder . '/' . $file, $levels - 1);
|
|
||||||
if ( $files2 )
|
// Skip hidden and excluded files.
|
||||||
|
if ( '.' === $file[0] || in_array( $file, $exclusions, true ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( is_dir( $folder . $file ) ) {
|
||||||
|
$files2 = list_files( $folder . $file, $levels - 1 );
|
||||||
|
if ( $files2 ) {
|
||||||
$files = array_merge($files, $files2 );
|
$files = array_merge($files, $files2 );
|
||||||
else
|
|
||||||
$files[] = $folder . '/' . $file . '/';
|
|
||||||
} else {
|
} else {
|
||||||
$files[] = $folder . '/' . $file;
|
$files[] = $folder . $file . '/';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$files[] = $folder . $file;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@closedir( $dir );
|
@closedir( $dir );
|
||||||
|
|
||||||
return $files;
|
return $files;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -193,32 +193,40 @@ function _get_plugin_data_markup_translate( $plugin_file, $plugin_data, $markup
|
||||||
function get_plugin_files( $plugin ) {
|
function get_plugin_files( $plugin ) {
|
||||||
$plugin_file = WP_PLUGIN_DIR . '/' . $plugin;
|
$plugin_file = WP_PLUGIN_DIR . '/' . $plugin;
|
||||||
$dir = dirname( $plugin_file );
|
$dir = dirname( $plugin_file );
|
||||||
$plugin_files = array($plugin);
|
|
||||||
if ( is_dir($dir) && $dir != WP_PLUGIN_DIR ) {
|
$data = get_plugin_data( $plugin_file );
|
||||||
$plugins_dir = @ opendir( $dir );
|
$label = isset( $data['Version'] )
|
||||||
if ( $plugins_dir ) {
|
? sanitize_key( 'files_' . $plugin . '-' . $data['Version'] )
|
||||||
while (($file = readdir( $plugins_dir ) ) !== false ) {
|
: sanitize_key( 'files_' . $plugin );
|
||||||
if ( substr($file, 0, 1) == '.' )
|
$transient_key = substr( $label, 0, 29 ) . md5( $label );
|
||||||
continue;
|
|
||||||
if ( is_dir( $dir . '/' . $file ) ) {
|
$plugin_files = get_transient( $transient_key );
|
||||||
$plugins_subdir = @ opendir( $dir . '/' . $file );
|
if ( false !== $plugin_files ) {
|
||||||
if ( $plugins_subdir ) {
|
return $plugin_files;
|
||||||
while (($subfile = readdir( $plugins_subdir ) ) !== false ) {
|
|
||||||
if ( substr($subfile, 0, 1) == '.' )
|
|
||||||
continue;
|
|
||||||
$plugin_files[] = plugin_basename("$dir/$file/$subfile");
|
|
||||||
}
|
|
||||||
@closedir( $plugins_subdir );
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if ( plugin_basename("$dir/$file") != $plugin )
|
|
||||||
$plugin_files[] = plugin_basename("$dir/$file");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@closedir( $plugins_dir );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$plugin_files = array( plugin_basename( $plugin_file ) );
|
||||||
|
|
||||||
|
if ( is_dir( $dir ) && WP_PLUGIN_DIR !== $dir ) {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters the array of excluded directories and files while scanning the folder.
|
||||||
|
*
|
||||||
|
* @since 4.9.0
|
||||||
|
*
|
||||||
|
* @param array $exclusions Array of excluded directories and files.
|
||||||
|
*/
|
||||||
|
$exclusions = (array) apply_filters( 'plugin_files_exclusions', array( 'CVS', 'node_modules', 'vendor', 'bower_components' ) );
|
||||||
|
|
||||||
|
$list_files = list_files( $dir, 100, $exclusions );
|
||||||
|
$list_files = array_map( 'plugin_basename', $list_files );
|
||||||
|
|
||||||
|
$plugin_files = array_merge( $plugin_files, $list_files );
|
||||||
|
$plugin_files = array_values( array_unique( $plugin_files ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
set_transient( $transient_key, $plugin_files, HOUR_IN_SECONDS );
|
||||||
|
|
||||||
return $plugin_files;
|
return $plugin_files;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -75,16 +75,16 @@ $file_types = wp_get_theme_file_editable_extensions( $theme );
|
||||||
foreach ( $file_types as $type ) {
|
foreach ( $file_types as $type ) {
|
||||||
switch ( $type ) {
|
switch ( $type ) {
|
||||||
case 'php':
|
case 'php':
|
||||||
$allowed_files += $theme->get_files( 'php', 1 );
|
$allowed_files += $theme->get_files( 'php', -1 );
|
||||||
$has_templates = ! empty( $allowed_files );
|
$has_templates = ! empty( $allowed_files );
|
||||||
break;
|
break;
|
||||||
case 'css':
|
case 'css':
|
||||||
$style_files = $theme->get_files( 'css' );
|
$style_files = $theme->get_files( 'css', -1 );
|
||||||
$allowed_files['style.css'] = $style_files['style.css'];
|
$allowed_files['style.css'] = $style_files['style.css'];
|
||||||
$allowed_files += $style_files;
|
$allowed_files += $style_files;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
$allowed_files += $theme->get_files( $type );
|
$allowed_files += $theme->get_files( $type, -1 );
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -984,10 +984,35 @@ final class WP_Theme implements ArrayAccess {
|
||||||
* being absolute paths.
|
* being absolute paths.
|
||||||
*/
|
*/
|
||||||
public function get_files( $type = null, $depth = 0, $search_parent = false ) {
|
public function get_files( $type = null, $depth = 0, $search_parent = false ) {
|
||||||
$files = (array) self::scandir( $this->get_stylesheet_directory(), $type, $depth );
|
// get and cache all theme files to start with.
|
||||||
|
$label = sanitize_key( 'files_' . $this->cache_hash . '-' . $this->get( 'Version' ) );
|
||||||
|
$transient_key = substr( $label, 0, 29 ) . md5( $label );
|
||||||
|
|
||||||
if ( $search_parent && $this->parent() )
|
$all_files = get_transient( $transient_key );
|
||||||
$files += (array) self::scandir( $this->get_template_directory(), $type, $depth );
|
if ( false === $all_files ) {
|
||||||
|
$all_files = (array) self::scandir( $this->get_stylesheet_directory(), null, -1 );
|
||||||
|
|
||||||
|
if ( $search_parent && $this->parent() ) {
|
||||||
|
$all_files += (array) self::scandir( $this->get_template_directory(), null, -1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
set_transient( $transient_key, $all_files, HOUR_IN_SECONDS );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter $all_files by $type & $depth.
|
||||||
|
$files = array();
|
||||||
|
if ( $type ) {
|
||||||
|
$type = (array) $type;
|
||||||
|
$_extensions = implode( '|', $type );
|
||||||
|
}
|
||||||
|
foreach ( $all_files as $key => $file ) {
|
||||||
|
if ( $depth >= 0 && substr_count( $key, '/' ) > $depth ) {
|
||||||
|
continue; // Filter by depth.
|
||||||
|
}
|
||||||
|
if ( ! $type || preg_match( '~\.(' . $_extensions . ')$~', $file ) ) { // Filter by type.
|
||||||
|
$files[ $key ] = $file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $files;
|
return $files;
|
||||||
}
|
}
|
||||||
|
@ -1107,8 +1132,9 @@ final class WP_Theme implements ArrayAccess {
|
||||||
* with `$relative_path`, with the values being absolute paths. False otherwise.
|
* with `$relative_path`, with the values being absolute paths. False otherwise.
|
||||||
*/
|
*/
|
||||||
private static function scandir( $path, $extensions = null, $depth = 0, $relative_path = '' ) {
|
private static function scandir( $path, $extensions = null, $depth = 0, $relative_path = '' ) {
|
||||||
if ( ! is_dir( $path ) )
|
if ( ! is_dir( $path ) ) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if ( $extensions ) {
|
if ( $extensions ) {
|
||||||
$extensions = (array) $extensions;
|
$extensions = (array) $extensions;
|
||||||
|
@ -1116,8 +1142,9 @@ final class WP_Theme implements ArrayAccess {
|
||||||
}
|
}
|
||||||
|
|
||||||
$relative_path = trailingslashit( $relative_path );
|
$relative_path = trailingslashit( $relative_path );
|
||||||
if ( '/' == $relative_path )
|
if ( '/' == $relative_path ) {
|
||||||
$relative_path = '';
|
$relative_path = '';
|
||||||
|
}
|
||||||
|
|
||||||
$results = scandir( $path );
|
$results = scandir( $path );
|
||||||
$files = array();
|
$files = array();
|
||||||
|
@ -1129,15 +1156,16 @@ final class WP_Theme implements ArrayAccess {
|
||||||
*
|
*
|
||||||
* @param array $exclusions Array of excluded directories and files.
|
* @param array $exclusions Array of excluded directories and files.
|
||||||
*/
|
*/
|
||||||
$exclusions = (array) apply_filters( 'theme_scandir_exclusions', array( 'CVS', 'node_modules' ) );
|
$exclusions = (array) apply_filters( 'theme_scandir_exclusions', array( 'CVS', 'node_modules', 'vendor', 'bower_components' ) );
|
||||||
|
|
||||||
foreach ( $results as $result ) {
|
foreach ( $results as $result ) {
|
||||||
if ( '.' == $result[0] || in_array( $result, $exclusions, true ) ) {
|
if ( '.' == $result[0] || in_array( $result, $exclusions, true ) ) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if ( is_dir( $path . '/' . $result ) ) {
|
if ( is_dir( $path . '/' . $result ) ) {
|
||||||
if ( ! $depth )
|
if ( ! $depth ) {
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
$found = self::scandir( $path . '/' . $result, $extensions, $depth - 1 , $relative_path . $result );
|
$found = self::scandir( $path . '/' . $result, $extensions, $depth - 1 , $relative_path . $result );
|
||||||
$files = array_merge_recursive( $files, $found );
|
$files = array_merge_recursive( $files, $found );
|
||||||
} elseif ( ! $extensions || preg_match( '~\.(' . $_extensions . ')$~', $result ) ) {
|
} elseif ( ! $extensions || preg_match( '~\.(' . $_extensions . ')$~', $result ) ) {
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
*
|
*
|
||||||
* @global string $wp_version
|
* @global string $wp_version
|
||||||
*/
|
*/
|
||||||
$wp_version = '4.9-beta1-41805';
|
$wp_version = '4.9-beta1-41806';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.
|
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.
|
||||||
|
|
Loading…
Reference in New Issue