11
3

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.

kintoneAdvent Calendar 2022

Day 11

kintoneでTailwindCSSを利用するための勘所

Last updated at Posted at 2022-12-10

この記事は kintone アドベントカレンダー 2022 の 11 日目の記事です。

はじめに

TailwindCSS、いいですよね。
学習コストは大した事ないくせにけっこう劇的な効果があって、これまで CSS の命名ルールとか運用ルールとかで汲々としていた現場にそこそこドラスティックな変化をもたらしてくれていると感じています。

TailwindCSS 自体をあまり詳しく知らないと言う方は、ちょっと調べれば参考になるサイトはたくさん見つかるでしょう。
特に以下の記事はメリット・デメリット双方に言及しておりかなり有用です。

tailwindcss を導入して感じたメリット・デメリットについて

この激オススメの CSS フレームワーク・TailwindCSS を kintone で利用するための勘所をお話ししようと言うのが今回の趣旨です。

早速作りましょう

仕上がりイメージはこんな感じ。
例によって kintone 新規アプリ作成の際に「おすすめのアプリ」としてラインナップされている「問い合わせ管理」をちょっといい感じに見せるやつです。
Trello などのような良くあるカンバン形式と少し趣向を変えて今回はカードを横に並べるやり方にしてみました。
01-result.png

プロジェクトを作る

今回は Vite + Vue3 + TypeScript に TailwindCSS を組み合わせる流れです。
基本的には Vite のスタートアップガイドTailwindCSS のガイドを参照すれば特段難しい事はないかと思います。

yarn create vite
? Project name: (任意のプロジェクト名)
? Select a framework: Vue
? Select a variant: TypeScript

できたプロジェクトフォルダを VS Code で開き、ターミナルを起動します。
ちなみに私はターミナルを Git Bash にしています。
TailwindCSS や関連ライブラリをインストールします。

yarn add -D tailwindcss postcss autoprefixer
npx tailwindcss init

tailwind.config.cjs と言うファイルができるかと思うので、拡張子は  .js にしておけばよいでしょう。

プロジェクト直下に postcss.config.js ファイルを作ります。

postcss.config.js
module.exports = {
  plugins: {
    tailwindcss: {},
    autoprefixer: {},
  }
}

流れで作成された tailwind.config.js を修正します。

tailwind.config.js
-   content: [],
+   content: ['./index.html', './src/**/*.{vue,js,ts,jsx,tsx}'],

この流れだと package.json"type": "module" の記述があり、ビルドに差し支えます。
ここを修正します。

package.json
-   "type": "module",

ここまでやって yarn dev と実行して開発サーバを立ち上げて動作確認をしましょう。
02-vite-vue.png
うむ。まあ普通の Vite + Vue3 のスタートアップページですね。

さらに、デフォルトで作成されている src/style.css について、ガイドに従い TailwindCSS を読み込めるようにします。

src/style.css
@tailwind base;
@tailwind components;
@tailwind utilities;

TypeScript 開発の準備

kintone で TypeScript 開発をするためには諸々準備が必要です。
まずは  @kintone/dts-gen をインストール。
ソース内で型参照をしたいので @kintone/rest-api-client も合わせてインストールします。

yarn add -D @kintone/dts-gen
yarn add @kintone/rest-api-client

src/types/kintone.d.ts と言うファイルを以下の内容で作成します。
ただこれはもっと良いやり方がある気がする。

src/types/kintone.d.ts
declare namespace kintone {
  interface Event {
    appId: number
    viewType: string
    viewId: number
    viewName: string
    records: Record[]
    offset: number
    size: number
    date: string
    type: string
  }
  interface Record {
    record: {
      $id: string
      [key?: string]: Field
    }
  }
  interface Field {
    type: string
    value: unknown
  }
}

この他、本来はアプリのレコードに関する  d.ts ファイルも作成する必要がありますが今回は省略。
詳しいやり方は こちら  や こちら をご覧ください。
今回は  src/types/Record.d.ts と言うファイルを作った事にします。

あとは、tsconfig.jsond.ts ファイルの参照を追加してやります。
ついでにソースコード中のインポートで @/ を  src/ に置き換えるショートハンドも登録してやります。(好みの問題)

tsconfig.json
{
  "compilerOptions": {
    (省略)
+     "suppressImplicitAnyIndexErrors": true,
+     "paths": {
+       "@/*": ["src/*"]
+     }
  },
+   "files": [
+     "node_modules/@kintone/dts-gen/kintone.d.ts",
+     "src/types/kintone.d.ts",
+     "src/types/Record.d.ts"
+   ],
-   "include": [ "src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue" ],
+   "include": [
+     "src/**/*.ts",
+     "src/**/*.d.ts",
+     "src/**/*.tsx",
+     "src/**/*.vue",
+     "node_modules/@kintone/rest-api-client/**/*.d.ts"
  ],
  "references": [{ "path": "./tsconfig.node.json" }]
}

がっしがっし作る

ここからは kintone の流儀。
src/main.ts をエントリポイントとして作り込んでいきます。
とりあえずカスタマイズビューに <div id="container" />  と言うコンテナを配置し、そこに UI をぶち込んでいく形で作りましょう。

src/main.ts
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
createApp(App).mount('#app')
kintone.events.on('app.record.index.show', (event: kintone.Event) => {
  const mountTarget = document.getElementById('container')
  if (!mountTarget) return event
  createApp(App, { records: event.records }).mount(mountTarget)
})

あとはひたすら良い感じのビジュアルになるよう頑張ればよろしい。
表示要素単位でコンポーネントを作り込み、それに対しユーティリティクラスの積み上げでデザインを煮詰めていくのは積み木をやっているみたいな感覚に近いものがあります。
ユーティリティクラスと言う特性上、使いこなすには素の CSS の事は一通り知っておく必要があり、つまり TailwindCSS があれば CSS を知らなくても素敵なビジュアルを自由自在に組み上げられるぜと言う事にはなりません。
TailwindCSS は BulmaVuetify のような UI ツールキットではないので、コンポーネント的な使い方をするなら割と自力で頑張るか こちら で紹介されているようなところから程よいものを漁るとかになるでしょう。
自前で頑張ってもなんとかなると思いますけどね。

完成!

よしできたぜ!俺のこいつを見てくれ!!
03-invalid-all.png
ベネ。(良し)
良い感じにリストが並んで、その中にカードが配置されています。
カードをドラッグで別のリストに移動したりとかクリックでレコード詳細に遷移とかそう言うギミックが考えられますが、そう言うのはそう言う製品に任せればいいと思うので(ステマ)、とりあえず何時間かでやれるぶんとしてはいい塩梅じゃないですか?

・・・と思いきや

・・・あれ?なんか・・・カスタマイズビューの中は意図通りのビジュアルだけど・・・
ヘッダとか、なんか、おかしくない・・・?
04-invalid-header.png
アプリのタイトルのところとか、その上の黒い帯のボタンの並び方とか、一覧切り替えの隣のフィルタや集計ボタンもおかしいよね?

どころか、カスタマイズビューでない普通の一覧でもおかしいですね・・・?
05-invalid-list-view.png

なぜこんな事になっているのか

TailwindCSS で適用されるデフォルトの CSS はページ内の全要素に作用するようになっています。
デベロッパーツールで確認してみると、それぞれのタグに対して数多くのスタイルが設定されており、これが影響している事が分かります。
06-tailwindcss-styles.png

これはなんとかならんのか?

つまり、TailwindCSS のスタイルを全要素にではなくカスタマイズビューの枠内だけに収める事はできないのか?

ここまで長々と書いてきましたが、今回の記事の本題はまさにここから。

もう少し原因を深堀りする

TailwindCSS のスタートアップガイドに従って、src/style.css に以下のように記述しましたね。

src/style.css
@tailwind base;
@tailwind components;
@tailwind utilities;

そして、カスタマイズのエントリポイントとなる src/main.ts にはファイルの冒頭に以下のように記述されています。

src/main.ts
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'

これは kintone.events.on() イベントハンドラの外側に配置されているため、kintone のイベント処理とは無関係に実行される。
さらに言えば、アプリの一覧画面、詳細画面、編集画面など、JS カスタマイズが作用するあらゆる場面で実行される事になる。

要するにこれがいけない。
src/style.css の影響を局所化したいわけです。

実はすごく簡単です

原因と目的が分かれば、対処法は容易です。
スタイルを局所化すると言えば Vue の界隈では スコープ付き CSSと相場が決まっています。

なので、解決のためのアプローチとしては、main.ts でグローバルに インポートするのをやめ、個々のコンポーネントのスコープ付きの(style scoped な) CSS の中で src/style.css をインポートすると言う戦略となります。

src/main.ts
import { createApp } from 'vue'
- import './style.css'
import App from './App.vue'
src/App.vue
<style scoped>
@import url('@/style.css');
</style>

これによってグローバルな世界には影響を与えず、カスタマイズビュー(厳密には当該コンポーネント)の中だけに TailwindCSS のスタイルを適用できるようになるわけです。

注意点としては、スコープ付きスタイルは子コンポーネントには波及しないので、 TailwindCSS を使うすべてのコンポーネントで @import url('@/style.css'); する必要がある と言う点です。

これで、
07-valid-header.png
ヘッダ部のレイアウトも汝のあるべき姿に戻りました。

なお、一連のソース一式は こちら で公開しています。

まとめ

と言うわけで、複雑な命名規則や運用ルールを強いる事を余儀なくされる近年の  CSS 設計に一石を投じる  TailwindCSS を  kintone で利用するためのアプローチについて解説しました。
余談ながら今回取り上げた vite + Vue3 + TypeScript + TailwindCSS と言うスタックは現在絶賛開発中の弊社の新プラグインでも採用しているセットだったりします。
先日の Cybozu Circus 大阪でその一端をご覧いただいた方もいらっしゃいますかね!

kintone における JavaScript 開発は独特なお作法があったりして、新しい技術なんかは世の潮流から 1,2 歩遅れて入ってくる事がままあると感じます。
昨今は ブラウザでもモジュールが使えるようになってきている 事ですし、kintone のカスタマイズも  type="module" として読み込めますよ、と言う風になると良いなと願っております。

以上、お読み頂きありがとうございました!

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?