重要前提
安装AI Skills的关键前提是:必须科学上网,且开启TUN模式,这一点至关重要,直接决定安装能否顺利完成,在此郑重提醒三遍:科学上网,科学上网,科学上网。查看完整安装教程 →
umbraco-collection by umbraco/umbraco-cms-backoffice-skills
npx skills add https://github.com/umbraco/umbraco-cms-backoffice-skills --skill umbraco-collection集合在 Umbraco 后台显示一个实体列表,内置支持多种视图(表格、网格)、筛选、分页、选择和批量操作。集合连接到数据仓库,并提供了一种标准化的方式来浏览和与项目列表交互。
在实现之前,请始终获取最新的文档:
一个完整的集合由以下组件组成:
collection/
├── manifests.ts # 主集合清单
├── constants.ts # 别名常量
├── types.ts # 项目和筛选器类型
├── my-collection.context.ts # 集合上下文(继承 UmbDefaultCollectionContext)
├── my-collection.element.ts # 集合元素(继承 UmbCollectionDefaultElement)
├── repository/
│ ├── manifests.ts
│ ├── my-collection.repository.ts # 实现 UmbCollectionRepository
│ └── my-collection.data-source.ts # API 调用
├── views/
│ ├── manifests.ts
│ └── table/
│ └── my-table-view.element.ts # 表格视图
└── action/
├── manifests.ts
└── my-action.element.ts # 集合操作
广告位招租
在这里展示您的产品或服务
触达数万 AI 开发者,精准高效
Umbraco 源代码包含一个工作示例:
位置 : /Umbraco-CMS/src/Umbraco.Web.UI.Client/examples/collection/
此示例演示了一个包含仓库、视图和上下文的完整自定义集合。请研究此示例以了解生产模式。
umbraco-repository-patternumbraco-context-apiumbraco-state-managementUmbDefaultCollectionContextexport const MY_COLLECTION_ALIAS = 'My.Collection';
export const MY_COLLECTION_REPOSITORY_ALIAS = 'My.Collection.Repository';
export interface MyCollectionItemModel {
unique: string;
entityType: string;
name: string;
// 添加其他字段
}
export interface MyCollectionFilterModel {
skip?: number;
take?: number;
filter?: string;
orderBy?: string;
orderDirection?: 'asc' | 'desc';
// 添加自定义筛选器
}
import type { MyCollectionItemModel, MyCollectionFilterModel } from '../types.js';
import type { UmbCollectionDataSource } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
export class MyCollectionDataSource implements UmbCollectionDataSource<MyCollectionItemModel> {
#host: UmbControllerHost;
constructor(host: UmbControllerHost) {
this.#host = host;
}
async getCollection(filter: MyCollectionFilterModel) {
// 在此处调用你的 API
const response = await fetch(`/api/my-items?skip=${filter.skip}&take=${filter.take}`);
const data = await response.json();
const items: MyCollectionItemModel[] = data.items.map((item: any) => ({
unique: item.id,
entityType: 'my-entity',
name: item.name,
}));
return { data: { items, total: data.total } };
}
}
import type { MyCollectionFilterModel } from '../types.js';
import { MyCollectionDataSource } from './my-collection.data-source.js';
import { UmbRepositoryBase } from '@umbraco-cms/backoffice/repository';
import type { UmbCollectionRepository } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
export class MyCollectionRepository extends UmbRepositoryBase implements UmbCollectionRepository {
#dataSource: MyCollectionDataSource;
constructor(host: UmbControllerHost) {
super(host);
this.#dataSource = new MyCollectionDataSource(host);
}
async requestCollection(filter: MyCollectionFilterModel) {
return this.#dataSource.getCollection(filter);
}
}
export default MyCollectionRepository;
import { MY_COLLECTION_REPOSITORY_ALIAS } from '../constants.js';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'repository',
alias: MY_COLLECTION_REPOSITORY_ALIAS,
name: 'My Collection Repository',
api: () => import('./my-collection.repository.js'),
},
];
import type { MyCollectionItemModel, MyCollectionFilterModel } from './types.js';
import { UmbDefaultCollectionContext } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
// 默认视图别名 - 必须与你的某个 collectionView 别名匹配
const MY_TABLE_VIEW_ALIAS = 'My.CollectionView.Table';
export class MyCollectionContext extends UmbDefaultCollectionContext<
MyCollectionItemModel,
MyCollectionFilterModel
> {
constructor(host: UmbControllerHost) {
super(host, MY_TABLE_VIEW_ALIAS);
}
// 如果需要,重写或添加自定义方法
}
export { MyCollectionContext as api };
import { customElement } from '@umbraco-cms/backoffice/external/lit';
import { UmbCollectionDefaultElement } from '@umbraco-cms/backoffice/collection';
@customElement('my-collection')
export class MyCollectionElement extends UmbCollectionDefaultElement {
// 重写 renderToolbar() 以自定义标题
// protected override renderToolbar() {
// return html`<umb-collection-toolbar slot="header"></umb-collection-toolbar>`;
// }
}
export default MyCollectionElement;
export { MyCollectionElement as element };
declare global {
interface HTMLElementTagNameMap {
'my-collection': MyCollectionElement;
}
}
import type { MyCollectionItemModel } from '../../types.js';
import { UMB_COLLECTION_CONTEXT } from '@umbraco-cms/backoffice/collection';
import { css, customElement, html, state } from '@umbraco-cms/backoffice/external/lit';
import { UmbLitElement } from '@umbraco-cms/backoffice/lit-element';
import type { UmbTableColumn, UmbTableConfig, UmbTableItem } from '@umbraco-cms/backoffice/components';
@customElement('my-table-collection-view')
export class MyTableCollectionViewElement extends UmbLitElement {
@state()
private _tableItems: Array<UmbTableItem> = [];
@state()
private _selection: Array<string> = [];
#collectionContext?: typeof UMB_COLLECTION_CONTEXT.TYPE;
private _tableConfig: UmbTableConfig = {
allowSelection: true,
};
private _tableColumns: Array<UmbTableColumn> = [
{ name: 'Name', alias: 'name', allowSorting: true },
{ name: '', alias: 'entityActions', align: 'right' },
];
constructor() {
super();
this.consumeContext(UMB_COLLECTION_CONTEXT, (context) => {
this.#collectionContext = context;
// 重要:为工作区模态路由调用 setupView
context?.setupView(this);
this.#observeItems();
this.#observeSelection();
});
}
#observeItems() {
if (!this.#collectionContext) return;
this.observe(
this.#collectionContext.items,
(items) => {
this._tableItems = (items as MyCollectionItemModel[]).map((item) => ({
id: item.unique,
icon: 'icon-document',
entityType: item.entityType,
data: [
{ columnAlias: 'name', value: item.name },
{
columnAlias: 'entityActions',
value: html`<umb-entity-actions-table-column-view
.value=${{ entityType: item.entityType, unique: item.unique }}
></umb-entity-actions-table-column-view>`,
},
],
}));
},
'_observeItems',
);
}
#observeSelection() {
if (!this.#collectionContext) return;
this.observe(
this.#collectionContext.selection.selection,
(selection) => {
this._selection = selection as string[];
},
'_observeSelection',
);
}
#handleSelect(event: CustomEvent) {
event.stopPropagation();
const table = event.target as any;
this.#collectionContext?.selection.setSelection(table.selection);
}
#handleDeselect(event: CustomEvent) {
event.stopPropagation();
const table = event.target as any;
this.#collectionContext?.selection.setSelection(table.selection);
}
override render() {
return html`
<umb-table
.config=${this._tableConfig}
.columns=${this._tableColumns}
.items=${this._tableItems}
.selection=${this._selection}
@selected=${this.#handleSelect}
@deselected=${this.#handleDeselect}
></umb-table>
`;
}
}
export default MyTableCollectionViewElement;
declare global {
interface HTMLElementTagNameMap {
'my-table-collection-view': MyTableCollectionViewElement;
}
}
import { MY_COLLECTION_ALIAS } from '../constants.js';
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collectionView',
alias: 'My.CollectionView.Table',
name: 'My Table Collection View',
element: () => import('./table/my-table-view.element.js'),
weight: 200,
meta: {
label: 'Table',
icon: 'icon-list',
pathName: 'table',
},
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: MY_COLLECTION_ALIAS,
},
],
},
];
import { MY_COLLECTION_ALIAS } from '../constants.js';
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collectionAction',
kind: 'button',
alias: 'My.CollectionAction.Refresh',
name: 'Refresh Collection Action',
element: () => import('./refresh-action.element.js'),
weight: 100,
meta: {
label: 'Refresh',
},
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: MY_COLLECTION_ALIAS,
},
],
},
];
import { manifests as repositoryManifests } from './repository/manifests.js';
import { manifests as viewManifests } from './views/manifests.js';
import { manifests as actionManifests } from './action/manifests.js';
import { MY_COLLECTION_ALIAS, MY_COLLECTION_REPOSITORY_ALIAS } from './constants.js';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collection',
alias: MY_COLLECTION_ALIAS,
name: 'My Collection',
api: () => import('./my-collection.context.js'),
element: () => import('./my-collection.element.js'),
meta: {
repositoryAlias: MY_COLLECTION_REPOSITORY_ALIAS,
},
},
...repositoryManifests,
...viewManifests,
...actionManifests,
];
import { html, customElement } from '@umbraco-cms/backoffice/external/lit';
import { UmbLitElement } from '@umbraco-cms/backoffice/lit-element';
@customElement('my-dashboard')
export class MyDashboardElement extends UmbLitElement {
override render() {
return html`<umb-collection alias="My.Collection"></umb-collection>`;
}
}
集合系统自动提供以下功能:
| 功能 | 描述 |
|---|---|
| 选择 | context.selection 上的 UmbSelectionManager |
| 分页 | context.pagination 上的 UmbPaginationManager |
| 加载状态 | 通过 context.loading 可观察 |
| 项目 | 通过 context.items 可观察 |
| 总数 | 通过 context.totalItems 可观察 |
| 筛选 | 通过 context.setFilter() 方法 |
| 视图切换 | 使用 UmbCollectionViewManager 的多个视图 |
使用 UMB_COLLECTION_ALIAS_CONDITION 来定位你的集合:
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: 'My.Collection',
},
],
就是这样!请始终获取最新文档,保持示例简洁,生成完整的工作代码。
每周安装次数
70
仓库
GitHub 星标
14
首次出现
2026年2月4日
安全审计
安装于
github-copilot51
cursor24
opencode22
codex22
gemini-cli20
amp20
A Collection displays a list of entities in the Umbraco backoffice with built-in support for multiple views (table, grid), filtering, pagination, selection, and bulk actions. Collections connect to a repository for data and provide a standardized way to browse and interact with lists of items.
Always fetch the latest docs before implementing:
A complete collection consists of these components:
collection/
├── manifests.ts # Main collection manifest
├── constants.ts # Alias constants
├── types.ts # Item and filter types
├── my-collection.context.ts # Collection context (extends UmbDefaultCollectionContext)
├── my-collection.element.ts # Collection element (extends UmbCollectionDefaultElement)
├── repository/
│ ├── manifests.ts
│ ├── my-collection.repository.ts # Implements UmbCollectionRepository
│ └── my-collection.data-source.ts # API calls
├── views/
│ ├── manifests.ts
│ └── table/
│ └── my-table-view.element.ts # Table view
└── action/
├── manifests.ts
└── my-action.element.ts # Collection action
The Umbraco source includes a working example:
Location : /Umbraco-CMS/src/Umbraco.Web.UI.Client/examples/collection/
This example demonstrates a complete custom collection with repository, views, and context. Study this for production patterns.
Repository Pattern : Collections require a repository for data access
umbraco-repository-patternContext API : For accessing collection context in views
umbraco-context-apiState Management : For understanding observables and reactive data
umbraco-state-managementUmbDefaultCollectionContext if custom behavior neededexport const MY_COLLECTION_ALIAS = 'My.Collection';
export const MY_COLLECTION_REPOSITORY_ALIAS = 'My.Collection.Repository';
export interface MyCollectionItemModel {
unique: string;
entityType: string;
name: string;
// Add other fields
}
export interface MyCollectionFilterModel {
skip?: number;
take?: number;
filter?: string;
orderBy?: string;
orderDirection?: 'asc' | 'desc';
// Add custom filters
}
import type { MyCollectionItemModel, MyCollectionFilterModel } from '../types.js';
import type { UmbCollectionDataSource } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
export class MyCollectionDataSource implements UmbCollectionDataSource<MyCollectionItemModel> {
#host: UmbControllerHost;
constructor(host: UmbControllerHost) {
this.#host = host;
}
async getCollection(filter: MyCollectionFilterModel) {
// Call your API here
const response = await fetch(`/api/my-items?skip=${filter.skip}&take=${filter.take}`);
const data = await response.json();
const items: MyCollectionItemModel[] = data.items.map((item: any) => ({
unique: item.id,
entityType: 'my-entity',
name: item.name,
}));
return { data: { items, total: data.total } };
}
}
import type { MyCollectionFilterModel } from '../types.js';
import { MyCollectionDataSource } from './my-collection.data-source.js';
import { UmbRepositoryBase } from '@umbraco-cms/backoffice/repository';
import type { UmbCollectionRepository } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
export class MyCollectionRepository extends UmbRepositoryBase implements UmbCollectionRepository {
#dataSource: MyCollectionDataSource;
constructor(host: UmbControllerHost) {
super(host);
this.#dataSource = new MyCollectionDataSource(host);
}
async requestCollection(filter: MyCollectionFilterModel) {
return this.#dataSource.getCollection(filter);
}
}
export default MyCollectionRepository;
import { MY_COLLECTION_REPOSITORY_ALIAS } from '../constants.js';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'repository',
alias: MY_COLLECTION_REPOSITORY_ALIAS,
name: 'My Collection Repository',
api: () => import('./my-collection.repository.js'),
},
];
import type { MyCollectionItemModel, MyCollectionFilterModel } from './types.js';
import { UmbDefaultCollectionContext } from '@umbraco-cms/backoffice/collection';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
// Default view alias - must match one of your collectionView aliases
const MY_TABLE_VIEW_ALIAS = 'My.CollectionView.Table';
export class MyCollectionContext extends UmbDefaultCollectionContext<
MyCollectionItemModel,
MyCollectionFilterModel
> {
constructor(host: UmbControllerHost) {
super(host, MY_TABLE_VIEW_ALIAS);
}
// Override or add custom methods if needed
}
export { MyCollectionContext as api };
import { customElement } from '@umbraco-cms/backoffice/external/lit';
import { UmbCollectionDefaultElement } from '@umbraco-cms/backoffice/collection';
@customElement('my-collection')
export class MyCollectionElement extends UmbCollectionDefaultElement {
// Override renderToolbar() to customize header
// protected override renderToolbar() {
// return html`<umb-collection-toolbar slot="header"></umb-collection-toolbar>`;
// }
}
export default MyCollectionElement;
export { MyCollectionElement as element };
declare global {
interface HTMLElementTagNameMap {
'my-collection': MyCollectionElement;
}
}
import type { MyCollectionItemModel } from '../../types.js';
import { UMB_COLLECTION_CONTEXT } from '@umbraco-cms/backoffice/collection';
import { css, customElement, html, state } from '@umbraco-cms/backoffice/external/lit';
import { UmbLitElement } from '@umbraco-cms/backoffice/lit-element';
import type { UmbTableColumn, UmbTableConfig, UmbTableItem } from '@umbraco-cms/backoffice/components';
@customElement('my-table-collection-view')
export class MyTableCollectionViewElement extends UmbLitElement {
@state()
private _tableItems: Array<UmbTableItem> = [];
@state()
private _selection: Array<string> = [];
#collectionContext?: typeof UMB_COLLECTION_CONTEXT.TYPE;
private _tableConfig: UmbTableConfig = {
allowSelection: true,
};
private _tableColumns: Array<UmbTableColumn> = [
{ name: 'Name', alias: 'name', allowSorting: true },
{ name: '', alias: 'entityActions', align: 'right' },
];
constructor() {
super();
this.consumeContext(UMB_COLLECTION_CONTEXT, (context) => {
this.#collectionContext = context;
// IMPORTANT: Call setupView for workspace modal routing
context?.setupView(this);
this.#observeItems();
this.#observeSelection();
});
}
#observeItems() {
if (!this.#collectionContext) return;
this.observe(
this.#collectionContext.items,
(items) => {
this._tableItems = (items as MyCollectionItemModel[]).map((item) => ({
id: item.unique,
icon: 'icon-document',
entityType: item.entityType,
data: [
{ columnAlias: 'name', value: item.name },
{
columnAlias: 'entityActions',
value: html`<umb-entity-actions-table-column-view
.value=${{ entityType: item.entityType, unique: item.unique }}
></umb-entity-actions-table-column-view>`,
},
],
}));
},
'_observeItems',
);
}
#observeSelection() {
if (!this.#collectionContext) return;
this.observe(
this.#collectionContext.selection.selection,
(selection) => {
this._selection = selection as string[];
},
'_observeSelection',
);
}
#handleSelect(event: CustomEvent) {
event.stopPropagation();
const table = event.target as any;
this.#collectionContext?.selection.setSelection(table.selection);
}
#handleDeselect(event: CustomEvent) {
event.stopPropagation();
const table = event.target as any;
this.#collectionContext?.selection.setSelection(table.selection);
}
override render() {
return html`
<umb-table
.config=${this._tableConfig}
.columns=${this._tableColumns}
.items=${this._tableItems}
.selection=${this._selection}
@selected=${this.#handleSelect}
@deselected=${this.#handleDeselect}
></umb-table>
`;
}
}
export default MyTableCollectionViewElement;
declare global {
interface HTMLElementTagNameMap {
'my-table-collection-view': MyTableCollectionViewElement;
}
}
import { MY_COLLECTION_ALIAS } from '../constants.js';
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collectionView',
alias: 'My.CollectionView.Table',
name: 'My Table Collection View',
element: () => import('./table/my-table-view.element.js'),
weight: 200,
meta: {
label: 'Table',
icon: 'icon-list',
pathName: 'table',
},
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: MY_COLLECTION_ALIAS,
},
],
},
];
import { MY_COLLECTION_ALIAS } from '../constants.js';
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collectionAction',
kind: 'button',
alias: 'My.CollectionAction.Refresh',
name: 'Refresh Collection Action',
element: () => import('./refresh-action.element.js'),
weight: 100,
meta: {
label: 'Refresh',
},
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: MY_COLLECTION_ALIAS,
},
],
},
];
import { manifests as repositoryManifests } from './repository/manifests.js';
import { manifests as viewManifests } from './views/manifests.js';
import { manifests as actionManifests } from './action/manifests.js';
import { MY_COLLECTION_ALIAS, MY_COLLECTION_REPOSITORY_ALIAS } from './constants.js';
export const manifests: Array<UmbExtensionManifest> = [
{
type: 'collection',
alias: MY_COLLECTION_ALIAS,
name: 'My Collection',
api: () => import('./my-collection.context.js'),
element: () => import('./my-collection.element.js'),
meta: {
repositoryAlias: MY_COLLECTION_REPOSITORY_ALIAS,
},
},
...repositoryManifests,
...viewManifests,
...actionManifests,
];
import { html, customElement } from '@umbraco-cms/backoffice/external/lit';
import { UmbLitElement } from '@umbraco-cms/backoffice/lit-element';
@customElement('my-dashboard')
export class MyDashboardElement extends UmbLitElement {
override render() {
return html`<umb-collection alias="My.Collection"></umb-collection>`;
}
}
The collection system provides these features automatically:
| Feature | Description |
|---|---|
| Selection | UmbSelectionManager on context.selection |
| Pagination | UmbPaginationManager on context.pagination |
| Loading state | Observable via context.loading |
| Items | Observable via context.items |
Use UMB_COLLECTION_ALIAS_CONDITION to target your collection:
import { UMB_COLLECTION_ALIAS_CONDITION } from '@umbraco-cms/backoffice/collection';
conditions: [
{
alias: UMB_COLLECTION_ALIAS_CONDITION,
match: 'My.Collection',
},
],
That's it! Always fetch fresh docs, keep examples minimal, generate complete working code.
Weekly Installs
70
Repository
GitHub Stars
14
First Seen
Feb 4, 2026
Security Audits
Gen Agent Trust HubPassSocketPassSnykWarn
Installed on
github-copilot51
cursor24
opencode22
codex22
gemini-cli20
amp20
React 组合模式指南:Vercel 组件架构最佳实践,提升代码可维护性
122,000 周安装
Observable via context.totalItems |
| Filtering | Via context.setFilter() method |
| View switching | Multiple views with UmbCollectionViewManager |