3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

LCLAdvent Calendar 2022

Day 21

【Nuxt.js】コンテンツの高さが動的に変化するアコーディオンをいい感じに実装する

Last updated at Posted at 2022-12-22

この記事は「LCL Advent Calendar 2022」21日目の記事です。

あるある

アコーディオンを作りたい

下記みたいな感じで書いてみるが所望の挙動にならない
(大抵の場合、height:autoであることが原因)

height.css
.element {
  transition: height 400ms ease;
}


max-heightによる実装を行う

max_height.css
.content {
  transition: max-height 400ms ease;
  overflow: hidden;
  max-height: 0;

  &.open {
    max-height: 1000px; // 十分大きな値
  }
}

max-heightによる実装の課題

  • max-heightの値が不十分だと要素次第では最後まで表示されない
  • max-heightの値が実際の高さに対して大きすぎるとUXの面で不適切
    • すごい速度で開く
      • 上例の場合、実際の高さがどのくらいであれ、400msで0→1000pxのtransitionを行うため
    • 押下してからなかなか閉じない
      • 上例の場合、実際の高さがどのくらいであれ、400msで1000px→0のtransitionを行うため

実装

方法はいろいろあると思いますが、最近

  • アコーディオンを開いている間にコンテンツ(アコーディオンで開閉される要素)の高さが動的に変化する可能性がある
  • その際も、heightはアニメーションしてほしい
  • ただし、アコーディオン自体の高さには上限がほしい(コンテンツの高さがアコーディオンの高さを超えた場合はスクロールするようにする)

という仕様を実装する機会があったので、これも考慮して実装してみます。

結果

ezgif.com-gif-maker.gif

サンプルコード

アコーディオン

Accordion.vue
<template>
  <AccordionBase :contentHeight="contentHeight">
    <template v-slot:title>Accordion</template>
    <template v-slot:content>
      <ul ref="content" class="list">
        <li v-for="(item, index) in list" :key="index" class="item">
          {{ item }}
        </li>
      </ul>
    </template>
  </AccordionBase>
</template>

<script>
export default {
  props: {
    // 動的要素。「pages内でAPI requestした結果をpropsとして受け取る」等が多い
    list: { type: Array, required: true },
  },
  data() {
    return {
      contentHeight: 0, // アコーディオンコンテンツの高さ
    }
  },
  watch: {
    list: {
      handler() {
        this.$nextTick(() => {
          // listの変化に対してアコーディオンコンテンツの高さを再計算
          this.contentHeight = this.$refs.content.getBoundingClientRect().height
        })
      },
      deep: true,
    },
  },
}
</script>

<style lang="scss" scoped>
// 動作と無関係なcssは記載していません
</style>

アコーディオン実装の大元であるAccordionBase.vueでwrapする形で実装します。

ポイントは$nextTickです。これにより、DOMの更新完了後(listが変化してコンテンツの高さが変化した後)にcontentHeightの値が変化するようになります。

アコーディオン(本体)

AccordionBase.vue
<template>
  <div class="Accordion">
    <div class="Accordion__title" @click="toggle">
      <slot name="title" />
    </div>
    <transition
      @before-enter="beforeEnter"
      @enter="enter"
      @before-leave="beforeLeave"
      @leave="leave"
    >
      <div v-show="isOpen" ref="content" class="Accordion__content">
        <slot name="content" />
      </div>
    </transition>
  </div>
</template>

<script>
export default {
  props: {
    // コンテンツの高さ
    contentHeight: { type: Number, required: false, default: 0 },
  },
  data() {
    return {
      isOpen: false,
    }
  },
  watch: {
    contentHeight() {
      this.updateHeight(this.contentHeight)
    },
  },
  methods: {
    updateHeight(height) {
      this.$refs.content.style.height = `${height}px`
    },
    beforeEnter(el) {
      el.style.height = `0`
    },
    enter(el) {
      el.style.height = `${el.scrollHeight}px`
    },
    beforeLeave(el) {
      el.style.height = `${el.scrollHeight}px`
    },
    leave(el) {
      el.style.height = '0'
    },
    toggle() {
      this.isOpen = !this.isOpen
    },
  },
}
</script>
<style lang="scss" scoped>
// 動作と無関係なcssは記載していません

.Accordion {
  &__content {
    max-height: 400px; // コンテンツの最大高さ 400px以上の場合はscrollさせる
    overflow-y: scroll;
    transition: height 400ms ease-in-out;
  }
}
</style>

基本のトランジション+JavaScriptフックに加えて、コンテンツ高さが変化した場合もheightの値を変更します。
どちらの場合でもheightの値が明示的になるので、transitionによるアニメーションが有効になります。

ページ

pages/index.vue
<template>
  <div>
    <div>
      <button @click="push">push</button>
      <button @click="pop">pop</button>
    </div>
    <AccordionSample :list="accordionContents"></AccordionSample>
  </div>
</template>

<script>
export default {
  data() {
    return {
      accordionContents: ['init element'], // 動的に変化する要素
    }
  },
  methods: {
    push() {
      this.accordionContents.push(`element-${this.accordionContents.length}`)
    },
    pop() {
      this.accordionContents.pop()
    },
  },
}
</script>

<style lang="scss" scoped>
  // 動作と無関係なcssは記載していません
</style>

一応載せていますが、本筋とはあまり関係ないです。

他の書き方(コンテンツもコンポーネントとして定義したい場合)

スコープ付きスロットを使用します。

アコーディオン

AccordionSample.vue
<template>
  <AccordionBase :contentHeight="contentHeight">
    <template v-slot:title>Accordion</template>
    <template v-slot:content="contentProps">
      <AccordionContent :list="list" @updateHeight="contentProps.updateHeight" />
    </template>
  </AccordionBase>
</template>

<script>
export default {
  props: {
    list: { type: Array, required: true },
  },
  data() {
    return {
      contentHeight: 0,
    }
  },
}
</script>

<style lang="scss" scoped>
  // 動作と無関係なcssは記載していません
</style>

アコーディオンのコンテンツ部分(コンポーネント化)

AccordionContent.vue
<template>
  <ul ref="content" class="list">
    <li v-for="(item, index) in list" :key="index" class="item">
      {{ item }}
    </li>
  </ul>
</template>

<script>
export default {
  props: {
    list: { type: Array, required: true },
  },
  watch: {
    list: {
      handler() {
        this.$nextTick(() => {
          this.$emit('updateHeight', this.$el.getBoundingClientRect().height)
        })
      },
      deep: true,
    },
  },
}
</script>

<style lang="scss" scoped>
  // 動作と無関係なcssは記載していません
</style>

アコーディオン(本体)

AccordionBase.vue
<template>
  <div class="Accordion">
    <div class="Accordion__title" @click="toggle">
      <slot name="title" />
    </div>
    <transition
      @before-enter="beforeEnter"
      @enter="enter"
      @before-leave="beforeLeave"
      @leave="leave"
    >
      <div v-show="isOpen" ref="content" class="Accordion__content">
        <slot name="content" :updateHeight="updateHeight" />
      </div>
    </transition>
  </div>
</template>

<script>
export default {
  props: {
    contentHeight: { type: Number, required: false, default: 0 },
  },
  data() {
    return {
      isOpen: false,
    }
  },
  watch: {
    contentHeight() {
      this.updateHeight(this.contentHeight)
    },
  },
  methods: {
    updateHeight(height) {
      this.$refs.content.style.height = `${height}px`
    },
    beforeEnter(el) {
      el.style.height = `0`
    },
    enter(el) {
      el.style.height = `${el.scrollHeight}px`
    },
    beforeLeave(el) {
      el.style.height = `${el.scrollHeight}px`
    },
    leave(el) {
      el.style.height = '0'
    },
    toggle() {
      this.isOpen = !this.isOpen
    },
  },
}
</script>
<style lang="scss" scoped>
// 動作と無関係なcssは記載していません

.Accordion {
  &__content {
    max-height: 400px;
    overflow-y: scroll;
    transition: height 400ms ease-in-out;
  }
}
</style>

終わりに

heightの値を直接書き換える点がややイケてない気がしなくもないですが、挙動としてはいい感じになると思います。

3
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?