LoginSignup
2
2

More than 1 year has passed since last update.

フロントエンド開発: Webアプリケーションをコンポーネント化した話 - with Svelte

Last updated at Posted at 2022-12-01

なぜコンポーネント化しようと思ったのか

同じデザインのボタンがサイト上にたくさん出てくるのにもかかわらず、
それぞれのボタンに長いcssを割り当てていた。つまり冗長化しちゃってた、
(tailwindのあるある?)

<button class="ml-1 shadow bg-yellow-500 hover:bg-yellow-400 focus:shadow-outline focus:outline-none text-white font-bold rounded">
  ボタン①
</button>

<button class="ml-1 shadow bg-yellow-500 hover:bg-yellow-400 focus:shadow-outline focus:outline-none text-white font-bold rounded">
  ボタン②
</button>

<button class="ml-1 shadow bg-yellow-500 hover:bg-yellow-400 focus:shadow-outline focus:outline-none text-white font-bold rounded">
  ボタン③
</button>

そのためにデザインを変えたいときに全てのボタンで変える必要があり、とても手間だった。

共通部品化したい! => コンポーネント化してみよう!

開発の流れ

  1. とりあえず一つのページの部品(コンポーネント)を洗い出す
  2. それぞれのファイルを作成する
  3. 呼び出して使えるようにする
  4. 他のぺージでも使える部分へ適用する
  5. 他のページの部品(コンポーネント)を洗い出す
  6. 2へ戻る

(本来であれば画面設計の段階で考えるべきだったと後悔...)

実際に部品化してみる!

1. とりあえず一つのページの部品(コンポーネント)を洗い出す

  • ログイン画面をよく見てみる
  • 見えてきた!部品だ!
    image.png

ということでメールアドレス用フォーム、パスワード用、Submit用ボタンを作成することに

2. それぞれのファイルを作成する

(いろいろ省略)
- lib
   |- Login.svelte
   |- Parts
        |- Buttons
        |    |- Submit.svelte
        |- Forms
             |- Email.svelte
             |- Password.svelte

3. 呼び出して使えるようにする

ここで気が付く。
あれ、、ボタンデザインは使いまわしたいけど、テキストは毎回違うな...

ということでテキストやプレースホルダーは呼び出すときに渡して変数としよう!

Submit用ボタン

// Submit.svelte
<script lang="ts">
  export let text: string = '決定!';
</script>

<button
  class="w-full h-full shadow mx-auto bg-yellow-500 hover:bg-yellow-400 focus:shadow-outline focus:outline-none hover:border-yellow-400 text-white font-bold py-2 px-4 rounded"
  type="submit">{text}
</button>
// Login.svelte
<script lang="ts">
  import SubmitButton from './Parts/Buttons/Submit.svelte'
</script>

<SubmitButton text='ログイン' />

export letで定義することで親から渡された変数を受け取れるんですね~!
便利ですね!初期値は決定!にしました。

あれ?
typeも変数にすればわざわざtypeごとにコンポーネントを分ける必要もないんじゃないか?

ダメでした。typeは変数にしてはいけないそうです。


EmailForm

emailのフォームは変数を渡すだけではなく、入力された値を親コンポーネントに渡す必要もあるので、dispatchを使って実現!
プレースホルダーも場合によっては親コンポーネントから受け取ってもいいかもしれません👍

// Email.svelte
<script lang="ts">
  import { createEventDispatcher } from 'svelte';
  export let value = '';

  const dispatch = createEventDispatcher();
  $: dispatch('updateChild', { value: value });
</script>

<input
  class="bg-gray-200 appearance-none border-2 border-gray-200 rounded w-full py-2 px-4 text-gray-700 leading-tight focus:outline-none focus:bg-white focus:border-purple-500"
  id="inline-full-name"
  placeholder="emailを入力..."
  name='email'
  type='email'
  required
  bind:value="{value}"
/>
<script lang="ts">
  import EmailForm from "../Parts/EmailForm.svelte"
  let email: string = '';

  const updateValue = (key, value) => {
    key = event.detail.value
  }
</script>

<EmailForm
  value={email}
  on:updateChild={(event) => {updateValue(email, event.detail.value)}}
/>

passwordのフォームも同じようにtype=passwordで同じように作りました!

4. 他のぺージでも使える部分へ適用する

パスワードを忘れた場合のメール送信画面の部品も洗い出したところ、、、

image.png

先ほど作ったやつそのまま適用できるじやないか!
コードが半分になりました!

こんな感じでコンポーネント化をしたお話しでした。

2
2
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
2
2