如何在 Laravel 7 中导入带有表关系的 Excel?

How to Import Excel with Table Relation in Laravel 7?

提问人:Hilmi Hidayat 提问时间:9/20/2021 更新时间:9/20/2021 访问量:2913

问:

我想问一下。

我想使用 laravel-excel 包制作 excel 导入功能。其中,我导入的数据具有与另一个表的关系值。

因此,当我导入 Excel 时,它会将数据保存在两个表中。

第一个表存储名称和文件路径等数据。第二个表保存导入的 excel 文件的详细信息,并将关系添加到第一个表中。

下面是我拥有的两个表格

booked_vouchers:

  • 编号
  • 名字
  • 路径

booked_voucher_details:

  • 编号
  • booked_voucher_id
  • voucher_code
  • course_code
  • user_name

以下是我在控制器、查看和导入文件中制作的代码。

形式

<form action="{{ route('booked.import')}}" method="POST" enctype="multipart/form-data">
    @csrf
    <div class="form-group">
        <label for="name">Name</label>
        <input name="name" type="text" class="form-control" required>
    </div>
    <div class="form-group">
        <label for="file">File</label>
        <input name="file" type="file" class="form-control" required>
    </div>
    <button type="submit" class="btn btn-primary">Import</button>
</form>

控制器

    public function import(Request $request)
    {
        $validator = Validator::make($request->all(), [
            'file' => 'required|mimes:csv,xlx,xls,xlsx'
        ]);

        if ($validator->fails()) {
            return back()->with('toast_error', $validator->messages()->all()[0])->withInput();
        }

        $data = new BookedVoucher();

        $data->name = $request->name;
        $fileName = time().'_'.$request->file->getClientOriginalName();
        $filePath = $request->file('file')->storeAs('reports', $fileName, 'public');

        $data->file = $filePath;
        $data->created_by = \Auth::user()->id;

        if ($data->save()) {
            Excel::import(new BookedVoucherDetailImport, $request->file('file'));
        }

        return redirect()->back()->with('success','Data have been imported');
    }

BookedVoucherDetailImport.php

<?php

namespace App\Imports;

use App\Model\BookedVoucher;
use App\Model\BookedVoucherDetail;
use Maatwebsite\Excel\Concerns\ToModel;
use Maatwebsite\Excel\Concerns\WithStartRow;

class BookedVoucherDetailImport implements ToModel, WithStartRow
{
    /**
     * @return int
     */
    public function startRow(): int
    {
        return 2;
    }
    /**
    * @param array $row
    *
    * @return \Illuminate\Database\Eloquent\Model|null
    */
    public function model(array $row)
    {
        $data = BookedVoucher::first();
        return new BookedVoucherDetail([
            'booked_voucher_id'     => $data->id,
            'course_code'           => $row[0], 
            'voucher_code'          => $row[1],
            'user_name'         => $row[2],
            'status'                => 'OK'
        ]);
    }
}

如何将booked_voucher_id值设置为与Excel导入过程之前保存的booked_voucher表中的id值相同?

*目前,如果我使用 BookedVoucherDetailImport.php 文件中的代码,则booked_voucher_details表中booked_voucher_id值的结果总是不正确的。

php laravel 软件包 laravel-7

评论


答:

1赞 Rwd 9/20/2021 #1

您可以将 传递给 Import 类:BookedVoucher

Excel::import(new BookedVoucherDetailImport($data), $request->file('file'));

然后将您的更新为:BookedVoucherDetailImport

<?php

namespace App\Imports;

use App\Model\BookedVoucher;
use App\Model\BookedVoucherDetail;
use Maatwebsite\Excel\Concerns\ToModel;
use Maatwebsite\Excel\Concerns\WithStartRow;

class BookedVoucherDetailImport implements ToModel, WithStartRow
{
    /**
     * @var BookedVoucher 
     */
    protected $bookedVoucher;

    /**
     * @param BookedVoucher $bookedVoucher
     */
    public function __construct(BookedVoucher $bookedVoucher)
    {
        $this->bookedVoucher = $bookedVoucher;
    }
    
    /**
     * @return int
     */
    public function startRow(): int
    {
        return 2;
    }

    /**
     * @param array $row
     *
     * @return \Illuminate\Database\Eloquent\Model|null
     */
    public function model(array $row)
    {
        return new BookedVoucherDetail([
            'booked_voucher_id' => $this->bookedVoucher->id,
            'course_code'       => $row[0],
            'voucher_code'      => $row[1],
            'user_name'         => $row[2],
            'status'            => 'OK',
        ]);
    }
}

评论

0赞 Hilmi Hidayat 9/21/2021
谢谢@Rwd,这真的很有帮助,我已经试过了,它工作正常。
0赞 Rwd 9/21/2021
@HilmiHidayat 很高兴我能帮上忙!:)