UNPKG

admin-bro

Version:
119 lines (104 loc) 3.72 kB
import { ResourceDecorator } from '..' import AdminBro from '../../../../admin-bro' import { BaseProperty, BaseResource } from '../../../adapters' import { PropertyDecorator } from '../../property' import { getPropertyByKey } from './get-property-by-key' import { pathToParts } from '../../../../utils/flat/path-to-parts' export type DecoratedProperties = {[key: string]: PropertyDecorator} const decorateDatabaseProperties = ( resource: BaseResource, admin: AdminBro, decorator: ResourceDecorator, ): DecoratedProperties => { const { options } = decorator return resource.properties().reduce((memo, property) => { const decoratedProperty = new PropertyDecorator({ property, admin, options: options.properties && options.properties[property.name()], resource: decorator, }) memo[property.name()] = decoratedProperty return memo }, {} as DecoratedProperties) } const decorateVirtualProperties = ( dbProperties: DecoratedProperties, admin: AdminBro, decorator: ResourceDecorator, ): DecoratedProperties => { const { options } = decorator if (options.properties) { return Object.keys(options.properties).reduce((memo, key) => { const existingProperty = getPropertyByKey(key, dbProperties) if (!existingProperty) { const property = new BaseProperty({ path: key, isSortable: false }) memo[key] = new PropertyDecorator({ property, admin, options: options.properties && options.properties[key], resource: decorator, isVirtual: true, }) return memo } return memo }, {} as DecoratedProperties) } return {} } /** * This function moves nested properties to existing mixed properties if there are any. * So that they could be printed as Section in the UI, and handled together as an Array if there * is a need for that. * * @param {DecoratedProperties} dbProperties * @param {DecoratedProperties} virtualProperties * @private * @hide */ const organizeNestedProperties = ( dbProperties: DecoratedProperties, virtualProperties: DecoratedProperties, ): DecoratedProperties => { const properties = { ...dbProperties, ...virtualProperties } const rootPropertyKeys = Object.keys(properties).filter((key) => { const property = properties[key] // reverse because we start by by finding from the longest path // and removes itself. (skips arrays) // changes 'root.nested.0.nested1' to [root.nested', 'root'] const parts = pathToParts(property.propertyPath, { skipArrayIndexes: true }).reverse().splice(1) if (parts.length) { const mixedPropertyPath = parts.find(part => ( properties[part] && properties[part].type() === 'mixed' )) if (mixedPropertyPath) { const mixedProperty = properties[mixedPropertyPath] mixedProperty.addSubProperty(property) // remove from the root properties return false } } return true }) return rootPropertyKeys.reduce((memo, key) => { memo[key] = properties[key] return memo }, {} as DecoratedProperties) } /** * Initializes PropertyDecorator for all properties within a resource. When * user passes new property in the options - it will be created as well. * * @returns {Object<string,PropertyDecorator>} * @private */ export function decorateProperties( resource: BaseResource, admin: AdminBro, decorator: ResourceDecorator, ): DecoratedProperties { const dbProperties = decorateDatabaseProperties(resource, admin, decorator) const virtualProperties = decorateVirtualProperties(dbProperties, admin, decorator) return organizeNestedProperties(dbProperties, virtualProperties) }