如何在 WooCommerce 商店页面中自动隐藏用户已购商品

如何在 WooCommerce 商店页面中自动隐藏用户已购商品

本文介绍如何通过 pre_get_posts 钩子动态过滤 woocommerce 商品列表,使已购买商品对当前登录用户完全不可见,既防止重复购买,又提升 lms 类站点(如 learndash + woocommerce)的课程展示体验。

在构建基于 WooCommerce 的在线学习平台(如 LearnDash 集成场景)时,一个常见且关键的用户体验需求是:当用户已成功购买某门课程(即对应可售商品)后,该商品不应再出现在商店首页(Shop 页面)的商品网格中。这不仅能避免用户误点重复下单,还能让界面更聚焦于“待购课程”,提升导航清晰度与转化效率。

实现这一目标的核心思路是:在 WooCommerce 主商品查询执行前,动态排除当前用户已购商品的 ID 列表。相比仅拦截加购(如 woocommerce_add_to_cart_validation),此方案更彻底——商品从源头上不参与查询、不渲染、不暴露,真正实现“视觉级隐藏”。

以下是推荐使用的完整解决方案,需将代码添加至当前主题的 functions.php 文件中:

add_action( 'pre_get_posts', 'hide_product_from_shop_page_if_user_already_purchased', 20 );

function hide_product_from_shop_page_if_user_already_purchased( $query ) {
    // 仅作用于前台主查询,跳过后台及非主循环
    if ( ! $query->is_main_query() || is_admin() || ! is_shop() ) {
        return;
    }

    $current_user = wp_get_current_user();
    if ( $current_user->ID === 0 ) { // 未登录用户不处理
        return;
    }

    // 查询该用户所有已完成/处理中的订单
    $customer_orders = get_posts( array(
        'posts_per_page' => -1,
        'post_type'      => 'shop_order',
        'post_status'    => array( 'wc-completed', 'wc-processing' ),
        'meta_key'       => '_customer_user',
        'meta_value'     => $current_user->ID,
        'fields'         => 'ids', // 仅获取订单 ID,提升性能
    ) );

    if ( empty( $customer_orders ) ) {
        return;
    }

    $purchased_product_ids = array();

    // 遍历每个订单,提取所购商品 ID(含变体)
    foreach ( $customer_orders as $order_id ) {
        $order = wc_get_order( $order_id );
        if ( ! $order ) {
            continue;
        }

        foreach ( $order->get_items() as $item ) {
            $product_id = $item->get_product_id();
            if ( $product_id ) {
                $purchased_product_ids[] = $product_id;
            }
        }
    }

    // 去重并确保为整数数组
    $purchased_product_ids = array_unique( array_map( 'absint', $purchased_product_ids ) );

    if ( ! empty( $purchased_product_ids ) ) {
        $query->set( 'post__not_in', $purchased_product_ids );
    }
}

关键优化说明:

PHP168  行业B2B

PHP168 行业B2B

解决问题如下:只列举最近用户提交问题,其余问题前面几次补丁已经解决,不在复述。1、解决搜索问题。以前搜索一定要确定到省下面的某个市,这个不符合用户体验。 现在改为,省–所有城市(默认为所有城市,也可以自己选择某个市)。2、解决首页推荐产品部显示问题。(以前没有考虑多个其他浏览器)3、解决供应、求购 今日产品显示问题。(理由同上)4、解决收藏商家、供应、求购问题。 (链接错误)5、解决后台分类过

下载

  • 使用 ‘fields’ => ‘ids’ 显著减少数据库负载;
  • absint() 确保 ID 安全转为正整数,防范潜在类型异常;
  • 严格限定 is_shop() 和 !is_admin(),避免影响分类页、搜索页或后台管理;
  • 仅对已登录用户生效,游客仍可见全部商品。

⚠️ 注意事项:

  • 此逻辑不影响商品详情页直接访问(如用户通过书签进入)。若需全局拦截,应额外结合 template_redirect 或产品模板内条件判断;
  • 对于变体商品(Variable Product),$item->get_product_id() 返回的是变体 ID(而非父商品 ID),因此变体购买后,仅该变体被隐藏——如需隐藏整个父商品,请改用 $item->get_variation_id() ? $item->get_parent_id() : $product_id;
  • 若使用对象缓存(如 Redis、Memcached),可能需清除相关查询缓存以确保实时性;
  • 建议搭配前端 JS 做二次校验(如禁用已购商品的“加入购物车”按钮),增强健壮性。

该方案简洁、高效、符合 WordPress/WooCommerce 最佳实践,已在 LearnDash + WooCommerce 多项目中稳定运行,是构建专业学习门户不可或缺的用户体验优化环节。

https://www.php.cn/faq/1965656.html

发表回复

Your email address will not be published. Required fields are marked *