0
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?

Vue3でBootStrapを利用する方法

Posted at

概要

Vue.js v3(以降、Vue3と略記)でBootstrapを使う方法を説明します。
ここでは、BootstrapVueではなく、オリジナルのBootstrap v5を利用する手法について解説します。
なお、本記事は自分用のメモとして作成しているため、細かい補足や個人向けの実装例を多く含んでいます。

目的

本記事の目的は以下の通りです。

  • Vue3上にてBootstrap v5を簡単に使えるようにする

背景

BootstrapをVueなどのフレームワーク上で利用する場合、DOM操作がフレームワークと競合する可能性があるため、フレームワーク固有のパッケージ(例: BootstrapVue)の利用が推奨されます1
しかし、BootstrapVueは現時点でVue3に未対応であるため、今回はオリジナルのBootstrap v5を直接利用する方法を採用します。
Bootstrapに不慣れな方がとりあえずVue3上でBootstrapを扱う方法、の一例として参考ください。

想定読者

この記事は以下のような方を対象としています。

  • Vue3で簡単なWebアプリ(ToDo管理など)を作成できる方
  • Composition API記法を使用している方
  • BootstrapVueの使用経験はあるが、オリジナルのBootstrapは未経験の方

動作環境(検証環境)

  "dependencies": {
    "bootstrap": "^5.3.3",
    "bootstrap-icons": "^1.11.3",
    "vue": "^3.5.13"
  },

サンプルコード

本記事に掲載のサンプルコードは抜粋のため、コード全体は以下を参照ください。

Vue3のプロジェクトとしてのルートは以下です。
ご自身の環境で手軽に動作確認する際には、こちらの配下を取得ください。


Vue3での作成手順

以下の手順で説明します。

  1. Vue3へのBootstrap v5の導入方法
  2. 静的コンポーネントの記述方法
  3. 動的コンポーネントの記述方法
  4. 動的コンポーネントの動作をカスタマイズする方法

1. Vue3へのBootstrap v5の導入方法

前提となるVue3のプロジェクト作成手順

任意のフォルダー上のコマンドラインから以下のコマンドを実行し、プロジェクトを作成します。

npm init vue@latest

この例では、プロジェクト名を「intro-bootstrap5-in-vue3」とし、その他の設定はデフォルト(ただし、ESLintのみ「y」に変更)で作成します。

実行例:

Need to install the following packages:
create-vue@3.14.0
Ok to proceed? (y) y

Vue.js - The Progressive JavaScript Framework

√ Project name: ... intro-bootstrap5-in-vue3
√ Add TypeScript? ... No / Yes
√ Add JSX Support? ... No / Yes
√ Add Vue Router for Single Page Application development? ... No / Yes
√ Add Pinia for state management? ... No / Yes
√ Add Vitest for Unit Testing? ... No / Yes
√ Add an End-to-End Testing Solution? ≫ No
√ Add ESLint for code quality? ≫ Yes
√ Add Prettier for code formatting? ... No / Yes

Done. Now run:

  cd intro-bootstrap5-in-vue3
  npm install
  npm run dev

作成したVue3へBootstrapを導入する手順

作成したプロジェクトフォルダー(例:intro-bootstrap5-in-vue3)内で、以下の手順に従います。

  1. BootstrapおよびBootstrap Iconsのインストール
    npmを使用して、BootstrapとBootstrap Iconsをインストールします2

    npm install bootstrap
    npm install bootstrap-icons 
    

    ※これにより、Bootstrap Iconsも含めて利用可能となります。

  1. JavaScriptとスタイルシートのインポート
    プロジェクト全体でBootstrapのスタイルと動作を利用できるよう、main.jsに以下の記述を追加します3

    import './assets/main.css'
    
    import { createApp } from 'vue'
    import App from './App.vue'
    
    // +++ for Bootstrap +++
    import 'bootstrap/dist/css/bootstrap.min.css'
    import 'bootstrap/dist/js/bootstrap.min.js'
    
    import 'bootstrap-icons/font/bootstrap-icons.css'
    // ----------------------
    
    createApp(App).mount('#app')
    

これで、プロジェクト全体でBootstrapのスタイルとJavaScriptが使用可能になります4


2. Bootstrapの静的コンポーネントの記述方法

最初に、静的なコンポーネント、つまりJavaScriptによる動的な表示を伴わず、Bootstrapの標準CSSクラスのみで見た目やレイアウトが実現されているコンポーネントを用いる方法について解説します。

例1: ボタン

以下のように、Bootstrapの定義済みCSSクラス(ここではbtn btn-primary form-controlが該当)を適用するだけで、Bootstrapのコンポーネントをとして利用できます。

<button class="btn btn-primary">Primary Button</button>

例2: インプットボックス

Vue3のリアクティブな変数(ref)と組み合わせる例です。

<script setup>
import { ref } from 'vue'
const inputText = ref('')
</script>

<template>
  <div>
    <h3>静的コンポーネント:インプット</h3>
    <input 
      type="text" 
      class="form-control"
      placeholder="テキストを入力してください"
      v-model="inputText"
    />
    <br><br>
    Debug: {{ inputText }}
  </div>
</template>

3. Bootstrapの動的コンポーネントの記述方法

続いて、動的な表示を伴なうコンポーネントを、Bootstrapの既定の動作仕様のままに利用する方法(動作カスタマイズを行わずに用いる場合の方法)を説明します。

Bootstrapの標準仕様の動作を利用する場合、追加でのJavaScriptの記述は不要です。

ここでは、ドロップダウンリスト(Dropdowns)とコラプス(collapse)を例にとります。

例1: ドロップダウンリスト(Dropdowns)

Dropdownsコンポーネントは、dropdownクラスが指定された要素の配下において、属性data-bs-toggleが付与された要素のクリックをトリガーとして同じ配下のdropdown-menuクラスが付与された要素の表示と非表示をトグルする仕様です。このため、トグル対象を明示するためのidなどを指定する必要はありません。

<template>
  <div class="dropdown">
    <button 
      class="btn btn-secondary dropdown-toggle" 
      type="button" 
      data-bs-toggle="dropdown" 
      aria-expanded="false"
    >
      ドロップダウンリスト
    </button>
    <ul class="dropdown-menu">
      <li class="dropdown-item">Action</li>
      <li class="dropdown-item">Another action</li>
      <li class="dropdown-item">Something else here</li>
    </ul>
  </div>
</template>

例2: コラプス(collapse)

コラプスとして開閉する要素を特定する(指定する)には、idを用います。Vueコンポーネントの再利用性を考慮するとidを即値で書くのは望ましくないので、動的に管理することで再利用時のid衝突を回避します。

この例では、コラプス要素のidがコンポーネント内でしか参照されない前提で、動的なUUIDを用いて実装しています5。開閉するボタン側のidはコンポーネント外から参照の可能性がありますが、コラプス要素自体はボタンからしか参照されないため、このようにしています。
なお、テストの観点から「コンポーネント外からも参照する」場合は、付与するidを親コンポーネントからprops経由で渡して子コンポーネントの再利用時のid衝突を回避がする(idを親コンポーネント側で管理する)など実装が望ましいです。

<script setup>
import { ref } from 'vue'
const _generateUUID = () => window.crypto.randomUUID()
const scopedIdCollapse1 = ref(_generateUUID())
</script>

<template>
  <div>
    <h3>動的コンポーネント:コラプス(collapse)</h3>
    <button 
      class="btn btn-primary" 
      type="button" 
      data-bs-toggle="collapse" 
      :data-bs-target="`#${scopedIdCollapse1}`" 
      aria-expanded="false" 
      :aria-controls="scopedIdCollapse1"
    >
      コラプスの開閉
    </button>
    <div class="collapse" :id="scopedIdCollapse1">
      <div class="card card-body">
        Some placeholder content for the collapse component...
      </div>
    </div>
  </div>
</template>

4. Bootstrapの動的コンポーネントの動作をカスタマイズする方法

動的コンポーネントの既定の挙動に対して、独自の処理を割り込ませたい場合の方法を紹介します。

ここでは、モーダルダイアログ(Modal)とドロップダウンリストを例にとります。

例1: モーダルダイアログ(Modal)

モーダルダイアログとして表示する要素を特定する(指定する)には、data-bs-target属性とidを用います。
その際に、Vueコンポーネントの再利用性を考慮して動的にidを指定する実装とするのは、コラプス(collapse)と同じです。

組み込みの閉じる操作を行うコントローラーにdata-bs-dismiss属性を付与することで、ここまではコラプスと同様に表示と非表示をJavaScriptを個別に実装すること無く実現できます。

異なるのは、OKボタンが押され時の動作です。こちらは、押された時のイベントを@click="onConfirmModalDialog1"で受け取り、そのメソッド内で明示的にbsModalDialog1.hide()を用いてダイアログを閉じる処理を行う実装としています。

bsModalDialog1インスタンスは、Modal.getOrCreateInstance()を用いて生成することで、idを経由して対象のモーダル要素に紐づけておきます。

<script setup>
import { ref, onMounted } from 'vue'
import { Modal } from 'bootstrap/dist/js/bootstrap.min.js'
const _generateUUID = () => window.crypto.randomUUID()
const scopedIdModalDialog1 = ref(_generateUUID())
const debugMsg4Dialog1 = ref('')
let bsModalDialog1 = null

const onConfirmModalDialog1 = () => {
  bsModalDialog1.hide()
  debugMsg4Dialog1.value = '保存ボタンが押されました。'
  setTimeout(() => {
    debugMsg4Dialog1.value = ''
  }, 3000)
}

onMounted(() => {
  bsModalDialog1 = Modal.getOrCreateInstance(document.getElementById(scopedIdModalDialog1.value))
})
</script>

<template>
  <div>
    <h3>動的コンポーネント:モーダルダイアログ(Modal)</h3>
    <button 
      type="button" 
      class="btn btn-primary" 
      data-bs-toggle="modal" 
      :data-bs-target="`#${scopedIdModalDialog1}`"
    >
      モーダルダイアログを表示
    </button>
    <div>
      [Debug]ダイアログの結果: {{ debugMsg4Dialog1 }}
    </div>
    <div 
      class="modal fade" 
      :id="scopedIdModalDialog1" 
      tabindex="-1" 
      aria-labelledby="exampleModalLabel" 
      aria-hidden="true"
    >
      <div class="modal-dialog">
        <div class="modal-content">
          <div class="modal-header">
            <h5 class="modal-title" id="exampleModalLabel">モーダルダイアログのタイトル</h5>
            <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
          </div>
          <div class="modal-body">
            モーダルダイアログの本文
          </div>
          <div class="modal-footer">
            <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">閉じる</button>
            <button 
              type="button" 
              class="btn btn-primary"
              @click="onConfirmModalDialog1"
            >
              保存
            </button>
          </div>
        </div>
      </div>
    </div>
  </div>
</template>

例2-1: ドロップダウンリスト(Dropdowns)

先に示した、デフォルト動作のDropdownsとは異なり、ドロップダウンリスト内部をクリックされた時に必ず閉じるのではなく、閉じる場合と閉じない場合に分岐させる実装例を説明します。

これは、リスト内部のクリック時を閉じる対象外としたうえで、閉じるときには自前のJavaScriptを用いて閉じる操作を行う事で実現できます6
この例では、Dropdownsではidによる指定が不要なので、操作用のDropdownsインスタンスを取得する際に、モーダルダイアログとは異なりidを経由せずにVueのref属性を用いて直接に要素を指定する方法を採用しています。

<script setup>
import { onMounted, ref, useTemplateRef } from 'vue';
import { Dropdown } from 'bootstrap/dist/js/bootstrap.min';

const dropdownItemsCustom = ref([
    {id: 1, name: 'Action(選択して閉じる)'},
    {id: 2, name: 'Another action(選択されるが閉じない)'},
    {id: 3, name: 'Something else here'},
]);
const selectedDropdownItemNameCustom = ref('');
const elemDropdownList = useTemplateRef('refDropdownList');
let bsDropdownList = null;
const onDropdownItemClickCustom = (selectedId) => {
    selectedDropdownItemNameCustom.value = dropdownItemsCustom.value.find(item => item.id === selectedId).name;
    if(selectedId !== 2){
        bsDropdownList.hide();
    }
};

onMounted(()=>{
    bsDropdownList = Dropdown.getOrCreateInstance(elemDropdownList.value);
});
</script>

<template>
    <div class="input-group dropdown">
        <input 
            type="text" 
            class="form-control" 
            placeholder="選択されたリスト要素を挿入"
            v-model="selectedDropdownItemNameCustom"
        ></input>
        <button
            type="button" 
            class="btn btn-secondary dropdown-toggle dropdown-toggle-split"
            data-bs-auto-close="outside"
            data-bs-toggle="dropdown" 
        ></button>
        <ul 
            class="dropdown-menu dropdown-menu-end" 
            ref="refDropdownList"
        >
            <li 
                v-for="item in dropdownItemsCustom" 
                :key="item.id"
                class="dropdown-item"
                @click="onDropdownItemClickCustom(item.id)"
            >
                {{item.name}}
            </li>
        </ul>
    </div>
</template>

例2-2: ドロップダウンリスト(Dropdowns)

次の例では、ドロップダウン自体の動作はカスタムしませんが、その際のイベントをフックして関連する処理を実行する(途中に割り込ませる)方法を説明します。
この場合は、Dropdownsインスタンスを取得する必要はありませんが、動作を割り込ませる対象の要素を特定する必要があるため、先の例と同様にVueのref属性を用いて直接に要素を指定する方法で実装しています。

<script setup>
import { onMounted, ref, useTemplateRef, onBeforeUnmount } from 'vue';

const debugMsgDropdownCustom2 = ref('');
const elemDropdownList2Parent = useTemplateRef('refDropdownListParent');
const _listenerHiddenBsDropdown = (event) => {
    console.log('hidden.bs.dropdown', event);
    debugMsgDropdownCustom2.value = 'ドロップダウンリストが閉じられました。';
    setTimeout(() => {
        debugMsgDropdownCustom2.value = '';
    }, 3000);
};

onMounted(()=>{
    elemDropdownList2Parent.value.addEventListener(
        'hide.bs.dropdown', 
        _listenerHiddenBsDropdown
    );
});
onBeforeUnmount(() => { 
    // https://ja.vuejs.org/api/composition-api-lifecycle#onunmounted
    // https://vueuse.org/core/useEventListener/ を使えば、umount時のこの考慮は
    // 不要となるが、そのためだけにimportするほどでもない、ので自前で実装する。
    elemDropdownList2Parent.value.removeEventListener(
        'hidden.bs.dropdown', 
        _listenerHiddenBsDropdown
    );
});
</script>

<template>
    <div class="dropdown">
        <button 
            class="btn btn-secondary dropdown-toggle" 
            type="button" 
            data-bs-toggle="dropdown" 
            aria-expanded="false"
            ref="refDropdownListParent"
        >
            ドロップダウンリスト
        </button>
        <ul class="dropdown-menu">
            <li 
                v-for="item in dropdownItems" 
                :key="item.id"
                class="dropdown-item"
            >
                {{item.name}}
            </li>
        </ul>
    </div>
    <div>
        [Debug]ドロップダウンリストの操作: {{debugMsgDropdownCustom2}}
    </div>
</template>

以上です。

  1. Bootstrap JavaScriptによるDOM操作が、各フレームワーク(Vue.jsなど)と競合する可能性があり、完全な互換性は保証されないためです。https://getbootstrap.jp/docs/5.3/getting-started/javascript/

  2. ここで、オプション--saveを記述してしませんが、これは、このオプションがnpm v5以降はデフォルトで有効となったためです。そのため省略できます。 npm install saves any specified packages into dependencies by default. https://blog.npmjs.org/post/161081169345/v500 , https://qiita.com/hvfnabndnd/items/c5beda8572aa8c1e6be6

  3. BootstrapのSCSSファイルをカスタマイズしたい場合に、src/scss/styles.scssを作成した上で、main.jsでSCSSインポートを行う方法がありますが、今回は割愛します。

  4. 実は、import 'bootstrap/dist/js/bootstrap.min.js'を記述しなくともimport 'bootstrap/dist/css/bootstrap.min.css'さえ記述すれば、本サンプルコードの範囲では問題なく動作する。 https://getbootstrap.jp/docs/5.0/getting-started/introduction/#components で「JavaScriptとPopperを明示的に必要としている」コンポーネントとして明記されている「Collapse」や「Dropdowns」を含んでいるのにもかかわらず! ・・・これの背景は不明(追ってない)。可能性としては、Vitaでのトランスパイル時にいい感じにjs側も補完解釈して読み込んでいる可能性が無いわけでは無いが、、、何故に動く???教えてエロい人!

  5. const _generateUUID = () => window.crypto.randomUUID()としてラッパーしてから利用しているのは、このIDの生成ロジックを後から差し替える可能性を考慮しているため。とりあえず windowオブジェクトのcryptoモジュールを用いているが、条件によっては(例えば単体テスト環境など)これ以外のモジュールを用いる方が望ましいことも少なくないため。

  6. デフォルト動作は「ドロップダウンリストの内外のいづれかがクリックされたら閉じる」であり属性data-bs-auto-close="true"が設定されます。これをdata-bs-auto-close="outside"と設定することで「外側がクリックされたら閉じるが、内側をクリックされても閉じない」動作となります。ここまではDropdownsの規定の動作オプションの範囲内です。 https://getbootstrap.jp/docs/5.3/components/dropdowns/#%E8%87%AA%E5%8B%95%E3%81%A7%E9%96%89%E3%81%98%E3%82%8B

0
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
0
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?