Skip to content
Sponsored by

MagicDraggable

MagicDraggable is a touch enabled, draggable component that can snap to configurable points inside its container.

Drag me
<template>
  <magic-draggable
    id="magic-draggable-demo"
    class="bg-surface-elevation-base w-full h-[20rem] rounded-md"
  >
    <div
      class="bg-surface-elevation-high aspect-square w-32 rounded-md flex items-center justify-center"
    >
      <m-badge>Drag me</m-badge>
    </div>
  </magic-draggable>
</template>

<style>
[data-id='magic-draggable-demo'] {
  --magic-draggable-position: relative;
  --magic-draggable-z-index: 50;
}
</style>

Overview

Anatomy

vue
<template>
  <magic-draggable id="your-draggable-id">
    <!-- your content -->
  </magic-draggable>
</template>

<script setup>
const { open } = useMagicDraggable('your-draggable-id')
</script>

Installation

CLI

Add @maas/vue-equipment to your dependencies.

sh
pnpm install @maas/vue-equipment
sh
npm install @maas/vue-equipment
sh
yarn add @maas/vue-equipment
sh
bun install @maas/vue-equipment

Vue

If you are using Vue, import and add MagicDraggablePlugin to your app.

js
import { createApp } from 'vue'
import { MagicDraggablePlugin } from '@maas/vue-equipment/plugins'

const app = createApp({})

app.use(MagicDraggablePlugin)

Nuxt

The component is available as a Nuxt module. In your Nuxt config file add @maas/vue-equipment/nuxt to your modules and add MagicDraggable to the plugins in your configuration.

js
export default defineNuxtConfig({
  modules: ['@maas/vue-equipment/nuxt'],
  vueEquipment: {
    plugins: ['MagicDraggable'],
  },
})

Direct Import

If you prefer a more granular approach, the component can also be directly imported into any Vue component.

vue
<script setup>
import { MagicDraggable } from '@maas/vue-equipment/plugins'
</script>

<template>
  <magic-draggable id="your-draggable-id">
    <!-- your content -->
  </magic-draggable>
</template>

Composable

In order to interact with the component from anywhere within your app, we provide a useMagicDraggable composable. Import it directly when needed.

js
import { useMagicDraggable } from '@maas/vue-equipment/plugins'

const { snapTo } = useMagicDraggable('your-draggable-id')

function handleClick() {
  snapTo('top-right')
}

TIP

If you have installed the component as a Nuxt module, the composable will be auto-imported and is automatically available in your Nuxt app.

Peer Dependencies

If you haven’t installed the required peer dependencies automatically, you’ll need to install the following packages manually.

Installation

sh
pnpm install @nuxt/kit @vueuse/core defu
sh
npm install @nuxt/kit @vueuse/core defu
sh
yarn add @nuxt/kit @vueuse/core defu
sh
bun install @nuxt/kit @vueuse/core defu

API Reference

Props

The component comes with a simple set of props. Only the id is required.

PropTypeRequired
id
MaybeRef<string>true
options
MagicDraggableOptionsfalse

Options

To customize the component override the necessary options. Any custom options will be merged with the default options.

OptionTypeDefault
tag
string
'div'
threshold.distance
number128
threshold.momentum
number1.5
threshold.idle
number250
threshold.lock
number0
animation.snap.duration
number500
animation.snap.easing
function
easeOutBack
initial.snapPoint
DraggableSnapPoint
'center'
snapPoints
DraggableSnapPoint[]
DraggableSnapPoint[]
scrollLock
boolean | objectobject
scrollLock.padding
booleantrue
disabled
booleanfalse

CSS Variables

In order to provide its basic functionality the component comes with some CSS. To ensure that the component is customizable, relevant values are available as CSS variables.

VariableDefault
--magic-draggable-positionfixed
--magic-draggable-width100%
--magic-draggable-height100%
--magic-draggable-z-index999
--magic-draggable-inset0
--magic-draggable-cursorgrab
--magic-draggable-cursor-dragginggrabbing

Examples

Resize

<template>
  <magic-draggable
    id="magic-draggable-demo"
    class="bg-surface-elevation-base w-full h-[20rem] rounded-md"
  >
    <div
      :class="[
        'bg-surface-elevation-high w-40 rounded-md flex items-center justify-center transition-all duration-150',
        {
          'aspect-square': square,
          'aspect-[2/3]': !square,
        },
      ]"
    >
      <m-button @click="toggle">Resize</m-button>
    </div>
  </magic-draggable>
</template>

<script lang="ts" setup>
import { ref } from 'vue'
import { MButton } from '@maas/mirror/vue'

const square = ref(true)

function toggle() {
  square.value = !square.value
}
</script>

<style>
[data-id='magic-draggable-demo'] {
  --magic-draggable-position: relative;
  --magic-draggable-z-index: 50;
}
</style>