admin-bro
Version:
Admin panel for apps written in node.js
119 lines (104 loc) • 3.72 kB
text/typescript
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)
}