mirror of
https://github.com/NyaaStudios/nyaabooru.git
synced 2025-12-10 05:42:58 +00:00
Compare commits
4 commits
1442dbfa60
...
b8032e9dec
| Author | SHA1 | Date | |
|---|---|---|---|
| b8032e9dec | |||
| 68d316cd86 | |||
|
|
87e535ff98 | ||
| cb086c4708 |
23 changed files with 131 additions and 9954 deletions
13
app/Livewire/App/Pagination.php
Normal file
13
app/Livewire/App/Pagination.php
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Livewire\App;
|
||||||
|
|
||||||
|
use Livewire\Component;
|
||||||
|
|
||||||
|
class Pagination extends Component
|
||||||
|
{
|
||||||
|
public function render()
|
||||||
|
{
|
||||||
|
return view('livewire.app.pagination');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -9,6 +9,6 @@ class Users extends Component
|
||||||
{
|
{
|
||||||
public function render()
|
public function render()
|
||||||
{
|
{
|
||||||
return view('livewire.pages.users', [ 'users' => User::all() ]);
|
return view('livewire.pages.users', [ 'users' => User::paginate(10) ]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -3,7 +3,9 @@
|
||||||
namespace App\Livewire\Posts;
|
namespace App\Livewire\Posts;
|
||||||
|
|
||||||
use App\Models\Post;
|
use App\Models\Post;
|
||||||
|
use App\Models\Tag;
|
||||||
use Livewire\Attributes\Title;
|
use Livewire\Attributes\Title;
|
||||||
|
use Livewire\Attributes\Url;
|
||||||
use Livewire\Component;
|
use Livewire\Component;
|
||||||
use Livewire\WithPagination;
|
use Livewire\WithPagination;
|
||||||
|
|
||||||
|
|
@ -14,7 +16,7 @@ class Index extends Component
|
||||||
#[Title('Posts')]
|
#[Title('Posts')]
|
||||||
public function render()
|
public function render()
|
||||||
{
|
{
|
||||||
return view('livewire.posts.index', [
|
return view('livewire.posts.index', [
|
||||||
'posts' => Post::orderBy('created_at', 'desc')->paginate(25),
|
'posts' => Post::orderBy('created_at', 'desc')->paginate(25),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,29 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Livewire;
|
|
||||||
|
|
||||||
use App\Models\User;
|
|
||||||
use Livewire\Attributes\Validate;
|
|
||||||
use Livewire\Component;
|
|
||||||
use Spatie\Searchable\ModelSearchAspect;
|
|
||||||
use Spatie\Searchable\Search as SpatieSearch;
|
|
||||||
|
|
||||||
class Search extends Component
|
|
||||||
{
|
|
||||||
#[Validate('string|min:3')]
|
|
||||||
public string $searchText = '';
|
|
||||||
|
|
||||||
public $searchResults = [];
|
|
||||||
|
|
||||||
public function render()
|
|
||||||
{
|
|
||||||
return view('livewire.search');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function updated()
|
|
||||||
{
|
|
||||||
$this->searchResults = (new SpatieSearch())
|
|
||||||
->registerModel(User::class, 'name')
|
|
||||||
->perform($this->searchText);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
13
app/Livewire/Tags/View.php
Normal file
13
app/Livewire/Tags/View.php
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Livewire\Tags;
|
||||||
|
|
||||||
|
use Livewire\Component;
|
||||||
|
|
||||||
|
class View extends Component
|
||||||
|
{
|
||||||
|
public function render()
|
||||||
|
{
|
||||||
|
return view('livewire.tags.view');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -78,6 +78,17 @@ class Post extends Model
|
||||||
return getimagesize($this->getFullUrl());
|
return getimagesize($this->getFullUrl());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getDimensionsStr(): string
|
||||||
|
{
|
||||||
|
[$width, $height] = $this->getDimensions();
|
||||||
|
return "$width x $height";
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getFileSize(): int
|
||||||
|
{
|
||||||
|
return Storage::size("posts/$this->id/full");
|
||||||
|
}
|
||||||
|
|
||||||
public function getAspectRatio(): string
|
public function getAspectRatio(): string
|
||||||
{
|
{
|
||||||
list($width, $height) = $this->getDimensions();
|
list($width, $height) = $this->getDimensions();
|
||||||
|
|
|
||||||
|
|
@ -5,8 +5,6 @@ namespace App\Models;
|
||||||
use MongoDB\Laravel\Eloquent\Model;
|
use MongoDB\Laravel\Eloquent\Model;
|
||||||
use MongoDB\Laravel\Relations\BelongsTo;
|
use MongoDB\Laravel\Relations\BelongsTo;
|
||||||
use MongoDB\Laravel\Relations\BelongsToMany;
|
use MongoDB\Laravel\Relations\BelongsToMany;
|
||||||
use MongoDB\Laravel\Relations\HasMany;
|
|
||||||
use MongoDB\Laravel\Relations\MorphToMany;
|
|
||||||
|
|
||||||
class Tag extends Model
|
class Tag extends Model
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -9,10 +9,8 @@ use Laravel\Sanctum\HasApiTokens;
|
||||||
use MongoDB\Laravel\Auth\User as Authenticatable;
|
use MongoDB\Laravel\Auth\User as Authenticatable;
|
||||||
use MongoDB\Laravel\Relations\HasMany;
|
use MongoDB\Laravel\Relations\HasMany;
|
||||||
use Overtrue\LaravelFavorite\Traits\Favoriter;
|
use Overtrue\LaravelFavorite\Traits\Favoriter;
|
||||||
use Spatie\Searchable\Searchable;
|
|
||||||
use Spatie\Searchable\SearchResult;
|
|
||||||
|
|
||||||
class User extends Authenticatable implements Searchable
|
class User extends Authenticatable
|
||||||
{
|
{
|
||||||
protected $connection = 'mongodb';
|
protected $connection = 'mongodb';
|
||||||
protected $table = 'users';
|
protected $table = 'users';
|
||||||
|
|
@ -48,16 +46,6 @@ class User extends Authenticatable implements Searchable
|
||||||
return $this->hasMany(Comment::class);
|
return $this->hasMany(Comment::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getSearchResult(): SearchResult
|
|
||||||
{
|
|
||||||
$url = url("/profiles/$this->id");
|
|
||||||
return new SearchResult(
|
|
||||||
$this,
|
|
||||||
$this->name,
|
|
||||||
$url
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getAvatarBase64(): string
|
public function getAvatarBase64(): string
|
||||||
{
|
{
|
||||||
$ext = $this->avatar_ext;
|
$ext = $this->avatar_ext;
|
||||||
|
|
|
||||||
|
|
@ -11,7 +11,6 @@
|
||||||
"require": {
|
"require": {
|
||||||
"php": "^8.2",
|
"php": "^8.2",
|
||||||
"ext-gmp": "*",
|
"ext-gmp": "*",
|
||||||
"intervention/image-driver-vips": "^1.0",
|
|
||||||
"intervention/image-laravel": "^1.5",
|
"intervention/image-laravel": "^1.5",
|
||||||
"laravel/framework": "^12.0",
|
"laravel/framework": "^12.0",
|
||||||
"laravel/octane": "^2.9",
|
"laravel/octane": "^2.9",
|
||||||
|
|
|
||||||
9852
composer.lock
generated
9852
composer.lock
generated
File diff suppressed because it is too large
Load diff
|
|
@ -4,7 +4,7 @@ return [
|
||||||
/**
|
/**
|
||||||
* Use uuid as primary key.
|
* Use uuid as primary key.
|
||||||
*/
|
*/
|
||||||
'uuids' => false,
|
'uuids' => true,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* User tables foreign key name.
|
* User tables foreign key name.
|
||||||
|
|
|
||||||
|
|
@ -65,12 +65,13 @@ return [
|
||||||
|
|
||||||
'temporary_file_upload' => [
|
'temporary_file_upload' => [
|
||||||
'disk' => null, // Example: 'local', 's3' | Default: 'default'
|
'disk' => null, // Example: 'local', 's3' | Default: 'default'
|
||||||
'rules' => ['required', 'file', 'max:12288'], // Example: ['file', 'mimes:png,jpg'] | Default: ['required', 'file', 'max:12288'] (12MB)
|
'rules' => ['file', 'max:81920'], // Example: ['file', 'mimes:png,jpg'] | Default: ['required', 'file', 'max:12288'] (12MB)
|
||||||
'directory' => null, // Example: 'tmp' | Default: 'livewire-tmp'
|
'directory' => null, // Example: 'tmp' | Default: 'livewire-tmp'
|
||||||
'middleware' => null, // Example: 'throttle:5,1' | Default: 'throttle:60,1'
|
'middleware' => null, // Example: 'throttle:5,1' | Default: 'throttle:60,1'
|
||||||
'preview_mimes' => [ // Supported file types for temporary pre-signed file URLs...
|
'preview_mimes' => [ // Supported file types for temporary pre-signed file URLs...
|
||||||
'png', 'gif', 'bmp', 'svg',
|
'png', 'gif', 'bmp', 'svg', 'wav', 'mp4',
|
||||||
'jpg', 'jpeg', 'webp',
|
'mov', 'avi', 'wmv', 'mp3', 'm4a',
|
||||||
|
'jpg', 'jpeg', 'mpga', 'webp', 'wma',
|
||||||
],
|
],
|
||||||
'max_upload_time' => 5, // Max duration (in minutes) before an upload is invalidated...
|
'max_upload_time' => 5, // Max duration (in minutes) before an upload is invalidated...
|
||||||
'cleanup' => true, // Should cleanup temporary uploads older than 24 hrs...
|
'cleanup' => true, // Should cleanup temporary uploads older than 24 hrs...
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
FROM dunglas/frankenphp
|
FROM dunglas/frankenphp
|
||||||
|
|
||||||
# Copy php.ini
|
# Copy php.ini
|
||||||
COPY ./deploy/php.ini $PHP_INI_DIR/php.ini
|
COPY ./deploy/php.ini $PHP_INI_DIR/php-app-config.ini
|
||||||
|
|
||||||
RUN apt update -y && apt install -y --no-install-recommends git nodejs npm
|
RUN apt update -y && apt install -y --no-install-recommends git nodejs npm
|
||||||
COPY --from=composer:lts /usr/bin/composer /usr/bin/composer
|
COPY --from=composer:lts /usr/bin/composer /usr/bin/composer
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,5 @@
|
||||||
upload_max_filesize = 64M
|
[php]
|
||||||
post_max_filesize = 72M
|
upload_max_filesize = 100M
|
||||||
|
post_max_filesize = 128M
|
||||||
|
memory_limit = 1G
|
||||||
|
max_execution_time = 120
|
||||||
|
|
|
||||||
|
|
@ -5,10 +5,6 @@
|
||||||
{{-- {{ config('app.name') }}--}}
|
{{-- {{ config('app.name') }}--}}
|
||||||
{{-- </h1>--}}
|
{{-- </h1>--}}
|
||||||
<h1>{{ config('app.name') }}</h1>
|
<h1>{{ config('app.name') }}</h1>
|
||||||
|
|
||||||
@auth
|
|
||||||
<livewire:search />
|
|
||||||
@endauth
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<livewire:post-feature lazy />
|
<livewire:post-feature lazy />
|
||||||
|
|
|
||||||
34
resources/views/livewire/app/pagination.blade.php
Normal file
34
resources/views/livewire/app/pagination.blade.php
Normal file
|
|
@ -0,0 +1,34 @@
|
||||||
|
<div class="wa-stack">
|
||||||
|
<wa-divider></wa-divider>
|
||||||
|
<div class="wa-split">
|
||||||
|
<span class="wa-caption-l">Showing {{ $paginator->firstItem() }}-{{ $paginator->lastItem() }} of {{ $paginator->total() }} results</span>
|
||||||
|
<div class="wa-cluster wa-gap-xs">
|
||||||
|
<wa-button
|
||||||
|
appearance="outlined"
|
||||||
|
@if($paginator->onFirstPage())
|
||||||
|
disabled
|
||||||
|
@else
|
||||||
|
wire:click="previousPage"
|
||||||
|
wire:loading.attr="disabled"
|
||||||
|
rel="prev"
|
||||||
|
@endif
|
||||||
|
>
|
||||||
|
<wa-icon name="arrow-left" slot="prefix"></wa-icon>
|
||||||
|
Previous
|
||||||
|
</wa-button>
|
||||||
|
<wa-button
|
||||||
|
appearance="outlined"
|
||||||
|
@if($paginator->onLastPage())
|
||||||
|
disabled
|
||||||
|
@else
|
||||||
|
wire:click="nextPage"
|
||||||
|
wire:loading.attr="disabled"
|
||||||
|
rel="prev"
|
||||||
|
@endif
|
||||||
|
>
|
||||||
|
Next
|
||||||
|
<wa-icon name="arrow-right" slot="suffix"></wa-icon>
|
||||||
|
</wa-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
@ -6,4 +6,6 @@
|
||||||
<livewire:app.user-card :$user />
|
<livewire:app.user-card :$user />
|
||||||
@endforeach
|
@endforeach
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{{ $users->links('livewire.app.pagination') }}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
||||||
|
|
@ -40,12 +40,14 @@
|
||||||
<span>Tags</span>
|
<span>Tags</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@foreach ($tags as $postTag)
|
<div class="wa-stack wa-gap-2xs" wire:poll>
|
||||||
<div class="wa-split">
|
@foreach ($tags as $postTag)
|
||||||
<span style="color: {{ $postTag->tagGroup->color }};">{{ $postTag->name }}</span>
|
<div class="wa-split">
|
||||||
<wa-icon-button name="times" appearance="plain" href="{{ url("/delete/post/$post->id/tag/$postTag->id") }}" wire:navigate></wa-icon-button>
|
<span style="color: {{ $postTag->tagGroup->color }};">{{ $postTag->name }}</span>
|
||||||
</div>
|
<wa-icon-button name="times" appearance="plain" href="{{ url("/delete/post/$post->id/tag/$postTag->id") }}" style="color: var(--wa-color-danger-on-quiet);" wire:navigate></wa-icon-button>
|
||||||
@endforeach
|
</div>
|
||||||
|
@endforeach
|
||||||
|
</div>
|
||||||
|
|
||||||
<wa-divider></wa-divider>
|
<wa-divider></wa-divider>
|
||||||
|
|
||||||
|
|
@ -63,24 +65,24 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{{-- Main content --}}
|
{{-- Main content --}}
|
||||||
<div class="wa-stack wa-gap-2xl">
|
<div>
|
||||||
<livewire:posts.image :$post lazy />
|
<livewire:posts.image :$post lazy />
|
||||||
</div>
|
|
||||||
|
|
||||||
{{-- Confirm deletion dialog --}}
|
{{-- Confirm deletion dialog --}}
|
||||||
<wa-dialog label="Confirm post deletion" without-header light-dismiss id="modal-confirm-post-delete" style="--width: 360px;">
|
<wa-dialog label="Confirm post deletion" without-header light-dismiss id="modal-confirm-post-delete" style="--width: 360px;" wire:ignore>
|
||||||
<div class="wa-stack wa-align-items-center">
|
<div class="wa-stack wa-align-items-center">
|
||||||
<span>Are you sure you want to delete this post?</span>
|
<span>Are you sure you want to delete this post?</span>
|
||||||
<div class="wa-split">
|
<div class="wa-split">
|
||||||
<wa-button variant="neutral" appearance="outlined" data-dialog="close">
|
<wa-button variant="neutral" appearance="outlined" data-dialog="close">
|
||||||
No, go back
|
No, go back
|
||||||
</wa-button>
|
</wa-button>
|
||||||
<wa-button href="{{ url("delete/post/$post->id") }}" appearance="outlined" variant="danger" wire:navigate>
|
<wa-button href="{{ url("delete/post/$post->id") }}" appearance="outlined" variant="danger" wire:navigate>
|
||||||
Yes, delete it
|
Yes, delete it
|
||||||
</wa-button>
|
</wa-button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</wa-dialog>
|
||||||
</wa-dialog>
|
</div>
|
||||||
|
|
||||||
@script
|
@script
|
||||||
<script>
|
<script>
|
||||||
|
|
|
||||||
|
|
@ -5,4 +5,5 @@
|
||||||
<livewire:posts.thumbnail :$post lazy />
|
<livewire:posts.thumbnail :$post lazy />
|
||||||
@endforeach
|
@endforeach
|
||||||
</div>
|
</div>
|
||||||
|
{{ $posts->links('livewire.app.pagination') }}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
<div class="wa-flank wa-align-items-start wa-gap-3xl" style="--flank-size: 20rem;">
|
<div class="wa-flank wa-align-items-start wa-gap-3xl" style="--flank-size: 20rem;">
|
||||||
|
|
||||||
{{-- Sidebar --}}
|
{{-- Sidebar --}}
|
||||||
<div class="wa-stack" wire:poll>
|
<div class="wa-stack">
|
||||||
|
|
||||||
{{-- Post navigation --}}
|
{{-- Post navigation --}}
|
||||||
<div class="wa-cluster">
|
<div class="wa-cluster">
|
||||||
|
|
@ -49,6 +49,12 @@
|
||||||
</wa-format-date>
|
</wa-format-date>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{{-- Post dimensions --}}
|
||||||
|
<div class="wa-cluster">
|
||||||
|
<wa-icon fixed-width name="image"></wa-icon>
|
||||||
|
<span>{{ $post->getDimensionsStr() }}, <wa-format-bytes value="{{ $post->getFileSize() }}"></wa-format-bytes> {{ $post->extension }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
{{-- Post rating --}}
|
{{-- Post rating --}}
|
||||||
<div class="wa-cluster">
|
<div class="wa-cluster">
|
||||||
<wa-icon fixed-width name="face-hand-peeking"></wa-icon>
|
<wa-icon fixed-width name="face-hand-peeking"></wa-icon>
|
||||||
|
|
@ -63,9 +69,11 @@
|
||||||
<span>Tags</span>
|
<span>Tags</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@foreach ($post->tags as $tag)
|
<div class="wa-stack wa-gap-2xs" wire:poll>
|
||||||
<span style="color: {{ $tag->tagGroup->color }};">{{ $tag->name }}</span>
|
@foreach ($post->tags as $tag)
|
||||||
@endforeach
|
<span style="color: {{ $tag->tagGroup->color }};">{{ $tag->name }}</span>
|
||||||
|
@endforeach
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,19 +0,0 @@
|
||||||
<form wire:submit>
|
|
||||||
<wa-input wire:model.live="searchText" type="text" placeholder="Search for posts, tags, users, etc.">
|
|
||||||
<wa-icon slot="prefix" name="magnifying-glass"></wa-icon>
|
|
||||||
</wa-input>
|
|
||||||
|
|
||||||
<wa-popup
|
|
||||||
placement="bottom-end"
|
|
||||||
distance="10"
|
|
||||||
sync="width"
|
|
||||||
auto-size="vertical"
|
|
||||||
auto-size-padding="10"
|
|
||||||
{{ $searchResults ? 'active' : '' }}>
|
|
||||||
<wa-card>
|
|
||||||
@isset($searchResults)
|
|
||||||
<pre><code>{{ var_dump($searchResults) }}</code></pre>
|
|
||||||
@endisset
|
|
||||||
</wa-card>
|
|
||||||
</wa-popup>
|
|
||||||
</form>
|
|
||||||
3
resources/views/livewire/tags/view.blade.php
Normal file
3
resources/views/livewire/tags/view.blade.php
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
<div>
|
||||||
|
{{-- To attain knowledge, add things every day; To attain wisdom, subtract things every day. --}}
|
||||||
|
</div>
|
||||||
|
|
@ -10,6 +10,7 @@ use App\Livewire\Posts\Edit as EditPost;
|
||||||
use App\Livewire\Posts\View as ViewPost;
|
use App\Livewire\Posts\View as ViewPost;
|
||||||
use App\Livewire\Tags\Index as TagsIndexPage;
|
use App\Livewire\Tags\Index as TagsIndexPage;
|
||||||
use App\Livewire\Tags\Groups as TagGroupsPage;
|
use App\Livewire\Tags\Groups as TagGroupsPage;
|
||||||
|
use App\Livewire\Tags\View as TagViewPage;
|
||||||
use App\Models\Post;
|
use App\Models\Post;
|
||||||
use Illuminate\Support\Facades\Route;
|
use Illuminate\Support\Facades\Route;
|
||||||
use Illuminate\Support\Facades\Storage;
|
use Illuminate\Support\Facades\Storage;
|
||||||
|
|
@ -21,6 +22,7 @@ Route::middleware('auth')->group(function () {
|
||||||
Route::get('/upload', UploadPage::class)->name('upload');
|
Route::get('/upload', UploadPage::class)->name('upload');
|
||||||
Route::get('/profiles/{user}', ProfilePage::class);
|
Route::get('/profiles/{user}', ProfilePage::class);
|
||||||
Route::get('/users', UsersPage::class)->name('users');
|
Route::get('/users', UsersPage::class)->name('users');
|
||||||
|
Route::get('/phpinfo', function () { return phpinfo(); });
|
||||||
});
|
});
|
||||||
|
|
||||||
// Post routes
|
// Post routes
|
||||||
|
|
@ -36,6 +38,7 @@ Route::middleware('auth')->prefix('posts')->group(function () {
|
||||||
// Tag routes
|
// Tag routes
|
||||||
Route::middleware('auth')->prefix('tags')->group(function () {
|
Route::middleware('auth')->prefix('tags')->group(function () {
|
||||||
Route::get('/', TagsIndexPage::class)->name('tags.home');
|
Route::get('/', TagsIndexPage::class)->name('tags.home');
|
||||||
|
Route::get('/view/{tag}', TagViewPage::class)->name('tags.view');
|
||||||
Route::get('/groups', TagGroupsPage::class)->name('tags.groups');
|
Route::get('/groups', TagGroupsPage::class)->name('tags.groups');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue