这里文章都是从个人的github博客直接复制过来的,排版可能有点乱. 原始地址
http://benq.im/2015/04/24/hexomd-03/
上一篇我们实现了文件的新建,保存,打开功能.
在这篇里我们将实现以下功能:
- 系统模块,包含一些软件的设置和存储功能
- 记录上次打开的文件
- 编辑器样式选择
系统模块
跟之前的studio
模块类似,我们在modules模块下增加system
目录.
比studio多了model.js
文件,用来实现系统模块的一些功能.
在app.js
里加载system
模块
1 2 3 4 5 6
|
angular.module('hmd', ['ui.router','hmd.directives','hmd.studio','hmd.system']) ... //引入模块 hmd.regModule('studio'); hmd.regModule('system'); ...
|
路由、导航栏
angular.js
用得不熟,导航栏的状态根据route来切换一直不知道怎么实现比较优雅.
我直接在app.js
里增加了一个导航栏切换的方法,每个route的onEnter事件里自行调用这个方法.
1 2 3 4 5 6
|
//TODO:更优雅的导航栏切换逻辑 hmd.changeStatus = function (state) { var $navList = $(' $navList.find('li').removeClass('active'); $navList.find('.' + state).addClass('active'); };
|
system/route.js
1 2 3 4 5 6 7 8 9 10 11
|
hmd.system.config(function ($stateProvider, $urlRouterProvider) { $stateProvider .state('system', { url: "/system", templateUrl: "modules/system/views/system.html", controller: 'system', onEnter: function () { hmd.changeStatus('system'); } }); });
|
studio/route.js
1 2 3 4 5
|
... onEnter: function () { hmd.changeStatus('studio'); } ...
|
然后在index.html
里配置好导航
1 2 3 4 5 6
|
... <ul class="nav navbar-nav" id="navlist" > <li class="studio"><a href="#/studio">编辑器</a></li> <li class="system"><a href="#/system">系统设置</a></li> </ul> ...
|
导航栏最终效果:
记录上次打开的文件
每次打开文件都会被记住,下次重新启动程序时将默认打开最后一次打开的文件.
system设置的读取和保存
我们先在system/model.js
实现保存和读取设置的功能.
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
|
var util = require('./helpers/util'), fs = require('fs'), system = hmd.system, dataFile = system.dataPath + '\system.json';
if (!fs.existsSync(system.dataPath)) { fs.mkdirSync(system.dataPath); }
var defaultSystemData = { lastFile: null };
system.get = function () { return $.extend(defaultSystemData,util.readJsonSync(dataFile)); };
system.save = function (data) { data = data || defaultSystemData; util.writeFileSync(dataFile, JSON.stringify(data)); };
system.setLastFile = function (filepath) { var systemData = system.get(); systemData.lastFile = filepath; system.save(systemData); };
|
system
实现了get
和save
方法,所有的设置都存储在一个简单的对象里,代码里并没有对这个对象做缓存,每次都是从文件里读取,因为这简单的文件还远达不到影响读取速度的情况.
然后我们修改editor
的setFile
方法,暴露setFiled
事件给外部使用.
1 2 3 4 5 6 7 8 9 10 11 12 13 14
|
setFile:function(filepath){ if(filepath && fs.existsSync(filepath)){ var txt = util.readFileSync(filepath); this.filepath = filepath; this.cm.setValue(txt); this.fire('setFiled',this.filepath); } else{ this.filepath = null; this.cm.setValue(''); this.fire('setFiled'); } }
|
最后修改studio/directives.js
的hmdEditor
,实现这个功能.
1 2 3 4 5 6 7 8 9 10
|
studio.directive('hmdEditor', function () { return function ($scope, elem) { var systemData = hmd.system.get(); hmd.editor.init({el:elem[0]},systemData.lastFile); hmd.editor.on('setFiled',function(filepath){ hmd.system.setLastFile(filepath); }); ...
|
编辑器样式选择
样式修改表单
样式文件在目录app/lib/codemirror/theme.
目录里每一个样式文件代表一种编辑器样式,还记得当初实现editor
的init
时,样式已经是通过配置传入的.
1 2 3 4 5
|
... if(options.theme != 'default'){ $('head').append('<link href="lib/codemirror/theme/'+options.theme+'.css" rel="stylesheet" />'); } ...
|
现在我们只要把theme参数存储到配置里,并提供给用户修改就可以.
在system/model.js
里的默认配置增加一个theme
字段.
1 2 3 4 5 6 7 8 9
|
... var defaultSystemData = { lastFile: null, theme:'ambiance' }; ...
|
修改system/views/system.html
模版,增加样式表单
1 2 3 4 5 6 7 8 9 10 11 12 13
|
<div class="content studio-wrap"> <form class="system-form" name="systemForm"> <div class="form-group"> <label>编辑器样式</label> <select ng-model="systemSetting.theme" name="theme"> <option value="ambiance">ambiance</option> <option value="mbo">mbo</option> <option value="neat">neat</option> </select> </div> <button type="submit" class="btn btn-default" ng-click="save(systemSetting)">保存</button> </form> </div>
|
这里的select控件我们先写了3个选项.现在先实现这个修改样式的功能,等完成这个功能后再把选项列表做成自动生成.
对应的system/controllers.js
(开发了3天了,终于第一次用到controller了)
1 2 3 4 5 6
|
system.controller('system', function ($scope) { $scope.systemSetting = system.get(); $scope.save = function (systemSetting) { system.save(systemSetting); }; });
|
controller
里读取system的数据,并赋值给$scope.systemSetting
,用于表单的数据绑定.由于angular
实现了数据的双向绑定,因此用户编辑表单时,绑定的数据也会跟着更新.这样我们的save
方法里只要将表单绑定的数据保存回system即可.
button按钮绑定save方法ng-click="save(systemSetting)"
.
这里可以稍微感受到angular
让我们节省了很多工作量.
自动生成select列表
修改controller
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
|
var system = hmd.system, fs = require('fs'); system.controller('system', function ($scope) { var files = fs.readdirSync('./app/lib/codemirror/theme'),themes={}; files.forEach(function (file) { if(~file.indexOf('.css')){ file = file.replace('.css',''); themes[file] = file; } }); $scope.themes = themes; $scope.systemSetting = system.get(); $scope.save = function (systemSetting) { system.save(systemSetting); }; });
|
从theme
目录里读取所有样式列表,生成键值对,然后赋值给$scope.themes
修改视图模版:
1 2
|
<select name="theme" ng-model="systemSetting.theme" ng-options="k as v for (k, v) in themes"> </select>
|
ng-options="k as v for (k, v) in themes"
是angular的绑定语法
这样我们就实现了样式列表的自动读取,用户如果想自定义样式,只要在app/lib/codemirror/theme
目录新增一个样式文件,并写上自己的样式就可以在系统设置里选择自定义的样式了.
总结
今天实现了记忆最后一次打开的文件以及样式选择的功能,并且第一次使用了angular
的controller
,感受到了angular
双向数据绑定的强大.
我们的程序又更好用一些了(但是随着界面变多,又更丑了,太为难了).
最终效果截图
附件
本篇程序打包
项目地址