STUDY/VUE
[vue.js] veux 사용시 computed property 의 반복적인 get/set 선언을 줄이고 싶다 (using vuex-map-fields)
simongs
2020. 9. 19. 20:45
▣ 고민사항
- back-end 만 하다가 오랜만에 front-end 쪽을 보니까 신세계이다.
- 한 페이지 내에 아래와 같이 vuex와 동기화 처리할 변수(상태 변수)가 여러개 이다.
- 만약 name, age, city... 10개의 항목이 있다면 유사한 코드 블럭이 10번 반복된다.
- 이를 축약시켜줄 수 있는 라이브러리가 있을까?
- 유사한 고민 : Using get/set Computed property with mapState, mapMutation in Vuex
<div>
<input v-model="name"> // computed property (with vuex) 의 양방향 바인딩
<div>
<script>
computed: {
name: {
get() {
return this.$store.state.namespace_001.name;
},
set(value) {
this.$store.commit("namespace_001/name", value);
}
},
// 늘어납니다.
age: {
get() {},
set() {},
},
// 늘어납니다.
city : {
get() {},
set() {},
}
}
</script>
▣ computed property 에 대한 간단 정리
Define Style 및 특징
- sum() 은 함수이지만 sum 과 같은 속성(property) 접근 방식을 사용해도 정상 실행이 된다.
- (Define Style 1) 참고
- 그러한 이유로 Computed Property 라고 부른다.
- Vue 인스턴스의 모든 Option정보를 다루는 $options 를 통해 함수를 확인가능하다.
- computed라는 단어가 주는 느낌으로 계산되어진 정보를 가져오기만 할거 같은 느낌이지만 set메소드를 지정하면 쓰기 작업도 가능하다.
- (Define Style 2) 참고
즉 이 기능을 활용해서 특정 computed property 의 2-way(양방향) 바인딩 구현이 가능하다.
- 요 스타일로 여러 변수를 선언할 때 코드가 늘어지는 것을 해결해보고 싶었다.
- vuejs 공식사이트 - computed 속성의 setter 함수
- 가장 중요한 특징은 cache 가 되어서 내부 값이 변하기 전에는 캐싱된 값을 리턴합니다.
Sample Source
<div id="sample">
{{ sum }} 원
{{ amount }} 원
</div>
var vm = new Vue({
el : "#example",
data : { num : 0 },
computed : {
// (Define Style 1) vm.sum에 대한 getter 함수 정의
sum : function() {
var sum = 합계;
return sum;
},
// (Define Style 2) vm.amount 에 대한 getter/setter 함수 정의
amount : {
get : function() {
return this.num * 2;
},
set : function(amt) {
this.num = amt;
}
}
}
▣ 해결방안
- vuex-map-fields 라이브러리를 사용한다
vuex-map-fields 란
- Vuex Store에 저장된 feild 에 대해 양방향으로 데이터 바인딩을 가능하게 한다.
- 아래 Reference 에 정의되어 있는 github 주소를 통해서 자세한 내용을 확인 가능합니다.
- 사용법 중 활용한 일부케이스에 대한 설명만 적어봅니다.
namespace 로 구분된 vuex 에서 사용법
- 기본적으로는 모듈 내의 mutations, getters는 global namespace 하위에 등록이 된다.
- 모듈 간의 mutations, getters 의 네이밍 충돌을 각 모듈별로 "namespaced" 속성을 활성화 할수 있다.
Store (fooModule namespace)
- state에 정의된 field 들에 대한 getField(getter), updateField(setter)를 만들어준다.
import Vue from 'vue';
import Vuex from 'vuex';
import { getField, updateField } from 'vuex-map-fields';
Vue.use(Vuex);
export default new Vuex.Store({
// ...
modules: {
fooModule: {
namespaced: true,
state: {
foo: '',
test: '',
name: '',
},
getters: {
getField,
},
mutations: {
updateField,
},
},
},
});
Component (Style 1)
- createHelpers 를 통해서 fooModule namespace 의 getters, mustations 접근
<template>
<div id="app">
<input v-model="foo">
<input v-model="name">
</div>
</template>
<script>
import { createHelpers } from 'vuex-map-fields';
// `fooModule` is the name of the Vuex module.
const { mapFields } = createHelpers({
getterType: 'fooModule/getField',
mutationType: 'fooModule/updateField',
});
export default {
computed: {
...mapFields(['foo', 'name']),
},
};
</script>
Component (Style 2)
- 직접적인 명시를 통한 fooModule namespace 의 getters, mustations 접근
- 개인적으로는 요 스타일 선호.
<template>
<div id="app">
<input v-model="foo">
<input v-model="name">
</div>
</template>
<script>
import { mapFields } from 'vuex-map-fields';
export default {
computed: {
// `fooModule` is the name of the Vuex module.
...mapFields('fooModule', ['foo', 'name']),
},
};
</script>
혹시 틀린 부분이나 수정이 필요한 정보가 있다면 댓글남겨주시면 수정하도록 하겠습니다.