emberjs学习一(环境和第一个例子)

code {
margin: 0;
padding: 0;
white-space: pre;
border: none;
background: transparent;
}

code, pre tt {
background-color: transparent;
border: none;
}
-->

emberjs学习一(环境和第一个例子)

博客:http://www.cnblogs.com/xiangbing/p/emberjs-test.html

案例:http://www.lovewebgames.com/emberjs/example/test1/index.html

源码:https://github.com/tianxiangbing/emberjs-test

准备工作

首页我们要做的是从网上下载下来emberjs相关的文件,目前在1.x的版本中,ember是要依赖jquery(v1.7.1~2.2.0) 和handlebars(v1.x)的,有一个比较好的方式来得到这些资源,那就是用bower.bower的安装很简单:

npm install -g bower   

装完bower就可以使用bower来下载资源了,比如我们要下载jquery 1.9.1版本的,我们可以这样写

bower install jquery#1.9.1

它会在当前目录下建一个bower_components的目录,把jquery的文件下载到jquery文件目录下.我们也可以把git上的项目发布到bower上,然后再从bower安装到本地,具体方法可以看这里http://blog.fens.me/nodejs-bower-intro/

我的案例已经发布在bower上,你可以直接用以下命令获取

bower install emberjs-test

点开bower_components下的test1目录,执行

npm install

安装grunt-watch,方便开发

再下载ember相关的资源

bower.json:

"dependencies": {
"jquery": "~1.9.1",
"ember": "~1.13.10",
"handlebars":"~1.3.0",
"ember-data":"~1.13.11"
}

执行 bower install 查看资源列表 bower list如图所示

开始一个例子

然后我们要根据这些文件做一个简单的例子,我们新建一个html页面index.html,引用js文件,代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>emberjs 初学记要</title>
  <script type="text/javascript" src="http://localhost:9090/livereload.js"></script>
</head>
<body>
  <script type="text/javascript" src="bower_components/jquery/jquery.min.js"></script>
  <script type="text/javascript" src="bower_components/handlebars/handlebars.min.js"></script>
  <script type="text/javascript" src="bower_components/ember/ember.min.js"></script>
</body>
</html> 

这里看到我在head里多引入了一个livereload.js,这个文件是使用grunt watch时生成的,可以动态的改变页面,你可以使用chrome的插件livereload.

好了,准备工作做好了之后 ,我们就可以编写首页的模板了,在页面添加如下代码:

<script type="text/x-handlebars">
首页
</script>

然后新建一个js文件,这里命名为app.js,放在js目录下,引入.

window.App = Ember.Application.create();

这时会发现页面上什么都没有,然后在console里还报了一个错:

Uncaught Error: Cannot call `compile` without the template compiler loaded. Please load `ember-template-compiler.js` prior to calling `compile`.

这个错误的大致意思好像是说我没有ember-template-compiler.js文件,但奇了怪的是,所以解析模板时出错了;但是在更早期的版本里是不需要的,可能是后期为了移除handlebars所以独立出来了吧,在bower_components/ember中还真有这么个文件,好吧,引入它.

<script type="text/javascript" src="bower_components/jquery/jquery.min.js"></script>
<script type="text/javascript" src="bower_components/handlebars/handlebars.min.js"></script>
<script type="text/javascript" src="bower_components/ember/ember-template-compiler.js"></script>
<script type="text/javascript" src="bower_components/ember/ember.min.js"></script>
<script type="text/javascript" src="js/app.js"></script>

好了,页面终于出现内容了,然后我们在模板中加入outlet,这是一个其他模块的入口,然后我们在app.js中加入其他的模块路由.

App.Router.map(function(){
    this.resource(‘add‘,{path:‘add‘});
});

这里可以使用resource也可以直接用route

this.route(‘add‘)

据说它们间的区别就是resource可以有子路由,但route已经是最小的路由了,所以可以理解为,大的类目时用recource,终极时用route,你开心就好。 然后我们添加add的模板,这里add的模板id要和link-to的参数一致,并且跟resource或route的第一个参数一样,path参数指的是url里的hash值.

<script type="text/x-handlebars" id="add">
<table>
<tr>
  <td>标题</td>
  <td><input type="text" /></td>
</tr>
<tr>
  <td>内容</td>
  <td><textarea></textarea></td>
</tr>
<tr>
  <td></td>
  <td><input type="button" value="确认"></td>
</tr>
</table>
</script>

这里有一个操作,所以应该有个Controller,我们先在app.js里加上这个controller.

App.AddController=Ember.Controller.extend({
    actions:{
        new:function(){
            console.log(‘new‘)
            var title = $(‘#title‘).val();
            var content = $(‘#content‘).val();
            //do save
        }
    }
});

它的命名规则是单词首字母大写,然后ember会把它注册到add这个模块里,这样你就可以在add模板中使用这个控制器了。把按钮改掉

<input type="button" value="确认" {{action "new"}}>

由于我们现在还没有使用localStorage这类本地存储,所以我们可以使用变量来临时的保存一下,但实际的应用中,似乎更多的是与服务器的交互。 它义数组var data=[];然后再action/new下加入

data.push({title:title,content:content});
this.transitionToRoute(‘index‘);

transitionToRoute的意思就是转到另一个路由去。我们这里转到首页去,然后我们看到一片空白的首页,还不知道是什么情况,因为还没有写代码让刚才加入的记录显示出来,好,我们在首页的模板中把记录each出来.

<script type="text/x-handlebars" id="index">
    <ol>
        {{#each model}}
        <li>{{title}}</li>
        {{/each}}
    </ol>
</script>

这里each的是model,如果直接each .会导致不更新。然后我们添加一个IndexRoute来返回一个数据集.

App.IndexRoute = Ember.Route.extend({
    model: function() {
        return data
    }
});

这里的IndexRoute就会默认在路由到index首页时调用。

我们在添加一个查看的按钮查看详情,为了能找到当前记录,我们给每条记录加一个index属性记录当前索引:

data.push({
    title: title,
    content: content,
    index:data.length
});
<li>{{title}}{{#link-to "info" this}}查看{{/link-to}}</li>

不知道为什么在这里不能直接取@index索引,所以只能我们在每条记录里加个唯一值。然后在路由中加上resouce

this.resource(‘index‘, {
    path: ‘/‘
}, function() {
    this.resource(‘info‘, {
        path: ‘info/:index‘
    })
})

这里我们把info的路由放在了index的下面,这样,就可以达到index和info同时存在了。这里要在index的模板中加上一个{{outlet}}才行。

{{model.title}}

{{model.content}}

同样的情况出现了,我要加上model才行,不明白的为什么{{title}}可以取出来content一定要{{model.content}}才行。在InfoRoute中返回当前的数组值

App.InfoRoute = Ember.Route.extend({
    model: function(param) {
        console.log(data[param.index].content)
        return data[param.index]
    }
});

在详情模板中加入编辑按钮,我们根据一个值来判断是显示“编辑”还是“保存”.

<div style="float:left;">
<h1>{{model.title}}</h1>
<p>
{{model.content}}
</p>
{{#if isEditing}}
  <button {{action "save"}}>保存</button>
{{else}}
  <button {{action "edit"}}>编辑</button>
{{/if}}
</div>

这里的isEditing是在Controlleraction中新增的一个属性,在actions中新增save和edit方法

App.InfoController = Ember.ObjectController.extend({
    actions: {
        isEditing: false,
        edit: function() {
            this.set(‘isEditing‘,true);
        },
        save:function(){
            this.set(‘isEditing‘,false);
        }
    }
});

这里的值都要使用set方法来改变,否则不会导致页面的刷新模板.然后我们加入一个子模板来显示编辑的内容。

<script type="text/x-handlebars" id="post/edit">
  <table>
    <tr>
      <td>标题</td>
      <td>{{input type="text" value=title}}</td>
    </tr>
    <tr>
      <td>内容</td>
      <td>{{textarea value=model.content}}</td>
    </tr>
    <tr>
      <td></td>
      <td><input type="button" value="确认" {{action "save"}}></td>
    </tr>
  </table>
  </script>

这里用的是{{input}}方式的控件,这样就可以实现传说中的双向绑定了,它是基于Ember.TextField类的,所以也是可以自定义一个Input控件的。如下官方例子所示input api:

Todos.EditTodoView = Ember.TextField.extend({
    didInsertElement: function() {
        this.$().focus();
    }
});
Ember.Handlebars.helper(‘edit-todo‘, Todos.EditTodoView);

然后就莫名其妙地完成了保存操作,好吧,我以为我还要save一下,官方的例子是使用的model,所以要对先this.get(‘model).save()一下才会有用。

最后我们要做的就是删除了,先加一个action为del的链接按钮:

{{#each model}}
<li>{{title}}{{#link-to "info" this}}查看{{/link-to}} <a href="javascript:void(0)" {{action "del" index}}>删除</a></li>
{{/each}}

然后在IndexController中新增del的action

App.IndexController = Ember.ArrayController.extend({
    actions: {
        del: function(index) {
            var d = this.get(‘model‘);
            console.log(d)
            var obj = d.findBy(‘index‘,index)
            d.removeObject(obj)
        }
    }
});

这里的问题是,我们要对model进行操作,才能实时的反应到页面上,看了下model的方法,可以使用findBy找出该元素,再进行removeObject

最终,我们就完成了一整个的增删改操作了。因为我也是在学习过程中,如果有什么不对的地方,请指出纠正吧!也欢迎加入我的Q群77813547

时间: 2024-10-11 12:18:27

emberjs学习一(环境和第一个例子)的相关文章

Hibernate学习的第一个例子

这是本人学习Hibernate的第一个例子: 1,先配置数据库驱动,点击myeclipse的myeclipse Database Explore,选择相应的数据库,进行配置,完成之后退出. 2,给项目添加支持Hibernate的功能,使用到的jar包有如下的: 进行了一系列的选项之后,到sessionfactory这一项,不选.然后点击完成. 3,配置表到java类的映射.点击右上方的myeclipse中的myeclipse database expore视图,选择数据驱动,选中要操作的表,然后

【转】APUE学习1:迈出第一步,编译myls.c

原文网址:http://blog.csdn.net/sddzycnqjn/article/details/7252444 注:以下写作风格均学习自潘云登前辈 /******************************************************************/   By:             聂强 Date:          2012-2-12 Email:         [email protected] Homepage: http://blog.c

深度学习服务器环境配置: Ubuntu17.04+Nvidia GTX 1080+CUDA 9.0+cuDNN 7.0+TensorFlow 1.3

本文来源地址:http://www.52nlp.cn/tag/cuda-9-0 一年前,我配置了一套"深度学习服务器",并且写过两篇关于深度学习服务器环境配置的文章:<深度学习主机环境配置: Ubuntu16.04+Nvidia GTX 1080+CUDA8.0> 和 <深度学习主机环境配置: Ubuntu16.04+GeForce GTX 1080+TensorFlow> , 获得了很多关注和引用. 这一年来,深度学习的大潮继续,特别是前段时间,吴恩达(And

Python学习总结之路--第一周

前言:35岁了,工作10年了,一直很排斥学软件编程,但是IT这条路一路走来发现如果不改变这辈子就这样了! 学Python是意外,也是缘分.前年的时候就看到有同事学这,那时他刚参加工作不久.今年他从我们公司走了, 去了甲方,年薪27W.然后我却没有进入我主导的项目甲方.双重打击让我郁闷了很久很久.无意中点开领导发的一次腾讯的免费课堂学习,既然是Python. 学习资料的来源.免费课程很坑,直播的时间我老是有事,录播的视频老是不给更新.所以我就从网上买了一份Python的学习视频,这就开启了我想学P

Android中关于JNI 的学习(零)简单的例子,简单地入门

Android中JNI的作用,就是让Java能够去调用由C/C++实现的代码,为了实现这个功能,需要用到Anrdoid提供的NDK工具包,在这里不讲如何配置了,好麻烦,配置了好久... 本质上,Java去调用C/C++的代码其实就是去调用C/C++提供的方法,所以,第一步,我们要创建一个类,并且定义一个Native方法,如下: JniTest类: public class JniTest { public native String getTestString(); } 可以看到,在这个方法的前

深度学习主机环境配置: Ubuntu16.04 + GeForce GTX 1070 + CUDA8.0 + cuDNN5.1 + TensorFlow

深度学习主机环境配置: Ubuntu16.04 + GeForce GTX 1070 + CUDA8.0 + cuDNN5.1 + TensorFlow 最近在公司做深度学习相关的学习和实验,原来一直在自己的电脑上安装虚拟机跑,速度实在太慢,主机本身性能太弱,独显都没有,物理安装Ubuntu也没多大意义,所以考虑用公司性能最强悍的游戏主机(i7 6700+GTX 1070) 做实验,这台主机平时是用来跑HTC VIVE的,现在归我用了o(*≧▽≦)ツ. 原本以为整个一套安装下来会很顺利,一路火花

(转)深度学习主机环境配置: Ubuntu16.04+Nvidia GTX 1080+CUDA8.0

深度学习主机环境配置: Ubuntu16.04+Nvidia GTX 1080+CUDA8.0 发表于2016年07月15号由52nlp 接上文<深度学习主机攒机小记>,这台GTX1080主机准备好之后,就是配置深度学习环境了,这里选择了比较熟悉Ubuntu系统,不过是最新的16.04版本,另外在Nvidia GTX1080的基础上安装相关GPU驱动,外加CUDA8.0,因为都比较新,所以踩了很多坑. 1. 安装Ubuntu16.04 不考虑双系统,直接安装 Ubuntu16.04,从ubun

JAVA基础学习——1.环境搭建 (系统Win10,64bit)

大学里虽然老师教过JAVA,但我没学.后来工作了,断断续续的也碰到了JAVA的项目,都是拉过来就干的节奏.也没有好好系统的学习一下. 从今天开始系统学习整理一下JAVA,以后再碰到JAVA项目的时候,心里也踏实点.咱也是正儿八经学过的人了,发发博客就当做个笔记,有不足的地方,大家多多指点,先在这里谢谢大家. 1.  环境搭建第一步,安装JDK. JDK 是Java开发工具包 (Java Development Kit ) 的缩写. JDK 是一系列工具的集合,这些工具是编译Java源码.运行Ja

Spring学习笔记--环境搭建和初步理解IOC

Spring框架是一个轻量级的框架,不依赖容器就能够运行,像重量级的框架EJB框架就必须运行在JBoss等支持EJB的容器中,核心思想是IOC,AOP,Spring能够协同Struts,hibernate等其他众多的框架. 下面介绍环境的搭建,首先下载好spring的完整包,我下载的是4.2.5版本的,在elipse中,新建java项目: 1,导入核心jar包: 2.在src目录下,创建spring配置文件(名称可以自定义,一般写为beans.xml): <?xml version="1.