使用node+vue.js实现SPA应用


Posted in Javascript onJanuary 28, 2016

业务需求

最近公司要求开发web版的app,由于app是偏向内容方面,而且带了一个聊天模块,所以一般的多页开发不是很适合,而且主要是手机浏览,对加载速度或者用户体验来说都比较苛刻。调研了很多框架和模式,最后自己东拼西凑搞出来了这么一个玩意。

服务端

毫无疑问使用node,使用typescript可以有效的在编码同时查错,强类型语言写服务端毫无压力。

#app.ts 只贴重要代码

var webpack = require('webpack')
var webpackDevMiddleware = require('webpack-dev-middleware')
var WebpackConfig = require('./webpack.config')

import * as index from "./routes/index";
import * as foo from "./routes/foo";
import * as bar from "./routes/bar";

var app = express();

//启动服务的时候 打包并监听客户端用到的文件,webpackDevMiddleware是开发模式,他会打包js在内存里面,你改了文件,它也会重新打包
app.use(webpackDevMiddleware(webpack(WebpackConfig), {
  publicPath: '/__build__/',
  stats: {
    colors: true
  }
}));

//一般的配置项
app.set('views', __dirname + '/views');
app.set('view engine', 'ejs');
app.set('view options', { layout: false });
app.use(bodyParser.urlencoded({ extended: true }));
app.use(bodyParser.json());
app.use(methodOverride());
app.use(express.static(__dirname + '/public'));

var env = process.env.NODE_ENV || 'development';
if (env === 'development') {
  app.use(errorHandler());
}

//路由配置
app.get('/', index.index);
app.get('/foo', foo.index);
app.get('/bar', bar.index);


app.listen(3000, function(){
  console.log("Demo Express server listening on port %d in %s mode", 3000, app.settings.env);
});

export var App = app;

服务端渲染页面

#index.ts
import express = require("express")
import vueServer = require("vue-server") //服务端渲染vue的插件

var Vue = new vueServer.renderer(); //创建一个服务端的vue

export function index(req: express.Request, res: express.Response) {

  //创建一个组件
  var vm = new Vue({
    template: `
    <p>This is index!</p>
    `
  });

  //等待html渲染完成,再返回给浏览器 vueServer.htmlReady是vue-server的自带事件
  vm.$on('vueServer.htmlReady', function(html:string) {
    //这里用的是ejs模板 可以把需要用到的数据设置成window下的全局变量,方便客户端的js访问。
    res.render('layout',{server_html:html,server_data:'window.cm_data = {name:"张三"}'})
  });

};
#layout.ejs 访问这个SPA的所有url返回的都是这个页面 <meta>标签都可以动态设置,只要传参数进来就可以
<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="utf-8">
  <title>Vue Router Example</title>
  <style>
    .v-link-active {
      color: red;
    }
  </style>
  <script>
    //定义一些前端需要用到的全局属性,文章ID或用户信息什么的
    //index.ts中传过来的是 window.cm_data = {name:"张三"}
    //前端就能访问到了
    <%-server_data%>
  </script>
</head>
<body>

//这里的id是前端需要用到的一个标识
<div id="app">
  <h1>Hello App!</h1>
  <p>
    <a v-link="{ path: '/foo' }">Go to Foo</a>
    <a v-link="{ path: '/bar' }">Go to Bar</a>
  </p>
  //router-view是客户端vue-router需要解析的dom
  //server_html是根据访问url地址生成的html,是做SEO的重点,不加载下面的app.js也可以看到内容
  <router-view> <%-server_html%> </router-view>
</div>
//webpack打包好的js,主要是路由配置
<script src="/__build__/app.js"></script>
</body>
</html>

客户端

#app.js 这个是/__build__/app.js,可以用es6编写,webpack会转换的

import Vue from './vue.min' //客户端的vue.js
import VueRouter from './vue-router.min' //vue的路由插件,配合webpack可以很简单实现懒加载

//懒加载路由 只有访问这个路由才会加载js
import Foo from 'bundle?lazy!../../components/foo' //配合webpack的bundle-loader,轻松实现懒加载
import Bar from 'bundle?lazy!../../components/bar'
import Index from 'bundle?lazy!../../components/index'

var App = Vue.extend({})

Vue.use(VueRouter)

var router = new VueRouter({
  //这里要好好说一下,一定要设置html5模式,不然前后端URL不统一会发生问题
  //比如访问 http://localhost:3000/ 服务端定义是访问index.ts这个路由文件
  //如果不是html5模式的话,经过客户端js运行之后会变成http://localhost:3000/#!/
  
  //在比如直接浏览器输入 http://localhost:3000/foo 服务端定义是访问.ts这个路由文件
  //如果不是html5模式的话,经过客户端js运行之后会变成 http://localhost:3000/foo/#!/
  
  //设置了html5模式后,加载完js后不会加上#!这2个类似锚点的字符,实现前后端路由统一如果用户刷新浏览器的话,服务端也能渲染出相应的页面。
  history: true, //html5模式 去掉锚点 
  saveScrollPosition: true //记住页面的滚动位置 html5模式适用
})

//定义路由,要和服务端路由路径定义的一样
router.map({
  '/'  : {
    component: Index //前端路由定义,
  },
  '/foo': {
    component: Foo
  },
  '/bar': {
    component: Bar
  }
})

//启动APP
router.start(App, '#app')

需要完善的地方

前后端统一模板,已经找到方法了把html分离出来,node端用fs.readFileSync方法获取,客户端用webpack的raw-loader获取html内容

不放源码都是瞎扯。

源码地址

https://github.com/yjj5855/node-vue-server-webpack

Javascript 相关文章推荐
javascript 语法基础 想学习js的朋友可以看看
Dec 16 Javascript
Javascript学习笔记1 数据类型
Jan 11 Javascript
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
May 24 Javascript
jQuery隔行变色与普通JS写法的对比
Apr 21 Javascript
IE的事件传递-event.cancelBubble示例介绍
Jan 12 Javascript
$.extend 的一个小问题
Jun 18 Javascript
JavaScript实现多种排序算法
Feb 24 Javascript
Bootstrap每天必学之级联下拉菜单
Mar 27 Javascript
js replace(a,b)之替换字符串中所有指定字符的方法
Aug 17 Javascript
js构造函数创建对象是否加new问题
Jan 22 Javascript
JavaScript实现随机五位数验证码
Sep 27 Javascript
详解Node.js使用token进行认证的简单示例
May 25 Javascript
jQuery+css实现的tab切换标签(兼容各浏览器)
Jan 28 #Javascript
javascript实现随机显示星星特效
Jan 28 #Javascript
基于javascript实现全国省市二级联动下拉选择菜单
Jan 28 #Javascript
JS实现动态生成表格并提交表格数据向后端
Nov 25 #Javascript
jQuery+css实现的时钟效果(兼容各浏览器)
Jan 27 #Javascript
jQuery实现的分子运动小球碰撞效果
Jan 27 #Javascript
jQuery+css3实现转动的正方形效果(附demo源码下载)
Jan 27 #Javascript
You might like
php项目开发中用到的快速排序算法分析
2016/06/25 PHP
node.js 一个简单的页面输出实现代码
2012/03/07 Javascript
js中精确计算加法和减法示例
2014/03/28 Javascript
浅析node连接数据库(express+mysql)
2015/11/30 Javascript
js如何改变文章的字体大小
2016/01/08 Javascript
javascript表单事件处理方法详解
2016/05/15 Javascript
js学习阶段总结(必看篇)
2016/06/16 Javascript
AngularJS解决ng界面长表达式(ui-set)的方法分析
2016/11/07 Javascript
100行代码理解和分析vue2.0响应式架构
2017/03/09 Javascript
jquery select插件异步实时搜索实例代码
2017/10/20 jQuery
Vue 报错TypeError: this.$set is not a function 的解决方法
2018/12/17 Javascript
如何自定义微信小程序tabbar上边框的颜色
2019/07/09 Javascript
layui实现三级联动效果
2019/07/26 Javascript
在Angular中实现一个级联效果的下拉框的示例代码
2020/05/20 Javascript
vuex页面刷新导致数据丢失的解决方案
2020/12/10 Vue.js
python 回调函数和回调方法的实现分析
2016/03/23 Python
利用Python获取操作系统信息实例
2016/09/02 Python
Pycharm无法显示动态图片的解决方法
2018/10/28 Python
Python面向对象程序设计之私有属性及私有方法示例
2019/04/08 Python
python快速排序的实现及运行时间比较
2019/11/22 Python
pytorch自定义二值化网络层方式
2020/01/07 Python
Python Selenium安装及环境配置的实现
2020/03/17 Python
Python 在局部变量域中执行代码
2020/08/07 Python
css3中的calc函数浅析
2018/07/10 HTML / CSS
html5利用canvas绘画二级树形结构图的示例
2017/09/27 HTML / CSS
使用canvas压缩图片上传的方法示例
2020/02/07 HTML / CSS
马来西亚银饰品牌:JEOEL
2017/12/15 全球购物
教育学专业实习生的自我鉴定
2013/11/26 职场文书
初中地理教学反思
2014/01/11 职场文书
建房协议书
2014/04/11 职场文书
中职生求职信
2014/07/01 职场文书
白酒营销策划方案
2014/08/17 职场文书
简历自荐信范文
2015/03/09 职场文书
筑梦中国心得体会
2016/01/18 职场文书
Javascript中async与await的捕捉错误详解
2022/03/03 Javascript
python turtle绘制多边形和跳跃和改变速度特效
2022/03/16 Python