Skip to content
Advertisement

How to toggle sidebar without page reload when router-link is clicked in Vue 3

I have a sidebar component that works similar to a modal. When a button is clicked, the sidebar translates into the viewport with nav links. These nav links are actually router-links that are wired up to a vue-router.

What I’m trying to accomplish

When I click on a router-link that is inside my sidebar component, I want the sidebar to transition off the viewport and I want the clicked router-link’s component to render without the page reloading.

What’s currently happening

When I click on the router-link, the sidebar is removed instantly from the DOM. It does not translate off the screen as intended. Also, the page is reloaded.

What else have I tried

I also tried moving the <transition> wrapper inside TheSidebar.vue component along with the associated CSS classes, and I passed sidebarIsVisible as a prop from App.vue to TheSidebar.vue.

My code

A Codesandbox demo can be found here

App.vue

JavaScript

TheSidebar.vue

JavaScript

main.js

JavaScript

Advertisement

Answer

There are a couple of things I’m unsure about here but I’ll try and explain what I think is happening.

Firstly, the click event on the router-link is what’s causing the page to reload, but I can’t find anything in the docs mentioning this as expected behaviour (it might be worth opening an issue on the GitHub repo).

The fix for this is to use event-delegation by moving the event-handler onto the ul and creating a method to determine if a link has been clicked (example below).

Secondly, and this is where things get weird, in VSCode, using kebab-case in the child components’ emitted event seems to prevent anything from being emitted, so changing them to camelCase fixes this. But, trying this in CodeSandbox simply doesn’t work, and ESLint complains that the emitted event should be kebab-case. So, in CodeSandbox, the opposite is true: the emitted event names should be kebab-case and the listener should be camelCase! Absolutely no idea why as this goes against what the docs say on casing:

…we recommend using kebab-cased event listeners when you are using in-DOM templates.

Again, I can’t find anything in the docs explicitly saying you need to use camelCase when emitting an event, it just says kebab-case is preferred when listening for an event.


So, all in all, for your code to work in VSCode and in a way which follows what is recommended by the docs, you need to change it to this:

JavaScript

Keep App.vue exactly as you have it already and it should work.


For your code to work in CodeSandbox, swap the casing:

JavaScript

App.vue:

JavaScript

Working example.


If anyone could shed some light on this, it’d be great as I’m completely stumped on what’s happening here.

User contributions licensed under: CC BY-SA
3 People found this is helpful
Advertisement