gallery support

This commit is contained in:
2023-07-18 18:35:58 +10:00
parent 64ef8e9acd
commit ef4efe723f
18 changed files with 447 additions and 119 deletions

View File

@@ -32,13 +32,14 @@ class ArticleConductor extends Conductor
* *
* @var string[] * @var string[]
*/ */
protected $includes = ['attachments', 'user']; protected $includes = ['attachments', 'user', 'gallery'];
/** /**
* Run a scope query on the collection before anything else. * Run a scope query on the collection before anything else.
* *
* @param Builder $builder The builder in use. * @param Builder $builder The builder in use.
* @return void
*/ */
public function scope(Builder $builder): void public function scope(Builder $builder): void
{ {
@@ -127,6 +128,19 @@ class ArticleConductor extends Conductor
}); });
} }
/**
* Include Gallery Field.
*
* @param Model $model Them model.
* @return mixed The model result.
*/
public function includeGallery(Model $model)
{
return $model->gallery()->get()->map(function ($item) {
return MediaConductor::includeModel(request(), 'gallery', $item->media);
});
}
/** /**
* Include User Field. * Include User Field.
* *

View File

@@ -2,9 +2,9 @@
namespace App\Conductors; namespace App\Conductors;
use App\Models\User;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Foundation\Auth\User;
class MediaConductor extends Conductor class MediaConductor extends Conductor
{ {

View File

@@ -39,14 +39,17 @@ class UserConductor extends Conductor
{ {
$user = auth()->user(); $user = auth()->user();
$data = $model->toArray(); $data = $model->toArray();
$limit = $this->fields($model);
// echo 'USER--' . implode(',', $limit) . "\n";
if ($user === null || ($user->hasPermission('admin/users') === false && strcasecmp($user->id, $model->id) !== 0)) { if ($user === null || ($user->hasPermission('admin/users') === false && strcasecmp($user->id, $model->id) !== 0)) {
$fields = ['id', 'display_name']; $limit = ['id', 'display_name'];
$data = arrayLimitKeys($data, $fields);
} else { } else {
$data['permissions'] = $user->permissions; $data['permissions'] = $user->permissions;
} }
$data = arrayLimitKeys($data, $limit);
return $data; return $data;
} }

View File

@@ -24,9 +24,13 @@ class ApiController extends Controller
* @param array $data Response data. * @param array $data Response data.
* @param integer $respondCode Response status code. * @param integer $respondCode Response status code.
* @param array $headers Response headers. * @param array $headers Response headers.
* @return JsonResponse
*/ */
public function respondJson(array $data, int $respondCode = HttpResponseCodes::HTTP_OK, array $headers = []): JsonResponse public function respondJson(
{ array $data,
int $respondCode = HttpResponseCodes::HTTP_OK,
array $headers = []
): JsonResponse {
return response()->json($data, $respondCode, $headers); return response()->json($data, $respondCode, $headers);
} }
@@ -34,9 +38,11 @@ class ApiController extends Controller
* Return forbidden message * Return forbidden message
* *
* @param string $message Response message. * @param string $message Response message.
* @return JsonResponse
*/ */
public function respondForbidden(string $message = 'You do not have permission to access the resource.'): JsonResponse public function respondForbidden(
{ string $message = 'You do not have permission to access the resource.'
): JsonResponse {
return response()->json(['message' => $message], HttpResponseCodes::HTTP_FORBIDDEN); return response()->json(['message' => $message], HttpResponseCodes::HTTP_FORBIDDEN);
} }
@@ -44,6 +50,7 @@ class ApiController extends Controller
* Return forbidden message * Return forbidden message
* *
* @param string $message Response message. * @param string $message Response message.
* @return JsonResponse
*/ */
public function respondNotFound(string $message = 'The resource was not found.'): JsonResponse public function respondNotFound(string $message = 'The resource was not found.'): JsonResponse
{ {
@@ -54,6 +61,7 @@ class ApiController extends Controller
* Return too large message * Return too large message
* *
* @param string $message Response message. * @param string $message Response message.
* @return JsonResponse
*/ */
public function respondTooLarge(string $message = 'The request entity is too large.'): JsonResponse public function respondTooLarge(string $message = 'The request entity is too large.'): JsonResponse
{ {
@@ -61,7 +69,9 @@ class ApiController extends Controller
} }
/** /**
* Return no content * Return no content.
*
* @return JsonResponse
*/ */
public function respondNoContent(): JsonResponse public function respondNoContent(): JsonResponse
{ {
@@ -69,7 +79,19 @@ class ApiController extends Controller
} }
/** /**
* Return created * Return no content
*
* @return JsonResponse
*/
public function respondNotImplmeneted(): JsonResponse
{
return response()->json([], HttpResponseCodes::HTTP_NOT_IMPLEMENTED);
}
/**
* Return created.
*
* @return JsonResponse
*/ */
public function respondCreated(): JsonResponse public function respondCreated(): JsonResponse
{ {
@@ -77,7 +99,9 @@ class ApiController extends Controller
} }
/** /**
* Return accepted * Return accepted.
*
* @return JsonResponse
*/ */
public function respondAccepted(): JsonResponse public function respondAccepted(): JsonResponse
{ {
@@ -89,9 +113,12 @@ class ApiController extends Controller
* *
* @param string $message Error message. * @param string $message Error message.
* @param integer $responseCode Resource code. * @param integer $responseCode Resource code.
* @return JsonResponse
*/ */
public function respondError(string $message, int $responseCode = HttpResponseCodes::HTTP_UNPROCESSABLE_ENTITY): JsonResponse public function respondError(
{ string $message,
int $responseCode = HttpResponseCodes::HTTP_UNPROCESSABLE_ENTITY
): JsonResponse {
return response()->json([ return response()->json([
'message' => $message 'message' => $message
], $responseCode); ], $responseCode);
@@ -102,9 +129,12 @@ class ApiController extends Controller
* *
* @param array $errors Error messages. * @param array $errors Error messages.
* @param integer $responseCode Resource code. * @param integer $responseCode Resource code.
* @return JsonResponse
*/ */
public function respondWithErrors(array $errors, int $responseCode = HttpResponseCodes::HTTP_UNPROCESSABLE_ENTITY): JsonResponse public function respondWithErrors(
{ array $errors,
int $responseCode = HttpResponseCodes::HTTP_UNPROCESSABLE_ENTITY
): JsonResponse {
$keys = array_keys($errors); $keys = array_keys($errors);
$error = $errors[$keys[0]]; $error = $errors[$keys[0]];
@@ -122,17 +152,20 @@ class ApiController extends Controller
/** /**
* Return resource data * Return resource data
* *
* @param array|Model|Collection $data Resource data. * @param array|Model|Collection $data Resource data.
* @param array $options Respond options. * @param array $options Respond options.
* @param callable|null $validationFn Optional validation function to check the data before responding.
* @return JsonResponse
*/ */
protected function respondAsResource( protected function respondAsResource(
mixed $data, mixed $data,
array $options = [], array $options = [],
$validationFn = null $validationFn = null
): JsonResponse { ): JsonResponse {
$isCollection = $options['isCollection'] ?? false; $isCollection = ($options['isCollection'] ?? false);
$appendData = $options['appendData'] ?? null; $appendData = ($options['appendData'] ?? null);
$resourceName = $options['resourceName'] ?? null; $resourceName = ($options['resourceName'] ?? '');
$transformResourceName = ($options['transformResourceName'] ?? true);
$respondCode = ($options['respondCode'] ?? HttpResponseCodes::HTTP_OK); $respondCode = ($options['respondCode'] ?? HttpResponseCodes::HTTP_OK);
if ($data === null || ($data instanceof Collection && $data->count() === 0)) { if ($data === null || ($data instanceof Collection && $data->count() === 0)) {
@@ -146,11 +179,11 @@ class ApiController extends Controller
} }
} }
if (is_null($resourceName) === true || empty($resourceName) === true) { if (empty($resourceName) === true) {
$resourceName = $this->resourceName; $resourceName = $this->resourceName;
} }
if (is_null($resourceName) === true || empty($resourceName) === true) { if (empty($resourceName) === true) {
$resourceName = get_class($this); $resourceName = get_class($this);
$resourceName = substr($resourceName, (strrpos($resourceName, '\\') + 1)); $resourceName = substr($resourceName, (strrpos($resourceName, '\\') + 1));
$resourceName = substr($resourceName, 0, strpos($resourceName, 'Controller')); $resourceName = substr($resourceName, 0, strpos($resourceName, 'Controller'));
@@ -163,15 +196,14 @@ class ApiController extends Controller
} elseif (is_array($data) === true) { } elseif (is_array($data) === true) {
$dataArray = $data; $dataArray = $data;
} elseif ($data instanceof Model) { } elseif ($data instanceof Model) {
$is_multiple = false;
$dataArray = $data->toArray(); $dataArray = $data->toArray();
} }
$resource = []; $resource = [];
if ($isCollection === true) { if ($isCollection === true) {
$resource = [Str::plural($resourceName) => $dataArray]; $resource = [$transformResourceName === true ? Str::plural($resourceName) : $resourceName => $dataArray];
} else { } else {
$resource = [Str::singular($resourceName) => $dataArray]; $resource = [$transformResourceName === true ? Str::singular($resourceName) : $resourceName => $dataArray];
} }
if ($appendData !== null) { if ($appendData !== null) {
@@ -180,4 +212,22 @@ class ApiController extends Controller
return response()->json($resource, $respondCode); return response()->json($resource, $respondCode);
} }
/**
* Get the Controller Model Class name.
*
* @return string
*/
public function getModelClass(): string
{
$controllerClass = static::class;
$modelName = 'App\\Models\\' . Str::replaceLast('Controller', '', Str::afterLast($controllerClass, '\\'));
if (class_exists($modelName) === false) {
return $modelName;
}
return $modelName;
}
} }

View File

@@ -8,6 +8,8 @@ use App\Enum\HttpResponseCodes;
use App\Http\Requests\ArticleRequest; use App\Http\Requests\ArticleRequest;
use App\Models\Media; use App\Models\Media;
use App\Models\Article; use App\Models\Article;
use App\Models\Gallery;
use App\Traits\HasGallery;
use Illuminate\Http\JsonResponse; use Illuminate\Http\JsonResponse;
use Carbon\Exceptions\InvalidFormatException; use Carbon\Exceptions\InvalidFormatException;
use Illuminate\Contracts\Container\BindingResolutionException; use Illuminate\Contracts\Container\BindingResolutionException;
@@ -17,6 +19,9 @@ use Illuminate\Http\Request;
class ArticleController extends ApiController class ArticleController extends ApiController
{ {
use HasGallery;
/** /**
* ApplicationController constructor. * ApplicationController constructor.
*/ */
@@ -73,14 +78,19 @@ class ArticleController extends ApiController
public function store(ArticleRequest $request) public function store(ArticleRequest $request)
{ {
if (ArticleConductor::creatable() === true) { if (ArticleConductor::creatable() === true) {
$article = Article::create($request->all()); $article = Article::create($request->except('gallery'));
if ($request->has('gallery') === true) {
$article->galleryAddMany($request->get('gallery'));
}
return $this->respondAsResource( return $this->respondAsResource(
ArticleConductor::model($request, $article), ArticleConductor::model($request, $article),
['respondCode' => HttpResponseCodes::HTTP_CREATED] ['respondCode' => HttpResponseCodes::HTTP_CREATED]
); );
} else { } else {
return $this->respondForbidden(); return $this->respondForbidden();
} }//end if
} }
/** /**
@@ -93,7 +103,12 @@ class ArticleController extends ApiController
public function update(ArticleRequest $request, Article $article) public function update(ArticleRequest $request, Article $article)
{ {
if (ArticleConductor::updatable($article) === true) { if (ArticleConductor::updatable($article) === true) {
$article->update($request->all()); if ($request->has('gallery') === true) {
$article->gallery()->delete();
$article->galleryAddMany($request->get('gallery'));
}
$article->update($request->except('gallery'));
return $this->respondAsResource(ArticleConductor::model($request, $article)); return $this->respondAsResource(ArticleConductor::model($request, $article));
} }
@@ -122,11 +137,8 @@ class ArticleController extends ApiController
* @param Request $request The user request. * @param Request $request The user request.
* @param Article $article The article model. * @param Article $article The article model.
* @return JsonResponse Returns the article attachments. * @return JsonResponse Returns the article attachments.
* @throws InvalidFormatException
* @throws BindingResolutionException
* @throws InvalidCastException
*/ */
public function getAttachments(Request $request, Article $article): JsonResponse public function attachmentIndex(Request $request, Article $article): JsonResponse
{ {
if (ArticleConductor::viewable($article) === true) { if (ArticleConductor::viewable($article) === true) {
$medium = $article->attachments->map(function ($attachment) { $medium = $article->attachments->map(function ($attachment) {
@@ -145,13 +157,11 @@ class ArticleController extends ApiController
* @param Request $request The user request. * @param Request $request The user request.
* @param Article $article The article model. * @param Article $article The article model.
* @return JsonResponse The response. * @return JsonResponse The response.
* @throws BindingResolutionException
* @throws MassAssignmentException
*/ */
public function storeAttachment(Request $request, Article $article): JsonResponse public function attachmentStore(Request $request, Article $article): JsonResponse
{ {
if (ArticleConductor::updatable($article) === true) { if (ArticleConductor::updatable($article) === true) {
if ($request->has("medium") && Media::find($request->medium)) { if ($request->has("medium") === true && Media::find($request->medium) !== null) {
$article->attachments()->create(['media_id' => $request->medium]); $article->attachments()->create(['media_id' => $request->medium]);
return $this->respondCreated(); return $this->respondCreated();
} }
@@ -167,10 +177,9 @@ class ArticleController extends ApiController
* *
* @param Request $request The user request. * @param Request $request The user request.
* @param Article $article The related model. * @param Article $article The related model.
* @throws BindingResolutionException * @return JsonResponse
* @throws MassAssignmentException
*/ */
public function updateAttachments(Request $request, Article $article): JsonResponse public function attachmentUpdate(Request $request, Article $article): JsonResponse
{ {
if (ArticleConductor::updatable($article) === true) { if (ArticleConductor::updatable($article) === true) {
$mediaIds = $request->attachments; $mediaIds = $request->attachments;
@@ -183,7 +192,7 @@ class ArticleController extends ApiController
// Delete attachments that are not in $mediaIds // Delete attachments that are not in $mediaIds
foreach ($attachments as $attachment) { foreach ($attachments as $attachment) {
if (!in_array($attachment->media_id, $mediaIds)) { if (in_array($attachment->media_id, $mediaIds) === false) {
$attachment->delete(); $attachment->delete();
} }
} }
@@ -193,13 +202,13 @@ class ArticleController extends ApiController
$found = false; $found = false;
foreach ($attachments as $attachment) { foreach ($attachments as $attachment) {
if ($attachment->media_id == $mediaId) { if ($attachment->media_id === $mediaId) {
$found = true; $found = true;
break; break;
} }
} }
if (!$found) { if ($found === false) {
$article->attachments()->create(['media_id' => $mediaId]); $article->attachments()->create(['media_id' => $mediaId]);
} }
} }
@@ -215,9 +224,9 @@ class ArticleController extends ApiController
* @param Request $request The user request. * @param Request $request The user request.
* @param Article $article The model. * @param Article $article The model.
* @param Media $medium The attachment medium. * @param Media $medium The attachment medium.
* @throws BindingResolutionException * @return JsonResponse
*/ */
public function deleteAttachment(Request $request, Article $article, Media $medium): JsonResponse public function attachmentDelete(Request $request, Article $article, Media $medium): JsonResponse
{ {
if (ArticleConductor::updatable($article) === true) { if (ArticleConductor::updatable($article) === true) {
$attachments = $article->attachments; $attachments = $article->attachments;
@@ -231,7 +240,7 @@ class ArticleController extends ApiController
} }
} }
if ($deleted) { if ($deleted === true) {
// Attachment was deleted successfully // Attachment was deleted successfully
return $this->respondNoContent(); return $this->respondNoContent();
} else { } else {

View File

@@ -117,7 +117,7 @@ class EventController extends ApiController
* @param Event $event The event model. * @param Event $event The event model.
* @return JsonResponse Returns the event attachments. * @return JsonResponse Returns the event attachments.
*/ */
public function getAttachments(Request $request, Event $event): JsonResponse public function attachmentIndex(Request $request, Event $event): JsonResponse
{ {
if (EventConductor::viewable($event) === true) { if (EventConductor::viewable($event) === true) {
$medium = $event->attachments->map(function ($attachment) { $medium = $event->attachments->map(function ($attachment) {
@@ -137,7 +137,7 @@ class EventController extends ApiController
* @param Event $event The event model. * @param Event $event The event model.
* @return JsonResponse The response. * @return JsonResponse The response.
*/ */
public function storeAttachment(Request $request, Event $event): JsonResponse public function attachmentStore(Request $request, Event $event): JsonResponse
{ {
if (EventConductor::updatable($event) === true) { if (EventConductor::updatable($event) === true) {
if ($request->has("medium") === true && Media::find($request->medium) !== null) { if ($request->has("medium") === true && Media::find($request->medium) !== null) {
@@ -157,7 +157,7 @@ class EventController extends ApiController
* @param Request $request The user request. * @param Request $request The user request.
* @param Event $event The related model. * @param Event $event The related model.
*/ */
public function updateAttachments(Request $request, Event $event): JsonResponse public function attachmentUpdate(Request $request, Event $event): JsonResponse
{ {
if (EventConductor::updatable($event) === true) { if (EventConductor::updatable($event) === true) {
$mediaIds = $request->attachments; $mediaIds = $request->attachments;
@@ -204,7 +204,7 @@ class EventController extends ApiController
* @param Event $event The model. * @param Event $event The model.
* @param Media $medium The attachment medium. * @param Media $medium The attachment medium.
*/ */
public function deleteAttachment(Request $request, Event $event, Media $medium): JsonResponse public function attachmentDelete(Request $request, Event $event, Media $medium): JsonResponse
{ {
if (EventConductor::updatable($event) === true) { if (EventConductor::updatable($event) === true) {
$attachments = $event->attachments; $attachments = $event->attachments;

View File

@@ -2,15 +2,18 @@
namespace App\Models; namespace App\Models;
use App\Traits\HasGallery;
use App\Traits\Uuids; use App\Traits\Uuids;
use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\MorphMany; use Illuminate\Database\Eloquent\Relations\MorphMany;
class Article extends Model class Article extends Model
{ {
use HasFactory; use HasFactory;
use Uuids; use Uuids;
use HasGallery;
/** /**
* The attributes that are mass assignable. * The attributes that are mass assignable.
@@ -29,6 +32,8 @@ class Article extends Model
/** /**
* Get the article user * Get the article user
*
* @return Illuminate\Database\Eloquent\Relations\BelongsTo
*/ */
public function user(): BelongsTo public function user(): BelongsTo
{ {
@@ -37,6 +42,8 @@ class Article extends Model
/** /**
* Get all of the article's attachments. * Get all of the article's attachments.
*
* @return Illuminate\Database\Eloquent\Relations\MorphMany
*/ */
public function attachments(): MorphMany public function attachments(): MorphMany
{ {

45
app/Models/Gallery.php Normal file
View File

@@ -0,0 +1,45 @@
<?php
namespace App\Models;
use App\Traits\Uuids;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\MorphTo;
class Gallery extends Model
{
use HasFactory;
use Uuids;
/**
* The attributes that are mass assignable.
*
* @var array<int, string>
*/
protected $fillable = [
'media_id',
];
/**
* Get gallery addendum model.
*
* @return Illuminate\Database\Eloquent\Relations\MorphTo Addenum model.
*/
public function addendum(): MorphTo
{
return $this->morphTo();
}
/**
* Get the media for this item.
*
* @return Illuminate\Database\Eloquent\Relations\BelongsTo The media model.
*/
public function media(): BelongsTo
{
return $this->belongsTo(Media::class);
}
}

View File

@@ -61,28 +61,27 @@ class RouteServiceProvider extends ServiceProvider
Route::macro('apiAddendumResource', function ($addendum, $uri, $controller) { Route::macro('apiAddendumResource', function ($addendum, $uri, $controller) {
$singularUri = Str::singular($uri); $singularUri = Str::singular($uri);
$pluralAddendumLC = strtolower(Str::plural($addendum)); $signularAddendum = Str::singular((strtolower($addendum)));
$pluralAddendumTC = ucfirst($pluralAddendumLC); $pluralAddendum = Str::plural($signularAddendum);
$singularAddendumTC = Str::singular($pluralAddendumTC);
Route::get("$uri/{{$singularUri}}/{{$pluralAddendumLC}}", [$controller, "get{{$pluralAddendumTC}}"]) Route::get("{$uri}/{{$singularUri}}/{$pluralAddendum}", [$controller, "{$signularAddendum}Index"])
->name("{{$singularUri}}.{{$pluralAddendumLC}}.index"); ->name("{$singularUri}.{$signularAddendum}.index");
Route::post("$uri/{{$singularUri}}/{{$pluralAddendumLC}}", [$controller, "store{{$singularAddendumTC}}"]) Route::post("{$uri}/{{$singularUri}}/{$pluralAddendum}", [$controller, "{$signularAddendum}Store"])
->name("{{$singularUri}}.{{$pluralAddendumLC}}.store"); ->name("{$singularUri}.{$signularAddendum}.store");
Route::match( Route::match(
['put', 'patch'], ['put', 'patch'],
"$uri/{{$singularUri}}/{{$pluralAddendumLC}}", "{$uri}/{{$singularUri}}/{$pluralAddendum}",
[$controller, "update{{$pluralAddendumTC}}"] [$controller, "{$signularAddendum}Update"]
) )
->name("{{$singularUri}}.{{$pluralAddendumLC}}.update"); ->name("{$singularUri}.{$signularAddendum}.update");
Route::delete( Route::delete(
"$uri/{{$singularUri}}/{{$pluralAddendumLC}}/{medium}", "{$uri}/{{$singularUri}}/{$pluralAddendum}/{medium}",
[$controller,"delete{{$singularAddendumTC}}"] [$controller,"{$signularAddendum}Delete"]
) )
->name("{{$singularUri}}.{{$pluralAddendumLC}}.destroy"); ->name("{$singularUri}.{$signularAddendum}.destroy");
}); });
} }
} }

63
app/Traits/HasGallery.php Normal file
View File

@@ -0,0 +1,63 @@
<?php
namespace App\Traits;
use App\Models\Media;
use Illuminate\Database\Eloquent\Relations\MorphMany;
trait HasGallery
{
/**
* Boot function from Laravel.
*
* @return void
*/
protected static function bootHasGallery(): void
{
static::deleting(function ($model) {
$model->gallery()->delete();
});
}
/**
* Add multiple gallery items to the model.
*
* @param array|string $ids The media ids to add.
* @param string $delimiter The split delimiter if $ids is a string.
* @param boolean $allowDuplicates Whether to allow duplicate media IDs or not.
* @return void
*/
public function galleryAddMany(array|string $ids, string $delimiter = ',', bool $allowDuplicates = false): void
{
if (is_array($ids) === false) {
$ids = explode($delimiter, $ids);
}
$ids = array_map('trim', $ids);
$existingIds = $this->gallery()->pluck('media_id')->toArray();
$galleryItems = [];
foreach ($ids as $id) {
if ($allowDuplicates === false && in_array($id, $existingIds) === true) {
continue;
}
$media = Media::find($id);
if ($media !== null) {
$galleryItems[] = ['media_id' => $id];
}
}
$this->gallery()->createMany($galleryItems);
}
/**
* Get the article's gallery.
*
* @return Illuminate\Database\Eloquent\Relations\MorphMany The gallery items
*/
public function gallery(): MorphMany
{
return $this->morphMany(\App\Models\Gallery::class, 'addendum');
}
}

View File

@@ -0,0 +1,32 @@
<?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::create('galleries', function (Blueprint $table) {
$table->uuid('id')->primary();
$table->uuid('media_id');
$table->uuidMorphs('addendum');
$table->timestamps();
$table->foreign('media_id')->references('id')->on('media')->onDelete('cascade');
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('galleries');
}
};

View File

@@ -1,17 +1,53 @@
<template> <template>
<div class="image-gallery" ref="gallery"> <div
:class="[
'flex',
'gap-4',
'my-4',
'select-none',
props.showEditor
? ['overflow-auto']
: ['flex-wrap', 'flex-justify-center'],
]">
<div <div
class="image-gallery-item" v-for="(image, index) in modelValue"
v-for="(image, index) in images" class="flex flex-col flex-justify-center relative sm-gallery-item p-1"
:key="index"> :key="index">
<img <img
:src="image as string" :src="mediaGetVariantUrl(image as Media, 'small')"
class="image-gallery-image" class="max-h-40 max-w-40 cursor-pointer"
@click="showModal(index)" /> @click="showGalleryModal(index)" />
<div
class="absolute rounded-5 bg-white -top-0.25 -right-0.25 hidden cursor-pointer item-delete"
@click="handleRemoveItem(image.id)">
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-6 w-6 block"
viewBox="0 0 24 24">
<path
d="M12,2C17.53,2 22,6.47 22,12C22,17.53 17.53,22 12,22C6.47,22 2,17.53 2,12C2,6.47 6.47,2 12,2M15.59,7L12,10.59L8.41,7L7,8.41L10.59,12L7,15.59L8.41,17L12,13.41L15.59,17L17,15.59L13.41,12L17,8.41L15.59,7Z"
fill="rgba(185,28,28,1)" />
</svg>
</div>
</div>
<div v-if="props.showEditor" class="flex flex-col flex-justify-center">
<div
class="flex flex-col flex-justify-center flex-items-center h-23 w-40 cursor-pointer bg-gray-300 text-gray-800 hover:text-gray-600"
@click="handleAddToGallery">
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-15 w-15"
viewBox="0 0 24 24">
<title>Add image</title>
<path
d="M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20,12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2M13,7H11V11H7V13H11V17H13V13H17V11H13V7Z"
fill="currentColor" />
</svg>
</div>
</div> </div>
</div> </div>
<div <div
v-if="showModalImage !== null" v-if="props.showEditor == false && showModalImage !== null"
:class="[ :class="[
'image-gallery-modal', 'image-gallery-modal',
{ 'image-gallery-modal-buttons': showButtons }, { 'image-gallery-modal-buttons': showButtons },
@@ -20,7 +56,7 @@
@mousemove="handleModalUpdateButtons" @mousemove="handleModalUpdateButtons"
@mouseleave="handleModalUpdateButtons"> @mouseleave="handleModalUpdateButtons">
<img <img
:src="images[showModalImage] as string" :src="mediaGetVariantUrl(modelValue[showModalImage] as Media)"
class="image-gallery-modal-image" /> class="image-gallery-modal-image" />
<div <div
class="image-gallery-modal-prev" class="image-gallery-modal-prev"
@@ -28,26 +64,46 @@
<div <div
class="image-gallery-modal-next" class="image-gallery-modal-next"
@click.stop="handleModalNextImage"></div> @click.stop="handleModalNextImage"></div>
<div class="image-gallery-modal-close" @click="hideModal">&times;</div> <div class="image-gallery-modal-close" @click="hideModal">
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-6 w-6"
viewBox="0 0 24 24">
<title>Close</title>
<path
d="M19,6.41L17.59,5L12,10.59L6.41,5L5,6.41L10.59,12L5,17.59L6.41,19L12,13.41L17.59,19L19,17.59L13.41,12L19,6.41Z"
fill="currentColor" />
</svg>
</div>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { onBeforeUnmount, onMounted, ref } from "vue"; import { onBeforeUnmount, onMounted, ref } from "vue";
import { Media } from "../helpers/api.types";
import { mediaGetVariantUrl } from "../helpers/media";
import { openDialog } from "./SMDialog";
import SMDialogMedia from "./dialogs/SMDialogMedia.vue";
const emits = defineEmits(["update:modelValue"]);
const props = defineProps({ const props = defineProps({
images: { modelValue: {
type: Array, type: Array,
default: () => [],
required: true, required: true,
}, },
showEditor: {
type: Boolean,
default: false,
required: false,
},
}); });
const gallery = ref(null);
const showModalImage = ref(null); const showModalImage = ref(null);
let showButtons = ref(false); let showButtons = ref(false);
let mouseMoveTimeout = null; let mouseMoveTimeout = null;
const showModal = (index) => { const showGalleryModal = (index) => {
showModalImage.value = index; showModalImage.value = index;
document.addEventListener("keydown", handleKeyDown); document.addEventListener("keydown", handleKeyDown);
}; };
@@ -87,7 +143,7 @@ const handleModalPrevImage = () => {
if (showModalImage.value > 0) { if (showModalImage.value > 0) {
showModalImage.value--; showModalImage.value--;
} else { } else {
showModalImage.value = props.images.length - 1; showModalImage.value = props.modelValue.length - 1;
} }
} }
}; };
@@ -96,7 +152,7 @@ const handleModalNextImage = () => {
handleModalUpdateButtons(); handleModalUpdateButtons();
if (showModalImage.value !== null) { if (showModalImage.value !== null) {
if (showModalImage.value < props.images.length - 1) { if (showModalImage.value < props.modelValue.length - 1) {
showModalImage.value++; showModalImage.value++;
} else { } else {
showModalImage.value = 0; showModalImage.value = 0;
@@ -104,6 +160,33 @@ const handleModalNextImage = () => {
} }
}; };
const handleAddToGallery = async () => {
let result = await openDialog(SMDialogMedia, {
allowUpload: true,
multiple: true,
});
if (result) {
const mediaResult = result as Media[];
let newValue = props.modelValue;
let galleryIds = new Set(newValue.map((item) => item.id));
mediaResult.forEach((item) => {
if (!galleryIds.has(item.id)) {
newValue.push(item);
galleryIds.add(item.id);
}
});
emits("update:modelValue", newValue);
}
};
const handleRemoveItem = async (id: string) => {
const newList = props.modelValue.filter((item) => item.id !== id);
emits("update:modelValue", newList);
};
onMounted(() => { onMounted(() => {
document.addEventListener("keydown", handleKeyDown); document.addEventListener("keydown", handleKeyDown);
}); });
@@ -114,30 +197,30 @@ onBeforeUnmount(() => {
</script> </script>
<style lang="scss"> <style lang="scss">
.image-gallery { // .image-gallery {
display: grid; // display: grid;
grid-template-columns: 1fr 1fr; // grid-template-columns: 1fr 1fr;
gap: 15px; // gap: 15px;
.image-gallery-image { // .image-gallery-image {
cursor: pointer; // cursor: pointer;
max-width: 100%; // max-width: 100%;
max-height: 100%; // max-height: 100%;
object-fit: contain; // object-fit: contain;
} // }
} // }
@media (min-width: 768px) { // @media (min-width: 768px) {
.image-gallery { // .image-gallery {
grid-template-columns: 1fr 1fr 1fr; // grid-template-columns: 1fr 1fr 1fr;
} // }
} // }
@media (min-width: 1024px) { // @media (min-width: 1024px) {
.image-gallery { // .image-gallery {
grid-template-columns: 1fr 1fr 1fr 1fr; // grid-template-columns: 1fr 1fr 1fr 1fr;
} // }
} // }
.image-gallery-modal { .image-gallery-modal {
position: fixed; position: fixed;
@@ -268,4 +351,8 @@ onBeforeUnmount(() => {
} }
} }
} }
.sm-gallery-item:hover .item-delete {
display: block;
}
</style> </style>

View File

@@ -357,18 +357,6 @@
" "
small small
class="bg-white bg-op-90 w-full h-full" /> class="bg-white bg-op-90 w-full h-full" />
<div
class="absolute border-1 border-1 rounded-5 bg-white -top-1.5 -right-1.5 hidden item-delete"
@click="handleRemoveItem(item.id)">
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-6 w-6 block"
viewBox="0 0 24 24">
<path
d="M12,2C17.53,2 22,6.47 22,12C22,17.53 17.53,22 12,22C6.47,22 2,17.53 2,12C2,6.47 6.47,2 12,2M15.59,7L12,10.59L8.41,7L7,8.41L10.59,12L7,15.59L8.41,17L12,13.41L15.59,17L17,15.59L13.41,12L17,8.41L15.59,7Z"
fill="rgba(185,28,28,1)" />
</svg>
</div>
</div> </div>
</li> </li>
</ul> </ul>

View File

@@ -96,6 +96,7 @@ export interface Article {
content: string; content: string;
publish_at: string; publish_at: string;
hero: Media; hero: Media;
gallery: Array<Media>;
attachments: Array<Media>; attachments: Array<Media>;
} }

View File

@@ -36,7 +36,12 @@
> >
</div> </div>
<SMHTML :html="article.content" /> <SMHTML :html="article.content" />
<SMAttachments :attachments="article.attachments || []" /> <SMImageGallery
v-if="article.gallery.length > 0"
:model-value="article.gallery" />
<SMAttachments
v-if="article.attachments.length > 0"
:attachments="article.attachments || []" />
</div> </div>
</template> </template>
</template> </template>
@@ -54,6 +59,7 @@ import { userHasPermission } from "../helpers/utils";
import SMLoading from "../components/SMLoading.vue"; import SMLoading from "../components/SMLoading.vue";
import SMPageStatus from "../components/SMPageStatus.vue"; import SMPageStatus from "../components/SMPageStatus.vue";
import SMHTML from "../components/SMHTML.vue"; import SMHTML from "../components/SMHTML.vue";
import SMImageGallery from "../components/SMImageGallery.vue";
const applicationStore = useApplicationStore(); const applicationStore = useApplicationStore();
@@ -61,8 +67,18 @@ const applicationStore = useApplicationStore();
* The article data. * The article data.
*/ */
let article: Ref<Article> = ref({ let article: Ref<Article> = ref({
id: "",
created_at: "",
updated_at: "",
title: "", title: "",
slug: "",
user_id: "",
user: { display_name: "" }, user: { display_name: "" },
content: "",
publish_at: "",
hero: {},
gallery: [],
attachments: [],
}); });
/** /**

View File

@@ -137,7 +137,7 @@ watch(
() => articlesPage.value, () => articlesPage.value,
() => { () => {
handleLoad(); handleLoad();
} },
); );
handleLoad(); handleLoad();

View File

@@ -38,6 +38,15 @@
class="mb-8" class="mb-8"
v-model:model-value="form.controls.content.value" /> v-model:model-value="form.controls.content.value" />
</div> </div>
<div class="mb-8">
<h3>Gallery</h3>
<p class="small">
{{ gallery.length }} image{{
gallery.length != 1 ? "s" : ""
}}
</p>
<SMImageGallery show-editor v-model:model-value="gallery" />
</div>
<SMInputAttachments v-model:model-value="attachments" /> <SMInputAttachments v-model:model-value="attachments" />
<div class="flex flex-justify-end"> <div class="flex flex-justify-end">
<input <input
@@ -70,6 +79,7 @@ import SMPageStatus from "../../components/SMPageStatus.vue";
import { userHasPermission } from "../../helpers/utils"; import { userHasPermission } from "../../helpers/utils";
import SMSelectImage from "../../components/SMSelectImage.vue"; import SMSelectImage from "../../components/SMSelectImage.vue";
import SMLoading from "../../components/SMLoading.vue"; import SMLoading from "../../components/SMLoading.vue";
import SMImageGallery from "../../components/SMImageGallery.vue";
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
@@ -78,6 +88,7 @@ let pageError = ref(200);
const authors = ref({}); const authors = ref({});
const attachments = ref([]); const attachments = ref([]);
const pageHeading = route.params.id ? "Edit Article" : "Create Article"; const pageHeading = route.params.id ? "Edit Article" : "Create Article";
const gallery = ref([]);
const form = reactive( const form = reactive(
Form({ Form({
@@ -85,12 +96,12 @@ const form = reactive(
slug: FormControl("", And([Required(), Min(6)])), slug: FormControl("", And([Required(), Min(6)])),
publish_at: FormControl( publish_at: FormControl(
route.params.id ? "" : new SMDate("now").format("d/M/yy h:mm aa"), route.params.id ? "" : new SMDate("now").format("d/M/yy h:mm aa"),
DateTime() DateTime(),
), ),
hero: FormControl(), hero: FormControl(),
user_id: FormControl(userStore.id), user_id: FormControl(userStore.id),
content: FormControl(), content: FormControl(),
}) }),
); );
const updateSlug = async () => { const updateSlug = async () => {
@@ -164,8 +175,10 @@ const loadData = async () => {
attachments.value = (data.article.attachments || []).map( attachments.value = (data.article.attachments || []).map(
function (attachment) { function (attachment) {
return attachment.id.toString(); return attachment.id.toString();
} },
); );
gallery.value = data.article.gallery;
} else { } else {
pageError.value = 404; pageError.value = 404;
} }
@@ -183,11 +196,12 @@ const handleSubmit = async () => {
title: form.controls.title.value, title: form.controls.title.value,
slug: form.controls.slug.value, slug: form.controls.slug.value,
publish_at: new SMDate( publish_at: new SMDate(
form.controls.publish_at.value as string form.controls.publish_at.value as string,
).format("yyyy/MM/dd HH:mm:ss", { utc: true }), ).format("yyyy/MM/dd HH:mm:ss", { utc: true }),
user_id: form.controls.user_id.value, user_id: form.controls.user_id.value,
content: form.controls.content.value, content: form.controls.content.value,
hero: form.controls.hero.value.id, hero: form.controls.hero.value.id,
gallery: gallery.value.map((item) => item.id),
}; };
let article_id = ""; let article_id = "";
@@ -280,7 +294,7 @@ const attachmentAdd = async (event) => {
}, },
progress: (progressEvent) => progress: (progressEvent) =>
event.attachment.setUploadProgress( event.attachment.setUploadProgress(
(progressEvent.loaded * progressEvent.total) / 100 (progressEvent.loaded * progressEvent.total) / 100,
), ),
}); });
@@ -292,7 +306,7 @@ const attachmentAdd = async (event) => {
event.preventDefault(); event.preventDefault();
alert( alert(
err.response?.data?.message || err.response?.data?.message ||
"An unexpected server error occurred" "An unexpected server error occurred",
); );
} }
} }

View File

@@ -44,10 +44,10 @@ Route::apiResource('media', MediaController::class);
Route::get('media/{medium}/download', [MediaController::class, 'download']); Route::get('media/{medium}/download', [MediaController::class, 'download']);
Route::apiResource('articles', ArticleController::class); Route::apiResource('articles', ArticleController::class);
Route::apiAddendumResource('attachment', 'articles', ArticleController::class); Route::apiAddendumResource('attachments', 'articles', ArticleController::class);
Route::apiResource('events', EventController::class); Route::apiResource('events', EventController::class);
Route::apiAddendumResource('attachment', 'events', EventController::class); Route::apiAddendumResource('attachments', 'events', EventController::class);
Route::get('/events/{event}/users', [EventController::class, 'userList']); Route::get('/events/{event}/users', [EventController::class, 'userList']);
Route::post('/events/{event}/users', [EventController::class, 'userAdd']); Route::post('/events/{event}/users', [EventController::class, 'userAdd']);