
本文讲解如何将 laravel 中扁平的权限字符串集合(如 “post access“)转换为嵌套结构数组,便于在 blade 模板中动态渲染权限复选框组。
在使用 spatie/laravel-permission 或自定义权限系统时,常需将类似 [“post access”, “post create”, “post read”, “publish access”] 这样的字符串集合,按资源(Resource)分组,转换为结构清晰、可直接绑定到 Blade 表单的数组格式:
[ [ 'name' => 'post', 'access' => false, 'create' => false, 'read' => false, 'update' => false, 'delete' => false, ], [ 'name' => 'publish', 'access' => false, ], ]
该结构天然适配 Blade 中的 @foreach 循环与复选框 name=”permissions[post][read]” 等命名约定,大幅提升表单处理与权限批量赋值的可维护性。
✅ 推荐实现方式(简洁、健壮、可扩展):
use IlluminateSupportStr; use IlluminateSupportCollection; $permissions = collect([ "post access", "post create", "post read", "post update", "post delete", "publish access", "user read", "user update" ]); // 步骤 1:按资源名分组(提取空格前部分) $grouped = $permissions->groupBy(fn ($item) => Str::before($item, ' ')); // 步骤 2:为每组生成结构化数组 $result = $grouped->map(function (Collection $items, string $resource) { // 定义所有可能的权限动作(可按需扩展) $actions = ['access', 'create', 'read', 'update', 'delete']; // 初始化基础结构 $row = ['name' => $resource]; // 遍历每个权限项,提取动作并设为 false foreach ($items as $permission) { $action = Str::after($permission, ' '); if (in_array($action, $actions)) { $row[$action] = false; } } return $row; })->values()->toArray(); // 输出结果(可用于 Blade 的 @foreach) dd($result);
? 关键说明:
- Str::before($item, ‘ ‘) 精准提取资源名(如 “post”),避免正则或模糊匹配风险;
- Str::after($item, ‘ ‘) 安全获取动作名(如 “create”),即使含空格也稳定;
- 使用 groupBy + map 是 Laravel Collection 的惯用范式,语义清晰、链式流畅;
- ->values()->toArray() 确保返回纯数字索引数组,兼容 Blade @foreach ($permissions as $group)。
⚠️ 注意事项:
- 原始答案中的 Str::contains([‘post’,’get’], $item) 逻辑错误(post 是资源而非关键词),且硬编码索引 $array[0] 易引发越界;
- 不建议手动管理数组下标(如 $array[0], $array[1]),应依赖 groupBy 动态识别资源;
- 若权限动作集固定,建议提取为常量或配置项,便于统一维护;
- 实际项目中,可将此逻辑封装为 PermissionTransformer 类或 Eloquent Accessor,提升复用性。
通过以上方式,你不仅能正确生成复选框所需的数据结构,还能轻松扩展支持新资源(如 category)或新动作(如 restore),真正实现权限 ui 的声明式开发。