3
1

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 3 years have passed since last update.

Nuxt.js + TypeScript のプロジェクトに CoreUI を手動クリーンインストール【後編】

Last updated at Posted at 2020-03-04

前編の続き。

https://coreui.io/vue/demo/3.0.0/legacy/#/dashboard
公式テンプレートのデモサイト↑ような、ヘッダー・フッター・サイドバーのあるレイアウトを組んでいく。

ただ、サイドバーは表示・非表示の切り替えと、あと画面全体をPCサイズで表示している場合は最小化が可能となっており、公式テンプレートはこのサイドバーの状態をVuexストアで管理しているので、まずは先にそっちの仕組みを作ってしまおうと思う。

ストアとサイドバー状態管理

普通のストア利用設定

CoreUIとは関係のない、一般的なNuxt.js+TypeScript環境におけるストア利用方法についてはNuxtの公式ドキュメントにあるストアのページを参考にしている。この記事ではクラスベースのアプローチを採用することにし、 vuex-module-decorators をインストールしておく。

npm i -D vuex-module-decorators

vuex-module-decorators をNuxt.jsで利用するときのマナーというか手法は https://github.com/championswimmer/vuex-module-decorators#accessing-modules-with-nuxtjs に紹介されているのだけど、 ~/utils/store-accessor.ts の存在と、let宣言した変数をそのままエクスポートしている実装がキモく感じたので、すこし形を変えさせてもらった。でもたいした違いはない。

ただ、自分はVueもNuxtもTypeScriptもド素人であるし、~/utils/ 以下に記述を分けている事情をわかっていないだけかもしれない。

サンプルとしてストアモジュールをひとつ作成

最終的には不要になるのだけれど、説明の手順上、ひとまずCoreUIとは完全に切り離してストアを利用するコードを書きたいので、そのためのサンプルモジュール ~/store/counter.ts を作成する。

// ~/store/counter.ts
import { VuexModule, Module, Mutation } from 'vuex-module-decorators'

@Module({
  stateFactory: true,
  namespaced: true,
  name: 'count'
})
export default class Counter extends VuexModule {

  count: number = 0

  @Mutation
  increment() {
    this.count++
  }
}

~/store/index.ts

続いて ~/store/index.ts 。今後、新たなストアモジュールを追加したときは、そのモジュールをこのファイルにも追記していく。

// ~/store/index.ts
import { Store } from 'vuex'
import { getModule } from 'vuex-module-decorators'
import Counter from '~/store/counter'

let counter: Counter

const initializer = (store: Store<any>): void => {
  counter = getModule(Counter, store)
}

export const plugins = [initializer]

export const store = new (class {
  get counter(): Counter {
    return counter
  }
})()

これで、Vueコンポーネント側では次のような記述でストアを利用することが可能になる。

import { store } from '~/store/'

store.counter.increment()

CoreUIサイドバー状態管理用モジュール追加

ここからは、一般的な話ではなくCoreUIに関する話に戻る。

CoreUIの公式テンプレートを参考に、サイドバーの状態管理用モジュールを書く。公式テンプレートの src/store.js ファイルの内容を vuex-module-decorators のマナーで書き直していく。

ついでに、状態をあらわす真偽値型変数名が minimize と動詞だったのがキモかったのですこし変えた。 display だって動詞っぽいじゃないかと言われそうだけど、これはCSSの display プロパティを連想するのでアリということにしとく。純粋な真偽値型でもないし。

// ~/store/sidebar.ts
import { VuexModule, Module, Mutation } from 'vuex-module-decorators'

@Module({
  stateFactory: true,
  namespaced: true,
  name: 'sidebar'
})
export default class Sidebar extends VuexModule {

  display: string | boolean = 'responsive'
  minimized: boolean = false

  @Mutation
  setDisplay(value: string | boolean) {
    this.display = value
  }

  @Mutation
  setMinimized(value: boolean) {
    this.minimized = value
  }

  @Mutation
  toggleSidebarDesktop() {
    const opened = [true, 'responsive'].includes(this.display)
    this.display = opened ? false : 'responsive'
  }

  @Mutation
  toggleSidebarMobile() {
    const closed = [false, 'responsive'].includes(this.display)
    this.display = closed ? true : 'responsive'
  }
}

そして ~/store/index.ts にこのモジュールを追加。やっぱりちょっと冗長。もうすこし短くはできると思うのでそこは好きに。下記のコードでは Counter も一応残しておくけれど、もちろん削除してくれてOK。

// ~/store/index.ts
import { Store } from 'vuex'
import { getModule } from 'vuex-module-decorators'
import Counter from '~/store/counter'
import Sidebar from '~/store/sidebar'

let counter: Counter
let sidebar: Sidebar

const initializer = (store: Store<any>): void => {
  counter = getModule(Counter, store)
  sidebar = getModule(Sidebar, store)
}

export const plugins = [initializer]

export const store = new (class {
  get counter(): Counter {
    return counter
  }
  get sidebar(): Sidebar {
    return sidebar
  }
})()

まだ動作確認はできないけれど、これでサイドバーの状態管理は可能になったはず。

各種コンポーネントの追加

ヘッダー・フッター・サイドバーをVueコンポーネントとしてプロジェクトに追加していく。公式テンプレートの src/containers ディレクトリにある TheHeader.vueTheHeaderDropdownAccnt.vueTheFooter.vueTheSidebar.vue の各ファイルをNuxtプロジェクトの ~/components/ 直下にコピーし、内容をTypeScript+vue-property-decoratorのマナーに書き換えていく。あとサイドバーについてはストアの状態名やミューテーションにすこし手を加えたので、それに合わせていく。

<!-- ~/components/Header.vue -->
<template>
  <CHeader fixed with-subheader light>
    <CToggler in-header class="ml-3 d-lg-none" @click="toggleSidebarMobile()" />
    <CToggler
      in-header
      class="ml-3 d-md-down-none"
      @click="toggleSidebarDesktop()"
    />
    <CHeaderBrand class="mx-auto d-lg-none" to="/">
      <CIcon name="logo" height="48" alt="Logo" />
    </CHeaderBrand>
    <CHeaderNav class="d-md-down-none mr-auto">
      <CHeaderNavItem class="px-3">
        <CHeaderNavLink to="/dashboard">Dashboard</CHeaderNavLink>
      </CHeaderNavItem>
      <CHeaderNavItem class="px-3">
        <CHeaderNavLink to="/users" exact>Users</CHeaderNavLink>
      </CHeaderNavItem>
      <CHeaderNavItem class="px-3">
        <CHeaderNavLink>Settings</CHeaderNavLink>
      </CHeaderNavItem>
    </CHeaderNav>
    <CHeaderNav class="mr-4">
      <CHeaderNavItem class="d-md-down-none mx-2">
        <CHeaderNavLink><CIcon name="cil-bell" /></CHeaderNavLink>
      </CHeaderNavItem>
      <CHeaderNavItem class="d-md-down-none mx-2">
        <CHeaderNavLink><CIcon name="cil-list" /></CHeaderNavLink>
      </CHeaderNavItem>
      <CHeaderNavItem class="d-md-down-none mx-2">
        <CHeaderNavLink><CIcon name="cil-envelope-open" /></CHeaderNavLink>
      </CHeaderNavItem>
      <HeaderDropdown />
    </CHeaderNav>
    <CSubheader class="px-3">
      <CBreadcrumbRouter class="border-0 mb-0" />
    </CSubheader>
  </CHeader>
</template>

<script lang="ts">
import { Vue, Component } from 'vue-property-decorator'
import HeaderDropdown from './HeaderDropdown.vue'
import { store } from '~/store/'

@Component({
  components: {
    HeaderDropdown
  }
})
export default class Header extends Vue {
  toggleSidebarMobile(): void {
    store.sidebar.toggleSidebarMobile()
  }
  toggleSidebarDesktop(): void {
    store.sidebar.toggleSidebarDesktop()
  }
}
</script>
<!-- ~/components/HeaderDropdown.vue -->
<template>
  <CDropdown
    in-nav
    class="c-header-nav-items"
    placement="bottom-end"
    add-menu-classes="pt-0"
  >
    <template #toggler>
      <CHeaderNavLink>
        <div class="c-avatar">
          <img src="/img/profile.jpg" class="c-avatar-img " />
        </div>
      </CHeaderNavLink>
    </template>
    <CDropdownHeader tag="div" class="text-center" color="light">
      <strong>Account</strong>
    </CDropdownHeader>
    <CDropdownItem>
      <CIcon name="cil-bell" /> Updates
      <CBadge color="info" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownItem>
      <CIcon name="cil-envelope-open" /> Messages
      <CBadge color="success" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownItem>
      <CIcon name="cil-task" /> Tasks
      <CBadge color="danger" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownItem>
      <CIcon name="cil-comment-square" /> Comments
      <CBadge color="warning" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownHeader tag="div" class="text-center" color="light">
      <strong>Settings</strong>
    </CDropdownHeader>
    <CDropdownItem><CIcon name="cil-user" /> Profile</CDropdownItem>
    <CDropdownItem><CIcon name="cil-settings" /> Settings</CDropdownItem>
    <CDropdownItem>
      <CIcon name="cil-dollar" /> Payments
      <CBadge color="secondary" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownItem>
      <CIcon name="cil-file" /> Projects
      <CBadge color="primary" class="ml-auto">{{ itemsCount }}</CBadge>
    </CDropdownItem>
    <CDropdownDivider />
    <CDropdownItem>
      <CIcon name="cil-shield-alt" /> Lock Account
    </CDropdownItem>
    <CDropdownItem><CIcon name="cil-lock-locked" /> Logout</CDropdownItem>
  </CDropdown>
</template>

<script lang="ts">
import { Vue, Component } from 'vue-property-decorator'

@Component
export default class HeaderDropdown extends Vue {
  itemsCount: number = 42
}
</script>

<style scoped>
.c-icon {
  margin-right: 0.3rem;
}
</style>
<!-- ~/components/Footer.vue -->
<template>
  <CFooter :fixed="false">
    <div>
      <a href="https://coreui.io" target="_blank">CoreUI</a>
      <span class="ml-1">
        &copy; {{ new Date().getFullYear() }} creativeLabs.
      </span>
    </div>
    <div class="ml-auto">
      <span class="mr-1">Powered by</span>
      <a href="https://coreui.io/vue" target="_blank">CoreUI for Vue</a>
    </div>
  </CFooter>
</template>

<script lang="ts">
import { Vue, Component } from 'vue-property-decorator'

@Component
export default class Footer extends Vue {}
</script>
<!-- ~/components/Sidebar.vue -->
<template>
  <CSidebar
    fixed
    :minimize="minimized"
    :show="display"
    @update:show="(value) => (display = value)"
  >
    <CSidebarBrand class="d-md-down-none" to="/">
      <CIcon
        class="d-block"
        name="logo"
        size="custom-size"
        :height="35"
        :view-box="`0 0 ${minimized ? 110 : 556} 134`"
      />
    </CSidebarBrand>
    <CRenderFunction flat :content-to-render="nav" />
    <CSidebarMinimizer
      class="d-md-down-none"
      @click.native="minimized = !minimized"
    />
  </CSidebar>
</template>

<script lang="ts">
import { Vue, Component } from 'vue-property-decorator'
import nav from './_nav'
import { store } from '~/store/'

@Component
export default class Sidebar extends Vue {
  nav = nav
  get display() {
    return store.sidebar.display
  }
  set display(value: boolean | string) {
    store.sidebar.setDisplay(value)
  }
  get minimized() {
    return store.sidebar.minimized
  }
  set minimized(value: boolean) {
    store.sidebar.setMinimized(value)
  }
}
</script>

あとは、HeaderDropdown.vue でURLべた書きしているプロフィール画像 /img/profile.jpg と、 Sidebar.vue でインポートしている、サイドメニュー項目を記述した ~/components/_nav.ts が必要。プロフィール画像は ~/static/img/profile.jpg として適当な画像ファイルを置けばよい。_nav.ts は公式テンプレートのままでもOK。一応、公式テンプレートを参考に、プロパティをなるべく使いつつ内容を短くしたものを次に貼り付けておく。

// ~/components/_nav.ts
export default [
  {
    _name: 'CSidebarNav',
    _children: [
      {
        _name: 'CSidebarNavItem',
        name: 'ダッシュボード',
        to: '/',
        icon: 'cil-speedometer',
        badge: {
          color: 'primary',
          text: 'NEW'
        }
      },
      {
        _name: 'CSidebarNavTitle',
        _children: ['管理メニュー']
      },
      {
        _name: 'CSidebarNavDropdown',
        name: 'ユーザ管理',
        route: '/users',
        icon: 'cil-user',
        items: [
          {
            name: 'ユーザ一覧',
            to: '/users/list'
          },
          {
            name: 'ユーザ新規登録',
            to: '/users/new'
          }
        ]
      },
      {
        _name: 'CSidebarNavItem',
        name: '設定',
        to: '/configs',
        icon: 'cil-puzzle'
      }
    ]
  }
]

レイアウトファイル編集

~/layouts/default.vue を、ヘッダー・フッター・サイドバーを用いたレイアウトに書き換える。

<!-- ~/layouts/default.vue -->
<template>
  <div class="c-app">
    <Sidebar />
    <CWrapper>
      <Header />
      <div class="c-body">
        <main class="c-main">
          <div class="container-fluid">
            <nuxt />
          </div>
        </main>
        <Footer />
      </div>
    </CWrapper>
  </div>
</template>

<script lang="ts">
import { Vue, Component } from 'vue-property-decorator'
import { iconsSet as icons } from '~/assets/icons/icons'
import Header from '~/components/Header.vue'
import Sidebar from '~/components/Sidebar.vue'
import Footer from '~/components/Footer.vue'

@Component({
  components: {
    Header,
    Sidebar,
    Footer
  }
})
export default class DefaultLayout extends Vue {
  created() {
    this.$root.$options.icons = icons
  }
  public name() {
    return this.$route.name
  }
  public list() {
    return this.$route.matched
  }
}
</script>

最後にページを追加

~/components/_nav.ts で定義した各ページのURLとNuxt.jsのルーティングの規約にしたがって、 ~/pages/users/list.vue~/pages/users/new.vue~/pages/configs/index.vue を作る。内容は好きに。

これで、CoreUIをフル利用したレイアウト完成。

3
1
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
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?