Build reusable UI components with Symfony UX Twig Components and Live Components for reactive interfaces
/plugin marketplace add MakFly/superpowers-symfony/plugin install makfly-superpowers-symfony@MakFly/superpowers-symfonyThis skill inherits all available tools. When active, it can use any tool Claude has access to.
composer require symfony/ux-twig-component
# For reactive components
composer require symfony/ux-live-component
<?php
// src/Twig/Components/Alert.php
namespace App\Twig\Components;
use Symfony\UX\TwigComponent\Attribute\AsTwigComponent;
#[AsTwigComponent]
class Alert
{
public string $type = 'info';
public string $message;
public bool $dismissible = false;
}
{# templates/components/Alert.html.twig #}
<div class="alert alert-{{ type }}{% if dismissible %} alert-dismissible{% endif %}" role="alert">
{{ message }}
{% if dismissible %}
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
{% endif %}
</div>
{# In any template #}
<twig:Alert type="success" message="Operation completed!" />
<twig:Alert type="danger" message="An error occurred" dismissible />
<?php
// src/Twig/Components/Card.php
namespace App\Twig\Components;
use Symfony\UX\TwigComponent\Attribute\AsTwigComponent;
#[AsTwigComponent]
class Card
{
public ?string $title = null;
public string $class = '';
}
{# templates/components/Card.html.twig #}
<div class="card {{ class }}">
{% if title %}
<div class="card-header">
<h5 class="card-title">{{ title }}</h5>
</div>
{% endif %}
<div class="card-body">
{% block content %}{% endblock %}
</div>
{% if block('footer') is not empty %}
<div class="card-footer">
{% block footer %}{% endblock %}
</div>
{% endif %}
</div>
<twig:Card title="User Profile">
<twig:block name="content">
<p>Name: {{ user.name }}</p>
<p>Email: {{ user.email }}</p>
</twig:block>
<twig:block name="footer">
<a href="{{ path('user_edit', {id: user.id}) }}" class="btn btn-primary">Edit</a>
</twig:block>
</twig:Card>
<?php
// src/Twig/Components/UserCard.php
namespace App\Twig\Components;
use App\Entity\User;
use App\Repository\PostRepository;
use Symfony\UX\TwigComponent\Attribute\AsTwigComponent;
#[AsTwigComponent]
class UserCard
{
public User $user;
public function __construct(
private PostRepository $postRepository,
) {}
public function getPostCount(): int
{
return $this->postRepository->countByAuthor($this->user);
}
public function getRecentPosts(): array
{
return $this->postRepository->findRecentByAuthor($this->user, 3);
}
}
{# templates/components/UserCard.html.twig #}
<div class="user-card">
<h3>{{ user.name }}</h3>
<p>{{ this.postCount }} posts</p>
<ul>
{% for post in this.recentPosts %}
<li>{{ post.title }}</li>
{% endfor %}
</ul>
</div>
<?php
// src/Twig/Components/Counter.php
namespace App\Twig\Components;
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
use Symfony\UX\LiveComponent\Attribute\LiveProp;
use Symfony\UX\LiveComponent\Attribute\LiveAction;
use Symfony\UX\LiveComponent\DefaultActionTrait;
#[AsLiveComponent]
class Counter
{
use DefaultActionTrait;
#[LiveProp(writable: true)]
public int $count = 0;
#[LiveAction]
public function increment(): void
{
$this->count++;
}
#[LiveAction]
public function decrement(): void
{
$this->count--;
}
}
{# templates/components/Counter.html.twig #}
<div {{ attributes }}>
<span>Count: {{ count }}</span>
<button data-action="live#action" data-live-action-param="decrement">-</button>
<button data-action="live#action" data-live-action-param="increment">+</button>
</div>
<?php
// src/Twig/Components/ProductSearch.php
namespace App\Twig\Components;
use App\Repository\ProductRepository;
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
use Symfony\UX\LiveComponent\Attribute\LiveProp;
use Symfony\UX\LiveComponent\DefaultActionTrait;
#[AsLiveComponent]
class ProductSearch
{
use DefaultActionTrait;
#[LiveProp(writable: true, url: true)]
public string $query = '';
#[LiveProp(writable: true)]
public int $page = 1;
public function __construct(
private ProductRepository $products,
) {}
public function getProducts(): array
{
if (strlen($this->query) < 2) {
return [];
}
return $this->products->search($this->query, $this->page);
}
}
{# templates/components/ProductSearch.html.twig #}
<div {{ attributes }}>
<input
type="search"
data-model="query"
placeholder="Search products..."
class="form-control"
>
<div class="results mt-3">
{% for product in this.products %}
<div class="product-card">
<h4>{{ product.name }}</h4>
<p>{{ product.price|format_currency('EUR') }}</p>
</div>
{% else %}
{% if query|length >= 2 %}
<p>No products found.</p>
{% endif %}
{% endfor %}
</div>
</div>
<?php
// src/Twig/Components/ContactForm.php
namespace App\Twig\Components;
use App\Form\ContactType;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\Form\FormInterface;
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
use Symfony\UX\LiveComponent\Attribute\LiveProp;
use Symfony\UX\LiveComponent\Attribute\LiveAction;
use Symfony\UX\LiveComponent\ComponentWithFormTrait;
use Symfony\UX\LiveComponent\DefaultActionTrait;
#[AsLiveComponent]
class ContactForm extends AbstractController
{
use ComponentWithFormTrait;
use DefaultActionTrait;
#[LiveProp]
public bool $submitted = false;
protected function instantiateForm(): FormInterface
{
return $this->createForm(ContactType::class);
}
#[LiveAction]
public function submit(): void
{
$this->submitForm();
if ($this->getForm()->isValid()) {
$data = $this->getForm()->getData();
// Process form...
$this->submitted = true;
}
}
}
{# templates/components/ContactForm.html.twig #}
<div {{ attributes }}>
{% if submitted %}
<div class="alert alert-success">Thank you for your message!</div>
{% else %}
{{ form_start(form) }}
{{ form_row(form.name) }}
{{ form_row(form.email) }}
{{ form_row(form.message) }}
<button
type="submit"
data-action="live#action"
data-live-action-param="submit"
class="btn btn-primary"
>
Send
</button>
{{ form_end(form) }}
{% endif %}
</div>
data-model="debounce(300)|query"url: true for bookmarkable stateApplies Anthropic's official brand colors and typography to any sort of artifact that may benefit from having Anthropic's look-and-feel. Use it when brand colors or style guidelines, visual formatting, or company design standards apply.
Creating algorithmic art using p5.js with seeded randomness and interactive parameter exploration. Use this when users request creating art using code, generative art, algorithmic art, flow fields, or particle systems. Create original algorithmic art rather than copying existing artists' work to avoid copyright violations.
Create beautiful visual art in .png and .pdf documents using design philosophy. You should use this skill when the user asks to create a poster, piece of art, design, or other static piece. Create original visual designs, never copying existing artists' work to avoid copyright violations.