摘要:在我的機(jī)子上,運(yùn)行于端口,以避免和其他默認(rèn)運(yùn)行于端口的沖突。我們可以使用命令連接數(shù)據(jù)庫(kù)查看定義應(yīng)用層次創(chuàng)建的模板應(yīng)用有一個(gè)問題,客戶端和服務(wù)器段的代碼是一樣的。在中加入然后添加問題模板注意我們使用了來確保用戶未登錄的情況下應(yīng)用。
編者注:我們發(fā)現(xiàn)了有趣的一系列文章《30天學(xué)習(xí)30種新技術(shù)》,正在翻譯中,一天一篇更新,年終禮包。下面是第15天的內(nèi)容。
到目前為止我們討論了Bower、AngularJS、GruntJS和PhoneGap等JavaScript技術(shù)。今天是“30天學(xué)習(xí)30種新技術(shù)”挑戰(zhàn)的第15天,我決定重返JavaScript,學(xué)習(xí)Meteor框架。雖然Meteor的文檔相當(dāng)好,但是它缺少為初學(xué)者準(zhǔn)備的教程。我覺得教程的學(xué)習(xí)效果更好,因?yàn)榻坛炭梢詭椭憧焖偕鲜忠环N技術(shù)。本文將介紹如何利用 Meteor 框架構(gòu)建一個(gè)epoll應(yīng)用。
Meteor是新一代的開發(fā)即時(shí)web應(yīng)用的開源框架,它能幫助你在最少的時(shí)間內(nèi)完成開發(fā)。它的理念和AngularJS、BackboneJS等框架大不相同。當(dāng)我們?cè)赽ackbone 和 angular 上工作時(shí),客戶端(Angular或Backbone)和REST后端通訊。我們可以用任何技術(shù)寫 REST 后端,例如 Java、NodeJS、PHP。
Meteor使用DDP(分布式數(shù)據(jù)協(xié)議)在客戶端和服務(wù)器間傳送數(shù)據(jù)??蛻舳薐avaScript開發(fā)者需要解決的首要問題是:向后端的數(shù)據(jù)庫(kù)發(fā)起查詢,發(fā)送數(shù)據(jù)到客戶端,當(dāng)數(shù)據(jù)庫(kù)變動(dòng)時(shí),推送變動(dòng)到客戶端。DDP是解決這一問題的標(biāo)準(zhǔn)做法。
Meteor應(yīng)用的后端基于Node和MongoDB。前端和后端的應(yīng)用同時(shí)使用Meteor的API。未來開發(fā)者可以選擇 MongoDB 之外的其他數(shù)據(jù)庫(kù)。
為什么使用Meteor?請(qǐng)閱讀Meteor的七大原則。
應(yīng)用案例本文中我們將搭建一個(gè) epoll 應(yīng)用,該應(yīng)用允許用戶發(fā)布問題并投票。這個(gè)應(yīng)用可以做到:
當(dāng)用戶訪問/時(shí),會(huì)看到一個(gè)問題列表。用戶需要通過Twitter登錄,以便投票或發(fā)布新問題。如下圖所示,由于未登錄,投票按鈕不可用。
當(dāng)用戶點(diǎn)擊Sign in with Twitter之后,他將授權(quán) epoll 應(yīng)用使用他的賬號(hào)。授權(quán)成功之后,用戶可以投票或發(fā)布新問題。
今天的示例應(yīng)用的代碼可以從GitHub取得。
安裝Meteor開始使用Meteor很容易。如果你使用Mac或Linux,只需輸入如下命令:
curl https://install.meteor.com | /bin/sh
Windows用戶請(qǐng)參閱文檔
創(chuàng)建Meteor應(yīng)用創(chuàng)建Meteor應(yīng)用很容易。安裝之后,運(yùn)行create命令即可。
meteor create epoll
這將創(chuàng)建epoll目錄,該目錄下有一些模板文件。項(xiàng)目結(jié)構(gòu)如下所示:
讓我們解釋下這個(gè)結(jié)構(gòu):
meteor文件夾下保存meteor特定的文件。.gitignore忽略存儲(chǔ)MongoDB數(shù)據(jù)庫(kù)文件和應(yīng)用文件的local文件夾。packages指明本應(yīng)用所需的包。你可以把它們看成是npm包。Meteor以包的形式提供功能。本文中會(huì)使用一些包。release保存了meteor版本。本文使用的版本是0.6.6.3。
epoll.css決定應(yīng)用的CSS樣式。
epoll.html是應(yīng)用的HTML標(biāo)記。目前meteor只支持handlebars模板引擎,不過未來可能支持其他模板引擎。
epoll.js是meteor應(yīng)用的核心。epoll.js同時(shí)部署在服務(wù)器段和客戶端。這允許開發(fā)者一次編寫、兩端使用。meteor創(chuàng)建的epoll.js模板如下所示:
if (Meteor.isClient) { Template.hello.greeting = function () { return "Welcome to epoll."; }; Template.hello.events({ "click input" : function () { // template data, if any, is available in "this" if (typeof console !== "undefined") console.log("You pressed the button"); } }); } if (Meteor.isServer) { Meteor.startup(function () { // code to run on server at startup }); }
Meteor.isServer和Meteor.isClient區(qū)分了服務(wù)器端和客戶端的代碼。
meteor命令可以運(yùn)行應(yīng)用:
cd epoll meteor
可以通過 http://localhost:3000 訪問應(yīng)用。點(diǎn)擊按鈕后,在chrome developer tools中你可以看到You pressed the button.信息。
修改epoll.js的歡迎部分:
Template.hello.greeting = function () { return "The Missing Meteor Tutorial!!!"; };
變動(dòng)會(huì)自動(dòng)應(yīng)用,頁(yè)面也會(huì)自動(dòng)刷新。
前面提到Meteor使用MongoDB來存儲(chǔ)數(shù)據(jù)。當(dāng)我們安裝meteor的時(shí)候,它同時(shí)會(huì)下載最新版的MongoDB。我們可以看到,MongoDB安裝在
; ps -ef|grep mongo 501 1704 1687 0 2:22PM ttys001 0:09.28 /Users/shekhargulati/.meteor/tools/0b2f28e18b/mongodb/bin/mongod --bind_ip 127.0.0.1 --smallfiles --nohttpinterface --port 3002 --dbpath /Users/shekhargulati/day15/epoll/.meteor/local/db
在我的機(jī)子上,MongoDB運(yùn)行于3002端口,以避免和其他默認(rèn)運(yùn)行于27017端口的MongoDB沖突。
智能的Meteor包管理前面提到Meteor以包的形式實(shí)現(xiàn)功能。這些包在瀏覽器和服務(wù)器上都能使用。運(yùn)行以下命令可以得知Meteor支持的所有包:
meteor list
使用meteor add和meteor remove命令來添加刪除包。
添加Twitter Bootstrap包我們將使用Twitter Bootstrap作為用戶界面的風(fēng)格。
meteor add bootstrap
注意,Meteor包不一定是最新版。
添加Twitter授權(quán)包在我們的應(yīng)用中,用戶需要首先通過Twitter授權(quán)才能投票或添加問題。Meteor提供了accounts-ui包,可以為我們的應(yīng)用添加登錄組件:
meteor add accounts-ui
然后我們添加授權(quán)提供者。在這個(gè)應(yīng)用中,我們使用Twitter,不過我們其實(shí)也可以使用facebook、github、google、weibo或meetup。
meteor add accounts-twitter
添加包之后,我們需要更新下epoll.html,添加Twitter登錄按鈕:
Epoll : Share your opinion online, anywhere, anytime {{> banner}}Sign in using Twitter to submit new questions or to vote on existing questions.
{{loginButtons}}
然后調(diào)整一下樣式,增加下面的代碼到epoll.css:
/* CSS declarations go here */ .login-display-name{color: white } .login-button{background-color: white} #main { padding-top:20px; }
應(yīng)用會(huì)自動(dòng)更新,你會(huì)見到這樣的頁(yè)面:
現(xiàn)在點(diǎn)擊Configure Twitter Login,會(huì)要求我們輸入 twitter 應(yīng)用的相關(guān)信息:
按照提示配置之后,我們可以使用twitter登錄了。
授權(quán)之后我們可以登錄應(yīng)用。使用完畢之后,我們可以登出。
MongoDB會(huì)在用戶集合內(nèi)創(chuàng)建新用戶。我們可以使用mongo命令連接數(shù)據(jù)庫(kù)查看:
; ~/.meteor/tools/0b2f28e18b/mongodb/bin/mongo --port 3002 MongoDB shell version: 2.4.6 connecting to: 127.0.0.1:3002/test > show dbs local 0.03125GB meteor 0.0625GB > use meteor switched to db meteor > show collections meteor_accounts_loginServiceConfiguration system.indexes users > db.meteor_accounts_loginServiceConfiguration.find() { "service" : "twitter", "consumerKey" : "xxx", "secret" : "xxx", "_id" : "xxx" } > > > db.users.find().pretty() { "createdAt" : ISODate("2013-11-11T18:03:23.488Z"), "_id" : "xx", "services" : { "twitter" : { "id" : "66993334", "screenName" : "shekhargulati", "accessToken" : "xxx-xxx", "accessTokenSecret" : "xxx", "profile_image_url" : "http://pbs.twimg.com/profile_images/378800000254412405/e4adcf8fb7800c3e5f8141c561cb57e4_normal.jpeg", "profile_image_url_https" : "https://pbs.twimg.com/profile_images/378800000254412405/e4adcf8fb7800c3e5f8141c561cb57e4_normal.jpeg", "lang" : "en" }, "resume" : { "loginTokens" : [ { "token" : "xxx", "when" : ISODate("2013-11-11T18:03:23.489Z") } ] } }, "profile" : { "name" : "Shekhar Gulati" } } >定義應(yīng)用層次
Meteor創(chuàng)建的模板應(yīng)用有一個(gè)問題,客戶端和服務(wù)器段的epoll.js代碼是一樣的。任何人的都可以使用瀏覽器的開發(fā)工具查看epoll.js。
如果我們不想將服務(wù)器端的特有代碼發(fā)送到客戶端,我們可以使用client和server目錄來分隔代碼。
cd epoll mkdir client server
在兩個(gè)目錄下分別創(chuàng)建epollclient.js和epollserver.js文件。
client/epollclient.js內(nèi)存放客戶端代碼:
Template.hello.greeting = function () { return "The Missing Meteor Tutorial!!!"; }; Template.hello.events({ "click input" : function () { // template data, if any, is available in "this" if (typeof console !== "undefined") console.log("You pressed the button"); } });
服務(wù)器端代碼存放在server/epollserver.js:
Meteor.startup(function () { // code to run on server at startup }); ``` 然后刪除`epoll.js`: ```sh rm -f epoll.js移除insecure包
每一個(gè)Meteor應(yīng)用預(yù)裝了insecure包。這個(gè)應(yīng)用讓用戶端可以在數(shù)據(jù)庫(kù)上實(shí)施一切操作。對(duì)于原型開發(fā)這很有用,但是通常不適合生產(chǎn)環(huán)境。
meteor remove insecure發(fā)布問題
現(xiàn)在我們添加一個(gè)功能,已登錄的用戶可以提交新問題。
Epoll : Share your opinion online, anywhere, anytime {{#if currentUser}} {{> addquestion}} {{/if}} {{#unless currentUser}} {{> banner}} {{/unless}}Sign in using Twitter to submit new questions or to vote on existing questions.
{{loginButtons}}
僅當(dāng)用戶登錄的時(shí)候才會(huì)渲染addQuestion模板。如果用戶登出,則不會(huì)見到添加新問題的文本框。
我們需要同時(shí)更新客戶端和服務(wù)器端的代碼以便實(shí)現(xiàn)這一功能。
在client/epollclient.js中加入:
Template.addquestion.events({ "click input.add-question" : function(event){ event.preventDefault(); var questionText = document.getElementById("questionText").value; Meteor.call("addQuestion",questionText,function(error , questionId){ console.log("added question with Id .. "+questionId); }); document.getElementById("questionText").value = ""; } });
以上代碼中:
我們首先將點(diǎn)擊input事件綁定到add-question類。
接著我們阻止默認(rèn)的點(diǎn)擊事件,從DOM中獲取問題文本。
然后我們調(diào)用Meteor服務(wù)器的方法addQuestion。由服務(wù)器負(fù)責(zé)插入、更新、刪除數(shù)據(jù)等有風(fēng)險(xiǎn)的操作。客戶端看不到實(shí)現(xiàn),也無法私自修改數(shù)據(jù)。
現(xiàn)在我們需要修改server/epollserver.js。我們首先定義一個(gè)名為Questions的新集合。然后我們會(huì)操作這個(gè)集合。Meteor使用minimongo作為API接口。參閱Meteor.Collection.documentation查看minimongo支持的所有操作。
Questions = new Meteor.Collection("questions"); Meteor.startup(function () { // code to run on server at startup }); Meteor.methods({ addQuestion : function(questionText){ console.log("Adding Question"); var questionId = Questions.insert({ "questionText" : questionText, "submittedOn": new Date(), "submittedBy" : Meteor.userId() }); return questionId; } });
現(xiàn)在訪問我們的應(yīng)用然后提交一個(gè)新問題:
查看下MongoDB中的數(shù)據(jù)
> db.questions.find().pretty() { "questionText" : "Is Sachin Tendulkar the greatest batsman of all time?", "submittedOn" : ISODate("2013-11-11T18:23:02.541Z"), "submittedBy" : "Jnu6oXoAZ2um57rZ8", "_id" : "nhqvgDcZqgZgLdDB7" }問題列表
我們接下來要實(shí)現(xiàn)的功能是問題列表。用戶不需登錄,就可以看到所有問題的列表。
在main div中加入:
{{> questions}}
然后添加問題模板:
All Questions
{{#each items}} {{> question}} {{/each}}{{questionText}} Yes {{yes}} No {{no}}
注意我們使用了unless來確保用戶未登錄的情況下應(yīng)用disabled css。
為了獲取所有問題,我們需要在客戶端使用Question集合來獲取所有文本。在client/epollclient.js添加如下代碼:
Questions = new Meteor.Collection("questions"); Template.questions.items = function(){ return Questions.find({},{sort:{"submittedOn":-1}}); };實(shí)現(xiàn)投票功能
最后我們需要實(shí)現(xiàn)投票功能。我們上面已經(jīng)在html文件中加入了相關(guān)的模板代碼,下面我們?cè)?b>client/epollclient.js加入如下代碼:
Template.question.events({ "click": function () { Session.set("selected_question", this._id); }, "click a.yes" : function (event) { event.preventDefault(); if(Meteor.userId()){ var questionId = Session.get("selected_question"); console.log("updating yes count for questionId "+questionId); Meteor.call("incrementYesVotes",questionId); } }, "click a.no": function(){ event.preventDefault(); if(Meteor.userId()){ var questionId = Session.get("selected_question"); console.log("updating no count for questionId "+questionId); Meteor.call("incrementNoVotes",questionId); } } });
上面的代碼實(shí)現(xiàn)了:
綁定點(diǎn)擊事件到問題模板。點(diǎn)擊任意問題的時(shí),在session中設(shè)置questionId。session提供了一個(gè)客戶端的全局對(duì)象,你可以在里面存儲(chǔ)任意的鍵值對(duì)。
當(dāng)用戶點(diǎn)擊Yes按鈕時(shí),我們會(huì)從session中取得選中的questionId,然后在服務(wù)器端調(diào)用incrementYesVotes方法。我們使用Meteor.userId()來確保用戶已經(jīng)登錄了。
當(dāng)用戶點(diǎn)擊No按鈕時(shí),我們?cè)诜?wù)器端調(diào)用incrementNoVotes函數(shù)。
最后我們?cè)?b>server/epollserver.js加入incrementYesVotes和incrementNoVotes函數(shù)。我們使用Meteor的集合更新功能來增加計(jì)數(shù)器。
incrementYesVotes : function(questionId){ console.log(questionId); Questions.update(questionId,{$inc : {"yes":1}}); }, incrementNoVotes : function(questionId){ console.log(questionId); Questions.update(questionId,{$inc : {"no":1}}); }
這樣每次用戶點(diǎn)擊yes或no按鈕之后,計(jì)數(shù)器會(huì)更新。你可以訪問 http://localhost:3000 試驗(yàn)一番。
部署Meteor應(yīng)用部署Meteor應(yīng)用有很多種方法。我們可以在Meteor提供的測(cè)試服務(wù)器上部署,也可以部署到OpenShift。
如果你打算部署到OpenShift上,請(qǐng)參閱Ryan的這篇博客。
運(yùn)行以下命令可以部署到Meteor測(cè)試服務(wù)器:
meteor deploy epoll
應(yīng)用可以通過 http://epoll.meteor.com/ 訪問。
今天就是這些了。歡迎繼續(xù)反饋。
原文:Day 15: Meteor——Building a Web App From Scratch in Meteor
翻譯整理: Segmentfault
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/77977.html
摘要:在我的機(jī)子上,運(yùn)行于端口,以避免和其他默認(rèn)運(yùn)行于端口的沖突。我們可以使用命令連接數(shù)據(jù)庫(kù)查看定義應(yīng)用層次創(chuàng)建的模板應(yīng)用有一個(gè)問題,客戶端和服務(wù)器段的代碼是一樣的。在中加入然后添加問題模板注意我們使用了來確保用戶未登錄的情況下應(yīng)用。 編者注:我們發(fā)現(xiàn)了有趣的一系列文章《30天學(xué)習(xí)30種新技術(shù)》,正在翻譯中,一天一篇更新,年終禮包。下面是第15天的內(nèi)容。 到目前為止我們討論了Bower...
摘要:全??蚣芑咎峁┝酥谱饕粋€(gè)移動(dòng)產(chǎn)品所有的框架和工具,從標(biāo)準(zhǔn)的,,應(yīng)用監(jiān)控,。指的是的名,一個(gè)對(duì)應(yīng)的默認(rèn)情況下會(huì)被復(fù)數(shù)化,比如的路徑是。再做個(gè)小廣告最近在做一個(gè)小班免費(fèi),專門教高中生技術(shù),直到達(dá)成可以自行參加的程度。 showImg(http://segmentfault.com/img/bVdnvZ); StrongLoop是一個(gè)基于Nodejs的強(qiáng)大框架,幾乎包含了移動(dòng)開發(fā)全棧所需...
摘要:請(qǐng)注意,觸發(fā)器將不會(huì)在上午點(diǎn)開始,僅在,,和請(qǐng)注意,一些調(diào)度要求太復(fù)雜,無法用單一觸發(fā)表示例如每上午至之間每分鐘,下午至晚上點(diǎn)之間每分鐘一次。在這種情況下的解決方案是簡(jiǎn)單地創(chuàng)建兩個(gè)觸發(fā)器,并注冊(cè)它們來運(yùn)行相同的作業(yè)。 表達(dá)式說明 Cron-Expressions 用于配置 CronTrigger的實(shí)例。Cron Expressions 是由七個(gè)子表達(dá)式組成的字符串,用于描述日程表的各個(gè)...
摘要:閑扯在翻譯之前我還是要簡(jiǎn)要介紹一下,這是一款跨時(shí)代的框架。后來開始使用動(dòng)態(tài)的腳本語(yǔ)言,比如后來以此構(gòu)成了框架。最為出名的自然是。通過來傳遞數(shù)據(jù),在本地起這種事情使得開發(fā)進(jìn)入了新的階段。如果你覺得我翻譯的很爛,來給我說吧郵件地址 閑扯 在翻譯之前我還是要簡(jiǎn)要介紹一下Meteor,這是一款跨時(shí)代的框架。真的,完全沒有騙你。 一開始的網(wǎng)站是純靜態(tài)頁(yè)面搭建的。 后來開始使用動(dòng)態(tài)的腳本語(yǔ)言,比如...
摘要:用定義視圖組件在中,視圖組件是被使用定義的。你的組件可以有任何你想要的方法。組件可以接收來自于父組件通過屬性傳過來的數(shù)據(jù)。因?yàn)楸窘坛虒W⒂诤?,所以你可以拷貝下面的代碼到文件中。嘿嘿,來添加上吧只要把改名成就好了 在 React 組件中定義視圖 在開始編寫React視圖庫(kù)之前,要先添加react包,這個(gè)包囊括了你在Meteor應(yīng)用中開始運(yùn)行React所需要所有東西。這個(gè)React庫(kù)自己可...
閱讀 2469·2019-08-30 15:53
閱讀 2583·2019-08-29 13:11
閱讀 2670·2019-08-29 12:45
閱讀 3497·2019-08-29 12:41
閱讀 2340·2019-08-26 10:14
閱讀 2167·2019-08-23 14:39
閱讀 2319·2019-08-23 12:38
閱讀 3384·2019-08-23 12:04