5 Commits

10 changed files with 1182 additions and 46 deletions

View File

@@ -29,3 +29,63 @@
align-items: center; align-items: center;
margin-top: 8px; margin-top: 8px;
} }
.groq-ai-bulk-panel {
margin: 16px 0;
padding: 16px;
background: #fff;
border: 1px solid #dcdcde;
}
.groq-ai-bulk-panel .description {
margin-top: 8px;
}
#groq-ai-bulk-status {
margin-top: 8px;
}
#groq-ai-bulk-status[data-status='error'] {
color: #b32d2e;
}
#groq-ai-bulk-status[data-status='success'] {
color: #008a20;
}
.groq-ai-bulk-log {
margin: 12px 0 0;
padding-left: 18px;
max-height: 220px;
overflow-y: auto;
font-size: 13px;
}
.groq-ai-bulk-log li {
margin-bottom: 4px;
}
.groq-ai-bulk-log li[data-status='error'] {
color: #b32d2e;
}
.groq-ai-bulk-log li[data-status='success'] {
color: #008a20;
}
.groq-ai-term-row.groq-ai-term-missing td {
background: #fff8e5;
}
.groq-ai-term-row.groq-ai-term-updated td {
animation: groqAiTermPulse 1.8s ease-out 1;
}
@keyframes groqAiTermPulse {
from {
background-color: #e3f8eb;
}
to {
background-color: transparent;
}
}

View File

@@ -13,6 +13,7 @@
const resultField = document.getElementById('groq-ai-output'); const resultField = document.getElementById('groq-ai-output');
const jsonCopyButton = modal.querySelector('.groq-ai-copy-json'); const jsonCopyButton = modal.querySelector('.groq-ai-copy-json');
const contextToggles = modal.querySelectorAll('.groq-ai-context-toggle'); const contextToggles = modal.querySelectorAll('.groq-ai-context-toggle');
const attributeToggles = modal.querySelectorAll('.groq-ai-attribute-toggle');
const resultFields = {}; const resultFields = {};
modal.querySelectorAll('.groq-ai-result-field').forEach((field) => { modal.querySelectorAll('.groq-ai-result-field').forEach((field) => {
const key = field.getAttribute('data-field'); const key = field.getAttribute('data-field');
@@ -60,6 +61,7 @@
promptField.value = GroqAIGenerator.defaultPrompt; promptField.value = GroqAIGenerator.defaultPrompt;
} }
resetContextToggles(); resetContextToggles();
resetAttributeToggles();
setTimeout(() => promptField.focus(), 50); setTimeout(() => promptField.focus(), 50);
} }
@@ -115,6 +117,7 @@
payload.append('prompt', prompt); payload.append('prompt', prompt);
payload.append('post_id', GroqAIGenerator.postId || 0); payload.append('post_id', GroqAIGenerator.postId || 0);
payload.append('context_fields', JSON.stringify(collectContextSelection())); payload.append('context_fields', JSON.stringify(collectContextSelection()));
payload.append('attribute_includes', JSON.stringify(collectAttributeSelection()));
toggleLoading(true); toggleLoading(true);
resultWrapper.hidden = true; resultWrapper.hidden = true;
@@ -458,6 +461,20 @@
}); });
} }
function resetAttributeToggles() {
const defaults = Array.isArray(GroqAIGenerator.attributeIncludesDefaults)
? GroqAIGenerator.attributeIncludesDefaults
: [];
attributeToggles.forEach((toggle) => {
const key = toggle.getAttribute('data-attribute');
if (!key) {
return;
}
toggle.checked = defaults.includes(key);
});
}
function collectContextSelection() { function collectContextSelection() {
const selected = []; const selected = [];
contextToggles.forEach((toggle) => { contextToggles.forEach((toggle) => {
@@ -467,4 +484,18 @@
}); });
return selected; return selected;
} }
function collectAttributeSelection() {
const selected = [];
attributeToggles.forEach((toggle) => {
if (!toggle.checked) {
return;
}
const key = toggle.getAttribute('data-attribute');
if (key) {
selected.push(key);
}
});
return selected;
}
})(jQuery); })(jQuery);

212
assets/js/category-bulk.js Normal file
View File

@@ -0,0 +1,212 @@
(function () {
const data = window.GroqAICategoryBulk || {};
const startButton = document.getElementById('groq-ai-bulk-generate');
const stopButton = document.getElementById('groq-ai-bulk-cancel');
const statusField = document.getElementById('groq-ai-bulk-status');
const logList = document.getElementById('groq-ai-bulk-log');
if (!startButton || !data.ajaxUrl) {
return;
}
let queue = [];
let totalCount = 0;
let processed = 0;
let successes = 0;
let isRunning = false;
let abortRequested = false;
function formatString(template, values) {
if (!template) {
return '';
}
let autoIndex = 0;
return template.replace(/%(\d+\$)?[sd]/g, (match, position) => {
let valueIndex;
if (position) {
valueIndex = parseInt(position, 10) - 1;
} else {
valueIndex = autoIndex;
autoIndex += 1;
}
const replacement = values[valueIndex];
return typeof replacement === 'undefined' ? '' : String(replacement);
});
}
function setStatus(message, type) {
if (!statusField) {
return;
}
statusField.textContent = message || '';
statusField.dataset.status = type || '';
}
function appendLog(message, type) {
if (!logList || !message) {
return;
}
const item = document.createElement('li');
item.textContent = message;
item.dataset.status = type || '';
logList.appendChild(item);
}
function resetLog() {
if (!logList) {
return;
}
logList.innerHTML = '';
}
function toggleButtons(running) {
isRunning = running;
startButton.disabled = running;
if (stopButton) {
stopButton.hidden = !running;
}
}
function getPendingTerms() {
if (!Array.isArray(data.terms)) {
return [];
}
return data.terms.filter((term) => !term.processed);
}
function updateRow(termId, words) {
const row = document.querySelector('[data-groq-ai-term-id="' + termId + '"]');
if (!row) {
return;
}
row.classList.remove('groq-ai-term-missing');
row.classList.add('groq-ai-term-updated');
const wordCell = row.querySelector('.groq-ai-word-count');
if (wordCell) {
wordCell.textContent = String(typeof words === 'number' ? words : wordCell.textContent);
}
}
function finish(state) {
const summaryTemplate =
state === 'done'
? data.strings && data.strings.statusDone
: state === 'stopped'
? data.strings && data.strings.statusStopped
: '';
const summary = summaryTemplate
? formatString(summaryTemplate, [successes])
: '';
const statusType = state === 'done' ? 'success' : state === 'stopped' ? 'info' : '';
setStatus(summary, statusType);
toggleButtons(false);
queue = [];
totalCount = 0;
abortRequested = false;
}
function processNext() {
if (abortRequested) {
finish('stopped');
return;
}
if (!queue.length) {
finish('done');
return;
}
const term = queue.shift();
const position = processed + 1;
const progressTemplate = data.strings && data.strings.statusProgress;
if (progressTemplate) {
setStatus(formatString(progressTemplate, [position, totalCount, term.name || '']), 'loading');
}
const payload = new URLSearchParams();
payload.append('action', 'groq_ai_bulk_generate_terms');
payload.append('nonce', data.nonce || '');
payload.append('taxonomy', data.taxonomy || 'product_cat');
payload.append('term_id', term.id);
fetch(data.ajaxUrl, {
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
},
body: payload.toString(),
})
.then((response) => response.json())
.then((json) => {
if (!json.success) {
const errorMessage = (json.data && json.data.message) || 'Onbekende fout';
appendLog(formatString((data.strings && data.strings.logError) || '%1$s: %2$s', [term.name || term.id, errorMessage]), 'error');
return;
}
term.processed = true;
successes += 1;
const words = json.data && typeof json.data.words !== 'undefined' ? json.data.words : 0;
updateRow(term.id, words);
appendLog(formatString((data.strings && data.strings.logSuccess) || '%1$s gevuld.', [term.name || term.id, words]), 'success');
})
.catch((error) => {
appendLog(
formatString((data.strings && data.strings.logError) || '%1$s: %2$s', [term.name || term.id, error && error.message ? error.message : 'Onbekende fout']),
'error'
);
})
.finally(() => {
processed += 1;
if (abortRequested) {
finish('stopped');
} else {
processNext();
}
});
}
function startBulk() {
if (isRunning) {
return;
}
const pending = getPendingTerms();
if (!pending.length) {
setStatus((data.strings && data.strings.statusEmpty) || '', 'info');
return;
}
queue = pending.slice();
totalCount = queue.length;
processed = 0;
successes = 0;
abortRequested = false;
resetLog();
toggleButtons(true);
setStatus((data.strings && data.strings.statusIdle) || '', 'info');
processNext();
}
startButton.addEventListener('click', startBulk);
if (stopButton) {
stopButton.addEventListener('click', () => {
if (!isRunning) {
return;
}
const confirmation = ! (data.strings && data.strings.confirmStop)
? window.confirm('Stoppen?')
: window.confirm(data.strings.confirmStop);
if (confirmation) {
abortRequested = true;
}
});
}
if (!Array.isArray(data.terms) || !data.terms.length) {
setStatus((data.strings && data.strings.statusEmpty) || '', 'info');
}
})();

View File

@@ -9,7 +9,11 @@
} }
const promptField = document.getElementById('groq-ai-term-prompt'); const promptField = document.getElementById('groq-ai-term-prompt');
const outputField = document.getElementById('groq-ai-term-generated'); const outputTopField = document.getElementById('groq-ai-term-generated-top');
const outputBottomField = document.getElementById('groq-ai-term-generated-bottom');
const outputMetaTitleField = document.getElementById('groq-ai-term-generated-meta-title');
const outputMetaDescriptionField = document.getElementById('groq-ai-term-generated-meta-description');
const outputFocusKeywordsField = document.getElementById('groq-ai-term-generated-focus-keywords');
const rawField = document.getElementById('groq-ai-term-raw'); const rawField = document.getElementById('groq-ai-term-raw');
const statusField = document.getElementById('groq-ai-term-status'); const statusField = document.getElementById('groq-ai-term-status');
const applyButton = document.getElementById('groq-ai-term-apply'); const applyButton = document.getElementById('groq-ai-term-apply');
@@ -47,11 +51,31 @@
if (applyButton) { if (applyButton) {
applyButton.addEventListener('click', () => { applyButton.addEventListener('click', () => {
const descriptionField = document.getElementById('description'); const descriptionField = document.getElementById('description');
if (!descriptionField || !outputField) { const bottomDescriptionField = document.getElementById('groq-ai-term-bottom-description');
const rankmathTitleField = document.getElementById('groq-ai-rankmath-title');
const rankmathDescriptionField = document.getElementById('groq-ai-rankmath-description');
const rankmathKeywordsField = document.getElementById('groq-ai-rankmath-keywords');
if (!outputTopField) {
return; return;
} }
descriptionField.value = outputField.value || '';
setStatus('Tekst ingevuld in het beschrijving-veld. Vergeet niet op "Opslaan" te klikken.', 'success'); if (descriptionField) {
descriptionField.value = outputTopField.value || '';
}
if (bottomDescriptionField && outputBottomField) {
bottomDescriptionField.value = outputBottomField.value || '';
}
if (rankmathTitleField && outputMetaTitleField) {
rankmathTitleField.value = outputMetaTitleField.value || '';
}
if (rankmathDescriptionField && outputMetaDescriptionField) {
rankmathDescriptionField.value = outputMetaDescriptionField.value || '';
}
if (rankmathKeywordsField && outputFocusKeywordsField) {
rankmathKeywordsField.value = outputFocusKeywordsField.value || '';
}
setStatus('Tekst ingevuld. Vergeet niet op "Opslaan" te klikken.', 'success');
}); });
} }
@@ -69,6 +93,12 @@
rawField.textContent = ''; rawField.textContent = '';
} }
if (outputTopField) outputTopField.value = '';
if (outputBottomField) outputBottomField.value = '';
if (outputMetaTitleField) outputMetaTitleField.value = '';
if (outputMetaDescriptionField) outputMetaDescriptionField.value = '';
if (outputFocusKeywordsField) outputFocusKeywordsField.value = '';
fetch(GroqAITermGenerator.ajaxUrl, { fetch(GroqAITermGenerator.ajaxUrl, {
method: 'POST', method: 'POST',
headers: { headers: {
@@ -83,8 +113,25 @@
throw new Error(errorMessage); throw new Error(errorMessage);
} }
if (outputField) { if (outputTopField) {
outputField.value = (json.data && json.data.description ? json.data.description : '').trim(); const top = json.data && (json.data.top_description || json.data.description) ? (json.data.top_description || json.data.description) : '';
outputTopField.value = String(top).trim();
}
if (outputBottomField) {
const bottom = json.data && json.data.bottom_description ? json.data.bottom_description : '';
outputBottomField.value = String(bottom).trim();
}
if (outputMetaTitleField) {
const metaTitle = json.data && json.data.meta_title ? json.data.meta_title : '';
outputMetaTitleField.value = String(metaTitle).trim();
}
if (outputMetaDescriptionField) {
const metaDescription = json.data && json.data.meta_description ? json.data.meta_description : '';
outputMetaDescriptionField.value = String(metaDescription).trim();
}
if (outputFocusKeywordsField) {
const keywords = json.data && json.data.focus_keywords ? json.data.focus_keywords : '';
outputFocusKeywordsField.value = String(keywords).trim();
} }
if (rawField) { if (rawField) {
rawField.textContent = (json.data && json.data.raw ? String(json.data.raw) : '').trim(); rawField.textContent = (json.data && json.data.raw ? String(json.data.raw) : '').trim();

View File

@@ -2,7 +2,7 @@
/** /**
* Plugin Name: SitiAI Product Teksten * Plugin Name: SitiAI Product Teksten
* Description: Genereer productteksten met diverse AI-aanbieders rechtstreeks vanuit WooCommerce. * Description: Genereer productteksten met diverse AI-aanbieders rechtstreeks vanuit WooCommerce.
* Version: 1.4.2 * Version: 1.6.0
* Author: SitiAI * Author: SitiAI
* Text Domain: siti-ai-product-content-generator * Text Domain: siti-ai-product-content-generator
* Domain Path: /languages * Domain Path: /languages

View File

@@ -59,6 +59,9 @@ class Groq_AI_Product_Text_Product_UI {
$post_id = ( $post && isset( $post->ID ) ) ? (int) $post->ID : 0; $post_id = ( $post && isset( $post->ID ) ) ? (int) $post->ID : 0;
$settings = $this->plugin->get_settings(); $settings = $this->plugin->get_settings();
$attribute_defaults = isset( $settings['product_attribute_includes'] ) && is_array( $settings['product_attribute_includes'] )
? array_values( array_unique( array_map( 'sanitize_key', $settings['product_attribute_includes'] ) ) )
: [];
wp_localize_script( wp_localize_script(
'groq-ai-admin', 'groq-ai-admin',
@@ -69,6 +72,7 @@ class Groq_AI_Product_Text_Product_UI {
'defaultPrompt' => $settings['default_prompt'], 'defaultPrompt' => $settings['default_prompt'],
'postId' => $post_id, 'postId' => $post_id,
'contextDefaults' => isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields(), 'contextDefaults' => isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields(),
'attributeIncludesDefaults' => $attribute_defaults,
] ]
); );
} }
@@ -82,6 +86,7 @@ class Groq_AI_Product_Text_Product_UI {
$settings = $this->plugin->get_settings(); $settings = $this->plugin->get_settings();
$rankmath_enabled = $this->plugin->is_rankmath_active() && $this->plugin->is_module_enabled( 'rankmath', $settings ); $rankmath_enabled = $this->plugin->is_rankmath_active() && $this->plugin->is_module_enabled( 'rankmath', $settings );
$attribute_options = $this->get_product_attribute_include_options();
?> ?>
<div id="groq-ai-modal" class="groq-ai-modal" aria-hidden="true"> <div id="groq-ai-modal" class="groq-ai-modal" aria-hidden="true">
<div class="groq-ai-modal__dialog" role="dialog" aria-modal="true" aria-labelledby="groq-ai-modal-title"> <div class="groq-ai-modal__dialog" role="dialog" aria-modal="true" aria-labelledby="groq-ai-modal-title">
@@ -109,6 +114,9 @@ class Groq_AI_Product_Text_Product_UI {
$context_definitions = $this->plugin->get_context_field_definitions(); $context_definitions = $this->plugin->get_context_field_definitions();
$context_defaults = isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields(); $context_defaults = isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields();
foreach ( $context_definitions as $context_key => $context_info ) : foreach ( $context_definitions as $context_key => $context_info ) :
if ( 'attributes' === $context_key ) {
continue;
}
$checked = ! empty( $context_defaults[ $context_key ] ); $checked = ! empty( $context_defaults[ $context_key ] );
?> ?>
<label class="groq-ai-context-option"> <label class="groq-ai-context-option">
@@ -122,6 +130,23 @@ class Groq_AI_Product_Text_Product_UI {
</label> </label>
<?php endforeach; ?> <?php endforeach; ?>
</div> </div>
<h3 style="margin-top:16px;"><?php esc_html_e( 'Attributen meesturen', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<p class="description"><?php esc_html_e( 'Selecteer welke productattributen je mee wilt geven aan de AI. Dit vervangt de oude alles-of-niets optie.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
<?php if ( empty( $attribute_options ) ) : ?>
<p class="description"><?php esc_html_e( 'Geen WooCommerce-attributen gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
<?php else : ?>
<div class="groq-ai-context-options__grid">
<?php foreach ( $attribute_options as $attr_key => $attr_label ) : ?>
<label class="groq-ai-context-option">
<input type="checkbox" class="groq-ai-attribute-toggle" data-attribute="<?php echo esc_attr( $attr_key ); ?>" />
<div>
<strong><?php echo esc_html( $attr_label ); ?></strong>
</div>
</label>
<?php endforeach; ?>
</div>
<?php endif; ?>
</div> </div>
</div> </div>
</form> </form>
@@ -225,4 +250,39 @@ class Groq_AI_Product_Text_Product_UI {
</div> </div>
<?php <?php
} }
private function get_product_attribute_include_options() {
$options = [
'__custom__' => __( 'Custom attributen (niet-taxonomie)', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
];
if ( function_exists( 'wc_get_attribute_taxonomies' ) ) {
$taxonomies = wc_get_attribute_taxonomies();
if ( is_array( $taxonomies ) ) {
foreach ( $taxonomies as $attr ) {
$name = isset( $attr->attribute_name ) ? sanitize_key( (string) $attr->attribute_name ) : '';
$label = isset( $attr->attribute_label ) ? sanitize_text_field( (string) $attr->attribute_label ) : '';
if ( '' === $name ) {
continue;
}
$taxonomy = 'pa_' . $name;
if ( '' === $label ) {
$label = function_exists( 'wc_attribute_label' ) ? wc_attribute_label( $taxonomy ) : $taxonomy;
}
$options[ $taxonomy ] = $label;
}
}
}
if ( count( $options ) > 1 ) {
$fixed = [
'__custom__' => $options['__custom__'],
];
unset( $options['__custom__'] );
asort( $options, SORT_NATURAL | SORT_FLAG_CASE );
$options = $fixed + $options;
}
return $options;
}
} }

View File

@@ -173,10 +173,44 @@ class Groq_AI_Product_Text_Settings_Page {
if ( is_wp_error( $terms ) ) { if ( is_wp_error( $terms ) ) {
$terms = []; $terms = [];
} }
$word_map = [];
$empty_terms = [];
foreach ( $terms as $term ) {
if ( ! $term || ! is_object( $term ) ) {
continue;
}
$word_count = $this->count_words( isset( $term->description ) ? $term->description : '' );
$word_map[ $term->term_id ] = $word_count;
if ( 0 === $word_count ) {
$empty_terms[] = $term;
}
}
?> ?>
<div class="wrap"> <div class="wrap">
<h1><?php esc_html_e( 'Categorie teksten', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h1> <h1><?php esc_html_e( 'Categorie teksten', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h1>
<p><?php esc_html_e( 'Klik op een categorie om teksten te genereren en instellingen te beheren. De tabel toont de huidige woordlengte van de categorie-omschrijving.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p> <p><?php esc_html_e( 'Klik op een categorie om teksten te genereren en instellingen te beheren. De tabel toont de huidige woordlengte van de categorie-omschrijving.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
<div class="groq-ai-bulk-panel">
<?php if ( ! empty( $empty_terms ) ) : ?>
<p>
<?php
printf(
/* translators: %d: amount of categories without description */
esc_html__( 'Er zijn %d categorieën zonder omschrijving. Klik op de knop hieronder om automatisch teksten te genereren.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
count( $empty_terms )
);
?>
</p>
<p>
<button type="button" class="button button-primary" id="groq-ai-bulk-generate"><?php esc_html_e( 'Genereer teksten voor lege categorieën', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button>
<button type="button" class="button" id="groq-ai-bulk-cancel" hidden><?php esc_html_e( 'Stop bulk generatie', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button>
</p>
<div id="groq-ai-bulk-status" class="description"></div>
<ol id="groq-ai-bulk-log" class="groq-ai-bulk-log"></ol>
<?php else : ?>
<p class="description"><?php esc_html_e( 'Alle categorieën hebben al een omschrijving.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
<?php endif; ?>
</div>
<table class="widefat striped"> <table class="widefat striped">
<thead> <thead>
<tr> <tr>
@@ -193,16 +227,24 @@ class Groq_AI_Product_Text_Settings_Page {
<?php foreach ( $terms as $term ) : ?> <?php foreach ( $terms as $term ) : ?>
<?php <?php
$link = $this->get_term_page_url( 'product_cat', $term->term_id ); $link = $this->get_term_page_url( 'product_cat', $term->term_id );
$words = $this->count_words( $term->description ); $words = isset( $word_map[ $term->term_id ] ) ? $word_map[ $term->term_id ] : 0;
$count = isset( $term->count ) ? absint( $term->count ) : 0; $count = isset( $term->count ) ? absint( $term->count ) : 0;
$row_classes = [ 'groq-ai-term-row' ];
if ( 0 === $words ) {
$row_classes[] = 'groq-ai-term-missing';
}
?> ?>
<tr> <tr class="<?php echo esc_attr( implode( ' ', $row_classes ) ); ?>" data-groq-ai-term-id="<?php echo esc_attr( (string) $term->term_id ); ?>">
<td> <td>
<a href="<?php echo esc_url( $link ); ?>"><strong><?php echo esc_html( $term->name ); ?></strong></a> <a href="<?php echo esc_url( $link ); ?>"><strong><?php echo esc_html( $term->name ); ?></strong></a>
</td> </td>
<td><?php echo esc_html( $term->slug ); ?></td> <td><?php echo esc_html( $term->slug ); ?></td>
<td><?php echo esc_html( (string) $count ); ?></td> <td><?php echo esc_html( (string) $count ); ?></td>
<td><?php echo esc_html( (string) $words ); ?></td> <td class="groq-ai-word-cell">
<span class="groq-ai-word-count" data-term-id="<?php echo esc_attr( (string) $term->term_id ); ?>">
<?php echo esc_html( (string) $words ); ?>
</span>
</td>
</tr> </tr>
<?php endforeach; ?> <?php endforeach; ?>
<?php endif; ?> <?php endif; ?>
@@ -319,10 +361,22 @@ class Groq_AI_Product_Text_Settings_Page {
$term_label = ( 'product_cat' === $taxonomy ) ? __( 'Categorie', GROQ_AI_PRODUCT_TEXT_DOMAIN ) : __( 'Term', GROQ_AI_PRODUCT_TEXT_DOMAIN ); $term_label = ( 'product_cat' === $taxonomy ) ? __( 'Categorie', GROQ_AI_PRODUCT_TEXT_DOMAIN ) : __( 'Term', GROQ_AI_PRODUCT_TEXT_DOMAIN );
$word_count = $this->count_words( $term->description ); $word_count = $this->count_words( $term->description );
$meta_prompt = get_term_meta( $term_id, 'groq_ai_term_custom_prompt', true ); $meta_prompt = get_term_meta( $term_id, 'groq_ai_term_custom_prompt', true );
$default_prompt = (string) $meta_prompt; $settings = $this->plugin->get_settings();
if ( '' === trim( $default_prompt ) ) { $bottom_meta_key = $this->resolve_term_bottom_description_meta_key( $term, $settings );
$default_prompt = __( 'Schrijf een SEO-vriendelijke categorieomschrijving in het Nederlands. Gebruik duidelijke tussenkoppen en <p>-tags. Voeg geen prijsinformatie toe.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); $effective_bottom_meta_key = '' !== $bottom_meta_key ? $bottom_meta_key : 'groq_ai_term_bottom_description';
$bottom_description = (string) get_term_meta( $term_id, $effective_bottom_meta_key, true );
$rankmath_module_enabled = $this->plugin->is_module_enabled( 'rankmath', $settings );
$rankmath_active = $this->plugin->is_rankmath_active();
$rankmath_title = '';
$rankmath_description = '';
$rankmath_focus_keywords = '';
if ( $rankmath_module_enabled ) {
$rankmath_keys = $this->resolve_rankmath_term_meta_keys( $term, $settings );
$rankmath_title = (string) get_term_meta( $term_id, $rankmath_keys['title'], true );
$rankmath_description = (string) get_term_meta( $term_id, $rankmath_keys['description'], true );
$rankmath_focus_keywords = (string) get_term_meta( $term_id, $rankmath_keys['focus_keyword'], true );
} }
$default_prompt = $this->get_term_prompt_text( $term, $meta_prompt );
?> ?>
<div class="wrap"> <div class="wrap">
<h1> <h1>
@@ -354,6 +408,24 @@ class Groq_AI_Product_Text_Settings_Page {
<p class="description"><?php esc_html_e( 'Dit is de standaard WordPress term-omschrijving (wordt o.a. gebruikt op categorie/merk paginas).', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p> <p class="description"><?php esc_html_e( 'Dit is de standaard WordPress term-omschrijving (wordt o.a. gebruikt op categorie/merk paginas).', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
</td> </td>
</tr> </tr>
<tr>
<th scope="row"><label for="groq-ai-term-bottom-description"><?php esc_html_e( 'Omschrijving (onderaan)', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th>
<td>
<textarea name="groq_ai_term_bottom_description" id="groq-ai-term-bottom-description" rows="8" class="large-text"><?php echo esc_textarea( (string) $bottom_description ); ?></textarea>
<p class="description">
<?php
printf(
/* translators: %s: meta key */
esc_html__( 'Deze tekst wordt opgeslagen in term meta (%s) en is bedoeld voor helemaal onderaan (LiveBetter customfields).', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
esc_html( $effective_bottom_meta_key )
);
if ( '' === $bottom_meta_key ) {
echo ' ' . esc_html__( 'Let op: stel de juiste LiveBetter meta key in via de plugin-instelling of via de filter groq_ai_term_bottom_description_meta_key.', GROQ_AI_PRODUCT_TEXT_DOMAIN );
}
?>
</p>
</td>
</tr>
<tr> <tr>
<th scope="row"><label for="groq-ai-term-custom-prompt"><?php esc_html_e( 'Prompt (optioneel, per term)', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th> <th scope="row"><label for="groq-ai-term-custom-prompt"><?php esc_html_e( 'Prompt (optioneel, per term)', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th>
<td> <td>
@@ -361,6 +433,34 @@ class Groq_AI_Product_Text_Settings_Page {
<p class="description"><?php esc_html_e( 'Laat leeg om de standaard prompt te gebruiken. Deze prompt wordt gebruikt wanneer je op de knop "Genereer" klikt.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p> <p class="description"><?php esc_html_e( 'Laat leeg om de standaard prompt te gebruiken. Deze prompt wordt gebruikt wanneer je op de knop "Genereer" klikt.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
</td> </td>
</tr> </tr>
<?php if ( $rankmath_module_enabled ) : ?>
<?php if ( ! $rankmath_active ) : ?>
<tr>
<th scope="row"><?php esc_html_e( 'Rank Math', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></th>
<td>
<p class="description"><?php esc_html_e( 'Rank Math plugin lijkt niet actief. Velden zijn wel invulbaar en worden opgeslagen in term meta, maar Rank Math gebruikt ze pas na activatie.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></p>
</td>
</tr>
<?php endif; ?>
<tr>
<th scope="row"><label for="groq-ai-rankmath-title"><?php esc_html_e( 'Rank Math meta title', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th>
<td>
<textarea name="groq_ai_rankmath_meta_title" id="groq-ai-rankmath-title" rows="2" class="large-text"><?php echo esc_textarea( (string) $rankmath_title ); ?></textarea>
</td>
</tr>
<tr>
<th scope="row"><label for="groq-ai-rankmath-description"><?php esc_html_e( 'Rank Math meta description', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th>
<td>
<textarea name="groq_ai_rankmath_meta_description" id="groq-ai-rankmath-description" rows="3" class="large-text"><?php echo esc_textarea( (string) $rankmath_description ); ?></textarea>
</td>
</tr>
<tr>
<th scope="row"><label for="groq-ai-rankmath-keywords"><?php esc_html_e( 'Rank Math focus keywords', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></label></th>
<td>
<textarea name="groq_ai_rankmath_focus_keywords" id="groq-ai-rankmath-keywords" rows="2" class="large-text" placeholder="<?php esc_attr_e( 'bijv. luxe massage apparaat, wellness cadeau', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?>"><?php echo esc_textarea( (string) $rankmath_focus_keywords ); ?></textarea>
</td>
</tr>
<?php endif; ?>
</table> </table>
<?php submit_button( __( 'Opslaan', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ); ?> <?php submit_button( __( 'Opslaan', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ); ?>
</form> </form>
@@ -384,11 +484,21 @@ class Groq_AI_Product_Text_Settings_Page {
<textarea id="groq-ai-term-prompt" class="large-text" rows="5"><?php echo esc_textarea( $default_prompt ); ?></textarea> <textarea id="groq-ai-term-prompt" class="large-text" rows="5"><?php echo esc_textarea( $default_prompt ); ?></textarea>
<p> <p>
<button type="submit" class="button button-primary"><?php esc_html_e( 'Genereer', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button> <button type="submit" class="button button-primary"><?php esc_html_e( 'Genereer', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button>
<button type="button" class="button" id="groq-ai-term-apply"><?php esc_html_e( 'Zet in omschrijving', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button> <button type="button" class="button" id="groq-ai-term-apply"><?php esc_html_e( 'Zet in velden', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></button>
</p> </p>
<div id="groq-ai-term-status" class="description" aria-live="polite"></div> <div id="groq-ai-term-status" class="description" aria-live="polite"></div>
<h3><?php esc_html_e( 'Gegenereerde tekst', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3> <h3><?php esc_html_e( 'Gegenereerde tekst (omschrijving, 1 alinea)', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<textarea id="groq-ai-term-generated" class="large-text" rows="10"></textarea> <textarea id="groq-ai-term-generated-top" class="large-text" rows="6"></textarea>
<h3><?php esc_html_e( 'Gegenereerde tekst (onderaan)', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<textarea id="groq-ai-term-generated-bottom" class="large-text" rows="10"></textarea>
<?php if ( $rankmath_module_enabled ) : ?>
<h3><?php esc_html_e( 'Gegenereerde Rank Math meta title', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<textarea id="groq-ai-term-generated-meta-title" class="large-text" rows="2"></textarea>
<h3><?php esc_html_e( 'Gegenereerde Rank Math meta description', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<textarea id="groq-ai-term-generated-meta-description" class="large-text" rows="3"></textarea>
<h3><?php esc_html_e( 'Gegenereerde Rank Math focus keywords', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<textarea id="groq-ai-term-generated-focus-keywords" class="large-text" rows="2"></textarea>
<?php endif; ?>
<h3><?php esc_html_e( 'Ruwe JSON-output', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3> <h3><?php esc_html_e( 'Ruwe JSON-output', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?></h3>
<pre id="groq-ai-term-raw" style="background:#fff;border:1px solid #ddd;padding:12px;max-height:240px;overflow:auto;"></pre> <pre id="groq-ai-term-raw" style="background:#fff;border:1px solid #ddd;padding:12px;max-height:240px;overflow:auto;"></pre>
</form> </form>
@@ -396,6 +506,86 @@ class Groq_AI_Product_Text_Settings_Page {
<?php <?php
} }
private function resolve_term_bottom_description_meta_key( $term, $settings ) {
$default_key = '';
if ( is_array( $settings ) && isset( $settings['term_bottom_description_meta_key'] ) ) {
$default_key = sanitize_key( (string) $settings['term_bottom_description_meta_key'] );
}
$key = apply_filters( 'groq_ai_term_bottom_description_meta_key', $default_key, $term, $settings );
$key = sanitize_key( (string) $key );
return $key;
}
private function resolve_rankmath_term_meta_keys( $term, $settings ) {
$keys = [
'title' => 'rank_math_title',
'description' => 'rank_math_description',
'focus_keyword' => 'rank_math_focus_keyword',
];
$keys = apply_filters( 'groq_ai_rankmath_term_meta_keys', $keys, $term, $settings );
if ( ! is_array( $keys ) ) {
$keys = [];
}
return [
'title' => isset( $keys['title'] ) ? sanitize_key( (string) $keys['title'] ) : 'rank_math_title',
'description' => isset( $keys['description'] ) ? sanitize_key( (string) $keys['description'] ) : 'rank_math_description',
'focus_keyword' => isset( $keys['focus_keyword'] ) ? sanitize_key( (string) $keys['focus_keyword'] ) : 'rank_math_focus_keyword',
];
}
private function get_term_prompt_text( $term, $custom_prompt = null ) {
$prompt = ( null !== $custom_prompt ) ? $custom_prompt : '';
if ( null === $custom_prompt && $term && isset( $term->term_id ) ) {
$prompt = get_term_meta( $term->term_id, 'groq_ai_term_custom_prompt', true );
}
$prompt = trim( (string) $prompt );
if ( '' !== $prompt ) {
return $prompt;
}
$default_prompt = __( 'Schrijf een SEO-vriendelijke categorieomschrijving in het Nederlands. Gebruik duidelijke tussenkoppen en <p>-tags. Voeg geen prijsinformatie toe.', GROQ_AI_PRODUCT_TEXT_DOMAIN );
return apply_filters( 'groq_ai_default_term_prompt', $default_prompt, $term );
}
private function get_terms_without_description_payload( $taxonomy ) {
$terms = get_terms(
[
'taxonomy' => $taxonomy,
'hide_empty' => false,
'orderby' => 'name',
'order' => 'ASC',
'number' => 0,
]
);
if ( is_wp_error( $terms ) ) {
return [];
}
$payloads = [];
foreach ( $terms as $term ) {
$description = isset( $term->description ) ? trim( wp_strip_all_tags( (string) $term->description ) ) : '';
if ( '' !== $description ) {
continue;
}
$payloads[] = [
'id' => isset( $term->term_id ) ? absint( $term->term_id ) : 0,
'name' => isset( $term->name ) ? (string) $term->name : '',
'slug' => isset( $term->slug ) ? (string) $term->slug : '',
'count' => isset( $term->count ) ? absint( $term->count ) : 0,
'url' => esc_url( $this->get_term_page_url( $taxonomy, isset( $term->term_id ) ? $term->term_id : 0 ) ),
];
}
return array_values( $payloads );
}
public function handle_save_term_content() { public function handle_save_term_content() {
if ( ! current_user_can( 'manage_options' ) ) { if ( ! current_user_can( 'manage_options' ) ) {
wp_die( esc_html__( 'Geen toestemming.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ); wp_die( esc_html__( 'Geen toestemming.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) );
@@ -406,7 +596,11 @@ class Groq_AI_Product_Text_Settings_Page {
$taxonomy = isset( $_POST['taxonomy'] ) ? sanitize_key( wp_unslash( $_POST['taxonomy'] ) ) : ''; $taxonomy = isset( $_POST['taxonomy'] ) ? sanitize_key( wp_unslash( $_POST['taxonomy'] ) ) : '';
$term_id = isset( $_POST['term_id'] ) ? absint( $_POST['term_id'] ) : 0; $term_id = isset( $_POST['term_id'] ) ? absint( $_POST['term_id'] ) : 0;
$description = isset( $_POST['description'] ) ? wp_kses_post( wp_unslash( $_POST['description'] ) ) : ''; $description = isset( $_POST['description'] ) ? wp_kses_post( wp_unslash( $_POST['description'] ) ) : '';
$bottom_description = isset( $_POST['groq_ai_term_bottom_description'] ) ? wp_kses_post( wp_unslash( $_POST['groq_ai_term_bottom_description'] ) ) : '';
$custom_prompt = isset( $_POST['groq_ai_term_custom_prompt'] ) ? sanitize_textarea_field( wp_unslash( $_POST['groq_ai_term_custom_prompt'] ) ) : ''; $custom_prompt = isset( $_POST['groq_ai_term_custom_prompt'] ) ? sanitize_textarea_field( wp_unslash( $_POST['groq_ai_term_custom_prompt'] ) ) : '';
$rankmath_meta_title = isset( $_POST['groq_ai_rankmath_meta_title'] ) ? sanitize_text_field( wp_unslash( $_POST['groq_ai_rankmath_meta_title'] ) ) : '';
$rankmath_meta_description = isset( $_POST['groq_ai_rankmath_meta_description'] ) ? sanitize_text_field( wp_unslash( $_POST['groq_ai_rankmath_meta_description'] ) ) : '';
$rankmath_focus_keywords = isset( $_POST['groq_ai_rankmath_focus_keywords'] ) ? sanitize_text_field( wp_unslash( $_POST['groq_ai_rankmath_focus_keywords'] ) ) : '';
if ( '' === $taxonomy || ! taxonomy_exists( $taxonomy ) || ! $term_id ) { if ( '' === $taxonomy || ! taxonomy_exists( $taxonomy ) || ! $term_id ) {
wp_safe_redirect( $this->get_settings_page_url() ); wp_safe_redirect( $this->get_settings_page_url() );
@@ -423,6 +617,21 @@ class Groq_AI_Product_Text_Settings_Page {
if ( ! is_wp_error( $result ) ) { if ( ! is_wp_error( $result ) ) {
update_term_meta( $term_id, 'groq_ai_term_custom_prompt', $custom_prompt ); update_term_meta( $term_id, 'groq_ai_term_custom_prompt', $custom_prompt );
$settings = $this->plugin->get_settings();
$term = get_term( $term_id, $taxonomy );
if ( $term && ! is_wp_error( $term ) ) {
$bottom_meta_key = $this->resolve_term_bottom_description_meta_key( $term, $settings );
$effective_bottom_meta_key = '' !== $bottom_meta_key ? $bottom_meta_key : 'groq_ai_term_bottom_description';
update_term_meta( $term_id, $effective_bottom_meta_key, $bottom_description );
$rankmath_module_enabled = $this->plugin->is_module_enabled( 'rankmath', $settings );
if ( $rankmath_module_enabled ) {
$rankmath_keys = $this->resolve_rankmath_term_meta_keys( $term, $settings );
update_term_meta( $term_id, $rankmath_keys['title'], $rankmath_meta_title );
update_term_meta( $term_id, $rankmath_keys['description'], $rankmath_meta_description );
update_term_meta( $term_id, $rankmath_keys['focus_keyword'], $rankmath_focus_keywords );
}
}
} }
wp_safe_redirect( $this->get_term_page_url( $taxonomy, $term_id ) ); wp_safe_redirect( $this->get_term_page_url( $taxonomy, $term_id ) );
@@ -554,6 +763,14 @@ class Groq_AI_Product_Text_Settings_Page {
'groq_ai_product_text_prompts' 'groq_ai_product_text_prompts'
); );
add_settings_field(
'groq_ai_term_bottom_description_meta_key',
__( 'Term-veld (onderaan) meta key', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
[ $this, 'render_term_bottom_description_meta_key_field' ],
'groq-ai-product-text-prompts',
'groq_ai_product_text_prompts'
);
add_settings_field( add_settings_field(
'groq_ai_context_fields', 'groq_ai_context_fields',
__( 'Standaard productcontext', GROQ_AI_PRODUCT_TEXT_DOMAIN ), __( 'Standaard productcontext', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
@@ -562,6 +779,14 @@ class Groq_AI_Product_Text_Settings_Page {
'groq_ai_product_text_prompts' 'groq_ai_product_text_prompts'
); );
add_settings_field(
'groq_ai_product_attribute_includes',
__( 'Productattributen meesturen', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
[ $this, 'render_product_attribute_includes_field' ],
'groq-ai-product-text-prompts',
'groq_ai_product_text_prompts'
);
add_settings_field( add_settings_field(
'groq_ai_response_format_compat', 'groq_ai_response_format_compat',
__( 'Response-format compatibiliteit', GROQ_AI_PRODUCT_TEXT_DOMAIN ), __( 'Response-format compatibiliteit', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
@@ -1486,6 +1711,22 @@ class Groq_AI_Product_Text_Settings_Page {
<?php <?php
} }
public function render_term_bottom_description_meta_key_field() {
$settings = $this->plugin->get_settings();
$value = isset( $settings['term_bottom_description_meta_key'] ) ? (string) $settings['term_bottom_description_meta_key'] : '';
?>
<input type="text"
name="<?php echo esc_attr( $this->plugin->get_option_key() ); ?>[term_bottom_description_meta_key]"
value="<?php echo esc_attr( $value ); ?>"
class="regular-text"
placeholder="bijv. bottom_description"
/>
<p class="description">
<?php esc_html_e( 'Dit is de term meta key van het extra customfields-veld dat onderaan de categorie/merk pagina wordt getoond (LiveBetter customfields). Laat leeg om alleen de standaard term-omschrijving te gebruiken.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?>
</p>
<?php
}
public function render_context_fields_field() { public function render_context_fields_field() {
$settings = $this->plugin->get_settings(); $settings = $this->plugin->get_settings();
$values = isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields(); $values = isset( $settings['context_fields'] ) ? $settings['context_fields'] : $this->plugin->get_default_context_fields();
@@ -1493,6 +1734,9 @@ class Groq_AI_Product_Text_Settings_Page {
?> ?>
<div class="groq-ai-context-defaults"> <div class="groq-ai-context-defaults">
<?php foreach ( $definitions as $key => $definition ) : <?php foreach ( $definitions as $key => $definition ) :
if ( 'attributes' === $key ) {
continue;
}
$checked = ! empty( $values[ $key ] ); $checked = ! empty( $values[ $key ] );
?> ?>
<label> <label>
@@ -1509,6 +1753,74 @@ class Groq_AI_Product_Text_Settings_Page {
<?php <?php
} }
public function render_product_attribute_includes_field() {
$settings = $this->plugin->get_settings();
$values = isset( $settings['product_attribute_includes'] ) && is_array( $settings['product_attribute_includes'] )
? $settings['product_attribute_includes']
: [];
$values = array_values( array_unique( array_map( 'sanitize_key', $values ) ) );
$options = $this->get_product_attribute_include_options();
?>
<div class="groq-ai-attribute-includes">
<p class="description" style="margin-top:0;">
<?php esc_html_e( 'Selecteer welke productattributen je als context mee wilt sturen naar de AI. Als je niets selecteert, worden attributen niet meegestuurd (tenzij je dit eerder al had ingeschakeld via de oude instelling).', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?>
</p>
<?php if ( empty( $options ) ) : ?>
<p class="description">
<?php esc_html_e( 'Geen WooCommerce-attributen gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); ?>
</p>
<?php else : ?>
<div style="display:grid;grid-template-columns:repeat(auto-fit,minmax(260px,1fr));gap:8px;">
<?php foreach ( $options as $key => $label ) :
$checked = in_array( $key, $values, true );
?>
<label style="display:flex;gap:8px;align-items:flex-start;">
<input type="checkbox" name="<?php echo esc_attr( $this->plugin->get_option_key() ); ?>[product_attribute_includes][]" value="<?php echo esc_attr( $key ); ?>" <?php checked( $checked ); ?> />
<span><?php echo esc_html( $label ); ?></span>
</label>
<?php endforeach; ?>
</div>
<?php endif; ?>
</div>
<?php
}
private function get_product_attribute_include_options() {
$options = [
'__custom__' => __( 'Custom attributen (niet-taxonomie)', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
];
if ( function_exists( 'wc_get_attribute_taxonomies' ) ) {
$taxonomies = wc_get_attribute_taxonomies();
if ( is_array( $taxonomies ) ) {
foreach ( $taxonomies as $attr ) {
$name = isset( $attr->attribute_name ) ? sanitize_key( (string) $attr->attribute_name ) : '';
$label = isset( $attr->attribute_label ) ? sanitize_text_field( (string) $attr->attribute_label ) : '';
if ( '' === $name ) {
continue;
}
$taxonomy = 'pa_' . $name;
if ( '' === $label ) {
$label = function_exists( 'wc_attribute_label' ) ? wc_attribute_label( $taxonomy ) : $taxonomy;
}
$options[ $taxonomy ] = $label;
}
}
}
if ( count( $options ) > 1 ) {
$fixed = [
'__custom__' => $options['__custom__'],
];
unset( $options['__custom__'] );
asort( $options, SORT_NATURAL | SORT_FLAG_CASE );
$options = $fixed + $options;
}
return $options;
}
public function render_response_format_compat_field() { public function render_response_format_compat_field() {
$settings = $this->plugin->get_settings(); $settings = $this->plugin->get_settings();
$is_enabled = ! empty( $settings['response_format_compat'] ); $is_enabled = ! empty( $settings['response_format_compat'] );
@@ -1560,7 +1872,7 @@ class Groq_AI_Product_Text_Settings_Page {
type="number" type="number"
id="groq-ai-rankmath-keywords" id="groq-ai-rankmath-keywords"
min="1" min="1"
max="99" max="100"
name="<?php echo esc_attr( $this->plugin->get_option_key() ); ?>[modules][rankmath][focus_keyword_limit]" name="<?php echo esc_attr( $this->plugin->get_option_key() ); ?>[modules][rankmath][focus_keyword_limit]"
value="<?php echo esc_attr( $keyword_limit ); ?>" value="<?php echo esc_attr( $keyword_limit ); ?>"
style="width: 80px;" style="width: 80px;"
@@ -1664,6 +1976,37 @@ class Groq_AI_Product_Text_Settings_Page {
); );
} }
if ( 'settings_page_groq-ai-product-text-categories' === $hook ) {
wp_enqueue_script(
'groq-ai-category-bulk',
plugins_url( 'assets/js/category-bulk.js', GROQ_AI_PRODUCT_TEXT_FILE ),
[],
GROQ_AI_PRODUCT_TEXT_VERSION,
true
);
wp_localize_script(
'groq-ai-category-bulk',
'GroqAICategoryBulk',
[
'ajaxUrl' => admin_url( 'admin-ajax.php' ),
'nonce' => wp_create_nonce( 'groq_ai_bulk_generate_terms' ),
'taxonomy' => 'product_cat',
'terms' => $this->get_terms_without_description_payload( 'product_cat' ),
'strings' => [
'statusIdle' => __( 'Klik op de knop om voor alle lege categorieën automatisch teksten te genereren.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'statusProgress' => __( 'Bezig met categorie %1$s van %2$s: %3$s', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'statusDone' => __( 'Klaar! %d categorieën bijgewerkt.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'statusStopped' => __( 'Bulk generatie gestopt. %d categorieën bijgewerkt.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'statusEmpty' => __( 'Geen categorieën zonder omschrijving gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'logSuccess' => __( '%1$s gevuld (%2$d woorden).', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'logError' => __( '%1$s mislukt: %2$s', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'confirmStop' => __( 'Weet je zeker dat je wilt stoppen? De huidige categorie kan onafgemaakt blijven.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
],
]
);
}
$current_settings = $this->plugin->get_settings(); $current_settings = $this->plugin->get_settings();
$data = [ $data = [
'optionKey' => $this->plugin->get_option_key(), 'optionKey' => $this->plugin->get_option_key(),

View File

@@ -10,6 +10,7 @@ class Groq_AI_Ajax_Controller {
add_action( 'wp_ajax_groq_ai_generate_text', [ $this, 'handle_generate_text' ] ); add_action( 'wp_ajax_groq_ai_generate_text', [ $this, 'handle_generate_text' ] );
add_action( 'wp_ajax_groq_ai_refresh_models', [ $this, 'handle_refresh_models' ] ); add_action( 'wp_ajax_groq_ai_refresh_models', [ $this, 'handle_refresh_models' ] );
add_action( 'wp_ajax_groq_ai_generate_term_text', [ $this, 'handle_generate_term_text' ] ); add_action( 'wp_ajax_groq_ai_generate_term_text', [ $this, 'handle_generate_term_text' ] );
add_action( 'wp_ajax_groq_ai_bulk_generate_terms', [ $this, 'handle_bulk_generate_terms_request' ] );
} }
public function handle_generate_term_text() { public function handle_generate_term_text() {
@@ -35,6 +36,101 @@ class Groq_AI_Ajax_Controller {
wp_send_json_error( [ 'message' => __( 'Term niet gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ], 404 ); wp_send_json_error( [ 'message' => __( 'Term niet gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ], 404 );
} }
$result = $this->run_term_generation(
$term,
$prompt,
[
'include_top_products' => $include_top_products,
'top_products_limit' => $top_products_limit,
]
);
if ( is_wp_error( $result ) ) {
wp_send_json_error( [ 'message' => $result->get_error_message() ], 500 );
}
wp_send_json_success( $result );
}
public function handle_bulk_generate_terms_request() {
if ( ! current_user_can( 'manage_options' ) ) {
wp_send_json_error( [ 'message' => __( 'Je hebt geen toestemming voor deze actie.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ], 403 );
}
check_ajax_referer( 'groq_ai_bulk_generate_terms', 'nonce' );
$taxonomy = isset( $_POST['taxonomy'] ) ? sanitize_key( wp_unslash( $_POST['taxonomy'] ) ) : '';
$term_id = isset( $_POST['term_id'] ) ? absint( $_POST['term_id'] ) : 0;
$force = ! empty( $_POST['force'] );
if ( '' === $taxonomy || ! taxonomy_exists( $taxonomy ) || ! $term_id ) {
wp_send_json_error( [ 'message' => __( 'Taxonomie en term_id zijn verplicht.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ], 400 );
}
$term = get_term( $term_id, $taxonomy );
if ( ! $term || is_wp_error( $term ) ) {
wp_send_json_error( [ 'message' => __( 'Term niet gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ], 404 );
}
$current_description = isset( $term->description ) ? trim( wp_strip_all_tags( (string) $term->description ) ) : '';
if ( '' !== $current_description && ! $force ) {
wp_send_json_error(
[
'message' => __( 'Categorie heeft al een omschrijving.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'code' => 'groq_ai_term_has_description',
],
400
);
}
$options = apply_filters(
'groq_ai_bulk_term_generation_options',
[
'include_top_products' => true,
'top_products_limit' => 10,
],
$term
);
$result = $this->run_term_generation( $term, $this->get_term_prompt_text( $term ), $options );
if ( is_wp_error( $result ) ) {
wp_send_json_error( [ 'message' => $result->get_error_message() ], 500 );
}
$settings = $this->plugin->get_settings();
$saved = $this->save_term_generation_result( $term, $result, $settings );
if ( is_wp_error( $saved ) ) {
wp_send_json_error( [ 'message' => $saved->get_error_message() ], 500 );
}
wp_send_json_success(
[
'term_id' => $term_id,
'name' => isset( $term->name ) ? (string) $term->name : '',
'words' => isset( $saved['words'] ) ? absint( $saved['words'] ) : 0,
'count' => isset( $term->count ) ? absint( $term->count ) : 0,
]
);
}
private function run_term_generation( $term, $prompt, $options = [] ) {
if ( ! $term || ! is_object( $term ) ) {
return new WP_Error( 'groq_ai_invalid_term', __( 'Term niet gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) );
}
$options = wp_parse_args(
$options,
[
'include_top_products' => true,
'top_products_limit' => 10,
]
);
$include_top_products = ! empty( $options['include_top_products'] );
$top_products_limit = isset( $options['top_products_limit'] ) ? absint( $options['top_products_limit'] ) : 10;
$top_products_limit = max( 1, min( 25, $top_products_limit ) );
$settings = $this->plugin->get_settings(); $settings = $this->plugin->get_settings();
$provider_manager = $this->plugin->get_provider_manager(); $provider_manager = $this->plugin->get_provider_manager();
$provider_key = $settings['provider']; $provider_key = $settings['provider'];
@@ -51,7 +147,6 @@ class Groq_AI_Ajax_Controller {
? $prompt_builder->build_term_system_prompt( $settings, $conversation_id, $term ) ? $prompt_builder->build_term_system_prompt( $settings, $conversation_id, $term )
: $prompt_builder->build_system_prompt( $settings, $conversation_id ); : $prompt_builder->build_system_prompt( $settings, $conversation_id );
$model = $this->plugin->get_selected_model( $provider, $settings );
$context_block = ''; $context_block = '';
if ( method_exists( $prompt_builder, 'build_term_context_block' ) ) { if ( method_exists( $prompt_builder, 'build_term_context_block' ) ) {
$context_block = $prompt_builder->build_term_context_block( $context_block = $prompt_builder->build_term_context_block(
@@ -78,6 +173,7 @@ class Groq_AI_Ajax_Controller {
$final_prompt = $prompt_builder->append_response_instructions( $prompt_with_context, $settings ); $final_prompt = $prompt_builder->append_response_instructions( $prompt_with_context, $settings );
} }
$model = $this->plugin->get_selected_model( $provider, $settings );
$result = $provider->generate_content( $result = $provider->generate_content(
[ [
'prompt' => $final_prompt, 'prompt' => $final_prompt,
@@ -91,7 +187,7 @@ class Groq_AI_Ajax_Controller {
); );
if ( is_wp_error( $result ) ) { if ( is_wp_error( $result ) ) {
wp_send_json_error( [ 'message' => $result->get_error_message() ], 500 ); return $result;
} }
$response_text = $this->extract_content_text( $result ); $response_text = $this->extract_content_text( $result );
@@ -105,12 +201,127 @@ class Groq_AI_Ajax_Controller {
]; ];
} }
wp_send_json_success( return [
[ 'top_description' => isset( $parsed['top_description'] ) ? $parsed['top_description'] : ( isset( $parsed['description'] ) ? $parsed['description'] : '' ),
'description' => isset( $parsed['description'] ) ? $parsed['description'] : '', 'bottom_description' => isset( $parsed['bottom_description'] ) ? $parsed['bottom_description'] : '',
'meta_title' => isset( $parsed['meta_title'] ) ? $parsed['meta_title'] : '',
'meta_description' => isset( $parsed['meta_description'] ) ? $parsed['meta_description'] : '',
'focus_keywords' => isset( $parsed['focus_keywords'] ) ? $parsed['focus_keywords'] : '',
'description' => isset( $parsed['description'] ) ? $parsed['description'] : ( isset( $parsed['top_description'] ) ? $parsed['top_description'] : '' ),
'raw' => $response_text, 'raw' => $response_text,
];
}
private function save_term_generation_result( $term, $result, $settings ) {
$term_id = isset( $term->term_id ) ? absint( $term->term_id ) : 0;
$taxonomy = isset( $term->taxonomy ) ? sanitize_key( (string) $term->taxonomy ) : '';
if ( ! $term_id || '' === $taxonomy ) {
return new WP_Error( 'groq_ai_invalid_term', __( 'Term niet gevonden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) );
}
$top_description = '';
if ( isset( $result['top_description'] ) && '' !== trim( (string) $result['top_description'] ) ) {
$top_description = (string) $result['top_description'];
} elseif ( isset( $result['description'] ) ) {
$top_description = (string) $result['description'];
}
if ( '' === trim( wp_strip_all_tags( $top_description ) ) ) {
return new WP_Error( 'groq_ai_missing_description', __( 'De AI gaf geen omschrijving terug.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) );
}
$update = wp_update_term(
$term_id,
$taxonomy,
[
'description' => wp_kses_post( $top_description ),
] ]
); );
if ( is_wp_error( $update ) ) {
return $update;
}
$bottom_key = $this->get_bottom_meta_key( $term, $settings );
if ( '' !== $bottom_key ) {
$bottom_description = isset( $result['bottom_description'] ) ? (string) $result['bottom_description'] : '';
update_term_meta( $term_id, $bottom_key, wp_kses_post( $bottom_description ) );
}
if ( $this->plugin->is_module_enabled( 'rankmath', $settings ) ) {
$rankmath_keys = $this->get_rankmath_term_meta_keys( $term, $settings );
update_term_meta( $term_id, $rankmath_keys['title'], sanitize_text_field( isset( $result['meta_title'] ) ? $result['meta_title'] : '' ) );
update_term_meta( $term_id, $rankmath_keys['description'], sanitize_text_field( isset( $result['meta_description'] ) ? $result['meta_description'] : '' ) );
update_term_meta( $term_id, $rankmath_keys['focus_keyword'], sanitize_text_field( isset( $result['focus_keywords'] ) ? $result['focus_keywords'] : '' ) );
}
return [
'words' => $this->count_words( $top_description ),
];
}
private function get_bottom_meta_key( $term, $settings ) {
$default_key = '';
if ( is_array( $settings ) && isset( $settings['term_bottom_description_meta_key'] ) ) {
$default_key = sanitize_key( (string) $settings['term_bottom_description_meta_key'] );
}
$key = apply_filters( 'groq_ai_term_bottom_description_meta_key', $default_key, $term, $settings );
$key = sanitize_key( (string) $key );
return '' !== $key ? $key : 'groq_ai_term_bottom_description';
}
private function get_rankmath_term_meta_keys( $term, $settings ) {
$defaults = [
'title' => 'rank_math_title',
'description' => 'rank_math_description',
'focus_keyword' => 'rank_math_focus_keyword',
];
$keys = apply_filters( 'groq_ai_rankmath_term_meta_keys', $defaults, $term, $settings );
if ( ! is_array( $keys ) ) {
$keys = $defaults;
}
return [
'title' => isset( $keys['title'] ) ? sanitize_key( (string) $keys['title'] ) : 'rank_math_title',
'description' => isset( $keys['description'] ) ? sanitize_key( (string) $keys['description'] ) : 'rank_math_description',
'focus_keyword' => isset( $keys['focus_keyword'] ) ? sanitize_key( (string) $keys['focus_keyword'] ) : 'rank_math_focus_keyword',
];
}
private function get_term_prompt_text( $term ) {
$prompt = '';
if ( $term && isset( $term->term_id ) ) {
$prompt = (string) get_term_meta( $term->term_id, 'groq_ai_term_custom_prompt', true );
}
$prompt = trim( $prompt );
if ( '' !== $prompt ) {
return $prompt;
}
$default_prompt = __( 'Schrijf een SEO-vriendelijke categorieomschrijving in het Nederlands. Gebruik duidelijke tussenkoppen en <p>-tags. Voeg geen prijsinformatie toe.', GROQ_AI_PRODUCT_TEXT_DOMAIN );
return apply_filters( 'groq_ai_default_term_prompt', $default_prompt, $term );
}
private function count_words( $text ) {
$text = wp_strip_all_tags( (string) $text );
$text = trim( preg_replace( '/\s+/u', ' ', $text ) );
if ( '' === $text ) {
return 0;
}
if ( preg_match_all( '/\pL[\pL\pN\']*/u', $text, $matches ) ) {
return count( $matches[0] );
}
return str_word_count( $text );
} }
public function handle_generate_text() { public function handle_generate_text() {
@@ -138,6 +349,23 @@ class Groq_AI_Ajax_Controller {
$system_prompt = $prompt_builder->build_system_prompt( $settings, $conversation_id ); $system_prompt = $prompt_builder->build_system_prompt( $settings, $conversation_id );
$model = $this->plugin->get_selected_model( $provider, $settings ); $model = $this->plugin->get_selected_model( $provider, $settings );
$context_fields = $prompt_builder->parse_context_fields_from_request( isset( $_POST['context_fields'] ) ? $_POST['context_fields'] : '', $settings ); $context_fields = $prompt_builder->parse_context_fields_from_request( isset( $_POST['context_fields'] ) ? $_POST['context_fields'] : '', $settings );
if ( array_key_exists( 'attribute_includes', $_POST ) ) {
$attribute_includes = [];
$attribute_raw = (string) wp_unslash( $_POST['attribute_includes'] );
$decoded = json_decode( $attribute_raw, true );
if ( is_array( $decoded ) ) {
foreach ( $decoded as $value ) {
$key = sanitize_key( (string) $value );
if ( '' === $key ) {
continue;
}
if ( in_array( $key, [ '__custom__', '__all__' ], true ) || 0 === strpos( $key, 'pa_' ) ) {
$attribute_includes[] = $key;
}
}
}
$settings['product_attribute_includes'] = array_values( array_unique( $attribute_includes ) );
}
$image_context_mode = $this->plugin->get_image_context_mode( $settings ); $image_context_mode = $this->plugin->get_image_context_mode( $settings );
$image_context_limit = $this->plugin->get_image_context_limit( $settings ); $image_context_limit = $this->plugin->get_image_context_limit( $settings );
@@ -163,7 +391,7 @@ class Groq_AI_Ajax_Controller {
} }
} }
$product_context_text = $prompt_builder->build_product_context_block( $post_id, $context_fields, $prompt_image_mode, $image_context_limit ); $product_context_text = $prompt_builder->build_product_context_block( $post_id, $context_fields, $prompt_image_mode, $image_context_limit, $settings );
$image_context_payloads = []; $image_context_payloads = [];
if ( $use_base64_payloads ) { if ( $use_base64_payloads ) {
$image_context_payloads = $prompt_builder->get_product_image_payloads( $post_id, $image_context_limit ); $image_context_payloads = $prompt_builder->get_product_image_payloads( $post_id, $image_context_limit );

View File

@@ -338,7 +338,7 @@ class Groq_AI_Prompt_Builder {
return $normalized; return $normalized;
} }
public function build_product_context_block( $post_id, $fields, $image_mode = 'url', $image_limit = 3 ) { public function build_product_context_block( $post_id, $fields, $image_mode = 'url', $image_limit = 3, $settings = null ) {
$post_id = absint( $post_id ); $post_id = absint( $post_id );
if ( ! $post_id ) { if ( ! $post_id ) {
@@ -368,8 +368,14 @@ class Groq_AI_Prompt_Builder {
} }
} }
if ( ! empty( $fields['attributes'] ) ) { $attribute_includes = [];
$attributes = $this->get_product_attributes_text( $post_id ); if ( is_array( $settings ) && isset( $settings['product_attribute_includes'] ) && is_array( $settings['product_attribute_includes'] ) ) {
$attribute_includes = array_values( array_unique( array_map( 'sanitize_key', $settings['product_attribute_includes'] ) ) );
}
$include_attributes = ! empty( $attribute_includes ) || ! empty( $fields['attributes'] );
if ( $include_attributes ) {
$attributes = $this->get_product_attributes_text( $post_id, $attribute_includes );
if ( $attributes ) { if ( $attributes ) {
$parts[] = sprintf( __( 'Attributen: %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), $attributes ); $parts[] = sprintf( __( 'Attributen: %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), $attributes );
} }
@@ -382,9 +388,66 @@ class Groq_AI_Prompt_Builder {
} }
} }
if ( ! empty( $fields['brands'] ) ) {
$brands_context = $this->get_product_brand_context_text( $post_id );
if ( '' !== $brands_context ) {
$parts[] = sprintf( __( 'Merken: %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), $brands_context );
}
}
return implode( "\n\n", array_filter( $parts ) ); return implode( "\n\n", array_filter( $parts ) );
} }
private function get_product_brand_context_text( $post_id ) {
$post_id = absint( $post_id );
$taxonomy = $this->detect_brand_taxonomy();
if ( ! $post_id || '' === $taxonomy || ! taxonomy_exists( $taxonomy ) ) {
return '';
}
$terms = get_the_terms( $post_id, $taxonomy );
if ( empty( $terms ) || is_wp_error( $terms ) ) {
return '';
}
$entries = [];
foreach ( $terms as $term ) {
if ( ! $term || ! is_object( $term ) ) {
continue;
}
$name = isset( $term->name ) ? trim( wp_strip_all_tags( (string) $term->name ) ) : '';
if ( '' === $name ) {
continue;
}
$description = isset( $term->description ) ? trim( wp_strip_all_tags( (string) $term->description ) ) : '';
if ( '' !== $description ) {
$entries[] = sprintf( '%s - %s', $name, $description );
} else {
$entries[] = $name;
}
}
$entries = array_values( array_unique( array_filter( $entries ) ) );
if ( empty( $entries ) ) {
return '';
}
$context = implode( '; ', $entries );
/**
* Filters the product brand context string added to prompts.
*
* @param string $context
* @param int $post_id
* @param array $terms
* @param string $taxonomy
*/
return (string) apply_filters( 'groq_ai_product_brand_context', $context, $post_id, $terms, $taxonomy );
}
public function prepend_context_to_prompt( $prompt, $context ) { public function prepend_context_to_prompt( $prompt, $context ) {
$context = trim( (string) $context ); $context = trim( (string) $context );
@@ -424,6 +487,16 @@ class Groq_AI_Prompt_Builder {
$parts[] = sprintf( __( 'Huidige omschrijving: %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), wp_strip_all_tags( (string) $term->description ) ); $parts[] = sprintf( __( 'Huidige omschrijving: %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), wp_strip_all_tags( (string) $term->description ) );
} }
$bottom_meta_key = $this->resolve_term_bottom_description_meta_key( $term, $settings );
$bottom_meta_key = '' !== $bottom_meta_key ? $bottom_meta_key : 'groq_ai_term_bottom_description';
if ( '' !== $bottom_meta_key && $term_id ) {
$bottom = (string) get_term_meta( $term_id, $bottom_meta_key, true );
$bottom = trim( wp_strip_all_tags( $bottom ) );
if ( '' !== $bottom ) {
$parts[] = sprintf( __( 'Huidige omschrijving (onderaan): %s', GROQ_AI_PRODUCT_TEXT_DOMAIN ), $bottom );
}
}
if ( $include_top_products ) { if ( $include_top_products ) {
$top_products = $this->get_top_products_for_term( $taxonomy, $term_id, $top_products_limit ); $top_products = $this->get_top_products_for_term( $taxonomy, $term_id, $top_products_limit );
if ( ! empty( $top_products ) ) { if ( ! empty( $top_products ) ) {
@@ -466,9 +539,14 @@ class Groq_AI_Prompt_Builder {
$desc_pixels = $this->settings_manager->get_rankmath_meta_description_pixel_limit( $settings ); $desc_pixels = $this->settings_manager->get_rankmath_meta_description_pixel_limit( $settings );
$properties = [ $properties = [
'description' => [ 'top_description' => [
'type' => 'string', 'type' => 'string',
'description' => __( 'HTML-omschrijving voor de categorie/term met paragrafen en eventueel lijstjes.', GROQ_AI_PRODUCT_TEXT_DOMAIN ), 'description' => __( 'Korte HTML-omschrijving (1 alinea) voor de standaard WordPress term description. Exact één alinea in <p>-tags.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'minLength' => 20,
],
'bottom_description' => [
'type' => 'string',
'description' => __( 'Uitgebreide HTML-omschrijving (helemaal onderaan), 24 alineas, met paragrafen en eventueel lijstjes.', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'minLength' => 20, 'minLength' => 20,
], ],
]; ];
@@ -506,7 +584,7 @@ class Groq_AI_Prompt_Builder {
$schema = [ $schema = [
'type' => 'object', 'type' => 'object',
'properties' => $properties, 'properties' => $properties,
'required' => [ 'description' ], 'required' => [ 'top_description', 'bottom_description' ],
'additionalProperties' => false, 'additionalProperties' => false,
]; ];
@@ -549,14 +627,25 @@ class Groq_AI_Prompt_Builder {
]; ];
} }
$description = isset( $decoded['description'] ) ? trim( (string) $decoded['description'] ) : ''; $top = isset( $decoded['top_description'] ) ? trim( (string) $decoded['top_description'] ) : '';
if ( '' === $description ) { $bottom = isset( $decoded['bottom_description'] ) ? trim( (string) $decoded['bottom_description'] ) : '';
return new WP_Error( 'groq_ai_parse_error', __( 'De AI-respons bevatte geen description veld.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) ); // Backward compatibility: older prompts only returned `description`.
if ( '' === $top && isset( $decoded['description'] ) ) {
$top = trim( (string) $decoded['description'] );
}
if ( '' === $top && '' === $bottom ) {
return new WP_Error( 'groq_ai_parse_error', __( 'De AI-respons bevatte geen top_description/bottom_description velden.', GROQ_AI_PRODUCT_TEXT_DOMAIN ) );
} }
$result = [ $result = [];
'description' => $description, if ( '' !== $top ) {
]; $result['top_description'] = $top;
// For backwards compatibility with existing UI, keep `description` alias.
$result['description'] = $top;
}
if ( '' !== $bottom ) {
$result['bottom_description'] = $bottom;
}
if ( isset( $decoded['meta_title'] ) ) { if ( isset( $decoded['meta_title'] ) ) {
$result['meta_title'] = $this->truncate_meta_field( (string) $decoded['meta_title'], 60 ); $result['meta_title'] = $this->truncate_meta_field( (string) $decoded['meta_title'], 60 );
@@ -583,7 +672,8 @@ class Groq_AI_Prompt_Builder {
private function get_term_structured_response_instructions( $settings = null ) { private function get_term_structured_response_instructions( $settings = null ) {
$schema_parts = [ $schema_parts = [
'"description":"..."', '"top_description":"..."',
'"bottom_description":"..."',
]; ];
$rankmath_enabled = $this->settings_manager->is_module_enabled( 'rankmath', $settings ); $rankmath_enabled = $this->settings_manager->is_module_enabled( 'rankmath', $settings );
@@ -600,11 +690,21 @@ class Groq_AI_Prompt_Builder {
$json_structure $json_structure
); );
$instruction .= ' ' . __( 'Zorg dat description geldige HTML bevat (gebruik minimaal <p>-tags en waar relevant lijstjes of benadrukking). Voeg geen extra tekst buiten het JSON-object toe.', GROQ_AI_PRODUCT_TEXT_DOMAIN ); $instruction .= ' ' . __( 'Zorg dat top_description en bottom_description geldige HTML bevatten. top_description moet exact één alinea zijn in <p>-tags. bottom_description moet 24 alineas bevatten.', GROQ_AI_PRODUCT_TEXT_DOMAIN );
$instruction .= ' ' . __( 'Als in de context een sectie "Interne links" staat, verwerk dan 25 van deze links natuurlijk in de description als HTML-links (<a href="URL">Anker</a>).', GROQ_AI_PRODUCT_TEXT_DOMAIN ); $instruction .= ' ' . __( 'Voeg geen extra tekst buiten het JSON-object toe.', GROQ_AI_PRODUCT_TEXT_DOMAIN );
$instruction .= ' ' . __( 'Als in de context een sectie "Interne links" staat, verwerk dan 25 van deze links natuurlijk in bottom_description als HTML-links (<a href="URL">Anker</a>).', GROQ_AI_PRODUCT_TEXT_DOMAIN );
return $instruction; return $instruction;
} }
private function resolve_term_bottom_description_meta_key( $term = null, $settings = null ) {
$default_key = '';
if ( is_array( $settings ) && isset( $settings['term_bottom_description_meta_key'] ) ) {
$default_key = sanitize_key( (string) $settings['term_bottom_description_meta_key'] );
}
$key = apply_filters( 'groq_ai_term_bottom_description_meta_key', $default_key, $term, $settings );
return sanitize_key( (string) $key );
}
private function get_top_products_for_term( $taxonomy, $term_id, $limit = 10 ) { private function get_top_products_for_term( $taxonomy, $term_id, $limit = 10 ) {
$taxonomy = sanitize_key( (string) $taxonomy ); $taxonomy = sanitize_key( (string) $taxonomy );
$term_id = absint( $term_id ); $term_id = absint( $term_id );
@@ -796,7 +896,7 @@ class Groq_AI_Prompt_Builder {
return substr( $text, 0, $limit ); return substr( $text, 0, $limit );
} }
private function get_product_attributes_text( $post_id ) { private function get_product_attributes_text( $post_id, $attribute_includes = [] ) {
if ( ! function_exists( 'wc_get_product' ) ) { if ( ! function_exists( 'wc_get_product' ) ) {
return ''; return '';
} }
@@ -813,14 +913,27 @@ class Groq_AI_Prompt_Builder {
return ''; return '';
} }
$attribute_includes = is_array( $attribute_includes ) ? array_values( array_unique( array_map( 'sanitize_key', $attribute_includes ) ) ) : [];
$include_all = empty( $attribute_includes ) || in_array( '__all__', $attribute_includes, true );
$include_custom = $include_all || in_array( '__custom__', $attribute_includes, true );
$lines = []; $lines = [];
foreach ( $attributes as $attribute ) { foreach ( $attributes as $attribute ) {
if ( $attribute->is_taxonomy() ) { if ( $attribute->is_taxonomy() ) {
$terms = wc_get_product_terms( $post_id, $attribute->get_name(), [ 'fields' => 'names' ] ); $taxonomy_name = sanitize_key( (string) $attribute->get_name() );
if ( ! $include_all && ! in_array( $taxonomy_name, $attribute_includes, true ) ) {
continue;
}
$terms = wc_get_product_terms( $post_id, $taxonomy_name, [ 'fields' => 'names' ] );
$value = implode( ', ', array_map( 'sanitize_text_field', (array) $terms ) ); $value = implode( ', ', array_map( 'sanitize_text_field', (array) $terms ) );
$label = wc_attribute_label( $attribute->get_name() ); $label = wc_attribute_label( $taxonomy_name );
} else { } else {
if ( ! $include_custom ) {
continue;
}
$options = $attribute->get_options(); $options = $attribute->get_options();
$value = implode( ', ', array_map( 'sanitize_text_field', (array) $options ) ); $value = implode( ', ', array_map( 'sanitize_text_field', (array) $options ) );
$label = sanitize_text_field( $attribute->get_name() ); $label = sanitize_text_field( $attribute->get_name() );

View File

@@ -33,6 +33,8 @@ class Groq_AI_Settings_Manager {
'store_context' => '', 'store_context' => '',
'default_prompt' => '', 'default_prompt' => '',
'max_output_tokens' => 2048, 'max_output_tokens' => 2048,
'product_attribute_includes' => [],
'term_bottom_description_meta_key' => '',
'groq_api_key' => '', 'groq_api_key' => '',
'openai_api_key' => '', 'openai_api_key' => '',
'google_api_key' => '', 'google_api_key' => '',
@@ -73,6 +75,10 @@ class Groq_AI_Settings_Manager {
$limit = isset( $settings['image_context_limit'] ) ? $this->sanitize_image_context_limit_value( $settings['image_context_limit'] ) : 3; $limit = isset( $settings['image_context_limit'] ) ? $this->sanitize_image_context_limit_value( $settings['image_context_limit'] ) : 3;
$settings['image_context_limit'] = $limit; $settings['image_context_limit'] = $limit;
$settings['product_attribute_includes'] = $this->sanitize_product_attribute_includes(
isset( $settings['product_attribute_includes'] ) ? $settings['product_attribute_includes'] : []
);
return $settings; return $settings;
} }
@@ -89,6 +95,8 @@ class Groq_AI_Settings_Manager {
'store_context' => '', 'store_context' => '',
'default_prompt' => '', 'default_prompt' => '',
'max_output_tokens' => 2048, 'max_output_tokens' => 2048,
'product_attribute_includes' => [],
'term_bottom_description_meta_key' => '',
'groq_api_key' => '', 'groq_api_key' => '',
'openai_api_key' => '', 'openai_api_key' => '',
'google_api_key' => '', 'google_api_key' => '',
@@ -149,6 +157,8 @@ class Groq_AI_Settings_Manager {
'store_context' => sanitize_textarea_field( $input['store_context'] ), 'store_context' => sanitize_textarea_field( $input['store_context'] ),
'default_prompt' => sanitize_textarea_field( $input['default_prompt'] ), 'default_prompt' => sanitize_textarea_field( $input['default_prompt'] ),
'max_output_tokens' => $max_output_tokens, 'max_output_tokens' => $max_output_tokens,
'product_attribute_includes' => $this->sanitize_product_attribute_includes( isset( $raw_input['product_attribute_includes'] ) ? $raw_input['product_attribute_includes'] : [] ),
'term_bottom_description_meta_key' => sanitize_key( (string) $input['term_bottom_description_meta_key'] ),
'groq_api_key' => sanitize_text_field( $input['groq_api_key'] ), 'groq_api_key' => sanitize_text_field( $input['groq_api_key'] ),
'openai_api_key' => sanitize_text_field( $input['openai_api_key'] ), 'openai_api_key' => sanitize_text_field( $input['openai_api_key'] ),
'google_api_key' => sanitize_text_field( $input['google_api_key'] ), 'google_api_key' => sanitize_text_field( $input['google_api_key'] ),
@@ -174,6 +184,33 @@ class Groq_AI_Settings_Manager {
]; ];
} }
private function sanitize_product_attribute_includes( $value ) {
if ( ! is_array( $value ) ) {
return [];
}
$clean = [];
foreach ( $value as $item ) {
$item = sanitize_key( (string) $item );
if ( '' === $item ) {
continue;
}
// Allow special tokens and attribute taxonomies.
if ( in_array( $item, [ '__all__', '__custom__' ], true ) || 0 === strpos( $item, 'pa_' ) ) {
$clean[] = $item;
}
}
$clean = array_values( array_unique( $clean ) );
// Hard cap to avoid overly large option payloads.
if ( count( $clean ) > 200 ) {
$clean = array_slice( $clean, 0, 200 );
}
return $clean;
}
public function get_context_field_definitions() { public function get_context_field_definitions() {
if ( null === $this->context_field_definitions ) { if ( null === $this->context_field_definitions ) {
$this->context_field_definitions = [ $this->context_field_definitions = [
@@ -197,6 +234,11 @@ class Groq_AI_Settings_Manager {
'description' => __( 'Voeg gestructureerde productattributen toe (zoals kleur, maat, materiaal).', GROQ_AI_PRODUCT_TEXT_DOMAIN ), 'description' => __( 'Voeg gestructureerde productattributen toe (zoals kleur, maat, materiaal).', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'default' => false, 'default' => false,
], ],
'brands' => [
'label' => __( 'Merken', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'description' => __( 'Voegt gekoppelde productmerken toe (detecteert WooCommerce merk-taxonomieën).', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'default' => true,
],
'images' => [ 'images' => [
'label' => __( 'Afbeeldingen', GROQ_AI_PRODUCT_TEXT_DOMAIN ), 'label' => __( 'Afbeeldingen', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
'description' => __( 'Voeg een korte lijst toe met productafbeeldingen (beschrijving + URL).', GROQ_AI_PRODUCT_TEXT_DOMAIN ), 'description' => __( 'Voeg een korte lijst toe met productafbeeldingen (beschrijving + URL).', GROQ_AI_PRODUCT_TEXT_DOMAIN ),
@@ -224,7 +266,7 @@ class Groq_AI_Settings_Manager {
$normalized = []; $normalized = [];
foreach ( $definitions as $key => $data ) { foreach ( $definitions as $key => $data ) {
$normalized[ $key ] = false; $normalized[ $key ] = ! empty( $data['default'] );
} }
if ( ! is_array( $fields ) ) { if ( ! is_array( $fields ) ) {
@@ -282,7 +324,7 @@ class Groq_AI_Settings_Manager {
$config = $this->get_module_config( 'rankmath', $settings ); $config = $this->get_module_config( 'rankmath', $settings );
$limit = isset( $config['focus_keyword_limit'] ) ? absint( $config['focus_keyword_limit'] ) : 3; $limit = isset( $config['focus_keyword_limit'] ) ? absint( $config['focus_keyword_limit'] ) : 3;
return max( 1, min( 10, $limit ) ); return max( 1, min( 100, $limit ) );
} }
public function get_rankmath_meta_title_pixel_limit( $settings = null ) { public function get_rankmath_meta_title_pixel_limit( $settings = null ) {
@@ -375,7 +417,7 @@ class Groq_AI_Settings_Manager {
if ( $limit <= 0 ) { if ( $limit <= 0 ) {
$limit = $module_default_config['focus_keyword_limit']; $limit = $module_default_config['focus_keyword_limit'];
} }
$result[ $module_key ]['focus_keyword_limit'] = max( 1, min( 10, $limit ) ); $result[ $module_key ]['focus_keyword_limit'] = max( 1, min( 100, $limit ) );
$title_pixel_limit = isset( $raw['meta_title_pixel_limit'] ) ? absint( $raw['meta_title_pixel_limit'] ) : ( isset( $current_config['meta_title_pixel_limit'] ) ? absint( $current_config['meta_title_pixel_limit'] ) : $module_default_config['meta_title_pixel_limit'] ); $title_pixel_limit = isset( $raw['meta_title_pixel_limit'] ) ? absint( $raw['meta_title_pixel_limit'] ) : ( isset( $current_config['meta_title_pixel_limit'] ) ? absint( $current_config['meta_title_pixel_limit'] ) : $module_default_config['meta_title_pixel_limit'] );
if ( $title_pixel_limit <= 0 ) { if ( $title_pixel_limit <= 0 ) {