Compare commits
376 Commits
master
...
ccb0399d91
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ccb0399d91 | ||
|
|
062d3a9141 | ||
|
|
9ba8f4acf4 | ||
|
|
e080d98e81 | ||
|
|
bd63a92604 | ||
|
|
dc05eb31ac | ||
|
|
5bee7c5e50 | ||
|
|
31500f3386 | ||
|
|
0f58dfe1f4 | ||
|
|
c0737e3bce | ||
|
|
7913576f1a | ||
|
|
d88b8c8a70 | ||
|
|
f2a87f4d12 | ||
|
|
c338a8afc7 | ||
|
|
29c21367d7 | ||
|
|
09b6c3cf28 | ||
|
|
a34905f34e | ||
|
|
8d8528f6fb | ||
|
|
e69487f73e | ||
|
|
a2da0a8a12 | ||
|
|
b7634e07c0 | ||
|
|
ece0b433fd | ||
|
|
c94d815e5a | ||
|
|
0d0e7c4652 | ||
|
|
0e9d758e6b | ||
|
|
883d2d634d | ||
|
|
1503f21b3b | ||
|
|
17fa80b96b | ||
|
|
92c9975e81 | ||
|
|
b214d8dc74 | ||
|
|
08fc49cb61 | ||
|
|
0f32386406 | ||
|
|
4023fd9ee4 | ||
|
|
1e32fd7cf1 | ||
|
|
870fc6c806 | ||
|
|
045641e687 | ||
|
|
38e477ee5f | ||
|
|
561cd2248c | ||
|
|
0a538ee1a2 | ||
|
|
e490c27837 | ||
|
|
e92f38f6c3 | ||
|
|
296dc4fde2 | ||
|
|
e4e2126ada | ||
|
|
44391e817c | ||
|
|
5359375210 | ||
|
|
c15a6e380f | ||
|
|
3acb69abe4 | ||
|
|
00fc978217 | ||
|
|
7a33245c1e | ||
|
|
927be4ea77 | ||
|
|
b0b87db86d | ||
|
|
53448d7907 | ||
|
|
e920157c0b | ||
|
|
8f905ce1eb | ||
|
|
b95da54240 | ||
|
|
e6ad147790 | ||
|
|
5fc8d3dc57 | ||
|
|
4caf069011 | ||
|
|
d0e97ccd08 | ||
|
|
38ce69b525 | ||
|
|
b67e91d228 | ||
|
|
94f84a7a85 | ||
|
|
54ab6b0231 | ||
|
|
9e30118088 | ||
|
|
6c88e43b74 | ||
|
|
5c20e6d5d0 | ||
|
|
79af996c63 | ||
|
|
c0c4146c9c | ||
|
|
84bc5f2e67 | ||
|
|
72f5da4555 | ||
|
|
349204307f | ||
|
|
36459de793 | ||
|
|
d432827bab | ||
|
|
930695166d | ||
|
|
8c39035cdc | ||
|
|
6a1d8ba80a | ||
|
|
6f8340b90b | ||
|
|
c5f06a608c | ||
|
|
50111787aa | ||
|
|
b80c2a8d41 | ||
|
|
2eea4ec73d | ||
|
|
88a57a9c32 | ||
|
|
7e5b1fea89 | ||
|
|
c053221662 | ||
|
|
25b8fc40c5 | ||
|
|
d33dd6e755 | ||
|
|
04f92c9695 | ||
|
|
7c8546e450 | ||
|
|
585d42264f | ||
|
|
9f4ef99ce2 | ||
|
|
501f3a5ab9 | ||
|
|
0ff39c1401 | ||
|
|
71ecb13af0 | ||
|
|
7561a8e8ba | ||
|
|
bbf4bc4232 | ||
|
|
5d5ea92127 | ||
|
|
a84995e0ce | ||
|
|
fcc564b6d8 | ||
|
|
be468304c9 | ||
|
|
b812a78f9f | ||
|
|
0ec0b8e85b | ||
|
|
9d520abfcf | ||
|
|
174c4ca0e2 | ||
|
|
216c408596 | ||
|
|
3cdb30a0dc | ||
|
|
5bcf265cda | ||
|
|
7ec1d3e89b | ||
|
|
741f389620 | ||
|
|
37f1ebbb59 | ||
|
|
81974d38ab | ||
|
|
cc9672ddca | ||
|
|
12903b6029 | ||
|
|
e6b15e2438 | ||
|
|
002644cb97 | ||
|
|
dfe7625995 | ||
|
|
53feef282f | ||
|
|
69478e3c74 | ||
|
|
ab8ab8eba1 | ||
|
|
de5f3d665d | ||
|
|
afaf0cc629 | ||
|
|
419a4c2b9a | ||
|
|
d750cdced7 | ||
|
|
01f9c9ae7c | ||
|
|
043b7d7215 | ||
|
|
f85df32c67 | ||
|
|
6f2a985edc | ||
|
|
5f1ff26196 | ||
|
|
ddb4520621 | ||
|
|
c619f540f8 | ||
|
|
39c80ce6d1 | ||
|
|
297dcc62d2 | ||
|
|
b011f40b2f | ||
|
|
fdbf819bf5 | ||
|
|
f4738377c7 | ||
|
|
99bdf09be7 | ||
|
|
b283363543 | ||
|
|
6649c32501 | ||
|
|
bc1cf1190b | ||
|
|
3dc6c70c4d | ||
|
|
6e4f93dd65 | ||
|
|
ba8f87cff4 | ||
|
|
efb3fe9670 | ||
|
|
23e6ca35ca | ||
|
|
695d23a139 | ||
|
|
3943fc033f | ||
|
|
bb77a199eb | ||
|
|
808e60e41b | ||
|
|
260eccc1f1 | ||
|
|
abed17b3f0 | ||
|
|
3ad83b870a | ||
|
|
241b255b1c | ||
|
|
d2ec87c374 | ||
|
|
24b86652aa | ||
|
|
923f988aba | ||
|
|
0778fdc10a | ||
|
|
900aa413a2 | ||
|
|
820a200e88 | ||
|
|
186e3de863 | ||
|
|
7722b73be9 | ||
|
|
685160ddf5 | ||
|
|
7449229ff7 | ||
|
|
c5ae71544a | ||
|
|
926996166b | ||
|
|
fd8e89e33c | ||
|
|
503efc6f25 | ||
|
|
e8ae460bab | ||
|
|
bb1e1a6b18 | ||
|
|
02b2d3e410 | ||
|
|
72870680cf | ||
|
|
f3289334f3 | ||
|
|
12e5dbb721 | ||
|
|
b42ae14428 | ||
|
|
ea5350a0a6 | ||
|
|
90d78bc3e3 | ||
|
|
a85e5f2921 | ||
|
|
5e6af2e5f8 | ||
|
|
11aaaa024c | ||
|
|
fd1ab5cf04 | ||
|
|
a03befbf44 | ||
|
|
ce7755ffe3 | ||
|
|
17792fce40 | ||
|
|
9698ba54d0 | ||
|
|
ae39681cb0 | ||
|
|
9895266798 | ||
|
|
c1a9c21e52 | ||
|
|
3a18679a1c | ||
|
|
6ef31f286e | ||
|
|
52193994a3 | ||
|
|
71c0489862 | ||
|
|
d33b5eea52 | ||
|
|
782809829b | ||
|
|
ebe7ba5f6c | ||
|
|
dae8156164 | ||
|
|
1880b25407 | ||
|
|
5819f51f79 | ||
|
|
01f56204b7 | ||
|
|
eadea3958d | ||
|
|
719e4481d7 | ||
|
|
eae2cdf345 | ||
|
|
ab145dd60f | ||
|
|
8054bffb43 | ||
|
|
d50ecd674e | ||
|
|
1703082b3e | ||
|
|
35310b049e | ||
|
|
4801ff4338 | ||
|
|
b2987e5eed | ||
|
|
8b334bc8ed | ||
|
|
79c717ae6c | ||
|
|
8abf391a53 | ||
|
|
cc2798a78e | ||
|
|
ce9f613b66 | ||
|
|
3370b8061c | ||
|
|
8c898bf63b | ||
|
|
fedf463f15 | ||
|
|
73508f6b46 | ||
|
|
c79facd0ca | ||
|
|
3ba5a833b2 | ||
|
|
c77db883c2 | ||
|
|
904110d10f | ||
|
|
3c8fab27da | ||
|
|
ee0954931f | ||
|
|
d399b72120 | ||
|
|
6fc6451f6f | ||
|
|
f3e9ccaa46 | ||
|
|
1fc860715f | ||
|
|
6837954fc9 | ||
|
|
483aa59750 | ||
|
|
f460865a57 | ||
|
|
4d31b1682c | ||
|
|
ee148a27ed | ||
|
|
79e5a6388a | ||
|
|
165262abfa | ||
|
|
2f77b5fc23 | ||
|
|
8afb3467f8 | ||
|
|
e3c60e7cde | ||
|
|
7187a312eb | ||
|
|
e208dc910b | ||
|
|
d4bb94a487 | ||
|
|
e77ed62066 | ||
|
|
8dddd4d99b | ||
|
|
edf43ff270 | ||
|
|
36c6402f04 | ||
|
|
8ba8d9a276 | ||
|
|
cdce15a29f | ||
|
|
c12b45f8ad | ||
|
|
081df4d5d0 | ||
|
|
2d86f162a0 | ||
|
|
c2b7315abf | ||
|
|
7b265d3b47 | ||
|
|
c90dd98319 | ||
|
|
193f5cf4ad | ||
|
|
930eeaede8 | ||
|
|
36a63f3b14 | ||
|
|
d9dba1c515 | ||
|
|
411b47f78d | ||
|
|
a550278f39 | ||
|
|
d715d726db | ||
|
|
60acbc7939 | ||
|
|
6a634c10ca | ||
|
|
308f226ca0 | ||
|
|
c0aeda7554 | ||
|
|
3633581ebf | ||
|
|
15539f27a6 | ||
|
|
0a9d03e214 | ||
|
|
36f16921bb | ||
|
|
cbe397be6a | ||
|
|
8dc409bd78 | ||
|
|
56659c73bb | ||
|
|
accc94f56b | ||
|
|
64f5663da5 | ||
|
|
c6ba4aaa24 | ||
|
|
020954a7cc | ||
|
|
61be4a6769 | ||
|
|
cf9439852f | ||
|
|
5bf3b7ec7b | ||
|
|
715e228b41 | ||
|
|
91675dc83b | ||
|
|
e0addd61ce | ||
|
|
66cc7a6054 | ||
|
|
2d6b092b64 | ||
|
|
9ab19b0e4d | ||
|
|
6110b42ce2 | ||
|
|
82a345c4c7 | ||
|
|
c0a80aa62b | ||
|
|
b325a44ee2 | ||
|
|
88d87ceaab | ||
|
|
ba04007f86 | ||
|
|
81dcd273ce | ||
|
|
d899d3dcaa | ||
|
|
9f6d21ef04 | ||
|
|
bdefa235eb | ||
|
|
545c242bba | ||
|
|
e241486143 | ||
|
|
2f81f1d61d | ||
|
|
4d3ffa2ca3 | ||
|
|
f237882757 | ||
|
|
090dcd6f24 | ||
|
|
6db343c6b2 | ||
|
|
d46347a66f | ||
|
|
ab8f68e4bc | ||
|
|
8c9fc66b89 | ||
|
|
2be53d581e | ||
|
|
26f3fc3d2c | ||
|
|
efff4f0341 | ||
|
|
b3f2aa36be | ||
|
|
cdc88b43df | ||
|
|
ecbb7b62c9 | ||
|
|
b1a2e70d12 | ||
|
|
b2f5cc4a45 | ||
|
|
8a1573d425 | ||
|
|
cb0b2e4aa0 | ||
|
|
7092cf23fc | ||
|
|
e434220c16 | ||
|
|
0858804095 | ||
|
|
5f7e90d22d | ||
|
|
6dce60d227 | ||
|
|
f6668a6dd3 | ||
|
|
dd396a86aa | ||
|
|
f3b6504723 | ||
|
|
41cab61bda | ||
|
|
c76996db86 | ||
|
|
f5716c6530 | ||
|
|
459d35e8a1 | ||
|
|
e9ce44481a | ||
|
|
18f1f8a13a | ||
|
|
e97f54f126 | ||
|
|
0d0e4deb16 | ||
|
|
da51da2530 | ||
|
|
c024bdc31e | ||
|
|
4ec0558cd8 | ||
|
|
48d89d338c | ||
|
|
0f23d5cd56 | ||
|
|
9380d99688 | ||
|
|
06fb42e7c7 | ||
|
|
66a830eec0 | ||
|
|
0e73c8109b | ||
|
|
5e4c9963c6 | ||
|
|
f9070b2ec8 | ||
|
|
8107078ea7 | ||
|
|
be3b6bc0a8 | ||
|
|
290602057a | ||
|
|
a1a3ccb227 | ||
|
|
e9002fb494 | ||
|
|
cffec143e0 | ||
|
|
385873a0cf | ||
|
|
8842b7eea7 | ||
|
|
0cd0e1f126 | ||
|
|
7c01366281 | ||
|
|
fa193a8089 | ||
|
|
27c5663ba1 | ||
|
|
cfd8086586 | ||
|
|
c3a05640ab | ||
|
|
ca691c8fdd | ||
|
|
47ff07451f | ||
|
|
5bbf63b33f | ||
|
|
768c7e88f9 | ||
|
|
23b8914187 | ||
|
|
aee50813e4 | ||
|
|
331b7647de | ||
|
|
b1494477a2 | ||
|
|
31b671bcdd | ||
|
|
9e2226a776 | ||
|
|
9ca510086b | ||
|
|
7ec40145de | ||
|
|
734ec87b89 | ||
|
|
0d421226fa | ||
|
|
d174fe1c81 | ||
|
|
64abc46d99 | ||
|
|
f4ab8e71a8 | ||
|
|
763577bf87 | ||
|
|
226153f744 | ||
|
|
e298320119 | ||
|
|
b90d633e6e | ||
|
|
31e60dee45 | ||
|
|
0dd3df2984 | ||
|
|
d53e9f5a4f |
@@ -43,7 +43,6 @@ COPY . /app
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
RUN mkdir -p /app/bootstrap/cache \
|
RUN mkdir -p /app/bootstrap/cache \
|
||||||
/app/storage/media-library/temp \
|
|
||||||
/app/storage/framework/cache \
|
/app/storage/framework/cache \
|
||||||
/app/storage/framework/views \
|
/app/storage/framework/views \
|
||||||
/app/storage/framework/sessions \
|
/app/storage/framework/sessions \
|
||||||
@@ -85,6 +84,8 @@ RUN apk add --no-cache xz
|
|||||||
# bring PHP app with vendor
|
# bring PHP app with vendor
|
||||||
COPY --from=phpdeps /app /app
|
COPY --from=phpdeps /app /app
|
||||||
|
|
||||||
|
# ensure required runtime dirs exist (empty is fine)
|
||||||
|
RUN mkdir -p storage/framework/{cache,views,sessions} bootstrap/cache
|
||||||
# create artifact (use tar + xz so we don't depend on GNU tar -J)
|
# create artifact (use tar + xz so we don't depend on GNU tar -J)
|
||||||
RUN mkdir -p /out \
|
RUN mkdir -p /out \
|
||||||
&& tar -C /app -cf /out/app.tar \
|
&& tar -C /app -cf /out/app.tar \
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
namespace App\Http\Controllers\Admin\Botanic;
|
namespace App\Http\Controllers\Admin\Botanic;
|
||||||
|
|
||||||
use App\Datatables\Botanic\SpeciesDataTable;
|
use App\Datatables\Botanic\SpeciesDataTable;
|
||||||
|
use App\Repositories\Botanic\Genres;
|
||||||
use App\Repositories\Botanic\Species;
|
use App\Repositories\Botanic\Species;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
|
||||||
@@ -20,7 +21,7 @@ class SpecieController extends Controller
|
|||||||
|
|
||||||
public function create()
|
public function create()
|
||||||
{
|
{
|
||||||
$data = Species::init();
|
$data = Genres::init();
|
||||||
|
|
||||||
return view('Admin.Botanic.Species.create', $data);
|
return view('Admin.Botanic.Species.create', $data);
|
||||||
}
|
}
|
||||||
@@ -35,7 +36,7 @@ class SpecieController extends Controller
|
|||||||
|
|
||||||
public function edit($id)
|
public function edit($id)
|
||||||
{
|
{
|
||||||
$data = Species::init();
|
$data = Genres::init();
|
||||||
$data['specie'] = Species::getFull($id);
|
$data['specie'] = Species::getFull($id);
|
||||||
|
|
||||||
return view('Admin.Botanic.Species.edit', $data);
|
return view('Admin.Botanic.Species.edit', $data);
|
||||||
|
|||||||
@@ -71,10 +71,7 @@ class PriceListValueController extends Controller
|
|||||||
|
|
||||||
public function addPrice($index)
|
public function addPrice($index)
|
||||||
{
|
{
|
||||||
$data = [
|
$data['index'] = $index;
|
||||||
'index' => $index,
|
|
||||||
'taxes' => Taxes::getOptions(),
|
|
||||||
];
|
|
||||||
|
|
||||||
return view('Admin.Shop.PriceListValues.partials.row_price', $data);
|
return view('Admin.Shop.PriceListValues.partials.row_price', $data);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
namespace App\Http\Requests\Admin\Shop;
|
namespace App\Http\Requests\Admin\Shop;
|
||||||
|
|
||||||
use Illuminate\Foundation\Http\FormRequest;
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
use Illuminate\Validation\Rule;
|
|
||||||
|
|
||||||
class StoreArticlePost extends FormRequest
|
class StoreArticlePost extends FormRequest
|
||||||
{
|
{
|
||||||
@@ -14,13 +13,8 @@ class StoreArticlePost extends FormRequest
|
|||||||
|
|
||||||
public function rules()
|
public function rules()
|
||||||
{
|
{
|
||||||
$articleId = $this->input('id');
|
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'ref' => [
|
'ref' => 'required|unique:shop_articles',
|
||||||
'required',
|
|
||||||
Rule::unique('shop_articles', 'ref')->ignore($articleId)->whereNull('deleted_at'),
|
|
||||||
],
|
|
||||||
'product_type' => 'required',
|
'product_type' => 'required',
|
||||||
'product_id' => 'required',
|
'product_id' => 'required',
|
||||||
'article_nature_id' => 'required',
|
'article_nature_id' => 'required',
|
||||||
|
|||||||
@@ -11,6 +11,9 @@ class AppServiceProvider extends ServiceProvider
|
|||||||
{
|
{
|
||||||
public function boot()
|
public function boot()
|
||||||
{
|
{
|
||||||
|
if (config('app.env') === 'production') {
|
||||||
|
\URL::forceScheme('https');
|
||||||
|
}
|
||||||
Schema::defaultStringLength(191);
|
Schema::defaultStringLength(191);
|
||||||
View::composer('Shop.layout.layout', LayoutComposer::class);
|
View::composer('Shop.layout.layout', LayoutComposer::class);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -65,9 +65,6 @@ class Varieties
|
|||||||
{
|
{
|
||||||
$images = $data['images'] ?? false;
|
$images = $data['images'] ?? false;
|
||||||
$tags = $data['tags'] ?? false;
|
$tags = $data['tags'] ?? false;
|
||||||
if (! array_key_exists('plus', $data) || $data['plus'] === null) {
|
|
||||||
$data['plus'] = '';
|
|
||||||
}
|
|
||||||
unset($data['images']);
|
unset($data['images']);
|
||||||
unset($data['tags']);
|
unset($data['tags']);
|
||||||
$variety = self::store($data);
|
$variety = self::store($data);
|
||||||
|
|||||||
@@ -2,8 +2,6 @@
|
|||||||
|
|
||||||
namespace App\Repositories\Core;
|
namespace App\Repositories\Core;
|
||||||
|
|
||||||
use Spatie\MediaLibrary\MediaCollections\Models\Media as MediaModel;
|
|
||||||
|
|
||||||
class Medias
|
class Medias
|
||||||
{
|
{
|
||||||
public static function getImage($model, $conversion = 'normal', $collection = 'images')
|
public static function getImage($model, $conversion = 'normal', $collection = 'images')
|
||||||
@@ -81,9 +79,13 @@ class Medias
|
|||||||
|
|
||||||
public static function getImageSrc($image)
|
public static function getImageSrc($image)
|
||||||
{
|
{
|
||||||
$media = self::resolveMedia($image);
|
if (! $image) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
$id = $image['id'];
|
||||||
|
$filename = self::getFilename($image);
|
||||||
|
|
||||||
return $media ? $media->getUrl() : null;
|
return "/storage/{$id}/{$filename}";
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getThumbSrc($image)
|
public static function getThumbSrc($image)
|
||||||
@@ -108,12 +110,6 @@ class Medias
|
|||||||
|
|
||||||
public static function getSrcByType($image, $type)
|
public static function getSrcByType($image, $type)
|
||||||
{
|
{
|
||||||
$media = self::resolveMedia($image);
|
|
||||||
|
|
||||||
if ($media) {
|
|
||||||
return $type ? $media->getUrl($type) : $media->getUrl();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $image ? '/storage/'.$image['id'].'/conversions/'.self::getFilename($image, $type) : false;
|
return $image ? '/storage/'.$image['id'].'/conversions/'.self::getFilename($image, $type) : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -128,48 +124,4 @@ class Medias
|
|||||||
{
|
{
|
||||||
return str_replace(['#', '/', '\\', ' '], '-', $name);
|
return str_replace(['#', '/', '\\', ' '], '-', $name);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static function resolveMedia($image): ?MediaModel
|
|
||||||
{
|
|
||||||
if ($image instanceof MediaModel) {
|
|
||||||
return $image;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_null($image)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_array($image)) {
|
|
||||||
return self::hydrateMedia($image);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_object($image)) {
|
|
||||||
if ($image instanceof \ArrayAccess) {
|
|
||||||
return self::hydrateMedia((array) $image);
|
|
||||||
}
|
|
||||||
|
|
||||||
$array = method_exists($image, 'toArray') ? $image->toArray() : (array) $image;
|
|
||||||
|
|
||||||
return self::hydrateMedia($array);
|
|
||||||
}
|
|
||||||
|
|
||||||
$id = data_get($image, 'id');
|
|
||||||
|
|
||||||
return $id ? MediaModel::query()->withoutGlobalScopes()->find($id) : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static function hydrateMedia(array $attributes): ?MediaModel
|
|
||||||
{
|
|
||||||
$id = data_get($attributes, 'id');
|
|
||||||
|
|
||||||
if (! $id) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$media = new MediaModel();
|
|
||||||
$media->forceFill($attributes);
|
|
||||||
$media->exists = true;
|
|
||||||
|
|
||||||
return $media;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,34 +27,21 @@ class ArticleTags
|
|||||||
|
|
||||||
switch ($article->product_type) {
|
switch ($article->product_type) {
|
||||||
case 'App\Models\Botanic\Variety':
|
case 'App\Models\Botanic\Variety':
|
||||||
$variety = $article->product;
|
$data += $article->product->tags->toArray();
|
||||||
if ($variety && $variety->tags) {
|
if ($article->product->specie ?? false) {
|
||||||
$data = array_merge($data, $variety->tags->toArray());
|
$data += $article->product->specie->tags->toArray();
|
||||||
}
|
|
||||||
if ($variety && $variety->specie && $variety->specie->tags) {
|
|
||||||
$data = array_merge($data, $variety->specie->tags->toArray());
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'App\Models\Botanic\Specie':
|
case 'App\Models\Botanic\Specie':
|
||||||
$specie = $article->product;
|
$data += $article->product->tags->toArray();
|
||||||
if ($specie && $specie->tags) {
|
|
||||||
$data = array_merge($data, $specie->tags->toArray());
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case 'App\Models\Shop\Merchandise':
|
case 'App\Models\Shop\Merchandise':
|
||||||
$merchandise = $article->product;
|
$data += $article->product->tags->toArray();
|
||||||
if ($merchandise && $merchandise->tags) {
|
$data += $article->product->producer->tags->toArray();
|
||||||
$data = array_merge($data, $merchandise->tags->toArray());
|
|
||||||
}
|
|
||||||
if ($merchandise && $merchandise->producer && $merchandise->producer->tags) {
|
|
||||||
$data = array_merge($data, $merchandise->producer->tags->toArray());
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
if ($article->tags) {
|
$data += $article->tags->toArray();
|
||||||
$data = array_merge($data, $article->tags->toArray());
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($data as $tag) {
|
foreach ($data as $tag) {
|
||||||
if (! isset($tags[$tag['group']][$tag['name']])) {
|
if (! isset($tags[$tag['group']][$tag['name']])) {
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ class Articles
|
|||||||
|
|
||||||
public static function autocomplete($str)
|
public static function autocomplete($str)
|
||||||
{
|
{
|
||||||
$data = Article::byAutocomplete($str)->visible()->orderBy('name')->limit(20)->pluck('name', 'id');
|
$data = Article::byAutocomplete($str)->orderBy('name')->limit(20)->pluck('name', 'id');
|
||||||
$export = [];
|
$export = [];
|
||||||
foreach ($data as $key => $name) {
|
foreach ($data as $key => $name) {
|
||||||
$export[] = ['value' => $key, 'text' => $name];
|
$export[] = ['value' => $key, 'text' => $name];
|
||||||
@@ -135,11 +135,8 @@ class Articles
|
|||||||
$data['specie'] = $article->product ? $article->product->description : '';
|
$data['specie'] = $article->product ? $article->product->description : '';
|
||||||
break;
|
break;
|
||||||
case 'App\Models\Shop\Merchandise':
|
case 'App\Models\Shop\Merchandise':
|
||||||
$merchandise = $article->product;
|
$data['merchandise'] = $article->product ? $article->product->description : '';
|
||||||
$data['merchandise'] = $merchandise ? ($merchandise->description ?? '') : '';
|
$data['producer'] = $article->product->producer->description;
|
||||||
if ($merchandise && $merchandise->producer) {
|
|
||||||
$data['producer'] = $merchandise->producer->description ?? '';
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
namespace App\Repositories\Shop;
|
namespace App\Repositories\Shop;
|
||||||
|
|
||||||
use App\Models\Shop\Offer;
|
use App\Models\Shop\Offer;
|
||||||
use App\Models\Shop\PriceList;
|
|
||||||
use App\Models\Shop\PriceListValue;
|
use App\Models\Shop\PriceListValue;
|
||||||
use App\Models\Shop\SaleChannel;
|
use App\Models\Shop\SaleChannel;
|
||||||
use App\Traits\Model\Basic;
|
use App\Traits\Model\Basic;
|
||||||
@@ -186,44 +185,20 @@ class Offers
|
|||||||
->with([
|
->with([
|
||||||
'article',
|
'article',
|
||||||
'tariff:id,status_id',
|
'tariff:id,status_id',
|
||||||
'variation',
|
|
||||||
])
|
])
|
||||||
->get();
|
->get();
|
||||||
|
|
||||||
return $channels->map(function ($channel) use ($offers) {
|
return $channels->map(function ($channel) use ($offers) {
|
||||||
$priceValue = null;
|
$priceValue = null;
|
||||||
$candidateOffer = null;
|
$candidateOffer = null;
|
||||||
$allOffersForChannel = [];
|
|
||||||
|
|
||||||
foreach ($offers as $offer) {
|
foreach ($offers as $offer) {
|
||||||
$priceCandidate = self::getPrice($offer->id, 1, $channel->id);
|
$priceCandidate = self::getPrice($offer->id, 1, $channel->id);
|
||||||
|
|
||||||
if ($priceCandidate && (float) $priceCandidate->price_taxed > 0) {
|
if ($priceCandidate && (float) $priceCandidate->price_taxed > 0) {
|
||||||
// Get price list name
|
$priceValue = $priceCandidate;
|
||||||
$priceListName = null;
|
$candidateOffer = $offer;
|
||||||
if ($priceCandidate) {
|
break;
|
||||||
$priceListModel = PriceList::find($priceCandidate->price_list_id);
|
|
||||||
$priceListName = $priceListModel ? $priceListModel->name : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Collect all offers with their details
|
|
||||||
$allOffersForChannel[] = [
|
|
||||||
'id' => $offer->id,
|
|
||||||
'variation_name' => $offer->variation ? $offer->variation->name : null,
|
|
||||||
'stock_current' => (int) $offer->stock_current,
|
|
||||||
'status_id' => (int) $offer->status_id,
|
|
||||||
'is_active' => (int) $offer->status_id === 1,
|
|
||||||
'tariff_id' => $offer->tariff_id ? (int) $offer->tariff_id : null,
|
|
||||||
'price_taxed' => (float) $priceCandidate->price_taxed,
|
|
||||||
'quantity' => (int) $priceCandidate->quantity,
|
|
||||||
'price_list_name' => $priceListName,
|
|
||||||
];
|
|
||||||
|
|
||||||
// Keep first valid offer as the main candidate
|
|
||||||
if (!$candidateOffer) {
|
|
||||||
$priceValue = $priceCandidate;
|
|
||||||
$candidateOffer = $offer;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -234,7 +209,6 @@ class Offers
|
|||||||
$offerHasStock = $candidateOffer && $candidateOffer->stock_current !== null
|
$offerHasStock = $candidateOffer && $candidateOffer->stock_current !== null
|
||||||
? (float) $candidateOffer->stock_current > 0
|
? (float) $candidateOffer->stock_current > 0
|
||||||
: null;
|
: null;
|
||||||
$offerTariffId = $candidateOffer && $candidateOffer->tariff_id ? (int) $candidateOffer->tariff_id : null;
|
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'id' => $channel->id,
|
'id' => $channel->id,
|
||||||
@@ -247,8 +221,6 @@ class Offers
|
|||||||
'offer_stock_current' => $offerStock,
|
'offer_stock_current' => $offerStock,
|
||||||
'offer_has_stock' => $offerHasStock,
|
'offer_has_stock' => $offerHasStock,
|
||||||
'tariff_status_id' => $offerTariffStatus,
|
'tariff_status_id' => $offerTariffStatus,
|
||||||
'tariff_id' => $offerTariffId,
|
|
||||||
'all_offers' => $allOffersForChannel,
|
|
||||||
];
|
];
|
||||||
})->toArray();
|
})->toArray();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -46,34 +46,12 @@ class PriceListValues
|
|||||||
{
|
{
|
||||||
foreach ($values as $value) {
|
foreach ($values as $value) {
|
||||||
$value['price_list_id'] = $price_list_id;
|
$value['price_list_id'] = $price_list_id;
|
||||||
if (self::hasPrice($value)) {
|
if ($value['price']) {
|
||||||
self::store($value);
|
self::store($value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function purgeRemovedValues($price_list_id, array $ids)
|
|
||||||
{
|
|
||||||
if (! count($ids)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
PriceListValue::byPriceList($price_list_id)
|
|
||||||
->whereIn('id', $ids)
|
|
||||||
->delete();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static function hasPrice($value): bool
|
|
||||||
{
|
|
||||||
if (! array_key_exists('price', $value)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$price = $value['price'];
|
|
||||||
|
|
||||||
return $price !== null && $price !== '';
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function getModel()
|
public static function getModel()
|
||||||
{
|
{
|
||||||
return PriceListValue::query();
|
return PriceListValue::query();
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ class PriceLists
|
|||||||
'taxes' => Taxes::getOptions(),
|
'taxes' => Taxes::getOptions(),
|
||||||
'price_list' => [
|
'price_list' => [
|
||||||
'tariff_id' => $tariffId,
|
'tariff_id' => $tariffId,
|
||||||
'price_list_values' => [],
|
'price_list_values' => array_fill(0, 3, ''),
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -50,8 +50,9 @@ class PriceLists
|
|||||||
public static function edit($id)
|
public static function edit($id)
|
||||||
{
|
{
|
||||||
$price_list = self::getFull($id)->toArray();
|
$price_list = self::getFull($id)->toArray();
|
||||||
if (count($price_list['price_list_values']) === 0) {
|
$n = count($price_list['price_list_values']);
|
||||||
$price_list['price_list_values'][] = [];
|
if ($n <= 3) {
|
||||||
|
$price_list['price_list_values'] += array_fill($n, 3 - $n, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
return $price_list;
|
return $price_list;
|
||||||
@@ -70,14 +71,9 @@ class PriceLists
|
|||||||
public static function store($data)
|
public static function store($data)
|
||||||
{
|
{
|
||||||
$id = $data['id'] ?? false;
|
$id = $data['id'] ?? false;
|
||||||
$price_list_values = $data['price_list_values'] ?? [];
|
$price_list_values = $data['price_list_values'] ?? false;
|
||||||
$deleted_values = array_map('intval', array_filter($data['deleted_price_list_value_ids'] ?? [], function ($value) {
|
|
||||||
return $value !== null && $value !== '';
|
|
||||||
}));
|
|
||||||
unset($data['price_list_values']);
|
unset($data['price_list_values']);
|
||||||
unset($data['deleted_price_list_value_ids']);
|
|
||||||
$price_list = $id ? self::update($data) : self::create($data);
|
$price_list = $id ? self::update($data) : self::create($data);
|
||||||
PriceListValues::purgeRemovedValues($price_list->id, $deleted_values);
|
|
||||||
PriceListValues::storePrices($price_list->id, $price_list_values);
|
PriceListValues::storePrices($price_list->id, $price_list_values);
|
||||||
|
|
||||||
return $price_list;
|
return $price_list;
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ class SaleChannels
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return self::getByCode('POSTE');
|
return self::getByCode('EXP');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getByCode($code)
|
public static function getByCode($code)
|
||||||
|
|||||||
@@ -8,14 +8,8 @@ class Searches
|
|||||||
{
|
{
|
||||||
public static function search($options)
|
public static function search($options)
|
||||||
{
|
{
|
||||||
// Get article IDs from Scout search
|
|
||||||
$searchResults = Article::search($options['search_name'])->get()->pluck('id');
|
|
||||||
|
|
||||||
// Filter to only include visible articles
|
|
||||||
$visibleArticleIds = Article::whereIn('id', $searchResults)->visible()->pluck('id');
|
|
||||||
|
|
||||||
return collect(Articles::getArticlesToSell([
|
return collect(Articles::getArticlesToSell([
|
||||||
'ids' => $visibleArticleIds,
|
'ids' => Article::search($options['search_name'])->get()->pluck('id'),
|
||||||
]))->sortBy('searchOrder')->toArray();
|
]))->sortBy('searchOrder')->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,35 +1,23 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
return [
|
return [
|
||||||
// Default configuration group for Gravatar
|
// --- The default avatar size
|
||||||
'default' => [
|
'size' => 80,
|
||||||
// --- The default avatar size
|
|
||||||
'size' => 80,
|
|
||||||
|
|
||||||
// --- The default avatar to display if we have no results
|
// --- The default avatar to display if we have no results
|
||||||
// (bool) false
|
// (bool) false
|
||||||
// (string) 404
|
// (string) 404
|
||||||
// (string) mm: (mystery-man) a simple, cartoon-style silhouetted outline of a person (does not vary by email hash).
|
// (string) mm: (mystery-man) a simple, cartoon-style silhouetted outline of a person (does not vary by email hash).
|
||||||
// (string) identicon: a geometric pattern based on an email hash.
|
// (string) identicon: a geometric pattern based on an email hash.
|
||||||
// (string) monsterid: a generated 'monster' with different colors, faces, etc.
|
// (string) monsterid: a generated 'monster' with different colors, faces, etc.
|
||||||
// (string) wavatar: generated faces with differing features and backgrounds.
|
// (string) wavatar: generated faces with differing features and backgrounds.
|
||||||
// (string) retro: awesome generated, 8-bit arcade-style pixelated faces.
|
// (string) retro: awesome generated, 8-bit arcade-style pixelated faces.
|
||||||
'fallback' => 'identicon',
|
'default' => 'identicon',
|
||||||
|
|
||||||
// --- Whether to use HTTPS protocol
|
// --- Set the type of avatars we allow to show
|
||||||
'secure' => false,
|
// - g: suitable for display on all websites with any audience type.
|
||||||
|
// - pg: may contain rude gestures, provocatively dressed individuals, the lesser swear words, or mild violence.
|
||||||
// --- Set the type of avatars we allow to show
|
// - r: may contain such things as harsh profanity, intense violence, nudity, or hard drug use.
|
||||||
// - g: suitable for display on all websites with any audience type.
|
// - x: may contain hardcore sexual imagery or extremely disturbing violence.
|
||||||
// - pg: may contain rude gestures, provocatively dressed individuals, the lesser swear words, or mild violence.
|
'maxRating' => 'g',
|
||||||
// - r: may contain such things as harsh profanity, intense violence, nudity, or hard drug use.
|
|
||||||
// - x: may contain hardcore sexual imagery or extremely disturbing violence.
|
|
||||||
'maximumRating' => 'g',
|
|
||||||
|
|
||||||
// --- Force default image to always display
|
|
||||||
'forceDefault' => false,
|
|
||||||
|
|
||||||
// --- Optional file extension appended to URL
|
|
||||||
'forceExtension' => 'jpg',
|
|
||||||
]
|
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -126,7 +126,7 @@ return [
|
|||||||
* The path where to store temporary files while performing image conversions.
|
* The path where to store temporary files while performing image conversions.
|
||||||
* If set to null, storage_path('media-library/temp') will be used.
|
* If set to null, storage_path('media-library/temp') will be used.
|
||||||
*/
|
*/
|
||||||
'temporary_directory_path' => env('MEDIA_LIBRARY_TEMP_PATH', storage_path('media-library/temp')),
|
'temporary_directory_path' => null,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The engine that should perform the image conversions.
|
* The engine that should perform the image conversions.
|
||||||
|
|||||||
@@ -1,30 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
use Illuminate\Database\Migrations\Migration;
|
|
||||||
use Illuminate\Database\Schema\Blueprint;
|
|
||||||
use Illuminate\Support\Facades\Schema;
|
|
||||||
|
|
||||||
return new class extends Migration
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Run the migrations.
|
|
||||||
*/
|
|
||||||
public function up(): void
|
|
||||||
{
|
|
||||||
Schema::table('shop_articles', function (Blueprint $table) {
|
|
||||||
$table->dropUnique('ref');
|
|
||||||
$table->unique(['ref', 'deleted_at'], 'shop_articles_ref_deleted_at_unique');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Reverse the migrations.
|
|
||||||
*/
|
|
||||||
public function down(): void
|
|
||||||
{
|
|
||||||
Schema::table('shop_articles', function (Blueprint $table) {
|
|
||||||
$table->dropUnique('shop_articles_ref_deleted_at_unique');
|
|
||||||
$table->unique('ref');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
@@ -15,9 +15,4 @@
|
|||||||
<td>
|
<td>
|
||||||
@include('components.form.inputs.money', ['name' => 'price_list_values[' . $index . '][price_taxed]', 'value' => $price_list_value['price_taxed'] ?? null, 'required' => true, 'class' => 'price_taxed'])
|
@include('components.form.inputs.money', ['name' => 'price_list_values[' . $index . '][price_taxed]', 'value' => $price_list_value['price_taxed'] ?? null, 'required' => true, 'class' => 'price_taxed'])
|
||||||
</td>
|
</td>
|
||||||
<td class="text-center align-middle">
|
|
||||||
<button type="button" class="btn btn-outline-danger btn-xs remove-price" title="{{ __('Supprimer') }}">
|
|
||||||
<i class="fa fa-trash"></i>
|
|
||||||
</button>
|
|
||||||
</td>
|
|
||||||
</tr>
|
</tr>
|
||||||
@@ -30,18 +30,12 @@
|
|||||||
<th>Unit. HT</th>
|
<th>Unit. HT</th>
|
||||||
<th>TVA</th>
|
<th>TVA</th>
|
||||||
<th>Unit. TTC</th>
|
<th>Unit. TTC</th>
|
||||||
<th class="text-center" style="width: 60px;">Actions</th>
|
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
@php($priceListValues = $price_list['price_list_values'] ?? [])
|
@foreach ($price_list['price_list_values'] as $price_list_value)
|
||||||
@php($nextIndex = count($priceListValues))
|
@include('Admin.Shop.PriceListValues.partials.row_price', ['index' => $loop->index])
|
||||||
|
|
||||||
@foreach ($priceListValues as $index => $price_list_value)
|
|
||||||
@include('Admin.Shop.PriceListValues.partials.row_price', ['index' => $index, 'price_list_value' => $price_list_value])
|
|
||||||
@endforeach
|
@endforeach
|
||||||
|
|
||||||
@include('Admin.Shop.PriceListValues.partials.row_price', ['index' => $nextIndex, 'price_list_value' => []])
|
|
||||||
</tbody>
|
</tbody>
|
||||||
<tfoot>
|
<tfoot>
|
||||||
<tr>
|
<tr>
|
||||||
@@ -55,28 +49,19 @@
|
|||||||
</tfoot>
|
</tfoot>
|
||||||
</table>
|
</table>
|
||||||
@endcomponent
|
@endcomponent
|
||||||
<div id="deleted-price-list-values"></div>
|
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
var priceRowIndex = 0;
|
|
||||||
var lastRemovedIndex = null;
|
|
||||||
|
|
||||||
function handleAddPrice() {
|
function handleAddPrice() {
|
||||||
$('#add_price').click( function () {
|
$('#add_price').click( function () {
|
||||||
addEmptyPriceRow();
|
var index = $('#prices-table tbody tr').length;
|
||||||
})
|
$.get("{{ route('Admin.Shop.PriceListValues.addPrice') }}/" + index, function(data) {
|
||||||
}
|
$("#prices-table").append(data);
|
||||||
|
})
|
||||||
function addEmptyPriceRow() {
|
|
||||||
var index = nextPriceRowIndex();
|
|
||||||
$.get("{{ route('Admin.Shop.PriceListValues.addPrice') }}/" + index, function(data) {
|
|
||||||
$("#prices-table tbody").append(data);
|
|
||||||
handlePrices();
|
handlePrices();
|
||||||
handleRemovePrice();
|
})
|
||||||
lastRemovedIndex = null;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function handle_prices() {
|
function handle_prices() {
|
||||||
@@ -116,82 +101,9 @@
|
|||||||
handle_prices_taxed();
|
handle_prices_taxed();
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleRemovePrice() {
|
|
||||||
$('#prices-table').off('click', '.remove-price').on('click', '.remove-price', function() {
|
|
||||||
var $row = $(this).closest('tr');
|
|
||||||
var idx = extractRowIndex($row);
|
|
||||||
var id = $row.find('input[name$="[id]"]').val();
|
|
||||||
if (id) {
|
|
||||||
registerDeletedPrice(id);
|
|
||||||
}
|
|
||||||
$row.remove();
|
|
||||||
lastRemovedIndex = idx;
|
|
||||||
ensureAtLeastOneRow();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function registerDeletedPrice(id) {
|
|
||||||
var $container = $('#deleted-price-list-values');
|
|
||||||
if ($container.find('input[value="' + id + '"]').length === 0) {
|
|
||||||
$container.append('<input type="hidden" name="deleted_price_list_value_ids[]" value="' + id + '">');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function ensureAtLeastOneRow() {
|
|
||||||
if ($('#prices-table tbody tr').length === 0) {
|
|
||||||
if (lastRemovedIndex !== null) {
|
|
||||||
$.get("{{ route('Admin.Shop.PriceListValues.addPrice') }}/" + lastRemovedIndex, function(data) {
|
|
||||||
$("#prices-table tbody").append(data);
|
|
||||||
handlePrices();
|
|
||||||
handleRemovePrice();
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
addEmptyPriceRow();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function computeStartingIndex() {
|
|
||||||
var maxIndex = -1;
|
|
||||||
$('#prices-table tbody tr').each(function() {
|
|
||||||
var $idInput = $(this).find('input[name$="[id]"]');
|
|
||||||
var name = $idInput.attr('name');
|
|
||||||
if (name) {
|
|
||||||
var matches = name.match(/price_list_values\[(\d+)\]\[id\]/);
|
|
||||||
if (matches && matches[1]) {
|
|
||||||
var idx = parseInt(matches[1], 10);
|
|
||||||
if (!isNaN(idx) && idx > maxIndex) {
|
|
||||||
maxIndex = idx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return maxIndex + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
function nextPriceRowIndex() {
|
|
||||||
return priceRowIndex++;
|
|
||||||
}
|
|
||||||
|
|
||||||
function extractRowIndex($row) {
|
|
||||||
var $idInput = $row.find('input[name$="[id]"]');
|
|
||||||
var name = $idInput.attr('name');
|
|
||||||
if (!name) {
|
|
||||||
return priceRowIndex;
|
|
||||||
}
|
|
||||||
|
|
||||||
var matches = name.match(/price_list_values\[(\d+)\]\[id\]/);
|
|
||||||
|
|
||||||
return matches && matches[1] ? parseInt(matches[1], 10) : priceRowIndex;
|
|
||||||
}
|
|
||||||
|
|
||||||
$(function() {
|
$(function() {
|
||||||
priceRowIndex = computeStartingIndex();
|
|
||||||
handleAddPrice();
|
handleAddPrice();
|
||||||
handlePrices();
|
handlePrices();
|
||||||
handleRemovePrice();
|
|
||||||
ensureAtLeastOneRow();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,59 +1,37 @@
|
|||||||
@php
|
@if ($article['offers']['semences'] ?? false)
|
||||||
// Check if article is not visible OR has no offers at all
|
@include('Shop.Articles.partials.addBasket', [
|
||||||
$hasNoOffers = empty($article['offers']['semences'] ?? false)
|
'data' => $article['offers']['semences'],
|
||||||
&& empty($article['offers']['plants'] ?? false)
|
'title' => 'Semences',
|
||||||
&& empty($article['offers']['legumes'] ?? false)
|
'model' => 'semences',
|
||||||
&& empty($article['offers']['marchandise'] ?? false);
|
'bgClass' => 'bg-green-light',
|
||||||
$shouldShowComingSoon = !($article['visible'] ?? true) || $hasNoOffers;
|
])
|
||||||
@endphp
|
@endif
|
||||||
|
|
||||||
@if ($shouldShowComingSoon)
|
@if ($article['offers']['plants'] ?? false)
|
||||||
{{-- Display "Coming Soon" box when article is not visible or has no offers --}}
|
@include('Shop.Articles.partials.addBasket', [
|
||||||
<div class="card border-info">
|
'data' => $article['offers']['plants'],
|
||||||
<div class="card-body text-center p-4">
|
'title' => 'Plants',
|
||||||
<h4 class="text-info mb-3">
|
'model' => 'plants',
|
||||||
<i class="fas fa-clock"></i>
|
'bgClass' => 'bg-green-light',
|
||||||
</h4>
|
])
|
||||||
<h5 class="card-title mb-0">Bientôt disponible</h5>
|
@endif
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@else
|
|
||||||
{{-- Display normal offers for visible articles with available offers --}}
|
|
||||||
@if ($article['offers']['semences'] ?? false)
|
|
||||||
@include('Shop.Articles.partials.addBasket', [
|
|
||||||
'data' => $article['offers']['semences'],
|
|
||||||
'title' => 'Semences',
|
|
||||||
'model' => 'semences',
|
|
||||||
'bgClass' => 'bg-green-light',
|
|
||||||
])
|
|
||||||
@endif
|
|
||||||
|
|
||||||
@if ($article['offers']['plants'] ?? false)
|
@if ($article['offers']['legumes'] ?? false)
|
||||||
@include('Shop.Articles.partials.addBasket', [
|
@include('Shop.Articles.partials.addBasket', [
|
||||||
'data' => $article['offers']['plants'],
|
'data' => $article['offers']['legumes'],
|
||||||
'title' => 'Plants',
|
'title' => 'Légumes',
|
||||||
'model' => 'plants',
|
'model' => 'legumes',
|
||||||
'bgClass' => 'bg-green-light',
|
'bgClass' => 'bg-green-light',
|
||||||
])
|
])
|
||||||
@endif
|
@endif
|
||||||
|
|
||||||
@if ($article['offers']['legumes'] ?? false)
|
@if ($article['offers']['marchandise'] ?? false)
|
||||||
@include('Shop.Articles.partials.addBasket', [
|
@include('Shop.Articles.partials.addBasket', [
|
||||||
'data' => $article['offers']['legumes'],
|
'data' => $article['offers']['marchandise'],
|
||||||
'title' => 'Légumes',
|
'title' => 'Marchandises',
|
||||||
'model' => 'legumes',
|
'model' => 'marchandise',
|
||||||
'bgClass' => 'bg-green-light',
|
'bgClass' => 'bg-green-light',
|
||||||
])
|
])
|
||||||
@endif
|
|
||||||
|
|
||||||
@if ($article['offers']['marchandise'] ?? false)
|
|
||||||
@include('Shop.Articles.partials.addBasket', [
|
|
||||||
'data' => $article['offers']['marchandise'],
|
|
||||||
'title' => 'Marchandises',
|
|
||||||
'model' => 'marchandise',
|
|
||||||
'bgClass' => 'bg-green-light',
|
|
||||||
])
|
|
||||||
@endif
|
|
||||||
@endif
|
@endif
|
||||||
|
|
||||||
@include('load.basket')
|
@include('load.basket')
|
||||||
|
|||||||
@@ -48,8 +48,8 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="col-lg-3 col-xs-12">
|
<div class="col-lg-3 col-xs-12">
|
||||||
@if (auth('web')->check() && !empty($article['available_sale_channels']))
|
@if (config('app.debug') && !empty($article['available_sale_channels']))
|
||||||
<div id="article-admin-offers" class="alert alert-info p-2 mb-3">
|
<div class="alert alert-info p-2 mb-3">
|
||||||
<strong class="d-block">Offres :</strong>
|
<strong class="d-block">Offres :</strong>
|
||||||
<ul class="list-unstyled mb-0 small">
|
<ul class="list-unstyled mb-0 small">
|
||||||
@php
|
@php
|
||||||
@@ -82,61 +82,16 @@
|
|||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
@if ($priceTaxed !== null)
|
@if ($priceTaxed !== null)
|
||||||
@php
|
<span class="ml-2 text-nowrap text-right {{ $nameClass }}">
|
||||||
$tariffId = $channel['tariff_id'] ?? null;
|
{{ number_format($priceTaxed, 2, ',', ' ') }} € TTC
|
||||||
@endphp
|
@if (! empty($quantity))
|
||||||
@if ($tariffId)
|
<span class="d-block text-muted" style="font-size: 0.85em;">Qté min. {{ $quantity }}</span>
|
||||||
<a href="{{ route('Admin.Shop.Tariffs.edit', $tariffId) }}" target="_blank" rel="noopener" class="ml-2 text-nowrap text-right {{ $nameClass }} text-decoration-none text-reset d-inline-block admin-link-group admin-price-link">
|
@endif
|
||||||
{{ number_format($priceTaxed, 2, ',', ' ') }} € TTC
|
</span>
|
||||||
@if (! empty($quantity))
|
|
||||||
<span class="d-block text-muted" style="font-size: 0.85em;">Qté min. {{ $quantity }}</span>
|
|
||||||
@endif
|
|
||||||
</a>
|
|
||||||
@else
|
|
||||||
<span class="ml-2 text-nowrap text-right {{ $nameClass }}">
|
|
||||||
{{ number_format($priceTaxed, 2, ',', ' ') }} € TTC
|
|
||||||
@if (! empty($quantity))
|
|
||||||
<span class="d-block text-muted" style="font-size: 0.85em;">Qté min. {{ $quantity }}</span>
|
|
||||||
@endif
|
|
||||||
</span>
|
|
||||||
@endif
|
|
||||||
@else
|
@else
|
||||||
<span class="ml-2 text-muted">–</span>
|
<span class="ml-2 text-muted">–</span>
|
||||||
@endif
|
@endif
|
||||||
</div>
|
</div>
|
||||||
@if (!empty($channel['all_offers']))
|
|
||||||
<ul class="list-unstyled mb-0 mt-1" style="padding-left: 0.75em;">
|
|
||||||
@foreach ($channel['all_offers'] as $offer)
|
|
||||||
@php
|
|
||||||
$isSelectedOffer = $offer['id'] === $channel['offer_id'];
|
|
||||||
$offerClass = $offer['is_active'] ? 'text-dark' : 'text-muted';
|
|
||||||
$stockClass = $offer['stock_current'] > 0 ? 'text-success' : 'text-danger';
|
|
||||||
@endphp
|
|
||||||
<li class="small {{ $offerClass }}" style="font-size: 0.85em;">
|
|
||||||
<a href="{{ route('Admin.Shop.Offers.edit', $offer['id']) }}" target="_blank" rel="noopener" class="text-decoration-none {{ $offerClass }} admin-link-group admin-offer-link">
|
|
||||||
<div class="d-flex justify-content-between align-items-start">
|
|
||||||
<div>
|
|
||||||
<span style="opacity: 0.5;">{{ $isSelectedOffer ? '▸' : '○' }}</span>
|
|
||||||
@if ($offer['variation_name'])
|
|
||||||
{{ $offer['variation_name'] }}
|
|
||||||
@endif
|
|
||||||
- Stock: <strong class="{{ $stockClass }}">{{ $offer['stock_current'] }}</strong>
|
|
||||||
@if (!$offer['is_active'])
|
|
||||||
<span class="text-muted">(inactive)</span>
|
|
||||||
@endif
|
|
||||||
</div>
|
|
||||||
<div class="text-right text-nowrap ml-2">
|
|
||||||
{{ number_format($offer['price_taxed'], 2, ',', ' ') }} €
|
|
||||||
@if ($offer['quantity'] > 1)
|
|
||||||
<span class="text-muted">(min {{ $offer['quantity'] }})</span>
|
|
||||||
@endif
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
@endforeach
|
|
||||||
</ul>
|
|
||||||
@endif
|
|
||||||
</li>
|
</li>
|
||||||
@endforeach
|
@endforeach
|
||||||
</ul>
|
</ul>
|
||||||
@@ -148,67 +103,3 @@
|
|||||||
@endsection
|
@endsection
|
||||||
|
|
||||||
@include('load.layout.modal')
|
@include('load.layout.modal')
|
||||||
|
|
||||||
@if (auth('web')->check() && !empty($article['available_sale_channels']))
|
|
||||||
@push('styles')
|
|
||||||
<style>
|
|
||||||
#article-admin-offers .admin-link-group {
|
|
||||||
transition: background-color 0.15s ease;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#article-admin-offers .admin-price-link {
|
|
||||||
display: inline-block;
|
|
||||||
padding: 2px 4px;
|
|
||||||
margin: -2px -4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#article-admin-offers .admin-offer-link {
|
|
||||||
display: block;
|
|
||||||
padding: 2px 4px;
|
|
||||||
margin: -2px -4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#article-admin-offers .admin-link-group:hover,
|
|
||||||
#article-admin-offers .admin-link-group:focus,
|
|
||||||
#article-admin-offers .admin-link-group.linked-hover {
|
|
||||||
background-color: rgba(0, 123, 255, 0.1);
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@endpush
|
|
||||||
|
|
||||||
@push('scripts')
|
|
||||||
<script>
|
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
|
||||||
const container = document.getElementById('article-admin-offers');
|
|
||||||
if (!container) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const links = Array.from(container.querySelectorAll('a.admin-link-group[href]'));
|
|
||||||
const grouped = new Map();
|
|
||||||
|
|
||||||
links.forEach((link) => {
|
|
||||||
const href = link.getAttribute('href');
|
|
||||||
if (!grouped.has(href)) {
|
|
||||||
grouped.set(href, []);
|
|
||||||
}
|
|
||||||
grouped.get(href).push(link);
|
|
||||||
});
|
|
||||||
|
|
||||||
grouped.forEach((group) => {
|
|
||||||
group.forEach((link) => {
|
|
||||||
const addHighlight = () => group.forEach((item) => item.classList.add('linked-hover'));
|
|
||||||
const removeHighlight = () => group.forEach((item) => item.classList.remove('linked-hover'));
|
|
||||||
|
|
||||||
link.addEventListener('mouseenter', addHighlight);
|
|
||||||
link.addEventListener('mouseleave', removeHighlight);
|
|
||||||
link.addEventListener('focus', addHighlight);
|
|
||||||
link.addEventListener('blur', removeHighlight);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
@endpush
|
|
||||||
@endif
|
|
||||||
|
|||||||
@@ -9,16 +9,14 @@
|
|||||||
var selector = (typeof(sel) == 'undefined') ? '.save' : sel;
|
var selector = (typeof(sel) == 'undefined') ? '.save' : sel;
|
||||||
$(selector).off().click(function(e) {
|
$(selector).off().click(function(e) {
|
||||||
if (typeof initValidator === 'function') {
|
if (typeof initValidator === 'function') {
|
||||||
|
console.log('click');
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
if ($(form).valid()) {
|
if ($(form).valid()) {
|
||||||
$(this).prop("disabled", true);
|
$(this).prop("disabled", true);
|
||||||
$(this).html($(this).data('loading-text'));
|
$(this).html($(this).data('loading-text'));
|
||||||
$(form).submit();
|
$(form).submit();
|
||||||
} else {
|
} else {
|
||||||
const $firstInvalid = $(form).find(':invalid, .error').first();
|
console.log('erreur');
|
||||||
if ($firstInvalid.length) {
|
|
||||||
$firstInvalid.focus();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$(form).submit();
|
$(form).submit();
|
||||||
|
|||||||
@@ -75,12 +75,12 @@
|
|||||||
if (typeof(tinyMCE) != 'undefined') {
|
if (typeof(tinyMCE) != 'undefined') {
|
||||||
tinyMCE.triggerSave();
|
tinyMCE.triggerSave();
|
||||||
}
|
}
|
||||||
getModalForm(form_id).trigger('submit');
|
$('form ' + form_id).submit();
|
||||||
status = 1;
|
status = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
function handlePostModal(form_id, url_save, callback) {
|
function handlePostModal(form_id, url_save, callback) {
|
||||||
getModalForm(form_id).off('submit').on('submit', function(e) {
|
$('form ' + form_id).submit(function(e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
var formData = new FormData(this);
|
var formData = new FormData(this);
|
||||||
$.ajax({
|
$.ajax({
|
||||||
@@ -98,15 +98,6 @@
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function getModalForm(form_id) {
|
|
||||||
var $form = $(form_id);
|
|
||||||
if (! $form.length) {
|
|
||||||
$form = $('form' + form_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $form;
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
@endpush
|
@endpush
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user