import type { LanguageName } from '@uiw/codemirror-extensions-langs'; import type { PropertiesSchema } from 'ajv/dist/types/json-schema'; import type { ComponentType, FunctionComponent, JSXElementConstructor, ReactElement, ReactNode, } from 'react'; import type { t, TranslateProps as ReactPolyglotTranslateProps } from 'react-polyglot'; import type { MediaFile as BackendMediaFile } from './backend'; import type { EditorControlProps } from './components/Editor/EditorControlPane/EditorControl'; import type { SORT_DIRECTION_ASCENDING, SORT_DIRECTION_DESCENDING, SORT_DIRECTION_NONE, } from './constants'; import type { formatExtensions } from './formats/formats'; import type { I18N_STRUCTURE } from './lib/i18n'; import type { AllowedEvent } from './lib/registry'; import type Cursor from './lib/util/Cursor'; import type AssetProxy from './valueObjects/AssetProxy'; export interface Pages { [collection: string]: { isFetching?: boolean; page?: number; ids: string[] }; } export type SortableField = | { key: string; name: string; label: string; } | ({ key: string; } & Field); export interface SortObject { key: string; direction: SortDirection; } export type SortMap = Record; export type Sort = Record; export type FilterMap = ViewFilter & { active?: boolean }; export type GroupMap = ViewGroup & { active?: boolean }; export type Filter = Record>; // collection.field.active export type Group = Record>; // collection.field.active export interface GroupOfEntries { id: string; label: string; value: string | boolean | undefined; paths: Set; } export type ObjectValue = { [key: string]: ValueOrNestedValue; }; export type ValueOrNestedValue = | string | number | (string | number)[] | boolean | ObjectValue | ValueOrNestedValue[] | null | undefined; export type EntryData = ObjectValue | undefined | null; export interface Entry { collection: string; slug: string; path: string; partial: boolean; raw: string; data: T | undefined | null; label: string | null; isModification: boolean | null; mediaFiles: MediaFile[]; author: string; updatedOn: string; status?: string; newRecord?: boolean; isFetching?: boolean; isPersisting?: boolean; isDeleting?: boolean; error?: string; i18n?: { [locale: string]: { data: EntryData; }; }; } export type Entities = Record; export interface FieldError { type: string; message?: string; } export interface FieldsErrors { [field: string]: FieldError[]; } export interface FieldValidationMethodProps { field: F; value: T | undefined | null; t: t; } export type FieldValidationMethod = ( props: FieldValidationMethodProps, ) => false | FieldError | Promise; export interface EntryDraft { entry: Entry; fieldsErrors: FieldsErrors; } export interface FilterRule { value: string; field: string; } export interface EditorConfig { preview?: boolean; frame?: boolean; } export interface CollectionFile { name: string; label: string; file: string; fields: Field[]; label_singular?: string; description?: string; media_folder?: string; public_folder?: string; i18n?: boolean | I18nInfo; editor?: EditorConfig; } interface Nested { summary?: string; depth: number; } export interface I18nSettings { currentLocale: string; defaultLocale: string; locales: string[]; } export type Format = keyof typeof formatExtensions; export interface i18nCollection extends Omit, 'i18n'> { i18n: Required>['i18n']; } export interface BaseCollection { name: string; description?: string; icon?: string; isFetching?: boolean; summary?: string; filter?: FilterRule; label_singular?: string; label: string; sortable_fields?: SortableFields; view_filters?: ViewFilter[]; view_groups?: ViewGroup[]; i18n?: boolean | I18nInfo; hide?: boolean; editor?: EditorConfig; identifier_field?: string; path?: string; extension?: string; format?: Format; frontmatter_delimiter?: string | [string, string]; slug?: string; media_folder?: string; public_folder?: string; } export interface FilesCollection extends BaseCollection { files: CollectionFile[]; } export interface FolderCollection extends BaseCollection { folder: string; fields: Field[]; create?: boolean; delete?: boolean; nested?: Nested; } export type Collection = | FilesCollection | FolderCollection; export type Collections = Record>; export interface MediaLibraryInstance { show: (args: { id?: string; value?: string | string[]; config: Record; allowMultiple?: boolean; imagesOnly?: boolean; }) => void; hide?: () => void; onClearControl?: (args: { id: string }) => void; onRemoveControl?: (args: { id: string }) => void; enableStandalone: () => boolean; } export type MediaFile = BackendMediaFile & { key?: string }; export interface DisplayURLState { isFetching: boolean; url?: string; err?: Error; } export type TranslatedProps = T & ReactPolyglotTranslateProps; /** * @deprecated Should use `useMediaAsset` React hook instead */ export type GetAssetFunction = ( path: string, field?: F, ) => Promise; export interface WidgetControlProps { collection: Collection; config: Config; entry: Entry; field: F; fieldsErrors: FieldsErrors; submitted: boolean; forList: boolean; /** * @deprecated Should use `useMediaAsset` React hook instead */ getAsset: GetAssetFunction; isDisabled: boolean; isFieldDuplicate: EditorControlProps['isFieldDuplicate']; isFieldHidden: EditorControlProps['isFieldHidden']; label: string; locale: string | undefined; mediaPaths: Record; onChange: (value: T | null | undefined) => void; clearMediaControl: EditorControlProps['clearMediaControl']; openMediaLibrary: EditorControlProps['openMediaLibrary']; removeInsertedMedia: EditorControlProps['removeInsertedMedia']; removeMediaControl: EditorControlProps['removeMediaControl']; i18n: I18nSettings | undefined; hasErrors: boolean; path: string; query: EditorControlProps['query']; t: t; value: T | undefined | null; } export interface WidgetPreviewProps { config: Config; collection: Collection; entry: Entry; field: RenderedField; /** * @deprecated Should use `useMediaAsset` React hook instead */ getAsset: GetAssetFunction; value: T | undefined | null; } export type WidgetPreviewComponent = // eslint-disable-next-line @typescript-eslint/no-explicit-any | ReactElement> | ComponentType>; export type WidgetsFor

= ( name: K, ) => P[K] extends Array ? { data: U | null; widgets: Record; }[] : { data: P[K] | null; widgets: Record; }; export interface TemplatePreviewProps { collection: Collection; fields: Field[]; entry: Entry; document: Document | undefined | null; window: Window | undefined | null; /** * @deprecated Should use `useMediaAsset` React hook instead */ getAsset: GetAssetFunction>; widgetFor: (name: T extends EntryData ? string : keyof T) => ReactNode; widgetsFor: WidgetsFor; } export type TemplatePreviewComponent< T = EntryData, EF extends BaseField = UnknownField, > = ComponentType>; export interface WidgetOptions { validator?: Widget['validator']; getValidValue?: Widget['getValidValue']; getDefaultValue?: Widget['getDefaultValue']; schema?: Widget['schema']; } export interface Widget { control: ComponentType>; preview?: WidgetPreviewComponent; validator: FieldValidationMethod; getValidValue: (value: T | undefined | null) => T | undefined | null; getDefaultValue?: (defaultValue: T | undefined | null, field: F) => T; schema?: PropertiesSchema; } export interface WidgetParam { name: string; controlComponent: Widget['control']; previewComponent?: Widget['preview']; options?: WidgetOptions; } export interface PersistOptions { newEntry?: boolean; commitMessage: string; collectionName?: string; status?: string; } export interface ImplementationEntry { data: string; file: { path: string; label?: string; id?: string | null; author?: string; updatedOn?: string }; } export interface DisplayURLObject { id: string; path: string; } export type DisplayURL = DisplayURLObject | string; export interface ImplementationMediaFile { name: string; id: string; size?: number; displayURL?: DisplayURL; path: string; draft?: boolean; url?: string; file?: File; field?: Field; } export interface DataFile { path: string; slug: string; raw: string; newPath?: string; } export interface BackendEntry { dataFiles: DataFile[]; assets: AssetProxy[]; } export interface Credentials { token: string | {}; refresh_token?: string; } export type User = Credentials & { backendName?: string; login?: string; name?: string; avatar_url?: string; }; export interface ImplementationFile { id?: string | null | undefined; label?: string; path: string; } export interface AuthenticatorConfig { site_id?: string; base_url?: string; auth_endpoint?: string; auth_token_endpoint?: string; auth_url?: string; app_id?: string; clearHash?: () => void; } export abstract class BackendClass { // eslint-disable-next-line @typescript-eslint/no-empty-function constructor(_config: Config, _options: BackendInitializerOptions) {} abstract authComponent(): (props: TranslatedProps) => JSX.Element; abstract restoreUser(user: User): Promise; abstract authenticate(credentials: Credentials): Promise; abstract logout(): Promise | void | null; abstract getToken(): Promise; abstract getEntry(path: string): Promise; abstract entriesByFolder( folder: string, extension: string, depth: number, ): Promise; abstract entriesByFiles(files: ImplementationFile[]): Promise; abstract getMediaDisplayURL(displayURL: DisplayURL): Promise; abstract getMedia(folder?: string, mediaPath?: string): Promise; abstract getMediaFile(path: string): Promise; abstract persistEntry(entry: BackendEntry, opts: PersistOptions): Promise; abstract persistMedia(file: AssetProxy, opts: PersistOptions): Promise; abstract deleteFiles(paths: string[], commitMessage: string): Promise; abstract allEntriesByFolder( folder: string, extension: string, depth: number, ): Promise; abstract traverseCursor( cursor: Cursor, action: string, ): Promise<{ entries: ImplementationEntry[]; cursor: Cursor }>; abstract isGitBackend(): boolean; abstract status(): Promise<{ auth: { status: boolean }; api: { status: boolean; statusPage: string }; }>; } export interface LocalePhrasesRoot { [property: string]: LocalePhrases; } export type LocalePhrases = string | { [property: string]: LocalePhrases }; export type CustomIcon = FunctionComponent; export type WidgetValueSerializer = { serialize: (value: ValueOrNestedValue) => ValueOrNestedValue; deserialize: (value: ValueOrNestedValue) => ValueOrNestedValue; }; export type MediaLibraryOptions = Record; export interface MediaLibraryInitOptions { options: Record | undefined; handleInsert: (url: string | string[]) => void; } export interface MediaLibraryExternalLibrary { name: string; config?: MediaLibraryOptions; init: ({ options, handleInsert }: MediaLibraryInitOptions) => Promise; } export interface MediaLibraryInternalOptions { allow_multiple?: boolean; choose_url?: boolean; } export type MediaLibrary = MediaLibraryExternalLibrary | MediaLibraryInternalOptions; export type BackendType = 'git-gateway' | 'github' | 'gitlab' | 'bitbucket' | 'test-repo' | 'proxy'; export type MapWidgetType = 'Point' | 'LineString' | 'Polygon'; export interface SelectWidgetOptionObject { label: string; value: string; } export type AuthScope = 'repo' | 'public_repo'; export type SlugEncoding = 'unicode' | 'ascii'; export type RenderedField = F & { renderedFields?: ReactNode[]; }; export interface BaseField { name: string; label?: string; required?: boolean; hint?: string; pattern?: [string, string]; i18n?: boolean | 'translate' | 'duplicate' | 'none'; comment?: string; widget: string; } export interface BooleanField extends BaseField { widget: 'boolean'; default?: boolean; } export interface CodeField extends BaseField { widget: 'code'; default?: string | { [key: string]: string }; default_language?: string; allow_language_selection?: boolean; keys?: { code: string; lang: string }; output_code_only?: boolean; code_mirror_config: { extra_keys?: Record; } & Record; } export interface ColorField extends BaseField { widget: 'color'; default?: string; allow_input?: boolean; enable_alpha?: boolean; } export interface DateTimeField extends BaseField { widget: 'datetime'; default?: string; format?: string; date_format?: boolean | string; time_format?: boolean | string; picker_utc?: boolean; } export interface FileOrImageField extends BaseField { widget: 'file' | 'image'; default?: string; media_library?: MediaLibrary; media_folder?: string; public_folder?: string; } export interface ObjectField extends BaseField { widget: 'object'; collapsed?: boolean; summary?: string; fields: Field[]; } export interface ListField extends BaseField { widget: 'list'; default?: ValueOrNestedValue[]; allow_add?: boolean; collapsed?: boolean; summary?: string; label_singular?: string; fields?: Field[]; max?: number; min?: number; add_to_top?: boolean; types?: ObjectField[]; type_key?: string; } export interface MapField extends BaseField { widget: 'map'; default?: string; decimals?: number; type?: MapWidgetType; height?: string; } export interface MarkdownField extends BaseField { widget: 'markdown'; default?: string; media_library?: MediaLibrary; media_folder?: string; public_folder?: string; } export interface NumberField extends BaseField { widget: 'number'; default?: string | number; value_type?: 'int' | 'float' | string; min?: number; max?: number; step?: number; } export interface SelectField extends BaseField { widget: 'select'; default?: string | number | (string | number)[]; options: (string | number)[] | SelectWidgetOptionObject[]; multiple?: boolean; min?: number; max?: number; } export interface RelationField extends BaseField { widget: 'relation'; default?: string | string[]; collection: string; value_field: string; search_fields: string[]; file?: string; display_fields?: string[]; multiple?: boolean; min?: number; max?: number; options_length?: number; } export interface HiddenField extends BaseField { widget: 'hidden'; default?: ValueOrNestedValue; } export interface StringOrTextField extends BaseField { // This is the default widget, so declaring its type is optional. widget: 'string' | 'text'; default?: string; } export interface UnknownField extends BaseField { widget: 'unknown'; } export type Field = | BooleanField | CodeField | ColorField | DateTimeField | FileOrImageField | ListField | MapField | MarkdownField | NumberField | ObjectField | RelationField | SelectField | HiddenField | StringOrTextField | EF; export interface ViewFilter { id: string; label: string; field: string; pattern: string; } export interface ViewGroup { id: string; label: string; field: string; pattern?: string; } export type SortDirection = | typeof SORT_DIRECTION_ASCENDING | typeof SORT_DIRECTION_DESCENDING | typeof SORT_DIRECTION_NONE; export interface SortableFieldsDefault { field: string; direction?: SortDirection; } export interface SortableFields { default?: SortableFieldsDefault; fields: string[]; } export interface Backend { name: BackendType; repo?: string; branch?: string; api_root?: string; site_domain?: string; base_url?: string; auth_endpoint?: string; app_id?: string; auth_type?: 'implicit' | 'pkce'; proxy_url?: string; large_media_url?: string; login?: boolean; use_large_media_transforms_in_media_library?: boolean; identity_url?: string; gateway_url?: string; auth_scope?: AuthScope; commit_messages?: { create?: string; update?: string; delete?: string; uploadMedia?: string; deleteMedia?: string; }; } export interface Slug { encoding?: SlugEncoding; clean_accents?: boolean; sanitize_replacement?: string; } export interface LocalBackend { url?: string; allowed_hosts?: string[]; } export interface Config { backend: Backend; collections: Collection[]; locale?: string; site_id?: string; site_url?: string; display_url?: string; base_url?: string; logo_url?: string; media_folder?: string; public_folder?: string; media_folder_relative?: boolean; media_library?: MediaLibrary; load_config_file?: boolean; slug?: Slug; i18n?: I18nInfo; local_backend?: boolean | LocalBackend; editor?: EditorConfig; search?: boolean; } export interface InitOptions { config: Config; } export interface BackendInitializerOptions { updateUserCredentials: (credentials: Credentials) => void; } export interface BackendInitializer { init: (config: Config, options: BackendInitializerOptions) => BackendClass; } export interface EventData { entry: Entry; author: { login: string | undefined; name: string }; } export type EventListenerOptions = Record; export type EventListenerHandler = ( data: EventData, options: EventListenerOptions, ) => Promise; export interface EventListener { name: AllowedEvent; handler: EventListenerHandler; } export interface AdditionalLinkOptions { icon?: string; } export interface AdditionalLink { id: string; title: string; data: string | FunctionComponent; options?: AdditionalLinkOptions; } export interface AuthenticationPageProps { onLogin: (user: User) => void; inProgress?: boolean; base_url?: string; siteId?: string; authEndpoint?: string; config: Config; error?: string | undefined; clearHash?: () => void; } export interface SearchResponse { entries: Entry[]; pagination: number; } export interface SearchQueryResponse { hits: Entry[]; query: string; } export interface EditorPersistOptions { createNew?: boolean; duplicate?: boolean; } export interface I18nInfo { locales: string[]; defaultLocale: string; structure?: I18N_STRUCTURE; } export interface ProcessedCodeLanguage { label: string; identifiers: string[]; codemirror_mode: LanguageName; codemirror_mime_type: string; } export interface FileMetadata { author: string; updatedOn: string; } export interface PreviewStyleOptions { raw?: boolean; } export interface PreviewStyle { value: string; raw: boolean; } export interface MarkdownPluginFactoryProps { config: Config; field: MarkdownField; mode: 'editor' | 'preview'; } // eslint-disable-next-line @typescript-eslint/no-explicit-any export type MarkdownPluginFactory = (props: MarkdownPluginFactoryProps) => any; export interface MarkdownToolbarItemsFactoryProps { // eslint-disable-next-line @typescript-eslint/no-explicit-any imageToolbarButton: any; } export type MarkdownToolbarItemsFactory = ( props: MarkdownToolbarItemsFactoryProps, // eslint-disable-next-line @typescript-eslint/no-explicit-any ) => (string | any)[][]; export interface MarkdownEditorOptions { // eslint-disable-next-line @typescript-eslint/no-explicit-any initialEditType?: any; height?: string; toolbarItems?: MarkdownToolbarItemsFactory; plugins?: MarkdownPluginFactory[]; } export type ShortcodeControlProps

= P & { onChange: (props: P) => void; controlProps: WidgetControlProps; }; export type ShortcodePreviewProps

= P & { previewProps: WidgetPreviewProps; }; export interface ShortcodeConfig

{ label?: string; openTag: string; closeTag: string; separator: string; toProps?: (args: string[]) => P; toArgs?: (props: P) => string[]; control: ComponentType; preview: ComponentType; } export enum CollectionType { FOLDER, FILES, } export type DeepPartial = T extends object ? { [P in keyof T]?: DeepPartial; } : T;