🎨 Add new Flux icons: implement multiple reusable icon components (e.g., hand-raised, hand-thumb-up, heart, hashtag, home) with variant support for improved UI consistency.

This commit is contained in:
HolgerHatGarKeineNode
2026-01-23 23:00:02 +01:00
parent 578e4f13fc
commit b30fec150c
792 changed files with 307541 additions and 117 deletions

View File

@@ -0,0 +1,81 @@
@blaze
@props([
'placeholder' => null,
'clearable' => true,
'closable' => null,
'icon' => null,
])
@php
// Clerable or closable, not both...
if ($closable !== null) $clearable = null;
$classes = Flux::classes()
->add('h-10 w-full flex items-center px-3 py-2')
->add('font-medium text-base sm:text-sm text-zinc-800 dark:text-white')
->add('ps-9') // Make room for magnifying glass icon...
->add('pe-9') // Make room for clear/clos button and loading indicator...
->add('outline-hidden')
->add('border-b border-zinc-200 dark:border-zinc-600')
->add('bg-white dark:bg-zinc-700')
// The below reverts styles added by Tailwind Forms plugin
->add('border-t-0 border-s-0 border-e-0 focus:ring-0 focus:border-zinc-200 dark:focus:border-zinc-600')
->add('data-invalid:text-red-500 dark:data-invalid:text-red-400')
;
$name = $attributes->whereStartsWith('wire:model')->first();
$invalid ??= ($name && $errors->has($name));
$loading = $attributes->whereStartsWith('wire:model.live')->isNotEmpty();
if ($loading) {
$attributes = $attributes->merge(['wire:loading.attr' => 'data-flux-loading']);
}
@endphp
<div class="relative flex grow mx-[-5px] mt-[-5px] mb-[5px]" data-flux-pillbox-search>
<div class="absolute top-0 bottom-0 flex items-center justify-center text-xs text-zinc-400 ps-3.5 start-0">
<?php if (is_string($icon)): ?>
<flux:icon :$icon variant="micro" />
<?php elseif ($icon): ?>
{{ $icon }}
<?php else: ?>
<flux:icon.magnifying-glass variant="micro" />
<?php endif; ?>
</div>
<input
type="text"
{{ $attributes->class($classes) }}
@if ($invalid) aria-invalid="true" data-invalid @endif
placeholder="{{ $placeholder ?? __('Search...') }}"
data-flux-pillbox-input
autofocus
/>
<?php if ($loading): ?>
<div class="opacity-0 [[data-flux-pillbox-search]:has([data-flux-loading])_&]:opacity-100 transition-opacity absolute top-0 bottom-0 flex items-center justify-center pe-2.5 end-0">
<flux:icon.loading class="text-zinc-400 [[data-flux-menu-item]:hover_&]:text-current" variant="mini" />
</div>
<?php endif; ?>
<?php if ($closable): ?>
<div class="[[data-flux-pillbox-search]:has([data-flux-loading])_&]:opacity-0 transition-opacity absolute top-0 bottom-0 flex items-center justify-center pe-1 end-0">
<ui-close>
<flux:button square variant="subtle" size="sm" aria-label="Clear search input">
<flux:icon.x-mark variant="micro" />
</flux:button>
</ui-close>
</div>
<?php elseif ($clearable): ?>
<div class="[[data-flux-pillbox-search]:has([data-flux-loading])_&]:opacity-0 transition-opacity absolute top-0 bottom-0 flex items-center justify-center pe-1 end-0 [[data-flux-pillbox-search]:has(input:placeholder-shown)_&]:hidden">
<flux:button square variant="subtle" size="sm" tabindex="-1" aria-label="Clear command input"
x-on:click="$el.closest('[data-flux-pillbox-search]').querySelector('input').value = ''; $el.closest('[data-flux-pillbox-search]').querySelector('input').dispatchEvent(new Event('input', { bubbles: false })); $el.closest('[data-flux-pillbox-search]').querySelector('input').focus()"
>
<flux:icon.x-mark variant="micro" />
</flux:button>
</div>
<?php endif; ?>
</div>