动手实践用Gulp+Webpack构建纯前端React应用

最近构思了一个练手项目,应用react实现程序的逻辑,从而进一步学习React框架开发。这次的例子是一个纯前端的简单应用,模拟实现图片上传和展示功能。应用中,使用sessionStorage存储图片数据,并且将sessionStorage中存储的图片数据,以列表的形式展示出来。

目录结构

用Webpack打包React应用

Webpack 是一个前端资源加载/打包工具,只需要相对简单的配置就可以提供前端工程化需要的各种功能。之前的例子中,对react的模块化,采用的是requirejs的AMD方式,使用Webpack之后,我们就可以使用nodejs的commonjs方式编写模块化的js代码。通过webpack,会将所有依赖的资源打包成一个单独的bundle,包含应用的逻辑代码,还有依赖的模块。

如果使用jsx格式编写混编的js和html代码,还需要安装相应的loader,才能成功打包,使用的是babel-loader。这里的loader除了jsx之外,根据需要,还可以支持sass等其他格式。

安装相应的npm依赖包

npm install react --save-dev
npm install react-dom --save-dev
npm install webpack --save-dev
npm install babel-loader --save-dev

跟所有的应用程序一样,react也需要有一个入口点,webpack会从入口点开始,对所有的依赖文件进行打包,在webpack.config.js配置文件中,进行相应的配置;entry就是入口点

module.exports = {
    entry:[
        './app/main.js'
    ],
    output: {
        path: __dirname + '/assets/',
        publicPath: "/assets/",
        filename: '[name].bundle.js'
    },
    resolve:{},
    module: {
        loaders: [
            /*{
                test: /\.scss$/,
                loaders: ["style", "css", "sass"]
            },*/{
                test: /\.js$/,
                loaders: ['babel-loader'],
                include: __dirname + '/app/'
            },
            { test: /\.jsx?$/, loaders: ['babel','jsx?harmony']}
        ]
    },
    plugins:[]
};

使用Gulp构建项目

为了开发和调试方便,引入构建工具gulp,主要使用到的插件包括:gulp-connectgulp-sassgulp-webpackgulp-uglify。分别使用npm安装对应的模块,npm install gulp --save-dev

项目开发中,简单的实现了几个Task,进行项目打包和构建。

var connect = require('gulp-connect');
var gulpWebpack = require('gulp-webpack');
var bower = require('gulp-bower');
var uglify = require('gulp-uglify');
var sass = require('gulp-sass');
var webpackConfig = require('./webpack.config');

gulp.task('bower', function() {
    return bower('./bower_components')
        .pipe(gulp.dest('./static/lib/'));
});

gulp.task('easy_webpack',function(){
    gulp.src('./app/main.js')
        .pipe(gulpWebpack(webpackConfig))
        .pipe(uglify())
        .pipe(gulp.dest('./assets/scripts/'))
});

gulp.task('easy_sass', function () {
    return gulp.src(path.SASS + '/**/*.scss')
        .pipe(sass().on('error', sass.logError))
        .pipe(gulp.dest('./assets/styles/'));
});

gulp.task('easy_build',['bower','easy_webpack','easy_sass'], function () {});

gulp.task('staticserver', function() {
    connect.server({
        port:3333,
    });
});

React的组件

搭建好环境之后,react的使用非常方便,把html根据需要拆分成不同的模块就可以直接在jsx语法的代码中混合编写,ReactDom.render就可以把模版转换成html代码,并且把html模块代码嵌入到指定的dom对象中,<代表往后是html代码,当作html解析,{代笔往后是js代码,当作js解析,需要注意的是,因为class是关键字,所以需要把class替换成className,把for替换成htmlFor。

var React = require('react');
var ReactDOM = require('react-dom');

module.exports = React.createClass({
    handleClick:function(event){
        console.log(event.target);
    },
    render: function() {
        return (
            <div className="img-item">
                <div className="img-view" onClick="{this.handleClick}">
                    <img className="picture" src={this.props.imgData.file}/>
                </div>
                <div className="img-info">
                    <span className="img-name">{this.props.imgData.name}</span>
                    <span className="img-type">{this.props.imgData.format}</span>
                </div>
            </div>
        );
    }
});

......

#使用的示例
var imageCtrl = new ImageCtrl();
var imageDataStore = imageCtrl.findAll();
ReactDOM.render(
    <div className="img-content card-mode">
        <ul className="js-img-list u-clearfix">
            {
                imageDataStore.map(function (item,index) {
                    return <li key={item.id}><ImgListItemComponent imgData={item}/></li>
                })
            }
        </ul>
    </div>,
    document.getElementById('imageList')
);

在React中,通过React.createClass方法就可以生成一个组件,通过这个方法,我们可以封装自己的组件,并且像普通的html标签一样在网页中插入。界面交互的大多数场景实现,就是在组织不同的组件,实现组件的复用和状态控制。对于组件,我们可以像普通的html标签一样,设置属性,然后通过this.props就可以读取对应的属性,不同于html标签的是,jsx的属性,支持所有js的数据类型而不是普通字符串。

React组件的props可以通过定义变量的方式,在组建类的外部访问,var imgList = <ImgListItemComponent imgData={item}/>,然后读取imgList.props.imgData,但是官方建议不要在外部对props进行修改,因为这个是不可控的。

React创新性的将组件看成是一个状态机,页面渲染的时候,根据state属性的初始状态来决定如何渲染,在组件有互动需要的时候,不同的交互会导致state的改变,从而触发重新渲染UI。这样的实现机制和Jquery的事件绑定机制,设计理念是完全不一样的。其中getInitialState返回的object会被赋值作为初始化的state。在交互过程中,如果有需要改变状态的时候,执行this.setState({fileContent:"",fileStatus:false,fileEntity:null});,在传入的对象中,制定的state对应的值就会改变,从而导致render中根据对应的state值来渲染。

......

module.exports = React.createClass({
    getInitialState: function() {
        return {fileTip: "",fileName: "",fileStatus:false,fileEntity:null,clicked: false,fileContent:"",fileNameShow:""};
    },
    render:function(){
        if (this.state.clicked) {
            return (
                <div className="simple-modal upload-modal">
            ...省略...
                </div>
            );
        }else {
            return <div />;
        }
    }

.......

this.props和this.state都是组件类提供的属性,可以通过组件直接读取,他们的区别是,props是固定的属性,已经设定就不再变化,而state根据交互的需要会发生变化。

React 使用驼峰命名规范的方式给组件绑定事件处理器。如果需要绑定点击事件,直接设置onClick就可以,onClick="{this.handleClick}"绑定成功,如果用户点击onClick就会调用React.createClass创建的对象中的handleClick函数,结合setState就可以动态的改变组件的状态,进而更新界面。

开发这个应用的过程中,使用了mixin实现的弹窗组件,该组件通过在body的最后面添加一个div标签来实现弹窗;基本够用,接下来的学习中,还会深入的学习mixin和弹窗的实现。khan学院的React modal

开发过程

在开发的过程中,因为最终要将html代码拆分成不同的模块,实现整个功能的过程中,可能涉及多个场景,所以将不同的场景,分别以纯静态页面的方式实现,这样就可以快速的调整页面布局,拆分之后保留页面原型。

保存图片的时候,存储格式是ImageData类,该类扮演了Schema的作用;图片通过File读取之后,以base64形式保存在sessionStorage中,保存的过程抽象成专门的模块,这样以后需要实现异步上传的时候,可以直接重新实现该模块。该模块使用mocha进行单元测试。

/*图片存储数据结构Schema*/
ImageData = function(initData){
    this.name = "";
    this.id = "";
    this.desc = "";
    this.url = "";
    this.visit = "";
    this.createTime = null;
    this.format = null;
    this.file = "";
    this.type = "";

    var dataTypeList = ImageData.getParamList();
    if(typeof initData == "object"){
        for(var key in dataTypeList){
            var keyName = dataTypeList[key];
            if(initData[keyName]){
                this[keyName] = initData[keyName];
            }
        }
    }
};
ImageData.getParamList = function(){
    var dataTypeList = ["name","id","desc","url","visit","createTime","format","file","type"];
    return dataTypeList;
};

/*数据存储相关代码*/
var DataHandler = {
    get:function(key,callback){
        if(typeof sessionStorage != "object"){
            return [];
        }

        var dataStr = sessionStorage.getItem(key);
        var keyData = [];
        if(dataStr){
            keyData = JSON.parse(dataStr);
        }
        typeof callback == "function" && callback(keyData);
        return keyData;
    },
    //以下代码省略
    ......

预览地址:http://pages.liuwill.com/
项目代码:https://github.com/liuwill/react-webpack-startup

下载源代码之后,执行npm install安装需要的依赖包,然后按照顺序执行gulp任务,就可以启动静态服务器预览效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容

  • 在现在的前端开发中,前后端分离、模块化开发、版本控制、文件合并与压缩、mock数据等等一些原本后端的思想开始...
    Charlot阅读 5,433评论 1 32
  • GitChat技术杂谈 前言 本文较长,为了节省你的阅读时间,在文前列写作思路如下: 什么是 webpack,它要...
    萧玄辞阅读 12,681评论 7 110
  • 最近看了一本关于学习方法论的书,强调了记笔记和坚持的重要性。这几天也刚好在学习React,所以我打算每天坚持一篇R...
    gaoer1938阅读 1,672评论 0 5
  • 一、现代前端开发 1.1 ES6 —— 新一代的JavaScript标准 1.1.1 语法特性 const、let...
    杀破狼real阅读 1,833评论 0 3
  • 我发现秋叶大叔恰好每次都能在红利期赶上潮流,这应该和秋叶大叔自己前期的累积和善于观察有关,教主也提到过秋叶大叔对自...
    有钱YQ阅读 911评论 0 0