First commit

This commit is contained in:
Kevin Adametz 2025-10-20 17:50:35 +02:00
commit 7cf3558ba7
12933 changed files with 1180047 additions and 0 deletions

View file

@ -0,0 +1,225 @@
<?php
namespace FluxCms\StarterComponents\Components;
use Livewire\Component;
use FluxCms\Core\FieldTypes\TextField;
use FluxCms\Core\FieldTypes\WysiwygField;
use FluxCms\Core\FieldTypes\MediaField;
use FluxCms\Core\FieldTypes\BooleanField;
class HeroSection extends Component
{
public array $content = [];
public function mount(array $content = [])
{
$this->content = $content;
}
public static function getCmsName(): string
{
return 'Hero Section';
}
public static function getCmsDescription(): string
{
return 'Large header section with headline, text and optional background image with call-to-action button';
}
public static function getCmsCategory(): string
{
return 'Layout';
}
public static function getCmsIcon(): string
{
return 'hero';
}
public static function getCmsTags(): array
{
return ['header', 'hero', 'banner', 'cta'];
}
public static function getCmsPreview(): string
{
return '<div class="bg-gradient-to-r from-blue-600 to-purple-700 text-white p-8 text-center rounded-lg">
<h1 class="text-3xl font-bold mb-4">Hero Headline</h1>
<p class="text-lg mb-6">Compelling subtitle text that draws attention</p>
<button class="bg-white text-blue-600 px-6 py-3 rounded-lg font-semibold">Call to Action</button>
</div>';
}
public static function getCmsFields(): array
{
return [
TextField::make('headline', 'Headline')
->translatable()
->required()
->maxLength(100)
->helpText('Main headline for the hero section'),
TextField::make('subheadline', 'Subheadline')
->translatable()
->maxLength(200)
->helpText('Supporting text below the main headline'),
WysiwygField::make('description', 'Description')
->translatable()
->basic()
->helpText('Additional descriptive text'),
TextField::make('cta_text', 'Button Text')
->translatable()
->maxLength(50)
->placeholder('Get Started'),
TextField::make('cta_link', 'Button Link')
->url()
->placeholder('https://example.com'),
BooleanField::make('cta_new_tab', 'Open Link in New Tab')
->default(false),
MediaField::make('background_image', 'Background Image')
->images()
->helpText('Optional background image for the hero section'),
BooleanField::make('dark_overlay', 'Dark Overlay')
->default(true)
->helpText('Add dark overlay for better text readability'),
TextField::make('text_alignment', 'Text Alignment')
->default('center')
->helpText('Text alignment: left, center, right'),
TextField::make('min_height', 'Minimum Height')
->default('500px')
->helpText('Minimum height of the hero section (e.g., 500px, 50vh)'),
];
}
public static function validateContent(array $content): array
{
$errors = [];
// Custom validation: If CTA text is provided, link must also be provided
$ctaText = $content['cta_text'] ?? [];
$ctaLink = $content['cta_link'] ?? '';
foreach ($ctaText as $locale => $text) {
if (!empty($text) && empty($ctaLink)) {
$errors["cta_link"] = ['Button link is required when button text is provided'];
break;
}
}
return $errors;
}
public function render()
{
return view('flux-cms-starter::components.hero-section');
}
// Helper methods for accessing content
protected function getHeadline(?string $locale = null): string
{
$locale = $locale ?? app()->getLocale();
return $this->content['headline'][$locale] ?? '';
}
protected function getSubheadline(?string $locale = null): string
{
$locale = $locale ?? app()->getLocale();
return $this->content['subheadline'][$locale] ?? '';
}
protected function getDescription(?string $locale = null): string
{
$locale = $locale ?? app()->getLocale();
return $this->content['description'][$locale] ?? '';
}
protected function getCtaText(?string $locale = null): string
{
$locale = $locale ?? app()->getLocale();
return $this->content['cta_text'][$locale] ?? '';
}
protected function getCtaLink(): string
{
return $this->content['cta_link'] ?? '';
}
protected function getTextAlignment(): string
{
return $this->content['text_alignment'] ?? 'center';
}
protected function getMinHeight(): string
{
return $this->content['min_height'] ?? '500px';
}
protected function hasCta(): bool
{
return !empty($this->getCtaText()) && !empty($this->getCtaLink());
}
protected function hasBackgroundImage(): bool
{
return !empty($this->content['background_image']);
}
protected function getBackgroundImageUrl(): ?string
{
if (!$this->hasBackgroundImage()) {
return null;
}
$mediaId = $this->content['background_image'];
$media = \Spatie\MediaLibrary\MediaCollections\Models\Media::find($mediaId);
return $media?->getUrl('large') ?? $media?->getUrl();
}
protected function hasDarkOverlay(): bool
{
return $this->content['dark_overlay'] ?? true;
}
protected function opensInNewTab(): bool
{
return $this->content['cta_new_tab'] ?? false;
}
protected function getContainerClasses(): string
{
$classes = ['hero-section', 'relative', 'flex', 'items-center', 'justify-center'];
// Text alignment
$alignment = $this->getTextAlignment();
if ($alignment === 'left') {
$classes[] = 'text-left';
} elseif ($alignment === 'right') {
$classes[] = 'text-right';
} else {
$classes[] = 'text-center';
}
return implode(' ', $classes);
}
protected function getTextClasses(): string
{
$classes = ['relative', 'z-10'];
if ($this->hasBackgroundImage()) {
$classes[] = 'text-white';
}
return implode(' ', $classes);
}
}