기본 콘텐츠로 건너뛰기

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의 운명은 과연 (쩜쩜쩜)

이 블로그의 인기 게시물

vulcanJS - 10. Posts Update/Delete

마지막으로 수정과 삭제를 구현해보면 목록 조회(List), 상세 조회, 쓰기, 수정, 삭제까지 모든 필요한 요소를 아우를 수 있을 것이다.
감이 좋은 분들은 눈치 챘을지도 모르겠지만 사실 수정이란 건 UI면에서 볼때 이미 양식이 채워져있는 신규 쓰기와 별반 다르지 않다.

먼저 해야할 것은 역시나 Component를 만드는 일이다.
$ vulcan g component
? Package name spectrum-simplebb
? Component name PostsEditComponent
? Component type Class Component
? Register component Yes
   create packages/spectrum-simplebb/lib/components/PostsEditComponent.jsx
 conflict packages/spectrum-simplebb/lib/components/index.js
? Overwrite packages/spectrum-simplebb/lib/components/index.js? overwrite
    force packages/spectrum-simplebb/lib/components/index.js PostsEditComponent를 만들었다.
route도 만들자. /posts/edit/:id 이렇게 경로를 만들면 좋겠다. 그러고보니 이전 글에서 만든 상세보기도 /posts/view/:id 형식으로 만들껄 그랬다.
$ vulcan g route
? Package name spectrum-simplebb
? Route name postsEdit
? Route path /posts/edit/:_id
? Component name PostsEditComponent
? Layout name
 conflict packages/spectrum-simplebb/lib/modules/routes.js
? Overwrite packages/spectrum-simplebb/lib/modules/routes.js? overwrite
   …

Troubleshooting - Meteor package가 적용이 되지 않을 때

버전 1.5 기준 package.js에서 Package.onUse 에 새 패키지를 추가했는데 인식하지 못하는 경우가 있다.
Package.onUse((api) => {
  api.use([
    'vulcan:core',
    'vulcan:forms',
    'vulcan:accounts' /* <-- 추가함! */
  ]);
...
} 내부패키지건 원격패키지건 안되는 안된다. 이럴 때 meteor add 후 meteor remove 해도 되지만 더 간단한 방법이 있다. meteor update vulcan:accounts 이렇게 update 해주는 방법이 있다. .meteor/package 파일을 건들지 않아서 좋다. 그래도 역시 좋지 않다. Meteor 스럽지 않다. https://github.com/meteor/meteor/issues/7721 현재 1.5.2에서도 해결이 안되었군요. 해결되어 적용되면 다시 글 올리겠습니다.

vulcanJS - 1. vulcan-cli 설치하고 프로젝트 만들기

vulcanJS를 학습하면서 알아낸 점을 기록하는 의미의 튜토리얼을 써본다.

Telescope도 그랬지만 forum 형태의 예제에서 embedly를 사용하는 URL 요소가 글에 들어가는 점이나 일반인들에게는 익숙하지 않은 Markdown 문법 등을 걷어내고 최소형태에서부터 접근해야할 필요성을 느껴서 단순 post만 목록 열람하고 읽고 쓰는 구현을 해보기로 했다.

먼저, vulcanJS를 github를 통해 clone 하고 meteor create --package를 통해 만들수도 있지만 Vulcan-Cli(https://github.com/VulcanJS/vulcanjs-cli)가 있어서 이걸 일단 사용해보기로 한다.

대충 내용을 읽어보니 Rails 생각이 나기도 하고 재밌어 보인다.
npm install -g vulcanjs-cli 혹은 meteor 에서만 한다면
meteor npm install -g vulcanjs-cli 이런 식으로 설치하고 쓰면 된다.
npm 에 global로 설치한 첫번째 경우 기준으로 진행해보겠다.
설치하고 난 뒤 vulcan 을 실행해보면
$ vulcan
vulcan usage:
Synopsis
  vulcan <action> <object> <...>
    <action>   Operation to perform
    <object>   Asset type (contextual to action)
    <...>      Parameters. If not provided, interactively entered
Project initialisation
  vulcan create <appName>
  vulcan init <appName>
Assets creation
  vulcan (generate|g) package <packageName>
  vulcan (generate|g) model <packageName>…