vr-shopxo-plugin/docs/SPEC_SELECTOR_DESIGN.md

12 KiB
Raw Blame History

VR 票务 spec 选择器 + 多座位选择 — 数据结构与实现说明

日期2026-04-21 背景:修正 COUNCIL_PHASE2_ASSESSMENT_CORRECTED.md 中 spec 描述的根本性错误


一、核心结论(先说清楚)

原报告里所有关于 spec 的示例都是错的,原因:spec 是 4 维度联合索引,而不是 1 个维度;前端目前根本拿不到完整的 4 维 spec 映射

我们实际上要做的产品形态是:

一个风格化、带座位图的多维度 spec 规格选择器

  • 界面同时具备原生 ShopXO spec 选择器的交互(场次/场馆/分区可选择,不在分支下的选项自动隐藏/变灰)
  • 又有自己的多座位视图(在座位图上直接点选多个座位)
  • 最终在 submit() 时,把选中座位的完整 4 维 spec 数组提交到 Buy 链路

二、ShopXO spec 的真实结构

2.1 四维 SPEC_DIMS

ShopXO 的每个 GoodsSpecBase 记录,通过 4 个维度的 spec 值联合确定:

const SPEC_DIMS = ['$vr-场馆', '$vr-分区', '$vr-座位号', '$vr-场次'];

2.2 一个座位的完整 spec 数组(示例)

以商品118的某个座位为例A排3座VIP区VR体验馆-1号厅场次15:00-16:59

{
  "spec_base_id": 1001,
  "price": 380.00,
  "inventory": 1,
  "spec": [
    { "type": "$vr-场馆",   "value": "VR体验馆-1号厅" },
    { "type": "$vr-分区",   "value": "VR体验馆-1号厅-VIP区" },
    { "type": "$vr-座位号", "value": "VR体验馆-1号厅-VIP区-A-1排3座" },
    { "type": "$vr-场次",   "value": "15:00-16:59" }
  ]
}

⚠️ 关键spec value 不是 "A_3""roomId_A_3" 这种短格式, 而是完整路径字符串 "VR体验馆-1号厅-VIP区-A-1排3座"。 这个字符串由 BatchGenerate 第131行构建 $val_seat = "{$venueName}-{$roomName}-{$char}-{$rowLabel}{$col}"

2.3 从前端座位元素到 spec_base_id 的正确路径

前端一个座位 DOM 元素,持有以下数据:

  • roomId = "room_001"(来自 template_snapshot.rooms[i].id
  • rowLabel = "A"(座位行标签)
  • colNum = 3座位列号
  • char = "A"(座位类型 char对应 sections[i].char

要找到对应的 GoodsSpecBase需要用以下映射关系

GoodsSpecBase.extends = {"seat_key": "room_001_A_3"}
       ↓  GetGoodsViewData() 动态构建
spec_base_id_map = {"room_001_A_3": 1001}
       ↓
前端 seatKey = room_001 + "_" + "A" + "_" + 3 = "room_001_A_3"
       ↓
spec_base_id_map["room_001_A_3"] = 1001 ✅

三、前端实际能拿到的数据(现状 vs 需要的)

3.1 当前 GetGoodsViewData() 返回的数据(不完整)

[
    'vr_seat_template' => [
        'seat_map'          => [...],   // template_snapshot.rooms座位图原始数据
        'spec_base_id_map'  => [...],   // ⚠️ 键名格式不对roomId_row_col且前端没有对应生成逻辑
    ],
    'goods_spec_data' => [
        ['spec_id' => 2001, 'spec_name' => '15:00-16:59', 'price' => 380],
        ['spec_id' => 2002, 'spec_name' => '18:00-20:59', 'price' => 480],
    ]
]

缺失的信息

  1. goods_spec_data 只有场次维度,缺少场馆/分区/座位号三个维度
  2. spec_base_id_map 的 key 格式(roomId_row_col)前端无法构造
  3. 前端不知道哪个 spec_id 对应哪个座位

3.2 前端需要的完整数据结构GetGoodsViewData 应返回)

[
    'vr_seat_template' => [
        'venue'    => $config['template_snapshot']['venue'],
        'rooms'    => $config['template_snapshot']['rooms'],  // 座位图原始数据
        'sessions' => $config['sessions'],                     // 场次列表
    ],

    // 【修复】重构后的 seatSpecMaproom_row_col → 完整规格信息
    // 用途:前端选中座位后,直接查表组装 4 维 spec 数组
    'seatSpecMap' => [
        'room_001_A_3' => [
            'spec_base_id' => 1001,
            'price'        => 380.00,
            'inventory'    => 1,
            'spec'         => [
                ['type' => '$vr-场馆',   'value' => 'VR体验馆-1号厅'],
                ['type' => '$vr-分区',   'value' => 'VR体验馆-1号厅-VIP区'],
                ['type' => '$vr-座位号', 'value' => 'VR体验馆-1号厅-VIP区-A-1排3座'],
                ['type' => '$vr-场次',   'value' => '15:00-16:59'],
            ],
            'rowLabel' => 'A',
            'colNum'   => 3,
            'roomId'   => 'room_001',
            'section'  => ['char' => 'A', 'name' => 'VIP区', 'color' => '#f06292'],
        ],
        'room_001_B_5' => [
            'spec_base_id' => 1002,
            'price'        => 180.00,
            // ... 同上
        ],
        // 每个可购座位一行
    ],

    // 当前商品的全部场次(用户需要先选场次)
    'sessions' => [
        ['spec_id' => 2001, 'spec_name' => '15:00-16:59', 'price' => 380, 'start' => '15:00', 'end' => '16:59'],
        ['spec_id' => 2002, 'spec_name' => '18:00-20:59', 'price' => 480, 'start' => '18:00', 'end' => '20:59'],
    ],
]

3.3 seatSpecMap 的生成逻辑(在 GetGoodsViewData 中实现)

// GetGoodsViewData() 中新增:
// 1. 查询当前商品所有 GoodsSpecBase含 extends.seat_key
$specs = Db::name('GoodsSpecBase')
    ->where('goods_id', $goodsId)
    ->where('inventory', '>', 0)
    ->select();

// 2. 查询每个 spec_base_id 对应的 4 维 GoodsSpecValue
$specValues = Db::name('GoodsSpecValue')
    ->whereIn('goods_spec_base_id', array_column($specs->toArray(), 'id'))
    ->select()
    ->toArray();

// 3. 按 spec_base_id 分组,构建 4 维 spec 数组
$specByBaseId = [];
foreach ($specValues as $sv) {
    $specByBaseId[$sv['goods_spec_base_id']][] = [
        'type'  => $sv['type'],
        'value' => $sv['value'],
    ];
}

// 4. 构建 seatSpecMapseat_key → 完整规格
$seatSpecMap = [];
foreach ($specs as $spec) {
    $extends = json_decode($spec['extends'] ?? '{}', true);
    $seatKey = $extends['seat_key'] ?? '';
    if (empty($seatKey)) continue;

    $seatSpecMap[$seatKey] = [
        'spec_base_id' => intval($spec['id']),
        'price'        => floatval($spec['price']),
        'inventory'    => intval($spec['inventory']),
        'spec'         => $specByBaseId[$spec['id']] ?? [],
    ];
}

四、前端 spec 选择器的完整交互

4.1 UI 结构

┌─────────────────────────────────────────────────┐
│  场次选择Tab 或下拉)                          │
│  [15:00-16:59] [18:00-20:59]                   │
│                                                 │
│  选完场次后 → 座位图自动切换到该场次的可用座位    │
│                                                 │
│  场馆选择(单选)                                │
│  [VR体验馆-1号厅 ✓] [VR体验馆-2号厅]              │
│                                                 │
│  分区选择(单选 / 多选)灰色表示不在分支内         │
│  [VIP区 ✓] [看台区] [普通区-已售罄]               │
│                                                 │
│  ─────────── 座位图(多选)────────────────    │
│  [舞台 - 固定位置]                               │
│  A排 [■■■■]   ← 可选座位VIP                  │
│  B排 [■■--■■] ← 部分已售                        │
│  C排 [已灰掉] ← 不在当前分区或已售                │
└─────────────────────────────────────────────────┘

4.2 spec 选择器的联动逻辑

维度优先级场次 > 场馆 > 演播室 > 分区

每次用户切换选项时,过滤规则:

场次切换 → 重置场馆/分区/座位 → 用 seatSpecMap 过滤出该场次所有座位
场馆切换 → 重置分区/座位 → 用 seatSpecMap 过滤出该场馆所有座位
分区切换 → 只高亮/过滤座位 → 用 seatSpecMap 过滤出该分区座位(灰色其他)
座位点击 → 选中/取消 → 更新 selectedSeats[]

灰色/隐藏逻辑(参考原生 ShopXO spec 选择器):

// 某座位"可亮"的条件:该座位的 spec 数组 包含 当前已选场次 + 当前已选场馆 + (当前已选分区 或 未选分区)
// 具体实现在 selectSession() / selectVenue() / selectSection() 中调用 filterSeatMap()
function filterSeatMap(sessionSpecId, venueId, sectionChar) {
    document.querySelectorAll('.vr-seat:not(.space)').forEach(function(el) {
        var seatKey = el.dataset.rowLabel + '_' + el.dataset.colNum;
        var seatInfo = seatSpecMap[seatKey];
        if (!seatInfo) { el.classList.add('disabled'); return; }

        var spec = seatInfo.spec;
        var hasSession = spec.some(s => s.type === '$vr-场次' && s.value === currentSessionSpec);
        var hasVenue   = spec.some(s => s.type === '$vr-场馆'   && s.value.includes(currentVenue));
        var hasSection = !sectionChar || spec.some(s => s.type === '$vr-分区' && s.value.includes(sectionChar));
        var isAvailable = seatInfo.inventory > 0;

        if (hasSession && hasVenue && hasSection && isAvailable) {
            el.classList.remove('disabled', 'sold');
        } else {
            el.classList.add(isAvailable ? 'disabled' : 'sold');
        }
    });
}

五、submit() 时如何组装 spec 数组

用户选了 2 个座位A排3座 + A排5座都是VIP区15:00场次VR体验馆-1号厅

// 选中座位后的 selectedSeats[] 数据结构
[
    { seatKey: 'room_001_A_3', price: 380, rowLabel: 'A', colNum: 3, section: {...} },
    { seatKey: 'room_001_A_5', price: 380, rowLabel: 'A', colNum: 5, section: {...} },
]

// submit() 构造 goods_data
var goodsDataList = selectedSeats.map(function(seat) {
    var seatInfo = seatSpecMap[seat.seatKey];  // 从注入的 seatSpecMap 查
    return {
        goods_id: self.goodsId,
        spec: seatInfo.spec,    // 4维完整 spec 数组不是1维
        stock: 1,
        order_base: {
            extension_data: {
                attendee: { real_name: '...', phone: '...', id_card: '...' }
            }
        }
    };
});

生成的 goods_data简化展示

[
  {
    "goods_id": 118,
    "spec": [
      { "type": "$vr-场馆",   "value": "VR体验馆-1号厅" },
      { "type": "$vr-分区",   "value": "VR体验馆-1号厅-VIP区" },
      { "type": "$vr-座位号", "value": "VR体验馆-1号厅-VIP区-A-1排3座" },
      { "type": "$vr-场次",   "value": "15:00-16:59" }
    ],
    "stock": 1,
    "order_base": {
      "extension_data": {
        "attendee": { "real_name": "张三", "phone": "13800138000", "id_card": "" }
      }
    }
  },
  {
    "goods_id": 118,
    "spec": [
      { "type": "$vr-场馆",   "value": "VR体验馆-1号厅" },
      { "type": "$vr-分区",   "value": "VR体验馆-1号厅-VIP区" },
      { "type": "$vr-座位号", "value": "VR体验馆-1号厅-VIP区-A-1排5座" },
      { "type": "$vr-场次",   "value": "15:00-16:59" }
    ],
    "stock": 1,
    "order_base": { "extension_data": { "attendee": { "real_name": "李四", ... } } }
  }
]

ShopXO 的 GoodsSpecificationsHandle 通过 4 个 type-value 组合在 GoodsSpecValue 表中精确匹配到对应的 GoodsSpecBase拿到 inventory=1price=380


六、需要修改的文件清单

文件 改动
SeatSkuService::GetGoodsViewData() 新增 seatSpecMap 生成逻辑,返回完整 4 维 spec 映射
Goods.php(票务判断块) MyViewAssign 中加入 seatSpecMapsessions
ticket_detail.html JS 新增 selectSession() / selectVenue() / selectSection() 联动逻辑;filterSeatMap() 过滤;submit() 使用 seatSpecMap 组装 spec
ticket_detail.html HTML 新增场次/场馆/分区选择器的 DOM 结构
ticket_detail.css spec 选择器样式(选中态/灰色态/隐藏态)

七、修复优先级

优先级 任务 依赖
P0 重构 GetGoodsViewData() 返回 seatSpecMap 后端
P0 前端用 seatSpecMap 替代错误的 specBaseIdMap 前端
P0 submit() 使用 seatSpecMap[seatKey].spec 前端
P1 实现场次/场馆/分区选择器 UI + 联动逻辑 前端
P1 filterSeatMap() 实现灰色/隐藏过滤 前端
P2 loadSoldSeats() → 使用 seatSpecMap + inventory 字段 前端