methods: {
onScroll () {
const currentScrollPosition = window.pageYOffset || document.documentElement.scrollTop
if (currentScrollPosition < 0) {
return
}
// Stop executing this function if the difference between
// current scroll position and last scroll position is less than some offset
if (Math.abs(currentScrollPosition - this.lastScrollPosition) < 60) {
return
}
this.showNavbar = currentScrollPosition < this.lastScrollPosition
this.lastScrollPosition = currentScrollPosition
}
}
처음 HTML에는 화면에 필요한 정보가 담긴 js링크와 id 루트 정보만 들어있다. 그래서 처음에 화면을 볼 때에는 빈화면이 보여지게 된다. 그 다음 링크된 자바스크립트를 서버로부터 받아오게 되는데 화면 구성에 필요한 모든 정보를 서버로 부터 받아온다. 추가로 필요한 정보가 있을 때에는 서버에 요청하여 데이터를 다시 받아오고 화면을 구성해서 사용자에게 보여주게 된다.
단점
처음 사용자에게 화면을 보여주기까지 시간이 오래걸리고 나쁜 SEO <Search Engine Optimization>가 문제로 꼽힌다.
여기서 SEO란 구글이나 네이버 같은 검색엔진들은 사이트를 돌아다니면서 타이틀과 Description을 분석하며 검색 엔진에 등록하는데 CSR로 작성된 HTML은 텅텅 비어있어서 웹페이지를 분석하는데 어려움이 있다.
흐름
1) 사이트에 접속
2) 서버에게서 인덱스 파일을 받아옴. (하지만 비어져 있기 때문에 사용자에게 아무것도 보이지가 않는다.)
3) 웹페이지에 필요한 모든 로직이 담겨 있는 자바스크립트를 서버에 요청.
4) 동적으로 웹사이트를 구성하고 나면 드디어 사용자가 화면을 볼 수 있고 클릭이 가능하게 된다.
TTV <Time To View>와 TTI <Time To Interaction> 이 동시에 가능하다.
2. SSR <Server Side Rendering>
CSR방식처럼 서버에서 필요한 정보를 가져와서 Client에서 구성하는 것이 아니라 이미 Server에서 완성된 형태의 HTML파일을 동적으로 제어할 수 있는 약간의 소스코드와 함께 Client에 보내주게 된다.
첫 번째 페이지의 로딩이 빨라지는 장점이 있고 모든 컨텐츠가 HTML에 담겨져 있기 때문에 효율적인 SEO가 가능하다.
단점
사용자가 클릭을 했을 때에 전체적인 웹사이트를 다시 받아오는 것이기 때문에 깜빡임 ISSUE가 발생한다. 그리고 서버에 과부하가 걸리기가 쉽다. 하지만 가장 치명적인 단점으로는 사용자가 빠르게 웹 사이트를 받아볼 수 있지만 동적으로 데이터를 처리하는 자바스크립트를 아직 다운로드 받지 못해서 이리저리 클릭했는데 반응이 없는 경우가 있을 수 있다.
흐름
1) 서버에서 이미 만들어진 인덱스 파일을 받아옴. (하지만 아직 동적으로 화면을 처리할 수 있는 JS를 받아오지 못했으므로 사용자가 화면을 클릭해도 처리할 수 없음.)
asyncData (비동기, blocking) [또는 완전히 정적 페이로드(static payload)가 로딩된 후]
beforeCreate & created (Vue 라이크사이클 메서드)
fetch (동기, non-blocking)
beforeMount & mounted
중간 정리
Nuxt의 라이프사이클은 전체적으로 서버와 클라이언트 사이드 부분으로 나뉘는 것을 알 수 있으며 각 부분에서 공통으로 실행되는 과정과 아닌 과정이 있음을 확인할 수 있었습니다. 또한, 정적 (Static) 어플리케이션를 만드느냐 SSR(Server-Side Rendering) 또는 유니버설(universal) 어플리케이션을 만드느냐에 따라 과정이 달라질 수 있다.
추가 공부
라이프사이클에 대해 공부 및 정리하면서 봤던 내용들에 대해 정리해보자.
1. 활용도가 높고 중요한 몇가지 훅
middleware(클라이언트, 서버)
클라이언트와 서버 모두 훅들이 본격적으로 호출되기 전에, 앱을 만드는 과정에서 미리 정의해놓은 미들웨어들이 먼저 동작합니다.
만약serverMiddlware을 정의했다면 서버 사이드에서만 렌더링 과정에서 일반 미들웨어가 동작하기 전에 먼저 동작합니다.
asyncData(서버 or 클라이언트)
서버 혹은 클라이언트 사이드에서 생명주기 통틀어 한번씩만 호출되는 훅입니다.
Vue 인스턴스의 생명주기 이전에 먼저 데이터를 가져와서 렌더링을 하고싶은 경우에 사용합니다.
컴포넌트를 로드하기 전에 항상 호출되며, 페이지 컴포넌트의 경우에만 사용할 수 있습니다.
asyncData 의 리턴값은 Vue 인스턴스의 data()와 병합됩니다.
beforeCreated, created(클라이언트, 서버)
서버에서 새로운 vue 인스턴스를 생성한 뒤 이를 프리랜더링 하기 때문에 Vue 인스턴스가 서버에서 만들어지는 시점에 created와 beforeCreated 훅이 호출됩니다.
또한 클라이언트에서도 인스턴스를 만들고 $mount 메소드가 호출될 때도 따라 호출됩니다.
즉 서버와 클라이언트 사이드 양쪽에서 호출되는 훅입니다.
beforeMount 이후의 훅(클라이언트)
클라이언트 사이드에서만, 하이드레이션 이후에 $mount 메소드가 호출될 때 나머지 Vue 생명주기 훅들이 실행됩니다.
출처: 참조 1
2. Nuxt.js 비동기 데이터
Nuxt.js에는 비동기데이터 가져 오기를 위해 설계된 3가지 후크가 있다.
nuxtServerInit : 모든 페이지에 호출되는 VueX 저장소를 미리 채우는 데 사용됩니다.
fetch : 페이지 내에서 호출 된 데이터로 VueX 저장소를 미리 채우는 데 사용됩니다.
asyncData : data 페이지의 객체와 동기 데이터를 병합하는데 사용된다. 비동기 방식으로 미들웨어를 사용할 수도 있습니다. 즉, 미들웨어를 사용하여 VueX 저장소를 채울 수 있습니다.
Nust-SSR
asyncData 메소드
서버사이드에서 데이터를 가져와서 렌더링하고 싶을 때, 이때 pages 컴포넌트를 로딩하기 전에 매번 호출되는asyncData를 사용하면 된다.
pages 컴포넌트에서만 지원됩니다.
서버사이드에서 호출되거나 페이지를 이동할 때 호출
컴포넌트가 렌더링 되기 전에 호출
context 인자 사용가능
nuxt.js가컴포넌트와 데이터를 자동으로 Merge하는 개념
!! asyncData 안에서 컴포넌트가 생성되기 전에 호출하기 때문에 this를 엑세스할 수 없다.
출처: 참조 2
3. asyncData()와 fetch()
두 개의 Hook은 매우 비슷해보이면서 목적은 다름
공통점: 컴포넌트가 로딩되기 전에 매번 호출된다는 점, 첫번째 인자로 컨텍스트 오브젝트를 받음
asyncData
페이지가 로딩되기전에 컴포넌트 data를 미리 지정하여 컴포넌트가 생성될 때 병합하는데에 목적
즉 필요한 컴포넌트의 data를 컴포넌트가 랜더링 되기 전에 미리 지정하여 보다 빠르게 구성한다는 데에 목적을 둠
fetch
주로 비동기 작업을 작성하게 되면, 컴포넌트가 랜더링 되기전에 이 작업을 기다리게 됨
예를들어 유저정보가 필요한 컴포넌트라면fetch부분에 유저정보를 가져오는 API를 호출하여 활용할 수 있음
요약하자면,
asyncData는 컴포넌트가 랜더링 되기전에컴포넌트 데이터를 구성하는데에 목적이 있고,
fetch는 컴포넌트가 랜더링 되기전에비동기로직을 호출하는데에 목적이 있다고 생각합니다!
4. Vue LifeCycle (CSR)
beforecreate()
인스턴스가 생성되기 전이다. 데이터에 접근할 수 없다.
created()
인스턴스가 생성된 후 동기적으로 호출된다. 이 단계에서는 data, computed속성 methods, watch/이벤트 콜백 등의 설정이 준비된다. 그러나 $el 속성을 사용할 수 없다. DOM을 제외한 초기 데이터 설정이 완료된 상태이다.
컴포넌트 초기에 외부에서 부여받은 설정에 의해 기본 값 변경이 필요하다면 이 단계에서 변경하는 것이 좋다.
beforeMount()
div태그 #app의 html 코드가 생성되기 전 상태
Mounted()
인스턴스가 마운트 된 후 호출된다. DOM이 생성되긴 했지만 모든 자식 컴포넌트가 마운트되었음을 보장하지는 않는다. 그래서 mounted $nextTick을 사용하기도 한다.
Vue에 강력한 기능이다. Component를 적절히 사용하면 화면을 구조화하여 일괄적인 패턴으로 개발 할 수 있으며, 코드를 쉽게 이해하고 재사용이 용이해진다.
컴포넌트 분류 방법
이름 규칙
kebab-case와 pascal-case로 정의한다.
Props
페이지에서 컴포넌트로 데이터를 전달할 때, props를 사용한다.
정적 전달
<blog-post title="Vue와 떠나는 여행"></blog-post>
동적 전달
<!-- 변수의 값을 동적으로 할당 -->
<blog-post :title="post.title"></blog-post>
<!-- 복잡한 표현을 포함한 동적 값 할당 -->
<blog-post :title="post.title + ' by ' + post.author.name"></blog-post>