feat: Inertia profile settings page, Studio edit redesign, EGS, Nova UI components\n\n- Redesign /dashboard/profile as Inertia React page (Settings/ProfileEdit)\n with SettingsLayout sidebar, Nova UI components (TextInput, Textarea,\n Toggle, Select, RadioGroup, Modal, Button), avatar drag-and-drop,\n password change, and account deletion sections\n- Redesign Studio artwork edit page with two-column layout, Nova components,\n integrated TagPicker, and version history modal\n- Add shared MarkdownEditor component\n- Add Early-Stage Growth System (EGS): SpotlightEngine, FeedBlender,\n GridFiller, AdaptiveTimeWindow, ActivityLayer, admin panel\n- Fix upload category/tag persistence (V1+V2 paths)\n- Fix tag source enum, category tree display, binding resolution\n- Add settings.jsx Vite entry, settings.blade.php wrapper\n- Update ProfileController with JSON response support for API calls\n- Various route fixes (profile.edit, toolbar settings link)"

This commit is contained in:
2026-03-03 20:57:43 +01:00
parent dc51d65440
commit b9c2d8597d
114 changed files with 8760 additions and 693 deletions

View File

@@ -0,0 +1,116 @@
<?php
use Illuminate\Foundation\Testing\RefreshDatabase;
uses(RefreshDatabase::class);
describe('Footer pages', function () {
it('shows the FAQ page', function () {
$this->get('/faq')
->assertOk()
->assertSee('Frequently Asked Questions');
});
it('shows the Rules & Guidelines page', function () {
$this->get('/rules-and-guidelines')
->assertOk()
->assertSee('Rules & Guidelines');
});
it('shows the Privacy Policy page', function () {
$this->get('/privacy-policy')
->assertOk()
->assertSee('Privacy Policy');
});
it('shows the Terms of Service page', function () {
$this->get('/terms-of-service')
->assertOk()
->assertSee('Terms of Service');
});
it('shows the bug report page to guests with login prompt', function () {
$this->get('/bug-report')
->assertOk()
->assertSee('Bug Report');
});
it('shows the bug report form to authenticated users', function () {
$user = \App\Models\User::factory()->create();
$this->actingAs($user)
->get('/bug-report')
->assertOk()
->assertSee('Subject');
});
it('submits a bug report as authenticated user', function () {
$user = \App\Models\User::factory()->create();
$this->actingAs($user)
->post('/bug-report', [
'subject' => 'Test subject',
'description' => 'This is a test bug report description.',
])
->assertRedirect('/bug-report');
$this->assertDatabaseHas('bug_reports', [
'user_id' => $user->id,
'subject' => 'Test subject',
]);
});
it('rejects bug report submission from guests', function () {
$this->post('/bug-report', [
'subject' => 'Test',
'description' => 'Test description',
])->assertRedirect('/login');
});
it('shows the staff page', function () {
$this->get('/staff')
->assertOk()
->assertSee('Meet the Staff');
});
it('shows the RSS feeds info page', function () {
$this->get('/rss-feeds')
->assertOk()
->assertSee('RSS')
->assertSee('Latest Uploads');
});
it('returns XML for latest uploads feed', function () {
$this->get('/rss/latest-uploads.xml')
->assertOk()
->assertHeader('Content-Type', 'application/rss+xml; charset=utf-8');
});
it('returns XML for latest skins feed', function () {
$this->get('/rss/latest-skins.xml')
->assertOk()
->assertHeader('Content-Type', 'application/rss+xml; charset=utf-8');
});
it('returns XML for latest wallpapers feed', function () {
$this->get('/rss/latest-wallpapers.xml')
->assertOk()
->assertHeader('Content-Type', 'application/rss+xml; charset=utf-8');
});
it('returns XML for latest photos feed', function () {
$this->get('/rss/latest-photos.xml')
->assertOk()
->assertHeader('Content-Type', 'application/rss+xml; charset=utf-8');
});
it('RSS feed contains valid XML', function () {
$response = $this->get('/rss/latest-uploads.xml');
$response->assertOk();
$xml = simplexml_load_string($response->getContent());
expect($xml)->not->toBeFalse();
expect((string) $xml->channel->title)->toContain('Skinbase');
});
});

View File

@@ -0,0 +1,194 @@
<?php
declare(strict_types=1);
/**
* Feature tests for the new canonical route families introduced by the
* Routing Unification spec (§3.2 Explore, §4 Blog/Pages, §6.1 redirects).
*/
// ── /explore routes ──────────────────────────────────────────────────────────
it('GET /explore returns 200', function () {
$this->get('/explore')->assertOk();
});
it('GET /explore contains "Explore" heading', function () {
$this->get('/explore')->assertOk()->assertSee('Explore', false);
});
it('GET /explore/wallpapers returns 200', function () {
$this->get('/explore/wallpapers')->assertOk();
});
it('GET /explore/skins returns 200', function () {
$this->get('/explore/skins')->assertOk();
});
it('GET /explore/photography returns 200', function () {
$this->get('/explore/photography')->assertOk();
});
it('GET /explore/artworks returns 200', function () {
$this->get('/explore/artworks')->assertOk();
});
it('GET /explore/other returns 200', function () {
$this->get('/explore/other')->assertOk();
});
it('GET /explore/wallpapers/trending returns 200', function () {
$this->get('/explore/wallpapers/trending')->assertOk();
});
it('GET /explore/wallpapers/latest returns 200', function () {
$this->get('/explore/wallpapers/latest')->assertOk();
});
it('GET /explore/wallpapers/best returns 200', function () {
$this->get('/explore/wallpapers/best')->assertOk();
});
it('GET /explore/wallpapers/new-hot returns 200', function () {
$this->get('/explore/wallpapers/new-hot')->assertOk();
});
it('/explore pages include canonical link tag', function () {
$html = $this->get('/explore')->assertOk()->getContent();
expect($html)->toContain('rel="canonical"');
});
it('/explore pages set robots index,follow', function () {
$html = $this->get('/explore')->assertOk()->getContent();
expect($html)->toContain('index,follow');
});
it('/explore pages include breadcrumb JSON-LD', function () {
$html = $this->get('/explore/wallpapers')->assertOk()->getContent();
expect($html)->toContain('BreadcrumbList');
});
// ── 301 redirects from legacy routes ─────────────────────────────────────────
it('GET /browse redirects to /explore with 301', function () {
$this->get('/browse')->assertRedirect('/explore')->assertStatus(301);
});
it('GET /discover redirects to /discover/trending with 301', function () {
$this->get('/discover')->assertRedirect('/discover/trending')->assertStatus(301);
});
// ── /blog routes ─────────────────────────────────────────────────────────────
it('GET /blog returns 200', function () {
$this->get('/blog')->assertOk();
});
it('/blog page contains Blog heading', function () {
$html = $this->get('/blog')->assertOk()->getContent();
expect($html)->toContain('Blog');
});
it('/blog page includes canonical link', function () {
$html = $this->get('/blog')->assertOk()->getContent();
expect($html)->toContain('rel="canonical"');
});
it('/blog/:slug returns 404 for non-existent post', function () {
$this->get('/blog/this-post-does-not-exist-xyz')->assertNotFound();
});
it('published blog post is accessible at /blog/:slug', function () {
$post = \App\Models\BlogPost::factory()->create([
'slug' => 'hello-world',
'title' => 'Hello World Post',
'body' => '<p>Test content.</p>',
'is_published' => true,
'published_at' => now()->subDay(),
]);
$html = $this->get('/blog/hello-world')->assertOk()->getContent();
expect($html)->toContain('Hello World Post');
});
it('unpublished blog post returns 404', function () {
\App\Models\BlogPost::factory()->create([
'slug' => 'draft-post',
'title' => 'Draft Post',
'body' => '<p>Draft.</p>',
'is_published' => false,
]);
$this->get('/blog/draft-post')->assertNotFound();
});
it('/blog post includes Article JSON-LD', function () {
\App\Models\BlogPost::factory()->create([
'slug' => 'schema-post',
'title' => 'Schema Post',
'body' => '<p>Content.</p>',
'is_published' => true,
'published_at' => now()->subHour(),
]);
$html = $this->get('/blog/schema-post')->assertOk()->getContent();
expect($html)->toContain('"Article"');
});
// ── /pages routes ─────────────────────────────────────────────────────────────
it('GET /pages/:slug returns 404 for non-existent page', function () {
$this->get('/pages/does-not-exist-xyz')->assertNotFound();
});
it('published page is accessible at /pages/:slug', function () {
\App\Models\Page::factory()->create([
'slug' => 'test-page',
'title' => 'Test Page Title',
'body' => '<p>Page content.</p>',
'is_published' => true,
'published_at' => now()->subDay(),
]);
$html = $this->get('/pages/test-page')->assertOk()->getContent();
expect($html)->toContain('Test Page Title');
});
it('/about returns 200 when about page exists', function () {
\App\Models\Page::factory()->create([
'slug' => 'about',
'title' => 'About Skinbase',
'body' => '<p>About us.</p>',
'is_published' => true,
'published_at' => now()->subDay(),
]);
$this->get('/about')->assertOk();
});
it('/legal/terms returns 200 when legal-terms page exists', function () {
\App\Models\Page::factory()->create([
'slug' => 'legal-terms',
'title' => 'Terms of Service',
'body' => '<p>Terms.</p>',
'is_published' => true,
'published_at' => now()->subDay(),
]);
$this->get('/legal/terms')->assertOk();
});
// ── tags index layout ─────────────────────────────────────────────────────────
it('/tags page renders with ContentLayout breadcrumbs', function () {
$html = $this->get('/tags')->assertOk()->getContent();
// ContentLayout should inject breadcrumb nav
expect($html)->toContain('Tags');
});
it('/tags page includes canonical and robots', function () {
$html = $this->get('/tags')->assertOk()->getContent();
expect($html)
->toContain('rel="canonical"')
->toContain('index,follow');
});