Querendo saber se você pode ter algumas idéias sobre esse problema. Estou no Google há dias, mas não consigo descobrir.
Aqui é onde eu estou:
Eu tenho uma caixa meta para os "produtos" do tipo de postagem do Woocommerce. Dentro da meta box, há um 'type' = > 'select'
que desejo preencher com uma lista de todos os 'taxonomy' = > 'product_cat'
disponíveis.
Eu posso obter a caixa de seleção para preencher e trabalhar com as categorias de postagem padrão, 'taxonomy' = > 'category'
usando o seguinte código:
function product_cats() {
$options = array();
$categories = get_terms( array( 'taxonomy' => 'category' ) );
foreach( $categories as $category ) {
$options[$category->term_id] = array(
'label' => $category->name,
'value' => $category->slug
);
}
// return array('options'=>$options);
return $options;
}
Tudo se desfaz quando tento usar ‘taxonomy' = > ‘product_cat’
ou qualquer outra taxonomia personalizada que eu tenha.
Eu pensei que o problema era que eu estava tentando acessar a taxonomia personalizada antes de ser registrada, então eu troquei algumas declarações/chamadas no meu arquivo function.php (aquelas que chamam o CPT, meta box e woocommece) para potencialmente mudar a ordem em que as coisas correm, mas sem sorte.
MAS, com base na pergunta e na resposta abaixo, posso agora confirmar que a função pode "ver" e exibir todos os termos, através de taxonomias. Se eu excluir o 'taxonomy =>
dos argumentos, ele retornará termos de todos os tipos de postagem personalizados e taxonomias.
Idealmente, a função básica seria:
function product_cats() {
$options = array();
$categories = get_terms( array( 'taxonomy' => 'product_cat' ) );
foreach( $categories as $category ) {
$options[$category->term_id] = array(
'label' => $category->name,
'value' => $category->slug
);
}
// return array('options'=>$options);
return $options;
}
Apenas querendo saber se você teve algum pensamento geral? Eu sei que é difícil sem ver toda a base de código, mas achei que valeria a pena perguntar.
Versão do Wordpress 4.7.2
Woocommerce versão 2.6.14
ATUALIZAÇÃO:
Lentamente, estou tentando identificar meu problema.
Parece que 'product_cat'
pode ser acessado depois de tudo (bom), mas está cuspindo uma matriz que não está sendo exibida corretamente.
Isso é confuso para mim como se eu simplesmente usasse get_terms()
sem nenhum parâmetro, ou especificando 'taxonomy' => 'category'
o código acima funciona perfeitamente
Os outros pedaços de código com os quais preciso trabalhar são:
A matriz que eu gostaria que a lista de opções para despejar em
array(
'label'=> 'Collections',
'desc' => 'Select the collection you would like to display',
'id' => $prefix.'collection',
'type' => 'select',
'options' => product_cats()
),
o código que gera a lista de seleção (usada para outros meta-campos)
// select
case 'select':
echo '<select name="'.$field['id'].'" id="'.$field['id'].'">';
foreach ($field['options'] as $option) {
echo '<option', $meta == $option['value'] ? ' selected="selected"' : '', ' value="'.$option['value'].'">'.$option['label'].'</option>';
}
echo '</select><br /><span class="description">'.$field['desc'].'</span>';
break;
Eu tenho zero problemas com quaisquer outros campos meta trabalhando ou exibindo, incluindo listas de seleção.
Eu prefiro não reescrever a meta box inteira com todos os seus campos, então estou tentando trabalhar com o que tenho no momento.
Para a vida de mim eu realmente quero fazer com que esse trabalho seja o certo caminho. Para a vida de mim eu não consigo entender a integração.
Anteriormente eu tinha olhado para wp_dropdown_categories()
e achei que era uma solução melhor (e mais fácil). Cheguei a trabalhar no problema acima porque não consegui descobrir como fazê-lo funcionar com a sintaxe de meta box existente.
Por enquanto eu decidi sobre a correção temporária encontrada abaixo. Não é ideal e certamente não é o melhor caminho, mas permite-me avançar com a chamada dos valores nos modelos que utilizarão este campo.
// Wrap all categories in a function
function product_cats() {
$output = array();
$categories = get_terms( array(
'orderby' => 'name',
'pad_counts' => false,
'hierarchical' => 1,
'hide_empty' => true,
) );
foreach( $categories as $category ) {
if ($category->taxonomy == 'product_cat' ) {
$output[$category->slug] = array(
'label' => $category->name,
'value' => $category->slug
);
}
}
//return array('options'=>$output);
return $output;
}
Eu atualizarei mais conforme me movo.
Isso provavelmente não vai consertar, mas queria compartilhar: Eu me deparei com o mesmo problema hoje e isso foi causado por não ter produtos nas minhas categorias. Se este for o seu caso, adicione 'hide_empty' => false
.
Dito isto. Quando você executa get_terms()
sem nenhum argumento. Qual é a saída?
Você está talvez executando uma versão antiga do WordPress (antes de 4.5).
Antes do WordPress 4.5.0, o primeiro parâmetro de get_terms () era uma taxonomia ou lista de taxonomias e desde 4.5.0, taxonomias deveriam ser passadas através do argumento 'taxonomy' no array $ args (que o que você está fazendo, deveria trabalhar assim).
Você encontrará todos os detalhes sobre essas mudanças na página de referência get_terms () .
UPDATE: Desculpe, eu verifico meu código, e uso get_categories () não get_terms, e isso certo get_terms () não funciona!
aqui está um exemplo de trabalho para listar todos os meus product_cat
$product_categories = get_categories( array(
'taxonomy' => 'product_cat',
'orderby' => 'name',
'pad_counts' => false,
'hierarchical' => 1,
'hide_empty' => false
) );
Espero que ajude !
Aqui está um exemplo totalmente funcional de uma meta box que exibe uma caixa de seleção de categoria de produto. A caixa meta aparecerá no tipo de postagem do produto.
add_action( 'add_meta_boxes', 'wpse256897_add_meta_box' );
add_action( 'save_post', 'wpse256897_save' );
/**
* Adds the meta box container.
*/
function wpse256897_add_meta_box( $post_type ) {
// Limit meta box to certain post types.
$post_types = array( 'product' );
if ( in_array( $post_type, $post_types ) ) {
add_meta_box(
'product_cat_selection',
__( 'Product Category Selection', 'textdomain' ),
'wpse256897_render_meta_box_content',
$post_type,
'advanced',
'high'
);
}
}
/**
* Save the meta when the post is saved.
*
* @param int $post_id The ID of the post being saved.
*/
function wpse256897_save( $post_id ) {
/*
* We need to verify this came from the our screen and with proper authorization,
* because save_post can be triggered at other times.
*/
// Check if our nonce is set.
if ( ! isset( $_POST['myplugin_inner_custom_box_nonce'] ) ) {
return $post_id;
}
$nonce = $_POST['myplugin_inner_custom_box_nonce'];
// Verify that the nonce is valid.
if ( ! wp_verify_nonce( $nonce, 'myplugin_inner_custom_box' ) ) {
return $post_id;
}
/*
* If this is an autosave, our form has not been submitted,
* so we don't want to do anything.
*/
if ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) {
return $post_id;
}
// Check the user's permissions.
if ( 'page' == $_POST['post_type'] ) {
if ( ! current_user_can( 'edit_page', $post_id ) ) {
return $post_id;
}
} else {
if ( ! current_user_can( 'edit_post', $post_id ) ) {
return $post_id;
}
}
/* OK, it's safe for us to save the data now. */
// Sanitize the user input.
$mydata = sanitize_text_field( $_POST['product_cat_selection'] );
// Update the meta field.
update_post_meta( $post_id, '_product_cat_selection', $mydata );
}
/**
* Render Meta Box content.
*
* @param WP_Post $post The post object.
*/
function wpse256897_render_meta_box_content( $post ) {
// Add an nonce field so we can check for it later.
wp_nonce_field( 'myplugin_inner_custom_box', 'myplugin_inner_custom_box_nonce' );
// Use get_post_meta to retrieve an existing value from the database.
$current_product_cat = get_post_meta( $post->ID, '_product_cat_selection', true );
// Display the form, using the current value.
$product_cats = wpse256897_product_cats();
if ( !empty ( $product_cats ) ) {
echo '<select name="product_cat_selection" id="product_cat_selection">';
foreach ( $product_cats as $product_cat_id => $product_cat ) { ?>
<option value="<?php echo esc_attr( $product_cat['value'] ); ?>" <?php if ( isset ( $current_product_cat ) ) selected( $current_product_cat, $product_cat['value'] ); ?>><?php echo esc_html( $product_cat['label'] ); ?></option><?php
}
echo '</select>';
}
}
function wpse256897_product_cats() {
$options = array();
$categories = get_terms( array( 'taxonomy' => 'product_cat' ) );
foreach( $categories as $category ) {
$options[$category->term_id] = array(
'label' => $category->name,
'value' => $category->slug
);
}
return $options;
}
Este não é o exemplo mais elegante (as convenções de nomenclatura poderiam ser melhores). Ele foi rapidamente adaptado das notas contribuídas na página de referência Adicionar Meta Box , mas demonstra que wpse256897_product_cats()
obtém as categorias de produto e que elas podem ser salvas e exibidas em uma caixa de seleção na página do produto dentro de uma meta caixa.
Eu também gostaria de acrescentar que vale a pena verificar a função wp_dropdown_categories()
. Que, apesar de seu nome, também trabalha com taxonomias personalizadas. Isso evita que você crie sua própria marcação suspensa de categoria.
Update: Parece que a estrutura da matriz retornada pela função product_cats()
não está combinando com a implementação da meta box. Observe que no meu exemplo acima, usei esta linha para percorrer as categorias ao gerar as opções para o elemento select:
foreach ( $product_cats as $product_cat_id => $product_cat ) { ?>
Isso ocorre porque $product_cats
é uma matriz associativa de IDs de categoria, cada uma contendo outra matriz contendo label
e slug
para cada ID de categoria.
Parece que você poderia usar essa versão alternativa de product_cats()
que formata as opções de valor de retorno $ de uma maneira compatível com seu código metabox:
function product_cats_alternate() {
$options = array();
$categories = get_terms( array( 'taxonomy' => 'product_cat' ) );
foreach( $categories as $category ) {
$options[] = array(
'label' => $category->name,
'value' => $category->slug
);
}
return $options;
}