如何使用Laravel eloquent / collections计算PHP中类似键的总量

I currently have the below json response that the API is returning. I am able to return it using Laravel Eloquent. There are several users and each user has a several receipts. A receipt has types and status. I want to try to get the total sum amount for each receipt that is related to its type and status. I was able to return the below json response using

$this->user->with('receipts')->has('receipts')->get(['id', 'name']);

I have tried using multiple laravel collections methods https://laravel.com/docs/5.8/collections#available-methods But I am still unable to get the desired response.

    {
        "id": 1,
        "name": "kent",
        "receipts": [
            {
                "id": 1,
                "user_id": 1,
                "type_id": 1,
                "status": 0,
                "amount": 100
            },
            {
                "id": 2,
                "user_id": 1,
                "type_id": 1,
                "status": 0,
                "amount": 100
            },
            {
                "id": 3,
                "user_id": 1,
                "type_id": 2,
                "status": 1,
                "amount": 50
            },
            {
                "id": 4,
                "user_id": 1,
                "type_id": 2,
                "status": 0,
                "amount": 30
            },
            {
                "id": 5,
                "user_id": 1,
                "type_id": 2,
                "status": 0,
                "amount": 30
            },
            {
                "id": 6,
                "user_id": 1,
                "type_id": 1,
                "status": 0,
                "amount": 20
            },
            {
                "id": 7,
                "user_id": 1,
                "type_id": 1,
                "status": 1,
                "amount": 10
            }
        ]
        },
        {
            "id": 2,
            "name": "allison",
            "receipts": [
                {
                    "id": 9,
                    "user_id": 2,
                    "type_id": 1,
                    "status": 0,
                    "amount": 20
                }
        ]
    }
]

I expect to get the below

    {
        "id": 1,
        "name": "kent",
        "receipts": [
            {
                "performance and deleted": 220,
                "performance and not deleted": 10,
                "project and deleted": 60,
                "project and deleted": 50
            }
        ]
    },
    {
        "id": 2,
        "name": "allison",
        "receipts": [
            {
                "performance and deleted": 20,
                "performance and not deleted": 0,
                "project and deleted": 0,
                "project and not deleted": 0
            }
        ]
    }
]

You should be able to get the sum of amount with

$this->user->with(['receipts' => function($query) {
    $query->selectRaw("SUM(amount) as amount, type_id, status, user_id")->groupBy('type_id', 'status', 'user_id');
}])->has('receipts')->get(['id', 'name']);

You can use collection methods to get the desired output

$this->user->with(['receipts' => function($query) {
    $query->selectRaw("SUM(amount) as amount, type_id, status, user_id")->groupBy('type_id', 'status', 'user_id');
}])->has('receipts')->get(['id', 'name'])
->each(function ($user) {
    $user->setRelation(
        'receipts',
        $user->receipts->mapWithKeys(function ($receipt) {
            return [
                $receipt->type_id . ' and ' . $receipt->status => $receipt->amount // format the key as you wish
            ];
        })
    );
})

You may use foreach and other loops to make the json you want! use This function to convert your collection to the json you want :

public function convert(Collection $collection) 
{
    $result_collection = $collection;
    $payment_collections = [];
    foreach ($result_collection->receipts as $receipt)
    {
        $payment["type_id${receipt->type_id} and status${$receipt->status}"] = $receipt->amount;
    }
    $result_collection->receipts = $payment_collections;
    return $result_collection;
}

And This is same way to get total amount. Just put an foreach and add each amount to a variable that initialized with 0;

and there is other ways like change toArray function in your Resource Collection.

I wrote the script assuming your data is a PHP array so you may change some part of my code. For example you may change:

$row['receipts']
// To 
$row->receipts

anyway

// The function in your controller
function index(){
  $users=User::whereHas('receipts')->with('receipts')->get()->toArray();
  return convert($users);
}
// The helper function 
function convert($data){
  $data=collect($data);
$allTypes=[];
$allStatuses=[];
return $data->each(function($row) use (&$allTypes,&$allStatuses){
    $types=collect($row['receipts'])->pluck('type_id')->toArray();
    $statuses=collect($row['receipts'])->pluck('status')->toArray();
    $allTypes=array_unique(array_merge($allTypes,$types));
    $allStatuses=array_unique(array_merge($allStatuses,$statuses));
})->map(function ($row,$index) use (&$allTypes,&$allStatuses){
    $result=[];
    $receipts=collect($row['receipts']);
    foreach ($allTypes as $type){
        foreach ($allStatuses as $status){
            $result["type_id {$type} and status {$status}: "]=$receipts->where('type_id',$type)->where('status',$status)->sum('amount');
        }
    }
    $row['receipts']=$result;
    return $row;
});
}

enter image description here