Widget Rendering
Learn about approaches to rendering widgets in Flexiboards.
Introduction
In Flexiboards, the FlexiWidget component allows you to create a widget with drag-and-drop functionality within the scope of the FlexiBoard. Within it, though, you’ll want to render some content.
As Flexiboards is headless, it does not make any assumptions about the appearance of the widget (except grid positioning). The FlexiWidget acts as an invisible wrapper around your content, providing the drag-and-drop functionality and positioning itself within the board.
In this article, we discuss how you can either use Svelte snippets or pass a component prop to render content inside a FlexiWidget.
Snippet-based
Using snippets (specifically, children) is the most intuitive approach to rendering a widget. Simply put, you can put any elements or components as the content markup of your widget, like any other container component.
Flexiboards pass parameters into the children snippet which you can access if you desire, but you do not have to. This looks as follows:
<!-- Without using the parameters (i.e. implicit children snippet) -->
<FlexiWidget>I'm a FlexiWidget!</FlexiWidget>
<!-- With the parameters (e.g. get widget reactive data) - we discuss component and componentProps later -->
<FlexiWidget>
	{#snippet children({ widget })}
		I'm a FlexiWidget at ({widget.x}, {widget.y})!
	{/snippet}
</FlexiWidget>With the first example, we do not need any data from the widget controller, so there’s no point in being explicit. Whereas, in the second example we’re getting the reactive x and y properties on the controller and showing these (for illustration - see FlexiWidget API for other properties like these).
As you can see, this approach is simple, and can work well. However, certain use cases lend themselves better to the approach we discuss next, which is using the component prop.
Component-based
Alternatively, you can use the component prop to specify any Svelte component of your choosing to render inside of the FlexiWidget.
Fundamentally, this is not dissimilar to the snippets approach; the main difference is that instead of needing some Svelte snippet in scope (or passed via a prop, for example), you just import the Svelte component. This might look as follows:
<script>
	import MyComponent from './my-component.svelte';
</script>
<FlexiWidget component={MyComponent} />Note that in this scenario, you cannot get the widget controller as simply as you can with the snippet approach (and it is not passed as a prop on the component). However, you can use the getFlexiwidgetCtx helper function to get the context of the widget:
<!-- my-component.svelte -->
<script>
	import { getFlexiwidgetCtx } from 'svelte-flexiboards';
	const widget = getFlexiwidgetCtx();
</script>Note that this uses the Svelte Context API under the hood, so it must be called from the top-level of the component (or some function that is called from the top-level).
Additionally, any Svelte component rendered inside of the FlexiWidget, whether via a snippet or a descendant component, will have access to the widget controller through the same mechanism.