Передача змінної до get_template_part


55

WP Кодекс говорить , щоб зробити це:

// You wish to make $my_var available to the template part at `content-part.php`
set_query_var( 'my_var', $my_var );
get_template_part( 'content', 'part' );

Але як мені echo $my_varвсередині частини шаблону? get_query_var($my_var)не працює для мене.

Я бачив тонни рекомендацій щодо використання locate_templateзамість цього. Це найкращий шлях?


Якщо б приблизно таке ж питання і отримав його на роботу з set_query_varі get_query_var, тим НЕ менш це було для використання значень в $argsмасиві , який передається до WP_Query. Може бути корисним для інших людей, які починають це вчити.
lowtechsun

Відповіді:


53

Оскільки публікації налаштовують свої дані через the_post()(відповідно через setup_postdata()) і тому доступні через API ( get_the_ID()наприклад,), припустимо, що ми перебираємо набір користувачів (як setup_userdata()заповнює глобальні змінні поточно зареєстрованого користувача та isn ' t корисно для цього завдання) і спробуйте відобразити метадані на кожного користувача:

<?php
get_header();

// etc.

// In the main template file
$users = new \WP_User_Query( [ ... ] );

foreach ( $users as $user )
{
    set_query_var( 'user_id', absint( $user->ID ) );
    get_template_part( 'template-parts/user', 'contact_methods' );
}

Потім у нашому wpse-theme/template-parts/user-contact_methods.phpфайлі нам потрібно отримати доступ до ідентифікатора користувача:

<?php
/** @var int $user_id */
$some_meta = get_the_author_meta( 'some_meta', $user_id );
var_dump( $some_meta );

Це воно.

Пояснення насправді саме вище частини, яку ви цитували у своєму запитанні:

Однак, load_template()який називається опосередковано, get_template_part()витягує всі WP_Queryзмінні запиту, в область завантаженого шаблону.

Нативна extract()функція PHP "витягує" змінні ( global $wp_query->query_varsвластивість) і ставить кожну частину у власну змінну, яка має точно те саме ім'я, що і ключ. Іншими словами:

set_query_var( 'foo', 'bar' );

$GLOBALS['wp_query'] (object)
    -> query_vars (array)
        foo => bar (string 3)

extract( $wp_query->query_vars );

var_dump( $foo );
// Result:
(string 3) 'bar'

1
як і раніше чудово працює
huraji

23

hm_get_template_partФункція від humanmade дуже хороший в цьому , і я використовую його весь час.

Ти телефонуєш

hm_get_template_part( 'template_path', [ 'option' => 'value' ] );

а потім всередині вашого шаблону ви використовуєте

$template_args['option'];

щоб повернути значення. Це кешування і все, хоча ви можете вийняти це, якщо хочете.

Ви навіть можете повернути наданий шаблон у вигляді рядка, перейшовши 'return' => trueв масив ключ / значення.

/**
 * Like get_template_part() put lets you pass args to the template file
 * Args are available in the tempalte as $template_args array
 * @param string filepart
 * @param mixed wp_args style argument list
 */
function hm_get_template_part( $file, $template_args = array(), $cache_args = array() ) {
    $template_args = wp_parse_args( $template_args );
    $cache_args = wp_parse_args( $cache_args );
    if ( $cache_args ) {
        foreach ( $template_args as $key => $value ) {
            if ( is_scalar( $value ) || is_array( $value ) ) {
                $cache_args[$key] = $value;
            } else if ( is_object( $value ) && method_exists( $value, 'get_id' ) ) {
                $cache_args[$key] = call_user_method( 'get_id', $value );
            }
        }
        if ( ( $cache = wp_cache_get( $file, serialize( $cache_args ) ) ) !== false ) {
            if ( ! empty( $template_args['return'] ) )
                return $cache;
            echo $cache;
            return;
        }
    }
    $file_handle = $file;
    do_action( 'start_operation', 'hm_template_part::' . $file_handle );
    if ( file_exists( get_stylesheet_directory() . '/' . $file . '.php' ) )
        $file = get_stylesheet_directory() . '/' . $file . '.php';
    elseif ( file_exists( get_template_directory() . '/' . $file . '.php' ) )
        $file = get_template_directory() . '/' . $file . '.php';
    ob_start();
    $return = require( $file );
    $data = ob_get_clean();
    do_action( 'end_operation', 'hm_template_part::' . $file_handle );
    if ( $cache_args ) {
        wp_cache_set( $file, $data, serialize( $cache_args ), 3600 );
    }
    if ( ! empty( $template_args['return'] ) )
        if ( $return === false )
            return false;
        else
            return $data;
    echo $data;
}

Включити до проекту 1300 рядків коду (від github HM), щоб передати один параметр шаблону? Не можу цього зробити у своєму проекті :(
Gediminas

11

Я озирався і знайшов різноманітні відповіді. Схоже, на рідному рівні Wordpress дозволяє отримати доступ до змінних у частинах шаблону. Я виявив, що використання include у поєднанні з locate_template дозволило доступ до області змінних у файлі.

include(locate_template('your-template-name.php'));

Використання includeне пройде перевірку теми .
lowtechsun

Нам дійсно потрібно щось, що схоже на перевірку W3C для WP Themes?
Fredy31

5
// you can use any value including objects.

set_query_var( 'var_name_to_be_used_later', 'Value to be retrieved later' );
//Basically set_query_var uses PHP extract() function  to do the magic.


then later in the template.
var_dump($var_name_to_be_used_later);
//will print "Value to be retrieved later"

Рекомендую прочитати про функцію PHP Extract ().


2

Я зіткнувся з цим самим питанням над проектом, над яким зараз працюю. Я вирішив створити свій власний невеликий плагін, який дозволяє більш чітко передавати змінні до get_template_part, використовуючи нову функцію.

Якщо вам це стане в нагоді, ось його сторінка на GitHub: https://github.com/JolekPress/Get-Template-Part-With-Variables

Ось приклад того, як це буде працювати:

$variables = [
    'name' => 'John',
    'class' => 'featuredAuthor',
];

jpr_get_template_part_with_vars('author', 'info', $variables);


// In author-info.php:
echo "
<div class='$class'>
    <span>$name</span>
</div>
";

// Would output:
<div class='featuredAuthor'>
    <span>John</span>
</div>

1

Мені подобається плагін Pods та їх функція pods_view . Це працює аналогічно hm_get_template_partфункції, зазначеній у відповіді djb. Я використовую додаткову функцію ( findTemplateу коді нижче), щоб спочатку шукати файл шаблону в поточній темі, і якщо його не знайдено, він повертає шаблон з тим самим іменем у /templatesпапці мого плагіна . Це приблизна ідея того, як я використовую pods_viewсвій плагін:

/**
 * Helper function to find a template
 */
function findTemplate($filename) {
  // Look first in the theme folder
  $template = locate_template($filename);
  if (!$template) {
    // Otherwise, use the file in our plugin's /templates folder
    $template = dirname(__FILE__) . '/templates/' . $filename;
  }
  return $template;
}

// Output the template 'template-name.php' from either the theme
// folder *or* our plugin's '/template' folder, passing two local
// variables to be available in the template file
pods_view(
  findTemplate('template-name.php'),
  array(
    'passed_variable' => $variable_to_pass,
    'another_variable' => $another_variable,
  )
);

pods_viewтакож підтримує кешування, але мені це не потрібно для моїх цілей. Більше інформації про аргументи функцій можна знайти на сторінках документації Pods. Див. Сторінки для перегляду pods_view та часткового керування сторінками та деталей інтелектуального шаблону з стручками .


1

На основі відповіді @djb з використанням коду від humanmade.

Це легка версія get_template_part, яка може приймати аргументи. Таким чином, змінні локально розміщуються до цього шаблону. Немає необхідності мати global, get_query_var, set_query_var.

/**
 * Like get_template_part() but lets you pass args to the template file
 * Args are available in the template as $args array.
 * Args can be passed in as url parameters, e.g 'key1=value1&key2=value2'.
 * Args can be passed in as an array, e.g. ['key1' => 'value1', 'key2' => 'value2']
 * Filepath is available in the template as $file string.
 * @param string      $slug The slug name for the generic template.
 * @param string|null $name The name of the specialized template.
 * @param array       $args The arguments passed to the template
 */

function _get_template_part( $slug, $name = null, $args = array() ) {
    if ( isset( $name ) && $name !== 'none' ) $slug = "{$slug}-{$name}.php";
    else $slug = "{$slug}.php";
    $dir = get_template_directory();
    $file = "{$dir}/{$slug}";

    ob_start();
    $args = wp_parse_args( $args );
    $slug = $dir = $name = null;
    require( $file );
    echo ob_get_clean();
}

Наприклад у cart.php:

<? php _get_template_part( 'components/items/apple', null, ['color' => 'red']); ?>

В apple.php:

<p>The apple color is: <?php echo $args['color']; ?></p>

0

Як щодо цього?

render( 'template-parts/header/header', 'desktop', 
    array( 'user_id' => 555, 'struct' => array( 'test' => array( 1,2 ) ) )
);
function render ( $slug, $name, $arguments ) {

    if ( $arguments ) {
        foreach ( $arguments as $key => $value ) {
                ${$key} = $value;
        }
    }

$name = (string) $name;
if ( '' !== $name ) {
    $templates = "{$slug}-{$name}.php";
    } else {
        $templates = "{$slug}.php";
    }

    $path = get_template_directory() . '/' . $templates;
    if ( file_exists( $path ) ) {
        ob_start();
        require( $path);
        ob_get_clean();
    }
}

Використовуючи ${$key}ви можете додати змінні в поточну область функцій. Працює для мене швидко і просто, і це не протікає і не зберігається в глобальному масштабі.


0

Для тих, хто виглядає дуже простим способом передачі змінних, ви можете змінити функцію, щоб вона включала:

включити (locate_template ('YourTemplate.php', false, false));

І тоді ви зможете використовувати всі змінні, визначені перед тим, як включити шаблон, не додаючи PASSING кожній для шаблону.

Кредити йдуть на: https://mekshq.com/passing-variables-via-get_template_part-wordpress/


Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.