2017년 7월 4일 화요일

spectrum에게 물어보세요. Meteor+Vue (feat HMR) + custom NPM 적용하려면?

vue를 쓰는 meteor project를 만들자.
이번엔 Blaze를 안쓸거다.
HMR(Hot module replace)도 된다고 하더라.
뭐 별로 중요하겠냐만 이런 거 좋아하는 사람도 있다.
meteor create --bare vueEx1
--bare 옵션을 주면 Blaze뿐만 아니라 어짜피 안쓸 기본 패키지도 빼준다. (jquery/autopublish/insecure 등등)

vue를 쓰기 위해 npm과 package를 하나씩 추가하자.
meteor add akryum:vue-component
meteor npm i -S vue
시작점이 중요하다.
html 먼저 작성하자.
아마 처음이자 마지막으로 작성하는 html일 것이다.

client/main.html
<head>
</head>
<body>
<app />
</body>
<app>에서 시작한다. 다른 이름이어도 상관없지만 $mount 할때 똑같이 맞추자.

client/main.js
import { Meteor } from 'meteor/meteor';
import Vue from 'vue';
import App from '/imports/ui/App.vue';
Meteor.startup(() => {
  new Vue({
    render: h => h(App),
  }).$mount('app');
});
main.js가 custom component인 <app/>을 먼저 바인딩하게 하고
imports 영역으로 최초 시작 vue component인 App.vue를 작성한다.

imports/ui/App.vue
<template>
<div class="app">
  <navhead />
  <contents />
</div>
</template>
여기까지 필수요소 되겠다.

이후로는 component를 client안에만 이름을 맞춰서 잘 넣으면 별 설정없이 잘 돌아간다.
여기선 navhead랑 contents를 썼으니 빠르게 확인해보자.

client/navhead.vue

<template>
  <div>
    <h1>header</h1>
    <input v-model="login"/>
    <div>
      <span>{{login}}</span>
    </div>
  </div>
</template>
<script>
  export default {
    data() {
      return {
        login: "guest"
      }
    }
  }
</script>
navhead 먼저 만들고

client/contents.vue
<template>
  <div>
    <ul>
      <li v-for="item in items">
        {{item}}
      </li>
    </ul>
  </div>
</template>
<script>
  export default {
    data: ()=> ({
      items: [ 2, 4, 6, 8, 10]
    })
  }
</script>
contents도 이렇게.


결과가 잘 나오는가?
전체적인 연결관계는 그려보면 아래와 같다.
파일 구조와 연결관계

약간 섬세한 초반 작업이 필요하지만 그 뒤론 vue 파일만 마구 쓰면 되니 아주 편하다.
설정은 우리 삶을 좀먹고 코드를 만들고자 하는 의욕을 떨어뜨린다.

한번 다른 vue package도 추가해보자. 
meteor npm install vue-flickity --save
그냥 npm 해도 되지만 node 조차 깔기 귀찮을 수 있다. 그냥 Meteor 만 설치했다면 이렇게 하자.
Meteor는 기본적으로 atmosphere라는 package 저장소를 사용하지만 1.3부터 meteor-node-stubs 
client/contents.vue에 <album /> 을 하나 추가하고 
<template>
  <div>
    <ul>
      <li v-for="item in items">
        {{item}}
      </li>
    </ul>
 <album />
  </div>
</template>
<script>
  export default {
    data: ()=> ({
      items: [ 2, 4, 6, 8, 10]
    })
  }
</script>
client/album.vue 를 잽싸게 만들어보자.
예제를 보고 template은 그대로 script는 export default에 맞게 조금 수정했다.
<template>
<div>
<flickity ref="flickity" :options="flickityOptions">
<div class="carousel-cell">1</div>
<div class="carousel-cell">2</div>
<div class="carousel-cell">3</div>
<div class="carousel-cell">4</div>
<div class="carousel-cell">5</div>
</flickity>
<!-- if you don't want to use the buttons Flickity provides -->
<button @click="previous()">Custom Previous Button</button>
<button @click="next()">Custom Next Button</button>
</div>
</template>
<script>
import Flickity from 'vue-flickity';
export default {
 components: {
      Flickity
 },
    data () {
      return {
        flickityOptions: {
          initialIndex: 3,
          prevNextButtons: false,
          pageDots: false,
          wrapAround: true
          // any options from Flickity can be used
        }
      }
    },
    methods: {
      next() {
        this.$refs.flickity.next();
      },
      previous() {
        this.$refs.flickity.previous();
      }
    }
}
</script>
저장하고 재적용이 잘 안되면 meteor를 다시 실행하자.
아마 css 가 적용이 안되서 잘 모르겠지만 특별히 오류도 없고 잘 작동하는 듯 하다.

솔직한 심정은 Blaze 2에서 나왔어야 할 것들이 Vue에서 대부분 구현되지 않았나 싶은데
기다리다 x된다는 말이 있는데 과연 Blaze 2의 운명은 과연 (쩜쩜쩜)