3
2

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

同じコンポーネントでStoreを使って状態を共有できるか確かめてみた話[Vue]

Last updated at Posted at 2019-12-01

はじめに

兄弟コンポーネント間での値のやり取りにはVuex自作のstoreを使えば、propsemitでバケツリレーをせずにやることが可能です。
ふと疑問に思ったのが同じコンポーネントを複数使い回して、その間でstoreを使えば状態の共有ができるのか?でした。
疑問に思ったらやってみよう!ということでやったみたのが今回の記事となります。

実験

まずは準備

今回使うコンポーネントとstoreモジュールを作っていきましょう。
最初にStoreの方から作っていきましょう。

store.js
export const Store = {
  state: {
    StoreValue: "Qiita"
  },
  setStore(value) {
    this.state.StoreValue= value;
  },
  getStore(){
    return this.state.StoreValue;
  }
};

状態を持つのはstateの部分で、実際に状態を取得したり設定したりするのもstoreにアクションを作っています。

次にVueのコンポーネントを作っていきます。
まずはHTML部分からです。

component.vue
<template>
  <div>
    <p>{{ ComponentValue }}</p>
    <button @click="onButtomClick">Click</button>
  </div>
</template>

結果を表示するためのPタグと、イベントを発火するためのButtonタグを配置しました。
次にScript部分。

component.vue
<script>
import { Store } from "./store.js";
export default {
  data() {
    return {
      StoreObject: Store.state,
    };
  },
  computed: {
    ComponentValue: {
      set(value) {
        Store.setStore(value);
      },
      get() {
        return Store.getStore();
      }
    }
  },
  methods: {
    onButtomClick(value) {
      this.ComponentValue = `${this.ComponentValue}!`;
    }
  }
};
</script>

全体としてはこのような感じになりました。
重要そうなとこを細かく見ていきましょう。

component.vue
  data() {
    return {
      StoreObject: Store.state,
    };
  },

dataオブジェクトで、読み込んだStoreの状態を持たせています。
ここでStoreをもたせておかないと、うまく更新されたものを反映できませんでした。

component.vue
  computed: {
    ComponentValue: {
      set(value) {
        Store.setStore(value);
      },
      get() {
        return Store.getStore();
      }
    }
  }

算出プロパティのほうはgetter/setterstoreに対して状態を取得と設定を行うようにしています。

component.vue
  methods: {
    onButtomClick(value) {
      this.ComponentValue = `${this.ComponentValue}!`;
    }
  }

ボタンをクリックした時に行うメソッドです。
算出プロパティを介してstoreの値を取得し、それを!マークと文字列結合をして、再度storeのアクションで状態を設定しにいく。
という流れになっています。

あとは親のほうで、作ったコンポーネントを複数配置すれば完了です。

component.vue
<template>
  <div>
    <qiita></qiita>
    <qiita></qiita>
  </div>
</template>

やってみた

ezgif-3-f54e35782a1d.gif

実際に動かしてみると、クリックしていないほうのも!マークが増えているので、同じコンポーネントでもstoreを介して状態を共有できることがわかりました。

本当にstoreを介してるのか?
ただたんに同じコンポーネントだから状態が共有されてるだけなのでは?という疑問もあったので、
dataのインスタンス変数を使った場合にどうなるかというのをやってみましょう。

インスタンス変数に変えてみる

component.vue
  data() {
    return {
      StoreObject: Store.state,
      childvalue:"Vue"
    };
  },
component.vue
  computed: {
    ComponentValue: {
      set(value) {
        // Store.setStore(value);
        this.childvalue= value;
      },
      get() {
        // return Store.getStore();
        return this.childvalue;
      }
    }
  }

dataにインスタンス変数を追加し、算出プロパティでのgetter/setterもそちらに対しての処理を行うように変更します。
この状態でどのように動くかを見てみましょう。

ezgif-3-3a6877c3dc30.gif

連動しなくなりました。
ということは、同じコンポーネントを使い回していても、コンポーネントはそれぞれ独立したインスタンスになっている、ということがわかります。
なので、上のほうのではstoreを介して状態が共有されている、と言えますね!

まとめ

同じコンポーネントでもstoreを使えば状態を共有できることが解りました。
ただしこれを使うシチュエーション自体がかなり限られてきますが、これを知っておくと似た構成のコンポーネントを増やして対応…
というのがなくなるかと思います!

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?