7/7/2022

Slot Props Vue

Have you ever pulled in a third-party UI component only to discover that because of one small tweak you need to make, you have to throw out the whole package?

  1. Vue Slot Props Method
  2. Vue Props Slot
  3. Slot Props Vue Online
  4. Slot Props Vue Free

Swiper Vue.js component receive all Swiper parameters as component props, plus some extra props. SwiperSlide component has the following slot props. Slot Content Vue implements a content distribution API that’s modeled after the current Web Components spec draft, using the element to serve as distribution outlets for content. This allows you to compose components like this. It's working great. I can pass anything I want into the slot like this: slot:item1='item1' slot The problem is that when I pass a function as a prop, it is undefined in the parent template. So this doesn't work: slot:my-method='myMethod' slot In this example, myMethod is a method defined on the child vue component. Regarding slots, they've been discussed here - pass the object syntax as children, it's even possible in react and is explained here, it's a similar pattern to render props (slots is just an object of named render functions). JSX should work straight away as an h/createVNode-wrapper/factory and TSX could even work without babel. Slot props allow us to turn slots into reusable templates that can render different content based on input props. This is most useful when you are designing a reusable component that encapsulates data logic while allowing the consuming parent component to customize part of its layout.

Custom controls like dropdowns, date pickers, or autocomplete fields can be very complex to build with a lot of unexpected edge cases to deal with.

There are a lot of libraries out there that do a great job handling this complexity, but they often come with a deal-breaking downside: it's hard or impossible to customize how they look.

Take this tags input control for example:

See the Pen Reusable Vue Components with Scoped Slots: Traditional Tags Input by Adam Wathan (@adamwathan) on CodePen.

This component wraps up a few interesting behaviors:

  • It doesn't let you add duplicates
  • It doesn't let you add empty tags
  • It trims whitespace from tags
  • Tags are added when the user presses enter
  • Tags are removed when the user clicks the × icon

If you needed a component like this in your project, pulling this in as a package and offloading that logic would certainly save you some time and effort.

But what if you needed it to look a little different?

This component has all of the same behavior as the previous component, but with a significantly different layout:

Slot

See the Pen Reusable Vue Components with Scoped Slots: Traditional Tags Input w/Alternate Layout by Adam Wathan (@adamwathan) on CodePen.

You could try and support both of these layouts with a single component through a combination of whacky CSS and component configuration options, but (thankfully) there's a better way.

Scoped Slots

In Vue.js, slots are placeholder elements in a component that are replaced by content passed in by the parent/consumer:

Scoped slots are just like regular slots but with the ability to pass parameters from the child component up to the parent/consumer.

Regular slots are like passing HTML to a component; scoped slots are like passing a callback that accepts data and returns HTML.

Parameters are passed up to the parent by adding props to the slot element in the child component, and the parent accesses these parameters by destructuring them out of the special slot-scope attribute.

Slot

Here's an example of a LinksList component that exposes a scoped slot for each list item, and passes the data for each item back to the parent through a :link prop:

By adding the :link prop to the slot element in the LinksList component, the parent can now access it through the slot-scope and make use of it in its slot template.

Types of Slot Props

You can pass anything to a slot, but I find it useful to think of every slot prop as belonging to one of three categories.

Data

The simplest type of slot prop is just data: strings, numbers, boolean values, arrays, objects, etc.

In our links example, link is an example of a data prop; it's just an object with some properties:

The parent can then render that data or use it to make decisions about what to render:

Actions

Action props are functions provided by the child component that the parent can call to invoke some behavior in the child component.

For example, we could pass a bookmark action to the parent that bookmarks a given link:

The parent could invoke this action when the user clicks a button next to an un-bookmarked link:

Bindings

Bindings are collections of attributes or event handlers that should be bound to a specific element using v-bind or v-on.

These are useful when you want to encapsulate implementation details about how interacting with a provided element should work.

For example, instead of making the consumer handle the v-show and @click behaviors for the bookmark button themselves, we could provide a bookmarkButtonAttrs binding and a bookmarkButtonEvents binding that move those details into the component itself:

Now if the consumer prefers, they can apply these bindings to the bookmark button blindly without having to know what they actually do:

Renderless Components

A renderless component is a component that doesn't render any of its own HTML.

Instead it only manages state and behavior, exposing a single scoped slot that gives the parent/consumer complete control over what should actually be rendered.

A renderless component renders exactly what you pass into it, without any extra elements:

So why is this useful?

Separating Presentation and Behavior

Since renderless components only deal with state and behavior, they don't impose any decisions about design or layout.

That means that if you can figure out a way to move all of the interesting behavior out of a UI component like our tags input control and into a renderless component, you can reuse the renderless component to implement any tags input control layout.

Vue Slot Props Method

Here's both tag input controls, but this time backed by a single renderless component:

See the Pen Reusable Vue Components with Scoped Slots: Headless Tags Input w/Multiple Layouts by Adam Wathan (@adamwathan) on CodePen.

So how does this work?

Renderless Component Structure

A renderless component exposes a single scoped slot where the consumer can provide the entire template they want to render.

The basic skeleton of a renderless component looks like this:

It doesn't have a template or render any HTML of its own; instead it uses a render function that invokes the default scoped slot passing through any slot props, then returns the result.

Any parent/consumer of this component can destructure exampleProp out of the slot-scope and use it in its template:

A Worked Example

Let's walkthrough building a renderless version of the tags input control from scratch.

We'll start with a blank renderless component that passes no slot props:

...and a parent component with a static, non-interactive UI that we pass in to the child component's slot:

Piece by piece, we'll make this component work by adding state and behavior to the renderless component and exposing it to our layout through the slot-scope.

Listing tags

First let's replace the static list of tags with a dynamic list.

The tags input component is a custom form control so like in the original example, the tags should live in the parent and be bound to the component using v-model.

We'll start by adding a value prop to the component and passing it up as a slot prop named tags:

Next, we'll add the v-model binding in the parent, fetch the tags out of the slot-scope, and iterate over them with v-for:

This slot prop is a great example of a simple data prop.

Removing tags

Next let's remove a tag when clicking the × button.

We'll add a new removeTag method to our component, and pass a reference to that method up to the parent as a slot prop:

Then we'll add a @click handler to the button in the parent that calls removeTag with the current tag:

This slot prop is an example of an action prop.

Adding new tags on enter

Adding new tags is a bit trickier than the last two examples.

To understand why, let's look at how it would be implement in a more traditional component:

We keep track of the new tag (before it's been added) in a newTag property, and we bind that property to the input using v-model.

Once the user presses enter, we make sure the tag is valid, add it to the list, then clear out the input field.

The question here is how do we pass a v-model binding through a scoped slot?

Well if you've dug into Vue deep enough, you might know that v-model is really just syntax sugar for a :value attribute binding, and an @input event binding:

That means we can handle this behavior in our renderless component by making a few changes:

  • Add a local newTag data property to the component
  • Pass back an attribute binding prop that binds :value to newTag
  • Pass back an event binding prop that binds @keydown.enter to addTag and @input to update newTag

Now we just need to bind those props to the input element in the parent:

Adding new tags explicitly

In our current layout, the user adds a new tag by typing it in the field and hitting the enter key. But it's easy to imagine a scenario where someone might want to provide a button that the user can click to add the new tag as well.

Making this possible is easy, all we need to do is pass a reference to our addTag method to the slot scope as well:

When designing renderless components like this, it's better to err on the side of 'too many slot props' than too few.

The consumer only needs to destructure out the props they actually need, so there's no cost to them if you give them a prop they aren't going to use.

Working Demo

Here's a working demo of the renderless tags input component that we've built so far:

See the Pen Reusable Vue Components with Scoped Slots: Renderless Tags Input by Adam Wathan (@adamwathan) on CodePen.

The actual component contains no HTML, and the parent where we define the template contains no behavior. Pretty neat right?

An Alternate Layout

Now that we have a renderless version of the tags input control, we can easily implement alternative layouts by writing whatever HTML we want and applying the provided slot props to the right places.

Here's what it would look like to implement the stacked layout from the beginning of the article using our new renderless component:

See the Pen Reusable Vue Components with Scoped Slots: Renderless Tags Input w/Alternate Layout by Adam Wathan (@adamwathan) on CodePen.

Creating Opinionated Wrapper Components

You might look some of these examples and think, 'wow, that sure is a lot of HTML to write every time I need to add another instance of this tags component!' and you'd be right.

It's definitely a lot more work to write this whenever you need a tags input:

...than this, which is what we started with in the beginning:

There's an easy fix though: create an opinionated wrapper component!

Vue Props Slot

Here's what it looks like to write our original <tags-input> component in terms of the renderless tags input:

Now you can use that component in one line of code anywhere you need that particular layout:

Getting Crazy

Once you realize that a component doesn't have to render anything and can instead be responsible solely for providing data, there's no limit to the type of behavior you can model with a component.

For example, here's a fetch-data component that takes a URL as a prop, fetches JSON from that URL, and passes the response back to the parent:

See the Pen Vue.js Fetch Component by Adam Wathan (@adamwathan) on CodePen.

Is this the right way to make every AJAX request? Probably not, but it's certainly interesting!

Slot

Conclusion

Splitting a component into a presentational component and a renderless component is an extremely useful pattern to master and can make code reuse a lot easier, but it's not always worth it.

Use this approach if:

  • You're building a library and you want to make it easy for users to customize how your component looks
  • You have multiple components in your project with very similar behavior but different layouts

Don't go down this path if you're working on a component that's always going to look the same everywhere it's used; it's considerably simpler to keep everything in a single component if that's all you need.

Learning More

Slot Props Vue Online

If you enjoyed this post, you might be interested in Advanced Vue Component Design, a video series I put together that goes deep into tons of useful component design patterns.

Slot Props Vue Free

Check out the website to learn more, or sign up below to watch two free preview lessons: