0

0

解决WooCommerce自定义模板分页加载重复内容的问题

花韻仙語

花韻仙語

发布时间:2025-09-28 13:33:01

|

301人浏览过

|

来源于php中文网

原创

解决woocommerce自定义模板分页加载重复内容的问题

本教程详细阐述了在WooCommerce自定义模板中,分页链接加载相同页面内容的问题及其解决方案。核心在于调整当前页码的获取方式,从get_query_var('paged')改为使用filter_input读取自定义GET参数,并相应修改paginate_links函数的format参数,确保分页链接能正确指向不同页面的内容。

问题描述

在开发自定义WooCommerce模板(例如,用于显示用户下载或订单历史的downloads.php或类似文件)时,开发者可能会遇到一个常见的分页问题:当点击分页链接(如“第2页”)时,URL会正确地更新(例如,从mywebsite.com/account/downloads变为mywebsite.com/account/downloads/2),但页面显示的内容却始终是第一页的数据。这表明尽管URL结构发生了变化,但后端查询并未正确识别并处理页码参数。

最初尝试使用get_query_var('paged')来获取当前页码,但这种方法在自定义wc_get_orders查询中可能无法按预期工作,因为它通常与WordPress主查询的重写规则更紧密相关。

理解WordPress与WooCommerce中的分页机制

WordPress默认的分页机制依赖于其重写规则来解析URL中的页码(例如/page/2/)。然而,当我们在自定义模板中使用wc_get_orders等函数构建独立于主查询的自定义查询时,WordPress的默认分页变量paged可能无法自动被正确识别和传递。为了确保自定义查询能够响应分页链接,我们需要明确地定义如何获取当前页码,以及如何构造分页链接的URL。

解决方案

解决此问题的关键在于两步:修改当前页码的获取方式调整paginate_links函数的format参数,使其协同工作。

步骤一:调整当前页码变量的获取方式

不再使用get_query_var('paged'),而是通过filter_input函数安全地从URL的GET参数中读取自定义的页码变量。

原代码示例(可能存在问题):

$current_page = ( get_query_var('paged') ) ? get_query_var('paged') : 1;

修改为:

$paged = max( 1, (int) filter_input( INPUT_GET, 'pagina' ) );

解释:

Napkin AI
Napkin AI

Napkin AI 可以将您的文本转换为图表、流程图、信息图、思维导图视觉效果,以便快速有效地分享您的想法。

下载
  • filter_input( INPUT_GET, 'pagina' ):这是一个更安全、更推荐的方式来获取GET请求参数。INPUT_GET指定从GET请求中获取,'pagina'是我们自定义的URL参数名称。
  • (int):将获取到的参数强制转换为整数类型,确保其为有效的页码。
  • max( 1, ... ):确保页码至少为1,避免出现负数或零页码的情况。
  • 这里我们将自定义的GET参数命名为pagina。您可以根据自己的偏好选择其他名称,但请确保在后续步骤中保持一致。

步骤二:配置paginate_links函数的format参数

为了让paginate_links函数生成包含我们自定义GET参数的正确分页URL,需要修改其format参数。

原代码示例(可能存在问题):

'format' => '%#%',

修改为:

'format' => '?pagina=%#%',

解释:

  • %#%是paginate_links函数中的一个占位符,它会被实际的页码替换。
  • 通过将format设置为'?pagina=%#%',我们指示paginate_links生成形如?pagina=2、?pagina=3这样的URL参数,而不是依赖WordPress默认的路径式分页(如/page/2/)。这样,当用户点击分页链接时,URL中会包含?pagina=N,我们的自定义查询就能通过filter_input( INPUT_GET, 'pagina' )正确读取到页码。

此外,在paginate_links的base参数中,也需要确保指向正确的端点URL。例如,如果您的自定义页面是WooCommerce我的账户下的一个端点,可以使用wc_get_endpoint_url('your-endpoint-slug')。在提供的解决方案中,'libreria'被用作端点。

'base' => esc_url( wc_get_endpoint_url( 'libreria') ) . '%_%',

完整示例代码

以下是整合了上述解决方案的自定义WooCommerce模板代码示例:

<?php

?><div class="orders-container"><?php

defined( 'ABSPATH' ) || exit;

$downloads     = WC()->customer->get_downloadable_products();
$has_downloads = (bool) $downloads;

do_action( 'woocommerce_before_account_downloads', $has_downloads );

if ( $has_downloads ) {

 ?><table class="table_orders heading"><tr>
  <td class="product_number">Ordine</td>
  <td class="product_name">Prodotto</td>
  <td class="product_data">Data</td>
  <td class="product_price">Totale</td>
  <td class="product_status">Stato</td>
   <td class="product_action">File</td>
 </tr></table><?php


    $order_statuses = array('wc-completed');
    // 步骤一:调整当前页码的获取方式
    $paged = max( 1, (int) filter_input( INPUT_GET, 'pagina' ) );

    $args = apply_filters('woocommerce_my_account_my_orders_query',array(
    'customer_id'     => get_current_user_id(),
    'post_status'     => $order_statuses,
    'paged'           => $paged, // 使用新的 $paged 变量
    'posts_per_page'  => 3, // 每页显示订单数,可根据需要调整
    'paginate'        => true,
    ));
    $customer_orders = wc_get_orders( $args );

    // Get Access $order variable Foreach - Definisci le variabili sotto per poi utilizzarle in echo
    foreach($customer_orders->orders as $order) {
     // Get $product object from $order / $order_id
     $orders_id = $order->get_id();
     $status =  wc_get_order_status_name( $order->get_status() );
     $date_created = $order->get_date_created()->date('d/m/Y');
     $payment_method = $order->get_payment_method_title();
     $order_total = $order->get_formatted_order_total();

        // Get Access Items & Product Variable Foreach
        foreach($order->get_items() as $item_id => $item) {
         $product_name = $item->get_name();
         $view_order = $order->get_view_order_url();

         // Get product image - https://www.businessbloomer.com/woocommerce-easily-get-product-info-title-sku-desc-product-object/
         $product = $item->get_product();
            if( $product instanceof WC_Product ){
             $order_img = $product->get_image();
            }

         //Get product download button 
         $downloads = $order->get_downloadable_items();
            if(is_array($downloads)) {
                foreach($downloads as $product){
                 $download_button = '<a href="'. $product['download_url'] .'" target="_blank">Download</a>';
                } 
            } 

            echo '
                <table class="table_orders">
                <tr class="table_row_items">
                    <td class="product_number">
                     <span class="mobile title">Ordine</span>
                     <span>#'. esc_attr($orders_id) .'</span>
                    </td>

                    <td class="product_name">
                     <span class="mobile title">Prodotto</span>
                     <a href="'. wp_kses_post($view_order) .'">'. wp_kses_post($product_name) .'</a>
                    </td>

                    <td class="product_data">
                     <span class="mobile title">Data</span>
                     <span>'. wp_kses_post($date_created) .'</span>
                    </td>

                    <td class="product_price">
                     <span class="mobile title">Prezzo</span>
                     <span>'. wp_kses_post($order_total) .'</span>
                    </td>

                    <td class="product_status">
                     <span class="mobile title">Stato</span>
                     <span class="label ' . $order->get_status() . '">'. wp_kses_post($status) .'</span>
                    </td>

                    <td class="product_action">
                     <span class="mobile title">File</span>
                     <a target=”_blank” href="'. esc_url($view_order) .'">Visualizza<i class="fa-duotone fa-eye"></i></a>
                    </td>
                </tr>    
                </table> 
            ';

            // Tasto download funzionante - if($downloads) { echo '<div class="container_orders_download"> '. $download_button .' </div>'; }
        }
    }


    // Pagination button - Responsabile dei bottoni e numerazione delle pagine della cronologia ordini
    ?><div class="container-pagination"><?php 
        $args = array(
         'base'          => esc_url( wc_get_endpoint_url( 'libreria') ) . '%_%', // 确保这里的端点名称与您的设置一致
         'format'        => '?pagina=%#%', // 步骤二:修改 format 参数
         'total'         => $customer_orders->max_num_pages,
         'current'       => $paged, // 使用新的 $paged 变量
         'show_all'      => false,
         'end_size'      => 3,
         'mid_size'      => 3,
         'prev_next'     => true,
         'prev_text' => __('<i class="fa-regular fa-angle-left"></i><span>Indietro</span>'), 
         'next_text' => __('<span>Avanti</span><i class="fa-regular fa-angle-right"></i>'),
         'type'          => 'plain',
         'add_args'      => false,
         'add_fragment'  => ''
        ); 
    echo paginate_links($args);   
}       
        else {
         ?><div class="msg_orders">La tua cronologia ordini è vuota!</div><?php
        } 
    ?></div><?php
    do_action( 'woocommerce_after_account_downloads', $has_downloads );

?>
</div>

注意事项

  1. 端点名称一致性: 在wc_get_endpoint_url()函数中使用的端点名称(例如'libreria')必须与您在WooCommerce设置中或通过代码注册的自定义账户端点名称完全一致。
  2. pagina参数: filter_input( INPUT_GET, 'pagina' )中的'pagina'与format参数中的'?pagina=%#%'必须保持一致。如果您选择其他名称,请确保两处都进行修改。
  3. 安全性: 使用filter_input是获取GET参数的推荐方式,因为它提供了数据过滤和验证的功能,有助于防止XSS等安全漏洞。
  4. 调试: 如果遇到问题,可以启用WordPress的调试模式(在wp-config.php中设置define( 'WP_DEBUG', true );)来查看是否有PHP错误或警告,这有助于定位问题。
  5. 缓存: 在进行此类更改后,请清除任何WordPress或服务器级别的缓存,以确保新的代码逻辑能够正确执行。

总结

在WooCommerce自定义模板中实现正确的分页,关键在于理解自定义查询与WordPress默认分页机制之间的差异。通过将当前页码的获取方式从get_query_var('paged')切换到使用filter_input读取自定义GET参数,并相应地调整paginate_links函数的format参数,我们可以确保分页链接能够正确地更新URL并加载对应页面的内容,从而提供无缝的用户体验。这种方法提供了一种灵活且健壮的解决方案,适用于需要精细控制分页逻辑的自定义场景。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

119

2023.09.26

define的用法
define的用法

define用法:1、定义常量;2、定义函数宏:3、定义条件编译;4、定义多行宏。更多关于define的用法的内容,大家可以阅读本专题下的文章。

390

2023.10.11

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

888

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

464

2024.06.27

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

615

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 13.5万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.3万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号