Component Best Practices
Good components need to be designed well. Taking the time to think of future use cases significantly enhances the components usability.
Sizing and Styling
Generally, it should be the implementation of the component that is responsible for the component's size. There are some exceptions. This means that the width of your component is generally full width and the parent or 'implementor' of the component will control the width.
Examples:
Page Header Component
Parent
this allows the header
Button Component
Component
Buttons generally don't take on the shapre of the parent elements. When you need to override a buttons width, you can pass in a class.
Toggle switch
Component
Generally this type of component woudl not adapt its size to the parent
Toggle switch Array
Parent
An array of toggles would wrap or flow based on the containing element
Actions
Lifecycle & Async
Use
onBeforeMountfor blocking setup (library load, data fetch); it awaits before render. UseonMountfor DOM work after render.Always
awaitasync work inside function actions to preserve order;bfcomponentlifecycle dispatch already awaitsACTIONS_QUEUE.Load external libraries with
BF.libraryLoadOnce(); no extra singleton guard needed for identical URLs.
Schema in Actions
Lifecycle actions receive the component schema on
action.schema(andaction.options._componentSchema).Template-triggered actions: pass schema explicitly when needed, e.g.
@click="namedAction('doThing', { schema, foo: 'bar' })".In functions, read with:
const schema = action.schema || action.options._componentSchema || action.options.schema || {};.
DOM & Refs
this.$refsis not available in namedAction functions. Use DOM queries:document.getElementById(...)orquerySelector(...).Give stable IDs in your HTML if you need to target elements from actions.
EventBus & async namedActions
runUtilityHook(and similar) support callbacks; wrap in a Promise:This keeps async actions serialized with lifecycle dispatch.
Context
Avoid
thisinside namedAction functions; rely on injected vars:model,app,document,window,BF,_, andschema(lifecycle only).Keep global/window writes idempotent; guard with
window.__bfSingletonsonly for truly global resources (e.g., registering Vue components).For third-party instances (charts, uploaders), prefer per-instance scoping instead of global singletons.
Multi-Instance Components
If the same component renders multiple times, add a unique
instanceId(or similar) in schema and namespace any globals/hosts/plugins with it.Example pattern:
const key = 'uppy_' + instanceId; window[key] = new Uppy.Uppy(...); hostId = 'bf-fileinput-host-' + instanceId;.
File Upload (Uppy + S3/FM)
Fetch presigned URLs via
runUtilityHook; await via the EventBus callback pattern above.Prefer dual URLs from backend:
url(PUT, short expiry) anddownloadUrl(GET, longer expiry) so users can view after upload.Namespace Uppy instances per
instanceId; create per-instance hidden hosts to avoid conflicts when multiple upload buttons exist.
Last updated
Was this helpful?