Aree Widget

In questa pagina

Le aree widget sono regioni nominate nei tuoi template dove gli amministratori possono posizionare blocchi di contenuto. Usale per barre laterali, colonne del piè di pagina, banner promozionali o qualsiasi sezione che gli editori dovrebbero controllare senza toccare il codice.

Interrogare le Aree Widget

Usa getWidgetArea() per recuperare un’area widget per nome:

---
import { getWidgetArea } from "emdash";

const sidebar = await getWidgetArea("sidebar");
---

{sidebar && sidebar.widgets.length > 0 && (
  <aside class="sidebar">
    {sidebar.widgets.map(widget => (
      <div class="widget">
        {widget.title && <h3>{widget.title}</h3>}
        <!-- Renderizza il contenuto del widget -->
      </div>
    ))}
  </aside>
)}

La funzione restituisce null se l’area widget non esiste.

Struttura dell’Area Widget

Un’area widget contiene metadati e un array di widget:

interface WidgetArea {
	id: string;
	name: string; // Unique identifier ("sidebar", "footer-1")
	label: string; // Display name ("Main Sidebar")
	description?: string;
	widgets: Widget[];
}

interface Widget {
	id: string;
	type: "content" | "menu" | "component";
	title?: string;
	// Type-specific fields
	content?: PortableTextBlock[]; // For content widgets
	menuName?: string; // For menu widgets
	componentId?: string; // For component widgets
	componentProps?: Record<string, unknown>;
}

Tipi di Widget

EmDash supporta tre tipi di widget:

Widget di Contenuto

Contenuto rich text memorizzato come Portable Text. Renderizza usando il componente PortableText:

---
import { PortableText } from "emdash/ui";
---

{widget.type === "content" && widget.content && (
  <div class="widget-content">
    <PortableText value={widget.content} />
  </div>
)}

Widget di Menu

Visualizza un menu di navigazione all’interno di un’area widget:

---
import { getMenu } from "emdash";

const menu = widget.menuName ? await getMenu(widget.menuName) : null;
---

{widget.type === "menu" && menu && (
  <nav class="widget-nav">
    <ul>
      {menu.items.map(item => (
        <li><a href={item.url}>{item.label}</a></li>
      ))}
    </ul>
  </nav>
)}

Widget di Componenti

Renderizza un componente registrato con props configurabili. EmDash include questi componenti principali:

Component IDDescrizioneProps
core:recent-postsElenco di post recenticount, showThumbnails, showDate
core:categoriesElenco categorieshowCount, hierarchical
core:tagsNuvola di tagshowCount, limit
core:searchModulo di ricercaplaceholder
core:archivesArchivi mensili/annualitype, limit

Renderizzare i Widget

Crea un componente di rendering dei widget riutilizzabile:

---
import { PortableText } from "emdash/ui";
import { getMenu } from "emdash";
import type { Widget } from "emdash";

// Import your widget components
import RecentPosts from "./widgets/RecentPosts.astro";
import Categories from "./widgets/Categories.astro";
import TagCloud from "./widgets/TagCloud.astro";
import SearchForm from "./widgets/SearchForm.astro";
import Archives from "./widgets/Archives.astro";

interface Props {
  widget: Widget;
}

const { widget } = Astro.props;

const componentMap: Record<string, any> = {
  "core:recent-posts": RecentPosts,
  "core:categories": Categories,
  "core:tags": TagCloud,
  "core:search": SearchForm,
  "core:archives": Archives,
};

const menu = widget.type === "menu" && widget.menuName
  ? await getMenu(widget.menuName)
  : null;
---

<div class="widget">
  {widget.title && <h3 class="widget-title">{widget.title}</h3>}

  {widget.type === "content" && widget.content && (
    <div class="widget-content">
      <PortableText value={widget.content} />
    </div>
  )}

  {widget.type === "menu" && menu && (
    <nav class="widget-menu">
      <ul>
        {menu.items.map(item => (
          <li><a href={item.url}>{item.label}</a></li>
        ))}
      </ul>
    </nav>
  )}

  {widget.type === "component" && widget.componentId && componentMap[widget.componentId] && (
    <Fragment>
      {(() => {
        const Component = componentMap[widget.componentId!];
        return <Component {...widget.componentProps} />;
      })()}
    </Fragment>
  )}
</div>

Esempi di Componenti Widget

Widget Post Recenti

Il seguente componente renderizza i post più recenti, con miniature e date opzionali:

---
import { getEmDashCollection } from "emdash";

interface Props {
  count?: number;
  showThumbnails?: boolean;
  showDate?: boolean;
}

const { count = 5, showThumbnails = false, showDate = true } = Astro.props;

const { entries: posts } = await getEmDashCollection("posts", {
  limit: count,
  orderBy: { publishedAt: "desc" },
});
---

<ul class="recent-posts">
  {posts.map(post => (
    <li>
      {showThumbnails && post.data.featured_image && (
        <img src={post.data.featured_image} alt="" class="thumbnail" />
      )}
      <a href={`/posts/${post.data.slug}`}>{post.data.title}</a>
      {showDate && post.data.publishedAt && (
        <time datetime={post.data.publishedAt.toISOString()}>
          {post.data.publishedAt.toLocaleDateString()}
        </time>
      )}
    </li>
  ))}
</ul>

Widget di Ricerca

Il seguente componente renderizza un modulo di ricerca che invia a una pagina di ricerca:

---
interface Props {
  placeholder?: string;
}

const { placeholder = "Search..." } = Astro.props;
---

<form action="/search" method="get" class="search-form">
  <input
    type="search"
    name="q"
    placeholder={placeholder}
    aria-label="Search"
  />
  <button type="submit">Search</button>
</form>

Usare le Aree Widget nei Layout

L’esempio seguente mostra un layout di blog con un’area widget nella barra laterale:

---
import { getWidgetArea } from "emdash";
import WidgetRenderer from "../components/WidgetRenderer.astro";

const sidebar = await getWidgetArea("sidebar");
---

<div class="layout">
  <main class="content">
    <slot />
  </main>

  {sidebar && sidebar.widgets.length > 0 && (
    <aside class="sidebar">
      {sidebar.widgets.map(widget => (
        <WidgetRenderer widget={widget} />
      ))}
    </aside>
  )}
</div>

<style>
  .layout {
    display: grid;
    grid-template-columns: 1fr 300px;
    gap: 2rem;
  }

  @media (max-width: 768px) {
    .layout {
      grid-template-columns: 1fr;
    }
  }
</style>

Elencare Tutte le Aree Widget

Usa getWidgetAreas() per recuperare tutte le aree widget con i loro widget:

import { getWidgetAreas } from "emdash";

const areas = await getWidgetAreas();
// Returns all areas with widgets populated

Creare Aree Widget

Crea aree widget tramite l’interfaccia di amministrazione su /_emdash/admin/widgets, o usa l’API di amministrazione:

POST /_emdash/api/widget-areas
Content-Type: application/json

{
  "name": "footer-1",
  "label": "Footer Column 1",
  "description": "First column in the footer"
}

Aggiungere un widget di contenuto:

POST /_emdash/api/widget-areas/footer-1/widgets
Content-Type: application/json

{
  "type": "content",
  "title": "About Us",
  "content": [
    {
      "_type": "block",
      "style": "normal",
      "children": [{ "_type": "span", "text": "Welcome to our site." }]
    }
  ]
}

Aggiungere un widget di componente:

POST /_emdash/api/widget-areas/sidebar/widgets
Content-Type: application/json

{
  "type": "component",
  "title": "Recent Posts",
  "componentId": "core:recent-posts",
  "componentProps": { "count": 5, "showDate": true }
}

Riferimento API

getWidgetArea(name)

Recupera un’area widget per nome con tutti i widget.

Parametri:

  • name — L’identificatore unico dell’area widget (string)

Restituisce: Promise<WidgetArea | null>

getWidgetAreas()

Elenca tutte le aree widget con i loro widget.

Restituisce: Promise<WidgetArea[]>

getWidgetComponents()

Elenca le definizioni dei componenti widget disponibili per l’interfaccia di amministrazione.

Restituisce: WidgetComponentDef[]