1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
namespace Backpack\CRUD\app\Library\Uploaders; |
4
|
|
|
|
5
|
|
|
use Backpack\CRUD\app\Library\CrudPanel\CrudPanelFacade as CRUD; |
6
|
|
|
use Illuminate\Database\Eloquent\Model; |
7
|
|
|
use Illuminate\Support\Arr; |
8
|
|
|
use Illuminate\Support\Facades\Storage; |
9
|
|
|
|
10
|
|
|
class MultipleFiles extends Uploader |
11
|
|
|
{ |
12
|
|
|
public static function for(array $field, $configuration) |
13
|
|
|
{ |
14
|
|
|
return (new self($field, $configuration))->multiple(); |
|
|
|
|
15
|
|
|
} |
16
|
|
|
|
17
|
|
|
public function uploadFile(Model $entry, $value = null) |
18
|
|
|
{ |
19
|
|
|
$filesToDelete = CRUD::getRequest()->get('clear_'.$this->getName()); |
|
|
|
|
20
|
|
|
$value = $value ?? CRUD::getRequest()->file($this->getName()); |
21
|
|
|
$previousFiles = $entry->getOriginal($this->getName()) ?? []; |
22
|
|
|
|
23
|
|
|
if (! is_array($previousFiles) && is_string($previousFiles)) { |
24
|
|
|
$previousFiles = json_decode($previousFiles, true); |
25
|
|
|
} |
26
|
|
|
|
27
|
|
|
if ($filesToDelete) { |
28
|
|
|
foreach ($previousFiles as $previousFile) { |
29
|
|
|
if (in_array($previousFile, $filesToDelete)) { |
30
|
|
|
Storage::disk($this->getDisk())->delete($previousFile); |
31
|
|
|
|
32
|
|
|
$previousFiles = Arr::where($previousFiles, function ($value, $key) use ($previousFile) { |
|
|
|
|
33
|
|
|
return $value != $previousFile; |
34
|
|
|
}); |
35
|
|
|
} |
36
|
|
|
} |
37
|
|
|
} |
38
|
|
|
|
39
|
|
|
foreach ($value ?? [] as $file) { |
40
|
|
|
if ($file && is_file($file)) { |
41
|
|
|
$fileName = $this->getFileName($file); |
42
|
|
|
|
43
|
|
|
$file->storeAs($this->getPath(), $fileName, $this->getDisk()); |
44
|
|
|
|
45
|
|
|
$previousFiles[] = $this->getPath().$fileName; |
46
|
|
|
} |
47
|
|
|
} |
48
|
|
|
|
49
|
|
|
return isset($entry->getCasts()[$this->getName()]) ? $previousFiles : json_encode($previousFiles); |
|
|
|
|
50
|
|
|
} |
51
|
|
|
|
52
|
|
|
public function uploadRepeatableFile(Model $entry, $files = null) |
53
|
|
|
{ |
54
|
|
|
$previousFiles = $this->getPreviousRepeatableValues($entry); |
55
|
|
|
$fileOrder = $this->getFileOrderFromRequest(); |
56
|
|
|
|
57
|
|
|
foreach ($files as $row => $files) { |
58
|
|
|
foreach ($files ?? [] as $file) { |
59
|
|
|
if ($file && is_file($file)) { |
60
|
|
|
$fileName = $this->getFileName($file); |
61
|
|
|
|
62
|
|
|
$file->storeAs($this->getPath(), $fileName, $this->getDisk()); |
63
|
|
|
$fileOrder[$row][] = $this->getPath().$fileName; |
64
|
|
|
} |
65
|
|
|
} |
66
|
|
|
} |
67
|
|
|
|
68
|
|
|
foreach ($previousFiles as $previousRow => $files) { |
69
|
|
|
foreach ($files ?? [] as $key => $file) { |
70
|
|
|
$key = array_search($file, $fileOrder, true); |
71
|
|
|
if ($key === false) { |
72
|
|
|
Storage::disk($this->getDisk())->delete($file); |
73
|
|
|
} |
74
|
|
|
} |
75
|
|
|
} |
76
|
|
|
|
77
|
|
|
return $fileOrder; |
78
|
|
|
} |
79
|
|
|
} |
80
|
|
|
|
In the issue above, the returned value is violating the contract defined by the mentioned interface.
Let's take a look at an example: