前言javascript
我還在攜程的作業務的時候,每一個看似簡單的移動頁面背後每每會隱藏5個以上的數據請求,其中最過複雜的當屬機票與酒店的訂單填寫業務代碼css
這裏先看看比較「簡單」的機票代碼:html
而後看看稍微複雜的酒店業務邏輯:前端
機票一個頁面的代碼量達到了5000行代碼,而酒店的代碼居然超過了8000行,這裏還不包括模板(html)文件!!!java
而後初略看了機票的代碼,就該頁面可能發生的接口請求有19個之多!!!而酒店的的交互DOM事件基本多到了使人髮指的地步:ios
固然,機票團隊的交互DOM事件已經多到了我筆記本不能截圖了:git
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
events: {
'click .js_check_invoice_type': 'checkInvoiceType', //切換髮票類型
'click .flight-hxtipshd': 'huiXuanDesc', //惠選說明
'click .js_ListReload': 'hideNetError',
'click #js_return': 'backAction', //返回列表頁
'click div[data-rbtType]': 'showRebate', //插爛返現說明
'click #paybtn .j_btn': 'beforePayAction', //提交訂單 //flightDetailsStore, passengerQueryStore, mdStore, postAddressStorage, userStore, flightDeliveryStore
'click .flight-loginbtn2': 'bookLogin', //登陸
'input #linkTel': 'setContact', //保存用戶輸入的聯繫人
'click #addPassenger .flight-labq': 'readmeAction',//姓名幫助
'click .jsDelivery': 'selDelivery', //選擇配送方式
'click #jsViewCoupons': 'viewCoupons', //查看消費券使用說明 //flightDetailsStore
// 'click .j_refundPolicy': 'fanBoxAction', //查看返現信息
//'click .flight-bkinfo-tgq .f-r': 'tgBoxAction', //查看退改簽
'click .js_del_tab': 'showDelListUI', //配送方式
// 'click .js_del_cost .flight-psf i': 'selectPaymentType', // 選擇快遞費用方式
'click #js_addrList': 'AddrListAction', //選擇地址
'click #date-picker': 'calendarAction', //取票日期 //airportDeliveryStore
'click #done-address': 'zqinairselect', //取票櫃檯
'click #selectCity': 'selectCityAction', //選擇城市
'click #date-zqtime': 'showZqTimeUI', //取票時間 //airportDeliveryStore
'click #jsinsure': 'viewInsure', //保險說明
'click #js_invoice_title': 'inTitleChangeWrp', //發票擡頭更改 // userStore, flightOrderInfoInviceStore, flightOrderStore //don't move outside
'click #js_invoice_title_div': 'inTitleChangeWrp',
'click .flight-icon-arrrht': 'showinTitleList', //‘+’號,跳轉發票擡頭列表 //userStore, invoiceURLStore
'focusin #linkTel': 'telInput',
'focusout #linkTel': 'telInputFinish',
'touchstart input': 'touchStartAction', // 處理Android手機上點擊不靈敏問題
'click #package .flight-arrrht': 'packageSelect',
'focusin input': 'hideErrorTips',
'click #dist_text_div': 'hideErrorTips',
'click .j_PackageNotice': 'toggletips',
'click .j_AnnouncementNotice': 'toggleNotice',
'click #travalPackageDesc': 'forwardToTravalPackage', //don't move into child modules
'click #airInsureDesc': 'showAirInsureDesc',
'click #paybtn': 'orderDetailAction',//價格明細
'click .J_retriveVerifyCodeBtn': 'getVerifyCode',
'click .J_toPay': 'toPayAction',
'click .J_closeVerifyCode': 'closeVerifyCodePopup',
'keyup .J_verifyCodePopup input': 'setToPayBtnStatus',
'click .js_flight_seat': 'selectRecommendCabin', // 選擇推薦倉位
'click .j_changeFlight': 'changeFlightAction', // 推薦航班彈層中更改航班
'focusin input:not([type=tel])': 'adjustInputPosition', // iphone5/5s ios8搜狗輸入法遮住input
'click .js_addr,#js_addr_div': 'editDeliverAddress',//報銷憑證,詳細地址編輯
'click .js_showUserInfo': 'showUserInfo', // add by hkhu v2.5.9
'click #logout': 'logout', // add by hkhu v2.5.9
'click #gotoMyOrder': 'gotoMyOrder', // add by hkhu v2.5.9
'touchstart #logout': function (e) { $(e.currentTarget).addClass('current'); },
'touchstart #gotoMyOrder': function (e) { $(e.currentTarget).addClass('current'); },
'click .js_buddypayConfirm': 'buddypayConfirmed',
'click .js_pickupTicket': 'viewPickUp', //261接送機券說明
'click .flt-bking-logintips': 'closelogintips'//關閉接送機券提示
},
|
就這種體量的頁面,若是須要迭代需求、打BUG補丁的話,我敢確定的說,一個BUG的修復很容易引發其它BUG,而上面還僅僅是其中一個業務頁面,後面還有強大而複雜的前端框架呢!如此複雜的前端代碼維護工做可不是開玩笑的!github
PS:說道此處,不得不爲攜程的前端水平點個贊,業內少有的單頁應用,一套代碼H5&Hybrid同時運行不說,還解決了SEO問題,嗯,很贊。web
如何維護這種頁面,如何設計這種頁面是咱們今天討論的重點,而上述是攜程合併後的代碼,他們兩個團隊的設計思路不便在此處展開。面試
今天,我這裏提供一個思路,認真閱讀此文可能在如下方面對你有所幫助:
1
2
3
4
|
①
如何將一個複雜的頁面拆分爲一個個獨立的頁面組件模塊
②
如何將分拆後的業務組件模塊從新合爲一個完整的頁面
③
從重構角度看組件化開發帶來的好處
④
從前端優化的角度看待組件化開發
|
文中是我我的的一些框架&業務開發經驗,但願對各位有用,也但願各位多多支持討論,指出文中不足以及提出您的一些建議。
因爲該項目涉及到了項目拆分與合併,基本屬於一個完整的前端工程化案例了,因此將之放到了github上:https://github.com/yexiaochai/mvc
其中工程化一塊的代碼,後續會由另外一位小夥伴持續更新,若是該文對各位有所幫助的話請各位給項目點個贊、加顆星:)
我相信若是是中級水平的前端,認真閱讀此文必定會對你有一點幫助滴。
一個實際的場景
演示地址
http://yexiaochai.github.io/mvc/webapp/bus/list.html
代碼倉促,可能會有BUG哦:)
代碼地址:https://github.com/yexiaochai/mvc/
頁面基本構成
由於訂單填寫頁通常有密度,我這裏挑選相對複雜而又沒有密度的產品列表頁來作說明,其中框架以及業務代碼已經作過抽離,不會包含敏感信息,一些優化後續會同步到開源blade框架中去。
咱們這裏列表頁的首屏頁面以下:
簡單來講組成以下:
① 框架級別UI組件UIHeader,頭部組件
② 點擊日期會出框架級別UI,日曆組件UICalendar
③ 點擊出發時段、出發汽車站、到達汽車站,皆會出框架級別UI
④ header下面的日期工具欄須要做爲獨立的業務模塊
⑤ 列表區域能夠做爲獨立的業務模塊,可是與主業務靠太近,不太適合
⑥ 出發時段、出發汽車站、到達汽車站皆是獨立的業務模塊
一個頁面被咱們拆分紅了若干個小模塊,咱們只須要關注模塊內部的交互實現,而包括業務模塊的通訊,業務模塊的樣式,業務模塊的重用,暫時有如下約定:
1
2
3
|
①
單個頁面的樣式所有寫在一個文件中,好比list裏面全部模塊對應的是list.css
②
模塊之間採用觀察者模式觀察數據實體變化,以數據爲媒介通訊
③
通常來講業務模塊不可重用,若是有重用的模塊,須要分離到common目錄中,由於咱們今天不考慮common重用,這塊暫時不予理睬
|
這裏有些朋友可能認爲單個模塊的CSS以及image也應該參與獨立,我這裏不太贊成,業務頁面樣式粒度太細的話會給設計帶來不小的麻煩,這裏再以通俗的話來講:尼瑪,我CSS功底通常,拆分的太細,對我來講難度過高……
很差的作法
很差的這個事情實際上是相對的,由於很差的作法通常是比較簡單的作法,對於一次性項目或者業務比較簡單的頁面來講反而是好的作法,好比這裏的業務邏輯能夠這樣寫:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
define(['AbstractView', 'list.layout.html', 'list.html', 'BusModel', 'BusStore', 'UICalendarBox', 'UILayerList', 'cUser', 'UIToast'],
function (AbstractView, layoutHtml, listTpl, BusModel, BusStore, UICalendarBox, UILayerList, cUser, UIToast) {
return _.inherit(AbstractView, {
propertys: function ($super) {
$super();
//一堆基礎屬性定義
//......
//交互業務邏輯
this.events = {
'click .js_pre_day': 'preAction', //點擊前一天觸發
'click .js_next_day': 'nextAction', //點擊後一天觸發
'click .js_bus_list li': 'toBooking', //點擊列表項目觸發
'click .js_show_calendar': 'showCalendar', //點擊日期項出日曆組件
'click .js_show_setoutdate': 'showSetoutDate', //篩選出發時段
'click .js_show_setstation': 'showStation', //篩選出發站
'click .js_show_arrivalstation': 'showArrivalStation', //篩選到達站
//迭代需求,增長其它頻道入口
'click .js-list-tip': function () {}
};
},
//初始化頭部標題欄
initHeader: function (t) { },
//首次dom渲染後,初始化後續會用到的全部dom元素,以避免重複獲取
initElement: function () {},
showSetoutDate: function () {},
showStation: function () {},
showArrivalStation: function () {},
showCalendar: function () {},
preAction: function (e) {},
nextAction: function () {},
toBooking: function (e) {},
listInit: function () {},
bindScrollEvent: function () {},
unbindScrollEvent: function () { },
addEvent: function () {
this.on('onShow', function () {
//當頁面渲染結束,須要作的初始化操做,好比渲染頁面
this.listInit();
//......
});
this.on('onHide', function () {
this.unbindScrollEvent();
});
}
});
});
|
根據以前的經驗,若是僅僅包含這些業務邏輯,這樣寫代碼問題不是很是大,代碼量預計在800行左右,可是爲了完成完整的業務邏輯,咱們這裏立刻產生了新的需求。
需求迭代
由於我這裏的班次列表,最初是沒有URL參數,因此根本沒法產出班次列表,頁面上全部組件模塊都是擺設,因而這裏新增一個需求:
1
|
當
url沒有出發-到達相關參數信息時,默認彈出出發城市到達城市選擇框
|
因而,咱們這裏會新增一個簡單的彈出層:
這個看似簡單的彈出層,背後卻隱藏了一個巨大的陷阱,由於點擊出發或者到達時會出城市列表,而城市列表自己就是一個比較複雜的業務:
因而頁面的組成發生了改變:
① 自己業務邏輯約800行代碼
② 新增出發到達篩選彈出層
③ 出發城市頁面,預計300行代碼
而彈出層的新增對業務自己形成了深遠的影響,原本url是不帶有業務參數的,可是點擊了彈出層的肯定按鈕,須要改變URL參數,而且刷新自己頁面的數據,因而簡單的一個彈出層新增直接將頁面的複雜程度提高了一倍。
因而該頁面代碼輕輕鬆鬆破千了,後續需求迭代js代碼量破2000僅僅是時間問題,到時候維護便複雜了,頁面複雜無規律的DOM操做將會令你焦頭爛額,這個時候組件化開發的優點便得以體現了,因而下面進入組件化開發的設計。
準備工做
整體架構
此次的代碼依賴於blade骨架,包括:
① MVC模塊,完成經過url獲取正確的page控制器,從而經過view.js完成渲染頁面的功能
② 數據請求模塊,完成接口請求
全站依賴於javascript的繼承功能,詳情見:【一次面試】再談javascript中的繼承,若是不太瞭解面向對象編程,文中代碼可能會有點吃力,也請各位多多瞭解。
整體業務架構如圖:
框架架構圖:
.
下面分別介紹下各個模塊,幫助各位在下文中能更好的瞭解代碼,首先是基本MVC的介紹,這裏請參考我這篇文章:簡單的MVC介紹
全局控制器
其實控制器可謂是變化萬千的一個對象,對於服務器端來講,控制器完成的功能是將本次請求分發到具體的代碼模塊,由代碼模塊處理後返回字符串給前端;
對於請求已經來到瀏覽器的前端來講,根據此次請求URL(或者其它判斷條件),判斷該次請求應該由哪一個前端js控制器執行,這是前端控制器乾的事情;
當真的此次處理邏輯進入一個具體的page後,這個page事實上也能夠做爲一個控制器存在……
咱們這裏的控制器,主要完成根據當前請求實例化View的功能,而且會提供一些view級別但願單例使用的接口:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
|
define([
'UIHeader',
'UIToast',
'UILoading',
'UIPageView',
'UIAlert'
], function (UIHeader, UIToast, UILoading, UIPageView, UIAlert) {
return _.inherit({
propertys: function () {
//view搜索目錄
this.viewRootPath = 'views/';
//默認view
this.defaultView = 'index';
//當前視圖路徑
this.viewId;
this.viewUrl;
//視圖集
this.views = {};
//是否開啓單頁應用
// this.isOpenWebapp = _.getHybridInfo().platform == 'baidubox' ? true : false;
this.isOpenWebapp = false;
this.viewMapping = {};
//UIHeader須要釋放出來
this.UIHeader = UIHeader;
this.interface = [
'forward',
'back',
'jump',
'showPageView',
'hidePageView',
'showLoading',
'hideLoading',
'showToast',
'hideToast',
'showMessage',
'hideMessage',
'showConfirm',
'hideConfirm',
'openWebapp',
'closeWebapp'
];
},
initialize: function (options) {
this.propertys();
this.setOption(options);
this.initViewPort();
this.initAppMapping();
//開啓fastclick
$.bindFastClick && $.bindFastClick();
},
setOption: function (options) {
_.extend(this, options);
},
//建立dom結構
initViewPort: function () {
this.d_header = $('#headerview');
this.d_state = $('#js_page_state');
this.d_viewport = $('#main');
//實例化全局使用的header,這裏好像有點不對
this.header = new this.UIHeader({
wrapper: this.d_header
});
//非共享資源,這裏應該引入app概念了
this.pageviews = {};
this.toast = new UIToast();
this.loading = new UILoading();
this.alert = new UIAlert();
this.confirm = new UIAlert();
},
openWebapp: function () {
this.isOpenWebapp = true;
},
closeWebapp: function () {
this.isOpenWebapp = false;
},
showPageView: function (name, _viewdata_, id) {
var view = null, k, scope = this.curViewIns || this;
if (!id) id = name;
if (!_.isString(name)) return;
// for (k in _viewdata_) {
// if (_.isFunction(_viewdata_[k])) _viewdata_[k] = $.proxy(_viewdata_[k], scope);
// }
view = this.pageviews[id];
var arr = name.split('/');
var getViewPath = window.getViewPath || window.GetViewPath;
if (!view) {
view = new UIPageView({
// bug fixed by zzx
viewId: arr[arr.length - 1] || name,
viewPath: getViewPath ? getViewPath(name) : name,
_viewdata_: _viewdata_,
onHide: function () {
scope.initHeader();
}
});
this.pageviews[id] = view;
} else {
view.setViewData(_viewdata_);
}
view.show();
},
hidePageView: function (name) {
if (name) {
if (this.pageviews[name]) this.pageviews[name].hide();
} else {
for (var k in this.pageviews) this.pageviews[k].hide();
}
},
showLoading: function () {
this.loading.show();
},
hideLoading: function () {
this.loading.hide();
},
showToast: function (msg, callback) {
this.toast.resetDefaultProperty();
this.toast.content = msg;
if (callback) this.toast.hideAction = callback;
this.toast.refresh();
this.toast.show();
},
hideToast: function () {
this.toast.hide();
},
showMessage: function (param) {
if (_.isString(param)) {
param = { content: param };
}
this.alert.resetDefaultProperty();
this.alert.setOption(param);
this.alert.refresh();
this.alert.show();
},
hideMessage: function () {
this.alert.hide();
},
showConfirm: function (params) {
if (!params) params = {};
if (typeof params == 'string') {
params = {
content: params
};
}
this.confirm.resetDefaultProperty();
//與showMessage不同的地方
this.confirm.btns = [
{ name: '取消', className: 'cm-btns-cancel js_cancel' },
{ name: '肯定', className: 'cm-btns-ok js_ok' }
];
this.confirm.setOption(params);
this.confirm.refresh();
this.confirm.show();
},
hideConfirm: function () {
this.confirm.hide();
},
//初始化app
initApp: function () {
//首次加載不須要走路由控制
this.loadViewByUrl();
//後面的加載所有要通過路由處理
if (this.isOpenWebapp === true)
$(window).on('popstate.app', $.proxy(this.loadViewByUrl, this));
},
loadViewByUrl: function (e) {
this.hidePageView();
var url = decodeURIComponent(location.href).toLowerCase();
var viewId = this.getViewIdRule(url);
viewId = viewId || this.defaultView;
this.viewId = viewId;
this.viewUrl = url;
this.switchView(this.viewId);
},
//@override
getViewIdRule: function (url) {
var viewId = '', hash = '';
var reg = /webapp\/.+\/(.+)\.html/;
var match = url.match(reg);
if (match && match[1]) viewId = match[1];
return viewId;
},
//@override
setUrlRule: function (viewId, param, replace, project) {
var reg = /(webapp\/.+\/)(.+)\.html/;
var url = window.location.href;
var match = url.match(reg);
var proj = project ? 'webapp/' + project : match[1];
var preUrl = '', str = '', i = 0, _k, _v;
//這裏這樣作有點過於業務了 *bug*
var keepParam = [
'us'
], p;
if (!viewId) return;
if (!match || !match[1]) {
preUrl = url + '/webapp/bus/' + viewId + '.html';
} else {
preUrl = url.substr(0, url.indexOf(match[1])) + proj + viewId + '.html'; ;
}
//特定的參數將會一直帶上去,渠道、來源等標誌
for (i = 0; i < keepParam.length; i++) {
p = keepParam[i];
if (_.getUrlParam()[p]) {
if (!param) param = {};
param[p] = _.getUrlParam()[p];
}
}
i = 0;
for (k in param) {
_k = encodeURIComponent(_.removeAllSpace(k));
_v = encodeURIComponent(_.removeAllSpace(param[k]));
if (i === 0) {
str += '?' + _k + '=' + _v;
i++;
} else {
str += '&' + _k + '=' + _v;
}
}
url = preUrl + str;
if (this.isOpenWebapp === false) {
window.location = url;
return;
}
if (replace) {
history.replaceState('', {}, url);
} else {
history.pushState('', {}, url);
}
},
switchView: function (id) {
var curView = this.views[id];
//切換前的當前view,立刻會隱藏
var tmpView = this.curView;
if (tmpView && tmpView != curView) {
this.lastView = tmpView;
}
//加載view樣式,權宜之計
// this.loadViewStyle(id);
//若是當前view存在,則執行請onload事件
if (curView) {
//若是當前要跳轉的view就是當前view的話便不予處理
//這裏具體處理邏輯要改*************************************
if (curView == this.curView) {
return;
}
this.curView = curView;
this.curView.show();
this.lastView && this.lastView.hide();
} else {
// this.showLoading();
this.loadView(id, function (View) {
//每次加載結束將狀態欄隱藏,這個代碼要改
// this.hideLoading();
this.curView = new View({
viewId: id,
refer: this.lastView ? this.lastView.viewId : null,
APP: this,
wrapper: this.d_viewport
});
//設置網頁上的view標誌
this.curView.$el.attr('page-url', id);
//保存至隊列
this.views[id] = this.curView;
this.curView.show();
this.lastView && this.lastView.hide();
});
}
},
//加載view
loadView: function (path, callback) {
var self = this;
requirejs([this.buildUrl(path)], function (View) {
callback && callback.call(self, View);
});
},
//override
//配置可能會有的路徑擴展,爲Hybrid與各個渠道作適配
initAppMapping: function () {
// console.log('該方法必須被重寫');
},
//@override
buildUrl: function (path) {
var mappingPath = this.viewMapping[path];
return mappingPath ? mappingPath : this.viewRootPath + '/' + path + '/' + path;
},
//此處須要一個更新邏輯,好比在index view再點擊到index view不會有反應,下次改**************************
forward: function (viewId, param, replace) {
if (!viewId) return;
viewId = viewId.toLowerCase();
this.setUrlRule(viewId, param, replace);
this.loadViewByUrl();
},
jump: function (path, param, replace) {
var viewId;
var project;
if (!path) {
return;
}
path = path.toLowerCase().split('/');
if (path.length <= 0) {
return;
}
viewId = path.pop();
project = path.length === 1 ? path.join('') + '/' : path.join('');
this.setUrlRule(viewId, param, replace, project);
this.loadViewByUrl();
},
back: function (viewId, param, replace) {
if (viewId) {
this.forward(viewId, param, replace)
} else {
if (window.history.length == 1) {
this.forward(this.defaultView, param, replace)
} else {
history.back();
}
}
}
});
});
abstract.app
|
這裏屬於框架控制器層面的代碼,與今天的主題不是很是相關,有興趣的朋友能夠詳細讀讀。
頁面基類
這裏的核心是頁面級別的處理,這裏會作比較多的介紹,首先咱們爲全部的業務級View提供了一個繼承的View:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
|
define([], function () {
'use strict';
return _.inherit({
_propertys: function () {
this.APP = this.APP || window.APP;
var i = 0, len = 0, k;
if (this.APP && this.APP.interface) {
for (i = 0, len = this.APP.interface.length; i < len; i++) {
k = this.APP.interface[i];
if (k == 'showPageView') continue;
if (_.isFunction(this.APP[k])) {
this[k] = $.proxy(this.APP[k], this.APP);
}
else this[k] = this.APP[k];
}
}
this.header = this.APP.header;
},
showPageView: function (name, _viewdata, id) {
this.APP.curViewIns = this;
this.APP.showPageView(name, _viewdata, id)
},
propertys: function () {
//這裏設置UI的根節點所處包裹層
this.wrapper = $('#main');
this.id = _.uniqueId('page-view-');
this.classname = '';
this.viewId = null;
this.refer = null;
//模板字符串,各個組件不一樣,如今加入預編譯機制
this.template = '';
//事件機制
this.events = {};
//自定義事件
//此處須要注意mask 綁定事件先後問題,考慮scroll.radio插件類型的mask應用,考慮組件通訊
this.eventArr = {};
//初始狀態爲實例化
this.status = 'init';
this._propertys();
},
getViewModel: function () {
//假若有datamodel的話,便直接返回,否則便重寫,這裏基本爲了兼容
if (_.isObject(this.datamodel)) return this.datamodel;
return {};
},
//子類事件綁定若想保留父級的,應該使用該方法
addEvents: function (events) {
if (_.isObject(events)) _.extend(this.events, events);
},
on: function (type, fn, insert) {
if (!this.eventArr[type]) this.eventArr[type] = [];
//頭部插入
if (insert) {
this.eventArr[type].splice(0, 0, fn);
} else {
this.eventArr[type].push(fn);
}
},
off: function (type, fn) {
if (!this.eventArr[type]) return;
if (fn) {
this.eventArr[type] = _.without(this.eventArr[type], fn);
} else {
this.eventArr[type] = [];
}
},
trigger: function (type) {
var _slice = Array.prototype.slice;
var args = _slice.call(arguments, 1);
var events = this.eventArr;
var results = [], i, l;
if (events[type]) {
for (i = 0, l = events[type].length; i < l; i++) {
results[results.length] = events[type][i].apply(this, args);
}
}
return results;
},
createRoot: function (html) {
//若是存在style節點,而且style節點不存在的時候須要處理
if (this.style && !$('#page_' + this.viewId)[0]) {
$('head').append($('<style id="page_' + this.viewId + '" class="page-style">' + this.style + '</style>'))
}
//若是具備fake節點,須要移除
$('#fake-page').remove();
//UI的根節點
this.$el = $('<div class="cm-view page-' + this.viewId + ' ' + this.classname + '" id="' + this.id + '">' + html + '</div>');
if (this.wrapper.find('.cm-view')[0]) {
this.wrapper.append(this.$el);
} else {
this.wrapper.html('').append(this.$el);
}
},
_isAddEvent: function (key) {
if (key == 'onCreate' || key == 'onPreShow' || key == 'onShow' || key == 'onRefresh' || key == 'onHide')
return true;
return false;
},
setOption: function (options) {
//這裏能夠寫成switch,開始沒有想到有這麼多分支
for (var k in options) {
if (k == 'events') {
_.extend(this[k], options[k]);
continue;
} else if (this._isAddEvent(k)) {
this.on(k, options[k])
continue;
}
this[k] = options[k];
}
// _.extend(this, options);
},
initialize: function (opts) {
//這種默認屬性
this.propertys();
//根據參數重置屬性
this.setOption(opts);
//檢測不合理屬性,修正爲正確數據
this.resetPropery();
this.addEvent();
this.create();
this.initElement();
window.sss = this;
},
$: function (selector) {
return this.$el.find(selector);
},
//提供屬性重置功能,對屬性作檢查
resetPropery: function () { },
//各事件註冊點,用於被繼承override
addEvent: function () {
},
create: function () {
this.trigger('onPreCreate');
//若是沒有傳入模板,說明html結構已經存在
this.createRoot(this.render());
this.status = 'create';
this.trigger('onCreate');
},
//實例化須要用到到dom元素
initElement: function () { },
render: function (callback) {
var data = this.getViewModel() || {};
var html = this.template;
if (!this.template) return '';
//引入預編譯機制
if (_.isFunction(this.template)) {
html = this.template(data);
} else {
html = _.template(this.template)(data);
}
typeof callback == 'function' && callback.call(this);
return html;
},
refresh: function (needRecreate) {
this.resetPropery();
if (needRecreate) {
this.create();
} else {
this.$el.html(this.render());
}
this.initElement();
if (this.status != 'hide') this.show();
this.trigger('onRefresh');
},
/**
* @description 組件顯示方法,首次顯示會將ui對象實際由內存插入包裹層
* @method initialize
* @param {Object} opts
*/
show: function () {
this.trigger('onPreShow');
// //若是包含就不要亂搞了
// if (!$.contains(this.wrapper[0], this.$el[0])) {
// //若是須要清空容器的話便清空
// if (this.needEmptyWrapper) this.wrapper.html('');
// this.wrapper.append(this.$el);
// }
this.$el.show();
this.status = 'show';
this.bindEvents();
this.initHeader();
this.trigger('onShow');
},
initHeader: function () { },
hide: function () {
if (!this.$el || this.status !== 'show') return;
this.trigger('onPreHide');
this.$el.hide();
this.status = 'hide';
this.unBindEvents();
this.trigger('onHide');
},
destroy: function () {
this.status = 'destroy';
this.unBindEvents();
this.$root.remove();
this.trigger('onDestroy');
delete this;
},
bindEvents: function () {
var events = this.events;
if (!(events || (events = _.result(this, 'events')))) return this;
this.unBindEvents();
// 解析event參數的正則
var delegateEventSplitter = /^(\S+)\s*(.*)$/;
var key, method, match, eventName, selector;
// 作簡單的字符串數據解析
for (key in events) {
method = events[key];
if (!_.isFunction(method)) method = this[events[key]];
if (!method) continue;
match = key.match(delegateEventSplitter);
eventName = match[1], selector = match[2];
method = _.bind(method, this);
eventName += '.delegateUIEvents' + this.id;
if (selector === '') {
this.$el.on(eventName, method);
} else {
this.$el.on(eventName, selector, method);
}
}
return this;
},
unBindEvents: function () {
this.$el.off('.delegateUIEvents' + this.id);
return this;
},
getParam: function (key) {
return _.getUrlParam(window.location.href, key)
},
renderTpl: function (tpl, data) {
if (!_.isFunction(tpl)) tpl = _.template(tpl);
return tpl(data);
}
});
});
abstract.view
|
一個Page級別的View會有如下幾個關鍵屬性&方法:
① template,html字符串,不包含請求的基礎模塊,會構成頁面的html骨架層
② events,全部的DOM事件定義處,以事件代理的方式定義,因此沒必要擔憂執行順序
③ addEvent,用於頁面級別各個階段的監控事件註冊點,通常來講用戶只須要關注不多幾個事件,好比:
1
2
3
4
5
6
7
8
9
|
//寫法
addEvent: function () {
//頁面渲染結束,並顯示時候觸發的事件
this.on('onShow', function () {
});
//離開頁面,頁面隱藏時候觸發的事件
this.on('onHide', function () {
});
}
|
一個頁面的基本寫法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
define(['AbstractView'], function (AbstractView) {
return _.inherit(AbstractView, {
propertys: function ($super) {
$super();
//一堆基礎屬性定義
//......
//交互業務邏輯
this.events = {
'click .js_pre_day': 'preAction'
};
},
preAction: function (e) { },
addEvent: function () {
this.on('onShow', function () {
//當頁面渲染結束,須要作的初始化操做,好比渲染頁面
//......
});
this.on('onHide', function () {
});
}
});
});
|
只要按照這種規則寫,便能展現頁面,而且具有DOM交互事件。
頁面模塊類
所謂頁面模塊類,即是用於拆分一個頁面爲單個組件模塊所用類,這裏有這些約定:
1
2
3
|
①
一個模塊類實例必定會依賴一個Page的基類實例
②
模塊類實例經過this.view能夠訪問到依賴類的一切資源
③
模塊類實例與模塊之間經過數據entity作通訊
|
這裏代碼能夠再優化,但不是咱們這裏關注的重點:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
|
define([], function () {
'use strict';
return _.inherit({
propertys: function () {
//這裏設置UI的根節點所處包裹層,必須設置
this.$el = null;
//用於定位dom的選擇器
this.selector = '';
//每一個moduleView必須有一個父view,頁面級容器
this.view = null;
//模板字符串,各個組件不一樣,如今加入預編譯機制
this.template = '';
//事件機制
this.events = {};
//實體model,跨模塊通訊的橋樑
this.entity = null;
},
setOption: function (options) {
//這裏能夠寫成switch,開始沒有想到有這麼多分支
for (var k in options) {
if (k == 'events') {
_.extend(this[k], options[k]);
continue;
}
this[k] = options[k];
}
// _.extend(this, options);
},
//@override
initData: function () {
},
//若是傳入了dom便
initWrapper: function (el) {
if (el && el[0]) {
this.$el = el;
return;
}
this.$el = this.view.$(this.selector);
},
initialize: function (opts) {
//這種默認屬性
this.propertys();
//根據參數重置屬性
this.setOption(opts);
this.initData();
this.initWithoutRender();
},
//處理dom已經存在,不須要渲染的狀況
initWithoutRender: function () {
if (this.template) return;
var scope = this;
this.view.on('onShow', function () {
scope.initWrapper();
if (!scope.$el[0]) return;
//若是沒有父view則不能繼續
if (!scope.view) return;
scope.initElement();
scope.bindEvents();
});
},
$: function (selector) {
return this.$el.find(selector);
},
//實例化須要用到到dom元素
initElement: function () { },
//@override
//收集來自各方的實體組成view渲染須要的數據,須要重寫
getViewModel: function () {
throw '必須重寫';
},
_render: function (callback) {
var data = this.getViewModel() || {};
var html = this.template;
if (!this.template) return '';
//引入預編譯機制
if (_.isFunction(this.template)) {
html = this.template(data);
} else {
html = _.template(this.template)(data);
}
typeof callback == 'function' && callback.call(this);
return html;
},
//渲染時必須傳入dom映射
render: function () {
this.initWrapper();
if (!this.$el[0]) return;
//若是沒有父view則不能繼續
if (!this.view) return;
var html = this._render();
this.$el.html(html);
this.initElement();
this.bindEvents();
},
bindEvents: function () {
var events = this.events;
if (!(events || (events = _.result(this, 'events')))) return this;
this.unBindEvents();
// 解析event參數的正則
var delegateEventSplitter = /^(\S+)\s*(.*)$/;
var key, method, match, eventName, selector;
// 作簡單的字符串數據解析
for (key in events) {
method = events[key];
if (!_.isFunction(method)) method = this[events[key]];
if (!method) continue;
match = key.match(delegateEventSplitter);
eventName = match[1], selector = match[2];
method = _.bind(method, this);
eventName += '.delegateUIEvents' + this.id;
if (selector === '') {
this.$el.on(eventName, method);
} else {
this.$el.on(eventName, selector, method);
}
}
return this;
},
unBindEvents: function () {
this.$el.off('.delegateUIEvents' + this.id);
return this;
}
});
});
module.view
|
數據實體類
這裏的數據實體對應着,MVC中的Model,由於以前已經使用model用做了數據請求相關的命名,這裏便使用Entity作該工做:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
define([], function () {
/*
一些原則:
init方法時,不可引發其它字段update
*/
var Entity = _.inherit({
initialize: function (opts) {
this.propertys();
this.setOption(opts);
},
propertys: function () {
//只取頁面展現須要數據
this.data = {};
//局部數據改變對應的響應程序,暫定爲一個方法
//能夠是一個類的實例,若是是實例必須有render方法
this.controllers = {};
this.scope = null;
},
subscribe: function (namespace, callback, scope) {
if (typeof namespace === 'function') {
scope = callback;
callback = namespace;
namespace = 'update';
}
if (!namespace || !callback) return;
if (scope) callback = $.proxy(callback, scope);
if (!this.controllers[namespace]) this.controllers[namespace] = [];
this.controllers[namespace].push(callback);
},
unsubscribe: function (namespace) {
if (!namespace) this.controllers = {};
if (this.controllers[namespace]) this.controllers[namespace] = [];
},
publish: function (namespace, data) {
if (!namespace) return;
if (!this.controllers[namespace]) return;
var arr = this.controllers[namespace];
var i, len = arr.length;
for (i = 0; i < len; i++) {
arr[i](data);
}
},
setOption: function (opts) {
for (var k in opts) {
this[k] = opts[k];
}
},
//首次初始化時,須要矯正數據,好比作服務器適配
//@override
handleData: function () { },
//通常用於首次根據服務器數據源填充數據
initData: function (data) {
var k;
if (!data) return;
//若是默認數據沒有被覆蓋可能有誤
for (k in this.data) {
if (data[k]) this.data[k] = data[k];
}
this.handleData();
this.publish('init', this.get());
},
//驗證data的有效性,若是無效的話,不該該進行如下邏輯,而且應該報警
//@override
validateData: function () {
return true;
},
//獲取數據前,能夠進行格式化
//@override
formatData: function (data) {
return data;
},
//獲取數據
get: function () {
if (!this.validateData()) {
//須要log
return {};
}
return this.formatData(this.data);
},
//數據跟新後須要作的動做,執行對應的controller改變dom
//@override
update: function (key) {
key = key || 'update';
var data = this.get();
this.publish(key, data);
}
});
return Entity;
});
abstract.entity
|
這裏的數據實體會以實例的方式注入給模塊類實例,他的工做是起一箇中樞左右,完成模塊之間的通訊,反正很是重要就是了
其它
數據請求統一使用abstract.model,數據前端緩存使用abstract.store,這裏由於目標是作頁面拆分,請求模塊不是關鍵,各位能夠把這段代碼看層一個簡單的ajax便可:
1
2
3
|
this.model.setParam({});
this.model.execute(function (data) {
});
|
業務入口
最後簡單說下業務入口文件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
(function () {
var project = './';
var viewRoot = 'pages';
require.config({
paths: {
//BUS相關模板根目錄
IndexPath: project + 'pages/index',
ListPath: project + 'pages/list',
BusStore: project + 'model/bus.store',
BusModel: project + 'model/bus.model'
}
});
require(['AbstractApp', 'UIHeader'], function (APP, UIHeader) {
window.APP = new APP({
UIHeader: UIHeader,
viewRootPath: viewRoot
});
window.APP.initApp();
});
})();
|
很簡單的代碼,指定了下require的path配置,最後咱們看看入口頁面的調用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
<!doctype html>
<html>
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no, minimal-ui" />
<meta content="yes" name="apple-mobile-web-app-capable" />
<meta content="black" name="apple-mobile-web-app-status-bar-style" />
<meta name="format-detection" content="telephone=no" />
<link href="../static/css/global.css" rel="stylesheet" type="text/css" />
<title>班次列表</title>
</head>
<body>
<div id="headerview">
<div class="cm-header">
<h1 class="cm-page-title js_title">
正在加載...
</h1>
</div>
</div>
<div class="cm-page-wrap">
<div class="cm-state" id="js_page_state">
</div>
<article class="cm-page" id="main">
</article>
</div>
<script type="text/javascript" src="../blade/libs/zepto.js"></script>
<script src="../blade/libs/fastclick.js" type="text/javascript"></script>
<script type="text/javascript" src="../blade/libs/underscore.js"></script>
<script src="../blade/libs/underscore.extend.js" type="text/javascript"></script>
<script type="text/javascript" src="../blade/libs/require.js"></script>
<script type="text/javascript" src="../blade/common.js"></script>
<script type="text/javascript" src="main.js"></script>
</body>
</html>
list.html
list.html
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
webapp
├─
blade //框架目錄
│
├─data
│
├─libs
│
├─mvc
│
└─ui
├─
bus
│
├─model //數據請求模塊,徹底可使用zepto ajax替換
│
└─pages
│
├─booking
│
├─index
│
└─list //demo代碼模塊
└─
static
|
接下來,讓咱們真實的開始拆分頁面吧。
組件式編程
骨架設計
首先,咱們進行最簡單的骨架設計,這裏依次是其js代碼與模板代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
define(['AbstractView', 'text!ListPath/list.css', 'text!ListPath/tpl.layout.html'], function (AbstractView, style, layoutHtml) {
return _.inherit(AbstractView, {
propertys: function ($super) {
$super();
this.style = style;
this.template = layoutHtml;
},
initHeader: function (name) {
var title = '班次列表';
this.header.set({
view: this,
title: title
});
},
addEvent: function () {
this.on('onShow', function () {
console.log('頁面渲染結束');
});
}
});
});
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
<div class="calendar-bar-wrapper js_calendar_wrapper">
日曆工具條模塊
</div>
<div class="none-data js_none_data" style="display: none;">
當前暫無班次可預訂</div>
<div class="js_list_wrapper">
列表模塊
</div>
<div class="js_list_loading" style="display: none; text-align: center; padding: 10px 0;">
正在加載...</div>
<ul class="bus-tabs list-filter">
<li class="tabs-item js_show_setoutdate">
<div class="line">
<i class="icon-time"></i>出發時段<i class="icon-sec"></i></div>
<div class="line js_day_sec">
全天</div>
</li>
<li class="tabs-item js_show_setstation">
<div class="line">
<i class="icon-circle icon-setout "></i>出發汽車站<i class="icon-sec"></i></div>
<div class="line js_start_sec">
所有車站</div>
</li>
<li class="tabs-item js_show_arrivalstation">
<div class="line">
<i class="icon-circle icon-arrival "></i>到達汽車站<i class="icon-sec"></i></div>
<div class="line js_arrival_sec">
所有車站</div>
</li>
</ul>
tpl.layout
|
頁面展現如圖:
日曆工具欄的實現
這裏要作的第一步是將日曆工具欄模塊實現,以數據爲先的思考,咱們先實現了一個與日曆業務有關的數據實體:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
|
define(['AbstractEntity'], function (AbstractEntity) {
var Entity = _.inherit(AbstractEntity, {
propertys: function ($super) {
$super();
var n = new Date();
var curTime = new Date(n.getFullYear(), n.getMonth(), n.getDate()).getTime();
this.data = {
date: curTime,
title: '當前日期'
};
},
set: function (date) {
if (!date) return;
if (_.isDate(date)) date = date.getTime();
if (typeof date === 'string') date = parseInt(date);
this.data.date = date;
this.update();
},
getDateStr: function () {
var date = new Date();
date.setTime(this.data.date);
var dateDetail = _.dateUtil.getDetail(date);
var name = dateDetail.year + '-' + dateDetail.month + '-' + dateDetail.day + ' ' + dateDetail.weekday + (dateDetail.day1 ? '(' + dateDetail.day1 + ')' : '');
return name;
},
nextDay: function () {
this.set(this.getDate() + 86400000);
return true;
},
getDate: function () {
return parseInt(this.data.date);
},
//是否可以再往前一天
canPreDay: function () {
var n = new Date();
var curTime = new Date(n.getFullYear(), n.getMonth(), n.getDate()).getTime();
//若是當前日期已是第一天,則不可預訂
if (curTime <= this.getDate() - 86400000) {
return true;
}
return false;
},
preDay: function () {
if (!this.canPreDay()) return false;
this.set(this.getDate() - 86400000);
return true;
}
});
return Entity;
});
en.date
|
裏面完成日期工具欄全部相關數據操做,而且不包含實際的業務邏輯。
而後這裏開始設計日期工具欄的模塊View:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
|
define(['ModuleView', 'UICalendarBox', 'text!ListPath/tpl.calendar.bar.html'], function (ModuleView, UICalendarBox, tpl) {
return _.inherit(ModuleView, {
//此處如果要使用model,處實例化時候必定要保證entity的存在,若是不存在即是業務BUG
initData: function () {
this.template = tpl;
this.events = {
'click .js_pre_day': 'preAction',
'click .js_next_day': 'nextAction',
'click .js_show_calendar': 'showCalendar'
};
//初始化時候須要執行的回調
this.dateEntity.subscribe('init', this.render, this);
this.dateEntity.subscribe(this.render, this);
},
initDate: function () {
var t = new Date().getTime();
//默認狀況下獲取當前日期,也有過了18.00就設置爲次日日期
//當時一旦url上有startdatetime參數的話,便須要使用之
if (_.getUrlParam().startdatetime) t = _.getUrlParam().startdatetime;
this.dateEntity.initData({
date: t
});
},
getViewModel: function () {
var data = this.dateEntity.get();
data.formatStr = this.dateEntity.getDateStr();
data.canPreDay = this.dateEntity.canPreDay();
return data;
},
preAction: function () {
if (this.dateEntity.preDay()) return;
this.view.showToast('前一天不可預訂');
},
nextAction: function () {
this.dateEntity.nextDay();
},
showCalendar: function () {
var scope = this, endDate = new Date();
var secDate = new Date();
secDate.setTime(this.dateEntity.getDate());
endDate.setTime(new Date().getTime() + 2592000000);
if (!this.calendar) {
this.calendar = new UICalendarBox({
endTime: endDate,
selectDate: secDate,
onItemClick: function (date, el, e) {
scope.dateEntity.set(date);
this.hide();
}
});
} else {
this.calendar.calendar.selectDate = secDate;
this.calendar.calendar.refresh();
}
this.calendar.show();
}
});
});
mod.date
|
這個組件模塊幹了幾個事情:
① 首先,dateEntity實體須要由list.js這個主view注入
② 這裏爲dateEntity註冊了兩個數據響應事件:
1
2
|
this.dateEntity.subscribe('init', this.render, this);
this.dateEntity.subscribe(this.render, this);
|
render方法繼承至基類,使用template與數據生成html,其中數據產生必須重寫父類一個方法:
1
2
3
4
5
6
|
getViewModel: function () {
var data = this.dateEntity.get();
data.formatStr = this.dateEntity.getDateStr();
data.canPreDay = this.dateEntity.canPreDay();
return data;
},
|
由於這裏的日曆數據,默認取當前時間,可是url參數可能傳遞日期參數,因此定義了一個數據初始化方法:
1
2
3
4
5
6
7
8
9
|
initDate: function () {
var t = new Date().getTime();
//默認狀況下獲取當前日期,也有過了18.00就設置爲次日日期
//當時一旦url上有startdatetime參數的話,便須要使用之
if (_.getUrlParam().startdatetime) t = _.getUrlParam().startdatetime;
this.dateEntity.initData({
date: t
});
},
|
該方法在主頁面渲染結束後會第一時間調用,這個時候日曆工具欄便渲染出來,其中日曆組件的使用便不予理睬了,主控制器的代碼改變以下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
define([
'AbstractView',
'text!ListPath/list.css',
'ListPath/en.date',
'ListPath/mod.date',
'text!ListPath/tpl.layout.html'
], function (
AbstractView,
style,
DateEntity,
DateModule,
layoutHtml
) {
return _.inherit(AbstractView, {
_initEntity: function () {
this.dateEntity = new DateEntity();
},
_initModule: function () {
this.dateModule = new DateModule({
view: this,
selector: '.js_calendar_wrapper',
dateEntity: this.dateEntity
});
},
propertys: function ($super) {
$super();
this._initEntity();
this._initModule();
this.style = style;
this.template = layoutHtml;
},
initHeader: function (name) {
var title = '班次列表';
this.header.set({
view: this,
title: title
});
},
addEvent: function () {
this.on('onShow', function () {
//初始化date數據
this.dateModule.initDate();
});
}
});
});
list.js
|
1
2
3
4
5
6
7
8
9
10
11
|
_initEntity: function () {
this.dateEntity = new DateEntity();
},
_initModule: function () {
this.dateModule = new DateModule({
view: this,
selector: '.js_calendar_wrapper',
dateEntity: this.dateEntity
});
},
|
1
2
3
4
5
6
7
|
addEvent: function () {
this.on('onShow', function () {
//初始化date數據
this.dateModule.initDate();
});
}
|
因而,整個界面變成了這個樣子:
這裏是對應的日曆工具模板文件tpl.calendar.html:
1
2
3
4
5
|
<ul class="bus-tabs calendar-bar">
<li class="tabs-item js_pre_day <%=!canPreDay ? 'disabled' : '' %>">前一天</li>
<li class="tabs-item js_show_calendar" style="-webkit-flex: 2; flex: 2;"><%=formatStr %></li>
<li class="tabs-item js_next_day">後一天</li>
</ul>
|
搜索工具欄的實現
咱們如今的頁面,就算不傳任何URL參數,已經能渲染出部分頁面了,可是下面出發站汽車等業務數據必須等待班次列表數據請求結束才能替換數據,可是這些數據若是沒有出發城市和到達城市是不能發起請求的,因此這裏先實現搜索工具欄功能:
在出發城市或者到達城市不存在的話便彈出搜索工具欄,引導用戶選擇城市,這裏新增彈出層須要在主頁面控制器(檢測主控制器)中使用一個UI組件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
|
define([
'AbstractView',
'text!ListPath/list.css',
'ListPath/en.date',
'ListPath/mod.date',
'text!ListPath/tpl.layout.html',
'text!ListPath/tpl.search.box.html',
'UIScrollLayer'
], function (
AbstractView,
style,
DateEntity,
DateModule,
layoutHtml,
searchBoxHtml,
UIScrollLayer
) {
return _.inherit(AbstractView, {
_initEntity: function () {
this.dateEntity = new DateEntity();
},
_initModule: function () {
this.dateModule = new DateModule({
view: this,
selector: '.js_calendar_wrapper',
dateEntity: this.dateEntity
});
},
propertys: function ($super) {
$super();
this._initEntity();
this._initModule();
this.style = style;
this.template = layoutHtml;
},
initHeader: function (name) {
var title = '班次列表';
this.header.set({
view: this,
title: title,
back: function () {
console.log('回退');
},
right: [
{
tagname: 'search-bar',
value: '搜索',
callback: function () {
console.log('彈出搜索框');
this.showSearchBox();
}
}
]
});
},
//搜索工具彈出層
showSearchBox: function () {
var scope = this;
if (!this.searchBox) {
this.searchBox = new UIScrollLayer({
title: '請選擇搜索條件',
html: searchBoxHtml,
events: {
'click .js-start': function () {
},
'click .js-arrive': function () {
},
'click .js_search_list': function () {
console.log('查詢列表');
}
}
});
}
this.searchBox.show();
},
addEvent: function () {
this.on('onShow', function () {
//初始化date數據
this.dateModule.initDate();
//這裏判斷是否須要彈出搜索彈出層
if (!_.getUrlParam().startcityid || !_.getUrlParam().arrivalcityid) {
this.showSearchBox();
return;
}
});
}
});
});
list.js
|
對應搜索彈出層html模板:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
<div class="c-row search-line" data-flag="start">
<div class="c-span3">
出發</div>
<div class="c-span9 js-start search-line-txt">
請選擇出發地</div>
</div>
<div class="c-row search-line" data-flag="arrive">
<div class="c-span3">
到達</div>
<div class="c-span9 js-arrive search-line-txt">
請選擇到達地</div>
</div>
<div class="c-row " data-flag="arrive">
<span class="btn-primary full-width js_search_list">查詢</span>
</div>
tpl.search.box.html
|
這裏核心代碼是:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
//搜索工具彈出層
showSearchBox: function () {
var scope = this;
if (!this.searchBox) {
this.searchBox = new UIScrollLayer({
title: '請選擇搜索條件',
html: searchBoxHtml,
events: {
'click .js-start': function () {
},
'click .js-arrive': function () {
},
'click .js_search_list': function () {
console.log('查詢列表');
}
}
});
}
this.searchBox.show();
},
|
因而當URL什麼參數都沒有的時候,就會彈出這個搜索框
這裏也迎來了一個難點,由於城市列表事實上應該是一個獨立的可訪問的頁面,可是這裏是想用彈出層的方式調用他,因此我在APP層實現了一個方法能夠用彈出層的方式調起一個獨立的頁面。
1
2
|
注意:
這裏
city城市列表未徹底採用組件化的方式開發,有興趣的朋友能夠本身嘗試着開發
|
這裏有一個不一樣的地方是,由於咱們點擊查詢的時候纔會作實體數據更新,這裏是單純的作DOM操做了,這裏不設置數據實體一個緣由就是:
這個搜索彈出層是一個頁面級DOM以外的部分,數據實體變化通常只應該影響Page級別的DOM,除非真的有兩個頁面級View會公用一個數據實體。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
|
define([
'AbstractView',
'text!ListPath/list.css',
'ListPath/en.date',
'ListPath/mod.date',
'text!ListPath/tpl.layout.html',
'text!ListPath/tpl.search.box.html',
'UIScrollLayer'
], function (
AbstractView,
style,
DateEntity,
DateModule,
layoutHtml,
searchBoxHtml,
UIScrollLayer
) {
return _.inherit(AbstractView, {
_initEntity: function () {
this.dateEntity = new DateEntity();
},
_initModule: function () {
this.dateModule = new DateModule({
view: this,
selector: '.js_calendar_wrapper',
dateEntity: this.dateEntity
});
},
propertys: function ($super) {
$super();
this._initEntity();
this._initModule();
this.style = style;
this.template = layoutHtml;
//主控制器業務屬性
this.urlData = {
start: {},
end: {}
};
},
initHeader: function (name) {
var title = '班次列表';
this.header.set({
view: this,
title: title,
back: function () {
console.log('回退');
},
right: [
{
tagname: 'search-bar',
value: '搜索',
callback: function () {
console.log('彈出搜索框');
this.showSearchBox();
}
}
]
});
},
//搜索工具彈出層
showSearchBox: function () {
var scope = this;
if (!this.searchBox) {
this.searchBox = new UIScrollLayer({
title: '請選擇搜索條件',
html: searchBoxHtml,
events: {
'click .js-start': function (e) {
scope._showCityView('start', $(e.currentTarget));
},
'click .js-arrive': function (e) {
scope._showCityView('end', $(e.currentTarget));
},
'click .js_search_list': function () {
var param = {};
if (!scope.urlData.start.id) {
scope.showToast('請先選擇出發城市');
return;
}
if (!scope.urlData.end.id) {
scope.showToast('請先選擇到達城市');
return;
}
//這裏必定會有出發城市與到達城市等數據
param.startcityid = scope.urlData.start.id;
param.arrivalcityid = scope.urlData.end.id;
param.startdatetime = scope.dateEntity.getDate();
param.startname = scope.urlData.start.name;
param.arrivename = scope.urlData.end.name;
if (scope.urlData.start.station) {
param.startstationid = scope.urlData.start.station
}
if (scope.urlData.end.station) {
param.arrivalstationid = end_station
}
scope.forward('list', param);
this.hide();
}
}
});
}
this.searchBox.show();
},
_showCityView: function (key, el) {
var scope = this;
if (key == 'end') {
//由於到達車站會依賴出發車站的數據,因此這裏得先作判斷
if (!this.urlData.start.id) {
this.showToast('請先選擇出發城市');
return;
}
}
this.showPageView('city', {
flag: key,
startId: this.urlData.start.id,
type: this.urlData.start.type,
onCityItemClick: function (id, name, station, type) {
scope.urlData[key] = {};
scope.urlData[key]['id'] = id;
scope.urlData[key]['type'] = type;
scope.urlData[key]['name'] = name;
if (station) scope.urlData[key]['name'] = station;
el.text(name);
scope.hidePageView();
},
onBackAction: function () {
scope.hidePageView();
}
});
},
addEvent: function () {
this.on('onShow', function () {
//初始化date數據
this.dateModule.initDate();
//這裏判斷是否須要彈出搜索彈出層
if (!_.getUrlParam().startcityid || !_.getUrlParam().arrivalcityid) {
this.showSearchBox();
return;
}
});
}
});
});
list.js
|
搜索功能完成後,咱們這裏即可以進入真正的數據請求功能渲染列表了。
其他模塊
在實現數據請求以前,我按照日期模塊的方式將下面三個模塊的功能也一併完成了,這裏惟一不一樣的是,這些模塊的DOM已經存在,咱們不須要渲染了,完成後的代碼大概是這樣的:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
|
define([
'AbstractView',
'text!ListPath/list.css',
'ListPath/en.station',
'ListPath/en.date',
'ListPath/en.time',
'ListPath/mod.date',
'ListPath/mod.time',
'ListPath/mod.setout',
'ListPath/mod.arrive',
'text!ListPath/tpl.layout.html',
'text!ListPath/tpl.search.box.html',
'UIScrollLayer'
], function (
AbstractView,
style,
StationEntity,
DateEntity,
TimeEntity,
DateModule,
TimeModule,
SetoutModule,
ArriveModule,
layoutHtml,
searchBoxHtml,
UIScrollLayer
) {
return _.inherit(AbstractView, {
_initEntity: function () {
this.dateEntity = new DateEntity();
this.timeEntity = new TimeEntity();
this.timeEntity.subscribe('init', this.renderTime, this);
this.timeEntity.subscribe(this.renderTime, this);
this.setoutEntity = new StationEntity();
this.setoutEntity.subscribe('init', this.renderSetout, this);
this.setoutEntity.subscribe(this.renderSetout, this);
this.arriveEntity = new StationEntity();
this.arriveEntity.subscribe('init', this.renderArrive, this);
this.arriveEntity.subscribe(this.renderArrive, this);
},
_initModule: function () {
this.dateModule = new DateModule({
view: this,
selector: '.js_calendar_wrapper',
dateEntity: this.dateEntity
});
this.timeModule = new TimeModule({
view: this,
selector: '.js_show_setoutdate',
timeEntity: this.timeEntity
});
this.setOutModule = new SetoutModule({
view: this,
selector: '.js_show_setstation',
setoutEntity: this.setoutEntity
});
this.arriveModule = new ArriveModule({
view: this,
selector: '.js_show_arrivalstation',
arriveEntity: this.arriveEntity
});
},
propertys: function ($super) {
$super();
this._initEntity();
this._initModule();
this.style = style;
this.template = layoutHtml;
//主控制器業務屬性
this.urlData = {
start: {},
end: {}
};
},
initHeader: function (name) {
var title = '班次列表';
this.header.set({
view: this,
title: title,
back: function () {
console.log('回退');
},
right: [
{
tagname: 'search-bar',
value: '搜索',
callback: function () {
console.log('彈出搜索框');
this.showSearchBox();
}
}
]
});
},
initElement: function () {
this.d_list_wrapper = this.$('.js_list_wrapper');
this.d_none_data = this.$('.js_none_data');
this.d_js_show_setoutdate = this.$('.js_show_setoutdate');
this.d_js_show_setstation = this.$('.js_show_setstation');
this.d_js_show_arrivalstation = this.$('.js_show_arrivalstation');
this.d_js_list_loading = this.$('.js_list_loading');
this.d_js_tabs = this.$('.js_tabs');
this.d_js_day_sec = this.$('.js_day_sec');
this.d_js_start_sec = this.$('.js_start_sec');
this.d_js_arrival_sec = this.$('.js_arrival_sec');
},
//搜索工具彈出層
showSearchBox: function () {
var scope = this;
if (!this.searchBox) {
this.searchBox = new UIScrollLayer({
title: '請選擇搜索條件',
html: searchBoxHtml,
events: {
'click .js-start': function (e) {
scope._showCityView('start', $(e.currentTarget));
|