Skip to content
Sponsored by

MagicPie

MagicPie renders a percentage based value as SVG.

0%
<template>
  <div class="flex flex-col gap-4 items-center">
    <magic-pie
      id="magic-pie-default-demo"
      class="w-16 border-2 rounded-full border-surface"
    />
    <m-badge>{{ mappedPercentage }}</m-badge>
    <div class="flex gap-2">
      <m-button @click="start">Start</m-button>
      <m-button @click="pause">Pause</m-button>
      <m-button @click="stop">Stop</m-button>
    </div>
  </div>
</template>

<script lang="ts" setup>
import { MButton, MBadge } from '@maas/mirror/vue'
import { useMagicPie } from '@maas/vue-equipment/plugins'
import { computed } from 'vue'

const {
  percentage,
  setPercentage,
  interpolatePercentage,
  cancelInterpolatePercentage,
} = useMagicPie('magic-pie-default-demo')

function start() {
  if (percentage.value === 100) {
    setPercentage(0)
  }

  interpolatePercentage({
    value: 100,
    duration: 2000,
  })
}

function pause() {
  cancelInterpolatePercentage()
}

function stop() {
  if (percentage.value === 100) {
    return
  }

  cancelInterpolatePercentage()
  setPercentage(0)
}

const mappedPercentage = computed(() => {
  return `${Math.round(percentage.value)}%`
})
</script>

<style>
[data-id='magic-pie-default-demo'] {
  --magic-pie-background: theme(
    'backgroundColor.surface.elevation.high.DEFAULT'
  );
}
</style>

Overview

Anatomy

vue
<template>
  <magic-pie id="your-pie-id" />
</template>

<script setup>
const { percentage } = useMagicPie('your-pie-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 MagicPiePlugin to your app.

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

const app = createApp({})

app.use(MagicPiePlugin)

Nuxt

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

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

Direct Import

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

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

<template>
  <magic-pie id="your-pie-id" />
</template>

Composable

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

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

const { setPercentage } = useMagicPie('your-pie-id')

onMounted(() => {
  setPercentage(50)
})

TIP

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

API Reference

Props

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

PropTypeRequired
id
MaybeRef<string>true
options
MagicPieOptionsfalse

Options

OptionTypeDefault
flip
booleanfalse

CSS Variables

VariableDefault
--magic-pie-backgroundtransparent
--magic-pie-foregroundcurrentColor

Examples

Flip

0%
<template>
  <div class="flex flex-col gap-4 items-center">
    <magic-pie
      id="magic-pie-flip-demo"
      class="w-16 border-2 rounded-full border-surface"
      :options="{ flip }"
    />
    <m-badge>{{ mappedPercentage }}</m-badge>
    <div class="flex gap-2">
      <m-button @click="start">Start</m-button>
      <m-button @click="pause">Pause</m-button>
      <m-button @click="stop">Stop</m-button>
    </div>
  </div>
</template>

<script lang="ts" setup>
import { ref, computed, watch } from 'vue'
import { MButton, MBadge } from '@maas/mirror/vue'
import { useMagicPie } from '@maas/vue-equipment/plugins'

const flip = ref(false)

const {
  percentage,
  setPercentage,
  interpolatePercentage,
  cancelInterpolatePercentage,
} = useMagicPie('magic-pie-flip-demo')

function start() {
  interpolatePercentage({
    value: 100,
    duration: 2000,
  })
}

function pause() {
  cancelInterpolatePercentage()
}

function stop() {
  if (percentage.value === 100) {
    return
  }

  cancelInterpolatePercentage()
  setPercentage(0)
}

const mappedPercentage = computed(() => {
  return `${Math.round(percentage.value)}%`
})

watch(
  () => percentage.value,
  (value) => {
    if (value === 100) {
      setPercentage(0)
      flip.value = !flip.value
      start()
    }
  }
)
</script>

<style>
[data-id='magic-pie-flip-demo'] {
  --magic-pie-background: theme(
    'backgroundColor.surface.elevation.high.DEFAULT'
  );
}
</style>