isDir() || $template_file->getExtension() !== 'php') {
continue;
}
// Получаем относительный путь (например, "pages/template-landing.php")
$relative_path = str_replace($custom_templates_dir, '', $template_file->getPathname());
// Формируем имя шаблона (убираем .php и заменяем слэши на дефисы)
$template_name = str_replace(['/', '.php'], [' - ', ''], $relative_path);
// Добавляем в список шаблонов
$templates['templates/' . $relative_path] = $template_name;
}
return $templates;
});
includeFilesFromFolder(get_template_directory() . '/global-functions');
// Add the function to the Timber context
add_filter('timber/context', function($context) {
$context['template_path'] = get_template_directory_uri();
$current_url = "http://" . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI'];
$context['current_url'] = htmlspecialchars($current_url);
return $context;
});
/**
* Подключает статические CSS и JS файлы из указанной директории
*
* @param string $dir_name Название директории в папке /static/
* @param array $options Дополнительные параметры:
* - 'css' => false - отключить подключение CSS
* - 'js' => false - отключить подключение JS
* - 'version' => string - версия файлов (по умолчанию использует время модификации файла)
*/
function enqueue_static_assets($dir_name, $options = []) {
// Устанавливаем пути к директориям
$static_dir = get_template_directory() . '/static/' . $dir_name;
$static_uri = get_template_directory_uri() . '/static/' . $dir_name;
// Подключаем CSS файлы
if (!isset($options['css']) || $options['css'] !== false) {
$css_dir = $static_dir . '/css/';
if (file_exists($css_dir)) {
$css_files = scandir($css_dir);
foreach ($css_files as $file) {
if (pathinfo($file, PATHINFO_EXTENSION) === 'css') {
$handle = $dir_name . '-' . pathinfo($file, PATHINFO_FILENAME);
$src = $static_uri . '/css/' . $file;
$ver = isset($options['version']) ? $options['version'] : filemtime($css_dir . $file);
wp_enqueue_style(
$handle,
$src,
array(),
$ver
);
}
}
}
}
// Подключаем JS файлы
if (!isset($options['js']) || $options['js'] !== false) {
$js_dir = $static_dir . '/js/';
if (file_exists($js_dir)) {
$js_files = scandir($js_dir);
foreach ($js_files as $file) {
if (pathinfo($file, PATHINFO_EXTENSION) === 'js') {
$handle = $dir_name . '-' . pathinfo($file, PATHINFO_FILENAME);
$src = $static_uri . '/js/' . $file;
$ver = isset($options['version']) ? $options['version'] : filemtime($js_dir . $file);
$in_footer = strpos($file, 'admin') === 0 ? false : true;
wp_enqueue_script(
$handle,
$src,
array(),
$ver,
$in_footer
);
}
}
}
}
}
add_action('init', function() {
if (function_exists('wpseo_load_textdomain')) {
wpseo_load_textdomain();
}
if (function_exists('pll_load_textdomain')) {
pll_load_textdomain();
}
add_filter('timber/context', function($context) {
$context['current_lang'] = pll_current_language();
return $context;
});
}, 1);
$modules_path = get_template_directory() . '/modules/*/editor-blocks/*/editor-block-controller.php';
foreach (glob($modules_path) as $file) {
require_once $file; // Подключаем каждый найденный файл
}
requireShortcodes(get_template_directory() . '/modules');
require_once('modules/blog/module-ajax-controller.php');
require_once('modules/forms/module-ajax-controller.php');
require_once('modules/author/module-ajax-controller.php');
include_module('forms');
include_module('layout');
class WooProduct extends Timber\Post {
protected $wc_product;
public function __construct ($pid = null) {
parent::__construct($pid);
$this->wc_product = wc_get_product($this->ID);
}
public function price() {
return $this->wc_product->get_price();
}
public function get_price_html() {
return $this->wc_product->get_price_html();
}
public function get_attr() {
return $this->wc_product->get_attribute('pa_compound');
}
public function get_test() {
return 'test';
}
};
add_filter('timber/post/classmap', function ($classmap) {
$custom_classmap = [
'product' => WooProduct::class,
];
return array_merge($classmap, $custom_classmap);
});
//Ajax подгрузка товаров в архиве
add_action( 'wp_ajax_nopriv_get_products', 'get_products' );
add_action( 'wp_ajax_get_products', 'get_products' );
function get_products() {
global $post;
if (function_exists('WC')) {
WC();
}
// Потом Timber
$context = Timber::context();
$context['get_page'] = empty($_POST['get_page']) ? 1 : $_POST['get_page'];
$context['criteria_for_new_product'] = date('Y-m-d', strtotime('-30 days'));
$context['get_category'] = isset($_POST['get_category']) ? $_POST['get_category'] : NULL;
$context['get_category_type'] = isset($_POST['get_category_type']) ? $_POST['get_category_type'] : NULL;
// Задаем количество постов для подзагрузки Ajax
$posts_per_page = 1;
$args = array(
'post_type' => 'product',
'post_status' => 'publish',
'posts_per_page' => $posts_per_page,
'paged' => $context['get_page'],
'has_password' => FALSE
);
$count_args = array(
'post_type' => 'product',
'post_status' => 'publish',
'posts_per_page' => -1,
'has_password' => FALSE
);
if ($context['get_category'] != NULL) {
$categories = [
'tax_query' => array(
array(
'taxonomy' => $context['get_category_type'],
'field' => 'term_id',
'terms' => array($context['get_category']),
'operator' => 'IN'
),
array(
'taxonomy' => 'product_visibility',
'field' => 'slug',
'terms' => 'exclude-from-catalog',
'operator' => 'NOT IN'
)
)
];
$args = array_merge($args, $categories);
$count_args = array_merge($count_args, $categories);
}
$products = new WP_Query($args);
$products = new Timber\PostQuery($products, 'Timber\Integrations\WooCommerce\Product');
$context['posts'] = $products;
$context['count'] = count(Timber::get_posts($count_args));
if ($context['count'] <= $context['get_page'] * 1) $context['ended'] = true;
Timber::render( 'woocommerce/archive-product/archive-product-ajaxload.twig', $context );
die();
}
function get_product_info ($id, $type) {
if (!$id) {
return '';
}
$product = wc_get_product($id);
if (!$product) {
return '';
}
if ($type == 'price') {
return $product->get_price();
} elseif ($type == 'weight') {
return $product->get_weight() ? $product->get_weight() . ' кг' : '';
}
return '';
}
function get_add_to_cart_button ($id) {
$product = wc_get_product( $id );
return ''. pll__('Добавить в корзину') .'';
}
function get_collection_siblings ($term) {
if (!$term) {
return [];
}
$args = array(
'posts_per_page' => -1,
'post_type' => 'product',
'order' => 'ASC',
'order_by' => 'name',
'tax_query' => [
[
'taxonomy' => 'pa_collection',
'terms' => $term,
'field' => 'id',
]
],
);
$siblings = get_posts($args);
$sibl_arr = [];
foreach( $siblings as $sibling ) {
$sibl_arr [] = $sibling;
}
return $sibl_arr;
}
register_sidebar( array(
'name' => 'Сайдбар для фильтров товаров',
'id' => 'sidebar_filters',
'before_widget' => '
',
'after_widget' => '
',
'before_title' => '',
) );
function add_comment_like() {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$wpdb->show_errors();
if (!is_user_logged_in()) {
wp_send_json_error('Необходимо авторизоваться');
die();
}
$comment_id = isset($_POST['comment_id']) ? intval($_POST['comment_id']) : 0;
$user_id = get_current_user_id();
if ($comment_id) {
$comment_exists = get_comment($comment_id);
if (!$comment_exists) {
echo '0';
die();
}
$existing_like = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE comment_id = %d AND user_id = %d",
$comment_id, $user_id
));
if (!$existing_like) {
$result = $wpdb->insert(
$table_name,
array(
'user_id' => $user_id,
'comment_id' => $comment_id,
'date_added' => current_time('mysql')
),
array('%d', '%d', '%s')
);
} else {
$result = $wpdb->delete(
$table_name,
array(
'user_id' => $user_id,
'comment_id' => $comment_id
),
array('%d', '%d')
);
}
$likes = get_comment_likes_count($comment_id);
wp_send_json(array(
'count' => $likes,
'is_liked' => !$existing_like
));
} else {
wp_send_json(array('count' => 0, 'is_liked' => false));
}
die();
}
add_action('wp_ajax_add_comment_like', 'add_comment_like');
function add_post_like() {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
if (!is_user_logged_in()) {
wp_send_json_error('Необходимо авторизоваться');
die();
}
$post_id = isset($_POST['post_id']) ? intval($_POST['post_id']) : 0;
$user_id = get_current_user_id();
if ($post_id) {
$existing_like = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE post_id = %d AND user_id = %d",
$post_id, $user_id
));
if (!$existing_like) {
$wpdb->insert(
$table_name,
array(
'user_id' => $user_id,
'post_id' => $post_id,
'date_added' => current_time('mysql')
),
array('%d', '%d', '%s')
);
} else {
$wpdb->delete(
$table_name,
array(
'user_id' => $user_id,
'post_id' => $post_id
),
array('%d', '%d')
);
}
$likes = get_post_likes_count($post_id);
wp_send_json(array(
'count' => $likes,
'is_liked' => !$existing_like
));
}
die();
}
add_action('wp_ajax_add_post_like', 'add_post_like');
function check_user_likes() {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
if (!is_user_logged_in()) {
wp_send_json_error('Необходимо авторизоваться');
die();
}
$user_id = get_current_user_id();
$liked_posts = $wpdb->get_col($wpdb->prepare(
"SELECT post_id FROM $table_name WHERE user_id = %d AND post_id > 0",
$user_id
));
$liked_comments = $wpdb->get_col($wpdb->prepare(
"SELECT comment_id FROM $table_name WHERE user_id = %d AND comment_id > 0",
$user_id
));
$response = array(
'posts' => $liked_posts,
'comments' => $liked_comments
);
echo json_encode($response);
die();
}
add_action('wp_ajax_check_user_likes', 'check_user_likes');
function get_post_likes_count($post_id) {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$count = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE post_id = %d",
$post_id
));
return $count ? $count : 0;
}
function get_comment_likes_count($comment_id) {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$count = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE comment_id = %d",
$comment_id
));
return $count ? $count : 0;
}
function is_user_liked_post($post_id) {
if (!is_user_logged_in()) {
return false;
}
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$user_id = get_current_user_id();
$result = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE post_id = %d AND user_id = %d",
$post_id, $user_id
));
return $result > 0;
}
function is_user_liked_comment($comment_id) {
if (!is_user_logged_in()) {
return false;
}
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$user_id = get_current_user_id();
$result = $wpdb->get_var($wpdb->prepare(
"SELECT COUNT(*) FROM $table_name WHERE comment_id = %d AND user_id = %d",
$comment_id, $user_id
));
return $result > 0;
}
add_filter('comment_form_logged_in', '__return_empty_string');
// Создание таблицы
function create_likes_table() {
global $wpdb;
$table_name = $wpdb->prefix . 'cosmopet_likes';
$charset_collate = $wpdb->get_charset_collate();
$sql = "CREATE TABLE IF NOT EXISTS $table_name (
id bigint(20) NOT NULL AUTO_INCREMENT,
user_id bigint(20) NOT NULL,
post_id bigint(20) DEFAULT '0',
comment_id bigint(20) DEFAULT '0',
date_added datetime DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (id),
KEY post_id (post_id),
KEY comment_id (comment_id),
KEY user_id (user_id),
UNIQUE KEY user_post (user_id, post_id, comment_id)
) $charset_collate;";
require_once(ABSPATH . 'wp-admin/includes/upgrade.php');
dbDelta($sql);
}
add_action('after_switch_theme', 'create_likes_table');
add_filter('woocommerce_product_data_tabs', function($tabs) {
$tabs['composition_tab'] = array(
'label' => 'Состав',
'target' => 'composition_product_data',
'class' => array('composition_tab'),
'priority' => 60,
);
$tabs['feeding_tab'] = array(
'label' => 'Рекомендации по кормлению',
'target' => 'feeding_product_data',
'class' => array('feeding_tab'),
'priority' => 61,
);
$tabs['important_tab'] = array(
'label' => 'Важно',
'target' => 'important_product_data',
'class' => array('important_tab'),
'priority' => 62,
);
return $tabs;
});
add_action('woocommerce_product_data_panels', function() {
global $post;
$composition = get_post_meta($post->ID, '_composition', true);
echo '';
woocommerce_wp_textarea_input([
'id' => '_composition',
'label' => 'Состав',
'desc_tip' => true,
'description' => 'Введите состав товара',
'value' => $composition
]);
echo '
';
});
add_action('woocommerce_product_data_panels', function() {
global $post;
$feeding = get_post_meta($post->ID, '_feeding_recommendations', true);
echo '';
woocommerce_wp_textarea_input([
'id' => '_feeding_recommendations',
'label' => 'Рекомендации по кормлению',
'desc_tip' => true,
'description' => 'Введите рекомендации по кормлению',
'value' => $feeding
]);
echo '
';
});
add_action('woocommerce_product_data_panels', function() {
global $post;
$important = get_post_meta($post->ID, '_important', true);
echo '';
woocommerce_wp_textarea_input([
'id' => '_important',
'label' => 'Важно',
'desc_tip' => true,
'description' => 'Введите важную информацию',
'value' => $important
]);
echo '
';
});
add_action('woocommerce_process_product_meta', function($post_id) {
if (isset($_POST['_composition'])) {
update_post_meta($post_id, '_composition', sanitize_textarea_field($_POST['_composition']));
}
if (isset($_POST['_feeding_recommendations'])) {
update_post_meta($post_id, '_feeding_recommendations', sanitize_textarea_field($_POST['_feeding_recommendations']));
}
if (isset($_POST['_important'])) {
update_post_meta($post_id, '_important', sanitize_textarea_field($_POST['_important']));
}
});
// Добавление поля для выбора рекомендуемых товаров
function register_recommended_products_acf_field() {
if (function_exists('acf_add_local_field_group')) {
acf_add_local_field_group(array(
'key' => 'group_recommended_products',
'title' => 'Рекомендуемые товары',
'fields' => array(
array(
'key' => 'field_recommended_products',
'label' => 'Выберите рекомендуемые товары',
'name' => 'recommended_products',
'type' => 'relationship',
'instructions' => 'Выберите товары, которые будут отображаться в секции "вашему питомцу может понравиться"',
'required' => 0,
'conditional_logic' => 0,
'post_type' => array(
0 => 'product',
),
'filters' => array(
0 => 'search',
1 => 'taxonomy',
),
'return_format' => 'object',
'min' => '',
'max' => 8,
),
),
'location' => array(
array(
array(
'param' => 'post_type',
'operator' => '==',
'value' => 'product',
),
),
),
'menu_order' => 0,
'position' => 'normal',
'style' => 'default',
'label_placement' => 'top',
'instruction_placement' => 'label',
'hide_on_screen' => '',
));
}
}
add_action('acf/init', 'register_recommended_products_acf_field');
add_action('wp_footer', 'remove_view_cart_button_js');
function remove_view_cart_button_js() {
?>