feat: Add Google Taxonomy integration and product editing enhancements

- Implemented Google Taxonomy loading via AJAX in main_view.php, allowing users to select categories for products.
- Enhanced product editing modal to include fields for product title, description, and Google category selection.
- Updated AJAX calls to save product data, including custom title, description, and selected Google category.
- Added character count validation for product title input.
- Integrated Select2 for improved category selection UI.
- Created google-taxonomy.php to fetch and cache Google Taxonomy data, ensuring efficient retrieval and fallback mechanisms.
- Removed outdated custom feed XML file.
- Updated layout-logged.php to include necessary Select2 styles and scripts.
This commit is contained in:
2025-11-20 23:46:21 +01:00
parent 92ce677f7c
commit f5db5263ab
11 changed files with 1424 additions and 38 deletions

View File

@@ -41,8 +41,8 @@
},
"class.Products.php": {
"type": "-",
"size": 9685,
"lmtime": 1755764898292,
"size": 11693,
"lmtime": 1760996915654,
"modified": false
},
"class.Site.php": {
@@ -73,8 +73,8 @@
},
"class.Products.php": {
"type": "-",
"size": 5233,
"lmtime": 1756241352457,
"size": 6067,
"lmtime": 1760996968673,
"modified": false
},
"class.Users.php": {

View File

@@ -305,6 +305,13 @@ class Cron
$results = $mdb -> query( 'SELECT * FROM products AS p INNER JOIN products_data AS pd ON p.id = pd.product_id WHERE p.client_id = ' . $client_id ) -> fetchAll( \PDO::FETCH_ASSOC );
// if empty results
if ( empty( $results ) )
{
echo json_encode( [ 'result' => "Brak produktów do wygenerowania pliku XML." ] );
exit;
}
$doc = new \DOMDocument('1.0', 'UTF-8');
$xmlRoot = $doc->createElement('rss');
$xmlRoot = $doc->appendChild($xmlRoot);
@@ -317,8 +324,10 @@ class Cron
$fieldMappings = [
'title' => 'g:title',
'description' => 'g:description',
'custom_label_4' => 'g:custom_label_4',
'custom_label_3' => 'g:custom_label_3',
'google_product_category' => 'g:google_product_category'
];
foreach ($results as $row)

View File

@@ -51,6 +51,16 @@ class Products
exit;
}
static public function get_product_data() {
$product_id = \S::get( 'product_id' );
$product_title = \factory\Products::get_product_data( $product_id, 'title' );
$product_description = \factory\Products::get_product_data( $product_id, 'description' );
echo json_encode( [ 'status' => 'ok', 'product_details' => [ 'title' => $product_title, 'description' => $product_description ] ] );
exit;
}
static public function get_products()
{
@@ -320,18 +330,25 @@ class Products
exit;
}
static public function save_custom_title()
static public function save_product_data()
{
$product_id = \S::get( 'product_id' );
$custom_title = \S::get( 'custom_title' );
$custom_description = \S::get( 'custom_description' );
$google_product_category = \S::get( 'google_product_category' );
if ( \factory\Products::set_product_data( $product_id, 'title', $custom_title ) )
{
\factory\Products::add_product_comment( $product_id, 'Zmiana nazwy produktu na: ' . $custom_title );
echo json_encode( [ 'status' => 'ok' ] );
}
else
echo json_encode( [ 'status' => 'error' ] );
if ( $product_id and $custom_title )
\factory\Products::set_product_data( $product_id, 'title', $custom_title );
if ( $product_id and $custom_description )
\factory\Products::set_product_data( $product_id, 'description', $custom_description );
if ( $product_id and $google_product_category )
\factory\Products::set_product_data( $product_id, 'google_product_category', $google_product_category );
\factory\Products::add_product_comment( $product_id, 'Zmiana tytułu i opisu produktu.' );
echo json_encode( [ 'status' => 'ok' ] );
exit;
}
}

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@@ -1335,4 +1335,75 @@ table {
.comment-form .hint {
font-size: 12px;
color: #666;
}
/* === Select2 w modalu "Edytuj produkt" === */
/* pełna szerokość i taki sam odstęp jak inne pola */
.jconfirm-box .form-group .select2-container {
width: 100% !important;
margin-top: 8px;
}
/* wygląd "inputa" */
.jconfirm-box .select2-container--default .select2-selection--single {
background-color: #fff;
border: 1px solid #ced4da;
/* jak bootstrapowe inputy */
border-radius: 3px;
min-height: 42px;
/* wysokość podobna do pola tytułu/opisu */
display: flex;
align-items: center;
padding: 4px 12px;
box-shadow: none;
transition: border-color .15s ease-in-out, box-shadow .15s ease-in-out;
font-size: 14px;
}
/* tekst w środku */
.jconfirm-box .select2-container--default .select2-selection--single .select2-selection__rendered {
padding-left: 0;
line-height: 1.4;
color: #495057;
}
/* placeholder */
.jconfirm-box .select2-container--default .select2-selection--single .select2-selection__placeholder {
color: #adb5bd;
}
/* strzałka po prawej wyrównanie */
.jconfirm-box .select2-container--default .select2-selection--single .select2-selection__arrow {
height: 100%;
right: 8px;
}
/* efekt hover/focus jak na form-control */
.jconfirm-box .select2-container--default.select2-container--focus .select2-selection--single,
.jconfirm-box .select2-container--default .select2-selection--single:hover {
border-color: #80bdff;
box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, .25);
outline: 0;
}
/* dropdown (lista kategorii) */
.jconfirm-box .select2-container .select2-dropdown {
border-color: #ced4da;
border-radius: 0 0 3px 3px;
font-size: 14px;
}
/* pole wyszukiwania w dropdownie */
.jconfirm-box .select2-container .select2-search--dropdown .select2-search__field {
padding: 6px 10px;
border-radius: 3px;
border: 1px solid #ced4da;
font-size: 14px;
}
/* podświetlenie zaznaczonej pozycji */
.jconfirm-box .select2-container--default .select2-results__option--highlighted[aria-selected] {
background-color: #007bff;
color: #fff;
}

File diff suppressed because one or more lines are too long

View File

@@ -56,6 +56,38 @@
</div>
</div>
<script type="text/javascript">
var GOOGLE_TAXONOMY_ENDPOINT = '/tools/google-taxonomy.php'; // dostosuj ścieżkę
var googleCategories = [];
// główna funkcja używaj jej w modalu
function loadGoogleCategories(callback) {
if (googleCategories.length) {
// już załadowane wcześniej
callback(googleCategories);
return;
}
$.ajax({
url: GOOGLE_TAXONOMY_ENDPOINT,
dataType: 'json',
success: function (res) {
if (res.status === 'ok') {
googleCategories = res.categories || [];
callback(googleCategories);
} else {
console.error('Błąd pobierania taksonomii:', res);
callback([]);
}
},
error: function (xhr, status, error) {
console.error('AJAX error przy pobieraniu taksonomii:', status, error);
callback([]);
}
});
}
$( function()
{
$( 'body' ).on( 'change', '#client_id', function()
@@ -161,13 +193,21 @@
$( 'body' ).on( 'click', '.edit-product-title', function(e)
{
$.confirm({
title: 'Edytuj tytuł',
title: 'Edytuj produkt',
content: '' +
'<form action="" class="formName">' +
'<div class="form-group">' +
'<input type="text" value="' + escapeHtml( $( this ).siblings( 'a' ).text().trim() ) + '" product_id="' + $( this ).attr( 'product_id' ) + '" class="name form-control" required />' +
'<input type="text" value="" product_id="' + $( this ).attr( 'product_id' ) + '" placeholder="Tytuł produktu" class="name form-control" required />' +
'<small>0/150 znaków</small>' +
'</div>' +
'<div class="form-group">' +
'<textarea class="form-control description" rows="4" placeholder="Opis produktu (opcjonalnie)"></textarea>' +
'</div>' +
'<div class="form-group">' +
' <select class="form-control google-category" id="google_category" style="width: 100%">' +
' <option value="">— wybierz kategorię —</option>' +
' </select>' +
'</div>' +
'</form>',
columnClass: 'col-md-8 col-md-offset-2 col-12',
theme: 'modern',
@@ -180,6 +220,7 @@
var jc = this;
var product_id = this.$content.find( '.name' ).attr( 'product_id' );
var customTitle = this.$content.find('.name').val();
var googleProductCategory = this.$content.find('.google-category').val(); // NOWOŚĆ
if ( !customTitle )
{
@@ -196,11 +237,13 @@
jc.showLoading(true);
$.ajax({
url: '/products/save_custom_title/',
url: '/products/save_product_data/',
type: 'POST',
data: {
product_id: product_id,
custom_title: customTitle
custom_title: customTitle,
custom_description: this.$content.find('.description').val(),
google_product_category: googleProductCategory // NOWOŚĆ zapis
},
success: function(response) {
data = JSON.parse(response);
@@ -208,7 +251,7 @@
if ( data.status == 'ok' )
{
$.alert( 'Tytuł został pomyślnie zapisany' );
$.alert( 'Dane produktu zostały zapisane.' );
jc.close();
}
else
@@ -231,26 +274,80 @@
},
},
onContentReady: function () {
var jc = this;
var jc = this;
var inputField = this.$content.find('.name');
var charCount = this.$content.find('small');
var $form = this.$content.find('form');
var $inputField = this.$content.find('.name');
var $charCount = this.$content.find('small').first();
var $description = this.$content.find('.description');
var $googleCategory = this.$content.find('.google-category');
inputField.on('input', function() {
var currentLength = $(this).val().length;
charCount.text(currentLength + '/150 znaków');
// 1) Pobierz dane produktu
var product_id = $inputField.attr('product_id');
if (currentLength > 150) {
$(this).addClass('is-invalid');
} else {
$(this).removeClass('is-invalid');
}
});
$.ajax({
url: '/products/get_product_data/',
type: 'POST',
data: { product_id: product_id },
success: function (response) {
var data = JSON.parse(response);
this.$content.find('form').on('submit', function (e) {
e.preventDefault();
jc.$$formSubmit.trigger('click');
});
if (data.status == 'ok') {
if (data.product_details.title) {
$inputField.val(data.product_details.title);
var currentLength = data.product_details.title.length;
$charCount.text(currentLength + '/150 znaków');
}
if (data.product_details.description) {
$description.val(data.product_details.description);
}
// zapamiętujemy ID kategorii z backendu
jc.preselectedGoogleCategory = data.product_details.google_product_category || "";
} else {
$.alert('Błąd: ' + response);
}
}.bind(this),
error: function () {
$.alert('Wystąpił błąd podczas pobierania szczegółów produktu. Spróbuj ponownie.');
}
});
// 2) Pobierz kategorie z Twojego endpointu (bez CORS) i zainicjuj Select2
loadGoogleCategories(function (cats) {
if (typeof $.fn.select2 !== 'undefined') {
$googleCategory.select2({
placeholder: 'Wpisz fragment nazwy kategorii...',
allowClear: true,
data: cats,
dropdownParent: jc.$content.closest('.jconfirm-box')
});
if (jc.preselectedGoogleCategory) {
$googleCategory.val(jc.preselectedGoogleCategory).trigger('change');
}
} else {
console.warn('Select2 nie jest załadowany pole kategorii nie będzie miało wyszukiwarki.');
}
});
// licznik znaków
$inputField.on('input', function () {
var currentLength = $(this).val().length;
$charCount.text(currentLength + '/150 znaków');
if (currentLength > 150) {
$(this).addClass('is-invalid');
} else {
$(this).removeClass('is-invalid');
}
});
$form.on('submit', function (e) {
e.preventDefault();
jc.$$formSubmit.trigger('click');
});
}
});
});

View File

@@ -26,6 +26,7 @@
<script src="/libraries/select2/js/select2.full.min.js"></script>
<script src="/libraries/functions.js"></script>
<link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/2.1.7/css/dataTables.bootstrap5.min.css">
<link href="https://cdn.jsdelivr.net/npm/select2@4.1.0-rc.0/dist/css/select2.min.css" rel="stylesheet" />
<link rel="Stylesheet" type="text/css" href="/libraries/framework/skin/default_skin/css/theme.css">
<link rel="stylesheet" type="text/css" href="/libraries/framework/vendor/plugins/datepicker/css/bootstrap-datetimepicker.css">
<link rel="Stylesheet" type="text/css" href="/libraries/framework/vendor/plugins/daterange/daterangepicker.css">
@@ -34,6 +35,7 @@
<link rel="stylesheet" type="text/css" href="/libraries/select2/css/select2.min.css">
<link rel="stylesheet" type="text/css" href="/libraries/select2/css/select2-bootstrap-5-theme.min.css">
<link rel="stylesheet" type="text/css" href="/layout/style.css">
<script src="https://cdn.jsdelivr.net/npm/select2@4.1.0-rc.0/dist/js/select2.min.js"></script>
</head>
<body class="logged">
<div class="top">

69
tools/google-taxonomy.php Normal file
View File

@@ -0,0 +1,69 @@
<?php
// tools/google-taxonomy.php
header('Content-Type: application/json; charset=utf-8');
const TAXONOMY_SOURCE_URL = 'https://www.google.com/basepages/producttype/taxonomy-with-ids.pl-PL.txt';
const TAXONOMY_CACHE_FILE = __DIR__ . '/cache/google-taxonomy-pl.json'; // zmień jak trzeba
const TAXONOMY_CACHE_TTL = 7 * 24 * 60 * 60; // 7 dni
// jeśli jest świeży cache zwracamy go od razu
if (file_exists(TAXONOMY_CACHE_FILE) && (time() - filemtime(TAXONOMY_CACHE_FILE) < TAXONOMY_CACHE_TTL)) {
readfile(TAXONOMY_CACHE_FILE);
exit;
}
// próba pobrania z Google
$txt = @file_get_contents(TAXONOMY_SOURCE_URL);
if ($txt === false) {
// jeśli nie ma cache i nie udało się pobrać błąd
if (!file_exists(TAXONOMY_CACHE_FILE)) {
http_response_code(500);
echo json_encode([
'status' => 'error',
'message' => 'Nie udało się pobrać taksonomii Google.'
], JSON_UNESCAPED_UNICODE);
exit;
}
// fallback stary cache (lepsze to niż nic)
readfile(TAXONOMY_CACHE_FILE);
exit;
}
// parsowanie TXT -> tablica {id, text}
$list = [];
$lines = explode("\n", $txt);
foreach ($lines as $line) {
$line = trim($line);
if ($line === '') {
continue;
}
// format linii: 2271 - Health & Beauty > Personal Care > Cosmetics
$parts = explode(' - ', $line, 2);
if (count($parts) === 2) {
$list[] = [
'id' => trim($parts[0]),
'text' => trim($parts[1]),
];
}
}
$response = [
'status' => 'ok',
'categories' => $list,
];
// zapisujemy cache na serwerze
$dir = dirname(TAXONOMY_CACHE_FILE);
if (!is_dir($dir)) {
@mkdir($dir, 0775, true);
}
file_put_contents(TAXONOMY_CACHE_FILE, json_encode($response, JSON_UNESCAPED_UNICODE));
// i zwracamy do frontu
echo json_encode($response, JSON_UNESCAPED_UNICODE);

View File

@@ -1,2 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:g="http://base.google.com/ns/1.0" version="2.0"><channel><title>Custom Feed</title><link>https://ads.pagedev.pl</link><item><g:id>shopify_pl_9049561235791_48689589289295</g:id><g:custom_label_4>pla</g:custom_label_4></item><item><g:id>shopify_pl_9615160082767_49821822353743</g:id><g:custom_label_4>pla</g:custom_label_4></item><item><g:id>shopify_pl_9458847154511_49357978468687</g:id><g:custom_label_4>pla</g:custom_label_4></item></channel></rss>