Smart Fields
This is the official documentation of the forestadmin/laravel-forestadmin
v2+ and forestadmin/symfony-forestadmin
PHP agents.
In legacy agents declaring a smart field was done in one big step.
In the new agent, the process was split into multiple steps, depending on the capabilities of the field (writing, filtering, sorting, etc.).
This was done to reduce the complexity of the code and to make it easier to understand, but also because it allows customers to reuse the same API when customizing the behavior of normal fields, and thus reduce the API surface that you need to learn.
You can find the full documentation of field customization here.
Code cheatsheet
Legacy agent | New agent |
---|---|
get: (record) => { ... } | getValues: (records) => { ... } |
set: (record, value) => { ... } | .replaceFieldWriting(...) |
filter: ({ condition, where }) => { ... } | .replaceFieldOperator(...) .emulateFieldOperator(...) .emulateFieldFiltering(...) |
type: 'String' | columnType: 'String' |
enums: ['foo', 'bar'] | columnType: 'Enum', enumValues: ['foo', 'bar'] |
reference: 'otherCollection.id' |
Do you still need a computed field?
Smart fields were a powerful tool, but they were also a performance bottleneck.
In the new agent, we have introduced two new concepts that can replace many of the use cases of smart fields:
If you were using a smart field to move a field from one collection to another or to create a link to another record in the UI, you can likely use one of these much simpler solutions.
Steps
Step 1: Implement a read-only version of the field
Computed fields in the new agent are declared by calling the addField
function on the appropriate collection.
Many changes have been made to the API.
Dependencies are explicit
You will notice that a new dependencies
property is required when declaring a computed field.
It is an array of field names that tells Forest Admin which fields the getValues()
function depends on. Unlike the legacy agent, the new agent will not automatically fetch the whole record.
You can fetch data from relations and fetch data from other computed fields.
Fields now work in batches
Even if it adds some complexity, exposing a batch API to our customers is a much better solution for performance.
The get
function is now called getValues
: it no longer takes a single record as its first argument, but an array of records, and must return an array of values, one for each record, in the same order.
Other changes
There are other minor changes to the API:
the
type
property was renamed tocolumnType
.The
field
property no longer exists. The field name is now the first argument of theaddField
function.The
reference
property no longer exists: use the smart relationships guide.The
enums
property was renamed toenumValues
.
Example
In the following example, we will port a field that fetches the full address of a user from a third-party service.
Note that when displaying a list of records, the new agent will only make one call to your handler, and then display the results for all records, instead of making one call per record.
Step 2: Implement write handler
If you want your computed field to be writable, you will need to call the .replaceFieldWriting()
function.
This part is very similar to the legacy agent. The API change is because this function can be used to make any field writable, not just computed fields, or to change the default writing behavior of a normal field.
Step 3: Implement the filters you use
Structure
Implementing filters in the new agent is done operator by operator, instead of using a single function.
This allows more fine-grained control over the behavior of each operator and makes it possible to implement only the operators you need.
Also note that unlike in the legacy agent, automatic operator replacement will be performed by the agent, so the number of operators that you need to implement to unlock all filtering is much lower.
Return value
Because the new Forest Admin agent is designed to work with multiple databases, the return value of the filter function is not a Sequelize or Mongoose condition anymore.
Instead, you'll be building a condition tree that will be translated to the appropriate database syntax by the agent.
Emulation
At the cost of performance, you can tell the agent to emulate the behavior of a given operator by calling the .emulateFieldOperator()
function.
Example
Last updated