import leanote ok

This commit is contained in:
life
2015-10-27 14:24:59 +08:00
parent fcecc47b65
commit 93f167e7eb
3 changed files with 470 additions and 6 deletions

View File

@@ -16,11 +16,10 @@
author: 'leanote', // 作者, 没用
createdTime: '2015-10-12 12:00:00',
updatedTime: '2015-10-12 12:00:00',
files: {
// fileId =>
'32323': {base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
'32323': {base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
}
files: [
{fileId: '', base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
{fileId: '', base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
]
}
]
}
@@ -110,6 +109,16 @@ define(function() {
}
me.fixFiles(note, function (content, files) {
content = $('<div>' + content + '</div>').html();
var filesArr = [];
files || (files = {});
for (var fileId in files) {
if (files.hasOwnProperty(fileId)) {
files[fileId].fileId = fileId;
filesArr.push(files[fileId]);
}
}
var noteInfo = {
title: note.Title,
content: content,
@@ -118,7 +127,7 @@ define(function() {
isMarkdown: note.IsMarkdown,
createdTime: me.getLeanoteTime(note.CreatedTime),
updatedTime: me.getLeanoteTime(note.UpdatedTime),
files: files || []
files: filesArr
};
info.notes.push(noteInfo);
callback(JSON.stringify(info, null, 2));

View File

@@ -0,0 +1,239 @@
var fs = require('fs');
var Evt = require('evt');
var File = require('file');
var Note = require('note');
var Web = require('web');
var Tag = require('tag');
var async = require('async');
var Common = require('common');
var Import = {
// 解析Leanote
/*
{
exportDate: '2015-10-12 12:00:00',
app: 'leanote.desktop.app.mac',
appVersion: '1.0',
notes: [
{
title: 'life',
content: 'laldfadf', // 图片, 附件链接为 leanote://api/file/getImage?fileId=xxxx, leanote://api/file/getAttach?fileId=3232323
tags: [1,2,3],
isMarkdown: true,
author: 'leanote', // 作者, 没用
createdTime: '2015-10-12 12:00:00',
updatedTime: '2015-10-12 12:00:00',
files: [
{fileId: '', base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
{fileId: '', base64: '', md5: '', type: 'png', 'isAttach': false, createdTime: '2031-12-31 12:12:32'}
]
}
]
}
*/
// callback 是全局的
// eachFileCallback是每一个文件的
// eachNoteFileCallback是每一个笔记的
// filePaths = []
importFromLeanote: function(notebookId, filePaths, callback,
eachFileCallback,
eachNoteCallback) {
var me = this;
// var filePaths = filePaths.split(';');
//
var filePaths = filePaths || [];
async.eachSeries(filePaths, function(path, cb) {
try {
var json = JSON.parse(fs.readFileSync(path));
me.parseLeanote(notebookId, json, function(ret) {
// 单个文件完成
eachFileCallback(ret, path)
cb();
},
// 单个笔记
function(ret) {
eachNoteCallback(ret);
});
} catch(e) {
cb();
return false;
}
}, function() {
// 全部完成
callback(true);
});
},
// 2015-12-12 12:00:00
parseLeanoteTime: function (str) {
if (!str || typeof str != 'string' || str.length != '2015-12-12 12:00:00'.length) {
return new Date();
}
var d = new Date(str);
// invalid
if (isNaN(d.getTime())) {
return new Date();
}
return d;
},
// 处理内容中的链接
fixContentLink: function (note, filesFixed) {
var me = this;
var content = note.content;
var allMatchs = [];
// 图片处理后, 可以替换内容中的链接了
// leanote://api/file/getImage?fileId=xxxx,
var reg = new RegExp('<img([^>]*?)src=["\']?leanote://api/file/getImage\\?fileId=([0-9a-zA-Z]{24})["\']?(.*?)>', 'g');
var matches = reg.exec(content);
while(matches) {
var all = matches[0];
var pre = matches[1]; // img与src之间
var fileId = matches[2];
var back = matches[3]; // src与>之间
allMatchs.push({
fileId: fileId,
pre: pre,
back: back,
all: all
});
// 下一个
matches = reg.exec(content);
}
// 处理附件
var reg = new RegExp('<a([^>]*?)href=["\']?leanote://api/file/getAttach\\?fileId=([0-9a-zA-Z]{24})["\']?(.*?)>([^<]*)</a>', 'g');
var matches = reg.exec(content);
// 先找到所有的
while(matches) {
var all = matches[0];
var pre = matches[1]; // a 与href之间
var fileId = matches[2];
var back = matches[3] // href与>之间
var title = matches[4];
allMatchs.push({
fileId: fileId,
title: title,
pre: pre,
back: back,
isAttach: true,
all: all
});
// 下一个
matches = reg.exec(content);
}
// 替换内容
for (var i = 0; i < allMatchs.length; ++i) {
var eachMatch = allMatchs[i];
var fileInfo = filesFixed[eachMatch.fileId];
var link;
if (!fileInfo) {
link = '';
}
else {
if (!eachMatch.isAttach) {
// 用新的FileId
var href = Api.evtService.localUrl + '/api/file/getImage?fileId=' + fileInfo.FileId;
link = '<img ' + eachMatch.pre + 'src="' + href + '"' + eachMatch.back + '>';
}
else {
var href = Api.evtService.localUrl + '/api/file/getAttach?fileId=' + fileInfo.FileId;
link = '<a ' + eachMatch.pre + 'href="' + href + '"' + eachMatch.back + '>' + eachMatch.title + '</a>';
}
}
content = content.replace(eachMatch.all, link);
}
note.content = content;
},
// 解析笔记
parseNote: function (notebookId, note, callback) {
var me = this;
// 先把files保存到本地
var files = note.files || [];
if (Common.isEmpty(files)) {
files = [];
}
var filesFixed = {}; // fileId(旧) => {fileId: (新fileId)}
var attachs = [];
async.eachSeries(files,
function(file, cb) {
var isAttach = file.isAttach;
File.writeBase64(file.base64, !isAttach, file.type, file.title, function(fileOk) {
if(fileOk) {
filesFixed[file.fileId] = fileOk;
if(isAttach) {
attachs.push(fileOk);
}
} else {
console.log('文件保存错误!');
}
cb();
});
}, function () {
me.fixContentLink(note, filesFixed);
// 添加到数据库中
var jsonNote = {
Title: note.title,
Content: note.content,
Tags: note.tags || [],
CreatedTime: me.parseLeanoteTime(note.createdTime),
UpdatedTime: me.parseLeanoteTime(note.updatedTime),
IsMarkdown: note.isMarkdown || false,
Attachs: attachs,
NotebookId: notebookId,
Desc: '',
NoteId: Common.objectId(),
IsNew: true
};
jsonNote._id = jsonNote.NoteId;
for(var h = 0; h < jsonNote.Tags.length; ++h) {
var tagTitle = jsonNote.Tags[h];
if (tagTitle) {
Tag.addOrUpdateTag(tagTitle, function(tag) {
Web.addTag(tag);
});
}
}
Note.updateNoteOrContent(jsonNote, function(insertedNote) {
callback && callback(insertedNote);
});
}
);
},
parseLeanote: function (notebookId, json, callback, eachCallback) {
var me = this;
var notes = json.notes || [];
if (Common.isEmpty(notes)) {
callback(true);
return;
}
async.eachSeries(notes, function(note, cb) {
me.parseNote(notebookId, note, function (insertedNote) {
eachCallback(insertedNote);
cb();
});
}, function () {
callback(true);
});
}
};
module.exports = Import;

View File

@@ -0,0 +1,216 @@
/**
* 导入leanote, 重构
* @author life@leanote.com
* @date 2015/04/09
*/
define(function() {
var importService = nodeRequire('./public/plugins/import_leanote/import');
var leanote = {
langs: {
'en-us': {
'importLeanote': 'Import Leanote',
},
'zh-cn': {
'importLeanote': '导入Leanote',
'Choose Leanote files(.enex)': '选择Leanote文件(.enex)',
'Close': "关闭",
'Import to': "导入至",
"Done! %s notes imported!": "完成, 成功导入 %s 个笔记!",
"Import file: %s Success!": "文件 %s 导入成功!",
"Import file: %s Failure, is leanote file ?": "文件 %s 导入失败! 是Leanote文件?",
"Import: %s Success!": "导入笔记: %s 成功!"
},
'zh-hk': {
'importLeanote': '導入Leanote',
'Choose Leanote files(.enex)': '選擇Leanote文件(.enex)',
'Close': "關閉",
"Import to": "導入至",
"Done! %s notes imported!": "完成, 成功導入 %s 個筆記!",
"Import file: %s Success!": "文件 %s 導入成功!",
"Import file: %s Failure, is leanote file ?": "文件 %s 導入失敗! 是Leanote文件?",
"Import: %s Success!": "導入筆記: %s 成功!"
}
},
_tpl: `
<style>
#importLeanoteDialog .tab-pane {
text-align: center;
padding: 10px;
padding-top: 20px;
}
#importLeanoteDialog .alert {
margin-top: 10px;
padding: 0;
border: none;
}
</style>
<div class="modal fade bs-modal-sm" id="importLeanoteDialog" tabindex="-1" role="dialog" aria-labelledby="mySmallModalLabel">
<div class="modal-dialog modal-sm">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
<h4 class="modal-title" class="modalTitle"><span class="lang">Import to</span> <span id="importDialogNotebook"></span></h4>
</div>
<div class="modal-body" id="">
<div role="tabpanel">
<!-- Nav tabs -->
<ul class="nav nav-tabs" role="tablist">
<li role="presentation" class="active"><a href="#leanoteTab" aria-controls="leanoteTab" role="tab" data-toggle="tab">Leanote</a></li>
</ul>
<!-- Tab panes -->
<div class="tab-content">
<div role="tabpanel" class="tab-pane active" id="leanoteTab">
<!-- import -->
<a id="chooseLeanoteFile" class="btn btn-success btn-choose-file">
<i class="fa fa-upload"></i>
<span class="lang">Choose Leanote files(.leanote)</span>
</a>
<!-- 消息 -->
<div id="importLeanoteMsg" class="alert alert-info">
<div class="curImportFile"></div>
<div class="curImportNote"></div>
<div class="allImport"></div>
</div>
</div>
<div role="tabpanel" class="tab-pane" id="youdaoTab">
<!-- 文件选择框 -->
<input id="importLeanoteInput" type="file" nwsaveas="" accept=".enex" multiple style="" style="display: none"/>
</div>
</div>
</div>
</div>
<div class="modal-footer ">
<button type="button" class="btn btn-default upgrade-cancel-btn lang" data-dismiss="modal">Close</button>
</div>
</div><!-- /.modal-content -->
</div><!-- /.modal-dialog -->
</div><!-- /.modal -->
`,
_importDialog: null,
_curNotebook: null,
_inited: false,
getMsg: function(txt, data) {
return Api.getMsg(txt, 'plugin.import_leanote', data)
},
init: function() {
var me = this;
me._inited = true;
$('body').append(me._tpl);
me._importDialog = $("#importLeanoteDialog");
me._importDialog.find('.lang').each(function() {
var txt = $.trim($(this).text());
$(this).text(me.getMsg(txt));
});
// 导入, 选择文件
$('#chooseLeanoteFile').click(function() {
Api.gui.dialog.showOpenDialog(Api.gui.getCurrentWindow(),
{
properties: ['openFile', 'multiSelections'],
filters: [
{ name: 'Leanote', extensions: ['leanote'] }
]
},
function(paths) {
if(!paths) {
return;
}
var notebookId = me._curNotebook.NotebookId;
var n = 0;
me.clear();
importService.importFromLeanote(notebookId, paths,
// 全局
function(ok) {
// $('#importLeanoteMsg .curImportFile').html("");
// $('#importLeanoteMsg .curImportNote').html("");
setTimeout(function() {
$('#importLeanoteMsg .allImport').html(me.getMsg('Done! %s notes imported!', n));
}, 500);
},
// 单个文件
function(ok, filename) {
if(ok) {
$('#importLeanoteMsg .curImportFile').html(me.getMsg("Import file: %s Success!", filename));
} else {
$('#importLeanoteMsg .curImportFile').html(me.getMsg("Import file: %s Failure, is leanote file ?", filename));
}
},
// 单个笔记
function(note) {
if(note) {
n++;
$('#importLeanoteMsg .curImportNote').html(me.getMsg("Import: %s Success!", note.Title));
// 插入到当前笔记中
Note.addSync([note]);
}
}
);
}
);
});
},
clear: function() {
$('#importLeanoteMsg .curImportFile').html("");
$('#importLeanoteMsg .curImportNote').html("");
$('#importLeanoteMsg .allImport').html('');
},
open: function(notebook) {
var me = this;
if(!notebook) {
return;
}
if(!me._inited) {
me.init();
}
me.clear();
$('#importDialogNotebook').html(notebook.Title);
me._curNotebook = notebook;
var notebookId = notebook.NotebookId;
me._importDialog.modal('show');
},
// 打开前要执行的
onOpen: function() {
var me = this;
var gui = Api.gui;
Api.addImportMenu({
label: Api.getMsg('plugin.import_leanote.importLeanote'),
click: (function() {
return function(notebook) {
me.open(notebook);
};
})()
});
},
// 打开后
onOpenAfter: function() {
},
// 关闭时需要运行的
onClose: function() {
}
};
return leanote;
});