mirror of
https://github.com/Jermolene/TiddlyWiki5
synced 2025-01-11 18:00:26 +00:00
First pass at a widget for importing tiddlers
This commit is contained in:
parent
a75fd3df34
commit
c287d9d6d0
196
core/modules/widgets/import.js
Normal file
196
core/modules/widgets/import.js
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
/*\
|
||||||
|
title: $:/core/modules/widget/import.js
|
||||||
|
type: application/javascript
|
||||||
|
module-type: widget
|
||||||
|
|
||||||
|
Implements the import widget.
|
||||||
|
|
||||||
|
```
|
||||||
|
<$import>
|
||||||
|
Import using the "browse..." button or drag files onto this text
|
||||||
|
</$import>
|
||||||
|
```
|
||||||
|
|
||||||
|
\*/
|
||||||
|
(function(){
|
||||||
|
|
||||||
|
/*jslint node: true, browser: true */
|
||||||
|
/*global $tw: false */
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var ImportWidget = function(renderer) {
|
||||||
|
// Save state
|
||||||
|
this.renderer = renderer;
|
||||||
|
// Generate child nodes
|
||||||
|
this.generate();
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.generate = function() {
|
||||||
|
// Get the parameters from the attributes
|
||||||
|
this.browse = this.renderer.getAttribute("browse","yes");
|
||||||
|
this["class"] = this.renderer.getAttribute("class");
|
||||||
|
// Compute classes
|
||||||
|
var classes = ["tw-import"];
|
||||||
|
if(this["class"]) {
|
||||||
|
$tw.utils.pushTop(classes,this["class"]);
|
||||||
|
}
|
||||||
|
// Create the file input and container elements
|
||||||
|
var fileInput = {
|
||||||
|
type: "element",
|
||||||
|
tag: "input",
|
||||||
|
attributes: {
|
||||||
|
type: {type: "string", value: "file"},
|
||||||
|
style: {type: "string", value: this.browse === "no" ? "display: none;" : "display: block;"}
|
||||||
|
},
|
||||||
|
events: [{name: "change", handlerObject: this, handlerMethod: "handleChangeEvent"}]
|
||||||
|
},
|
||||||
|
container = {
|
||||||
|
type: "element",
|
||||||
|
tag: "div",
|
||||||
|
children: this.renderer.parseTreeNode.children,
|
||||||
|
events: [
|
||||||
|
{name: "dragenter", handlerObject: this, handlerMethod: "handleDragEnterEvent"},
|
||||||
|
{name: "dragover", handlerObject: this, handlerMethod: "handleDragOverEvent"},
|
||||||
|
{name: "dragleave", handlerObject: this, handlerMethod: "handleDragLeaveEvent"},
|
||||||
|
{name: "drop", handlerObject: this, handlerMethod: "handleDropEvent"},
|
||||||
|
{name: "paste", handlerObject: this, handlerMethod: "handlePasteEvent"}]
|
||||||
|
};
|
||||||
|
// Set the return element
|
||||||
|
this.tag = "div";
|
||||||
|
this.attributes = {
|
||||||
|
"class": classes.join(" ")
|
||||||
|
};
|
||||||
|
this.children = this.renderer.renderTree.createRenderers(this.renderer.renderContext,[fileInput,container]);
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handleChangeEvent = function(event) {
|
||||||
|
event.stopPropagation();
|
||||||
|
this.importFiles(event.target.files);
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handleDragEnterEvent = function(event) {
|
||||||
|
this.renderer.domNode.classList.add("tw-dragover");
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handleDragOverEvent = function(event) {
|
||||||
|
event.stopPropagation();
|
||||||
|
event.preventDefault();
|
||||||
|
event.dataTransfer.dropEffect = 'copy'; // Explicitly show this is a copy.
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handleDragLeaveEvent = function(event) {
|
||||||
|
this.renderer.domNode.classList.remove("tw-dragover");
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handleDropEvent = function(event) {
|
||||||
|
this.renderer.domNode.classList.remove("tw-dragover");
|
||||||
|
event.stopPropagation();
|
||||||
|
event.preventDefault();
|
||||||
|
this.importFiles(event.dataTransfer.files);
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.handlePasteEvent = function(event) {
|
||||||
|
var self = this,
|
||||||
|
items = event.clipboardData.items;
|
||||||
|
for(var t = 0; t<items.length; t++) {
|
||||||
|
var item = items[t];
|
||||||
|
if(item.kind === "file") {
|
||||||
|
var file = item.getAsFile();
|
||||||
|
this.importFiles([file]);
|
||||||
|
} else if(item.kind === "string") {
|
||||||
|
item.getAsString(function(str) {
|
||||||
|
var fields = {
|
||||||
|
title: self.generateTitle("Untitled"),
|
||||||
|
text: str
|
||||||
|
}
|
||||||
|
self.renderer.renderTree.wiki.addTiddler(new $tw.Tiddler(fields));
|
||||||
|
self.openTiddler(fields.title);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
event.stopPropagation();
|
||||||
|
event.preventDefault();
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.openTiddler = function(title) {
|
||||||
|
$tw.utils.dispatchCustomEvent(this.renderer.domNode,"tw-navigate",{
|
||||||
|
navigateTo: title,
|
||||||
|
navigateFromNode: this.renderer.domNode,
|
||||||
|
navigateFromClientRect: this.renderer.domNode.getBoundingClientRect()
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.importFiles = function(files) {
|
||||||
|
var self = this,
|
||||||
|
importFile = function(file) {
|
||||||
|
// Get the type, falling back to the filename extension
|
||||||
|
var type = file.type;
|
||||||
|
if(type === "" || !type) {
|
||||||
|
var dotPos = file.name.lastIndexOf(".");
|
||||||
|
if(dotPos !== -1) {
|
||||||
|
var fileExtensionInfo = $tw.config.fileExtensionInfo[file.name.substr(dotPos)];
|
||||||
|
if(fileExtensionInfo) {
|
||||||
|
type = fileExtensionInfo.type;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Override HTML files so that they're recognised as TiddlyWiki files
|
||||||
|
if(type === "text/html") {
|
||||||
|
type = "application/vnd.tiddlywiki2";
|
||||||
|
}
|
||||||
|
// Figure out if we're reading a binary file
|
||||||
|
var contentTypeInfo = $tw.config.contentTypeInfo[type],
|
||||||
|
isBinary = contentTypeInfo ? contentTypeInfo.encoding === "base64" : false;
|
||||||
|
// Create the FileReader
|
||||||
|
var reader = new FileReader();
|
||||||
|
reader.onload = function(event) {
|
||||||
|
// Deserialise the file contents
|
||||||
|
var fields = {
|
||||||
|
title: file.name || "Untitled",
|
||||||
|
type: type};
|
||||||
|
// Are we binary?
|
||||||
|
if(isBinary) {
|
||||||
|
var commaPos = event.target.result.indexOf(",");
|
||||||
|
if(commaPos !== -1) {
|
||||||
|
fields.text = event.target.result.substr(commaPos+1);
|
||||||
|
self.renderer.renderTree.wiki.addTiddler(new $tw.Tiddler(fields));
|
||||||
|
self.openTiddler(fields.title);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var tiddlers = self.renderer.renderTree.wiki.deserializeTiddlers(type,event.target.result,fields);
|
||||||
|
if(!tiddlers) {
|
||||||
|
console.log("No tiddlers found in file ",file.name);
|
||||||
|
} else {
|
||||||
|
$tw.utils.each(tiddlers,function(tiddlerFields) {
|
||||||
|
var title = self.generateTitle(tiddlerFields.title);
|
||||||
|
self.renderer.renderTree.wiki.addTiddler(new $tw.Tiddler(tiddlerFields,{title: title}));
|
||||||
|
self.openTiddler(title);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if(isBinary) {
|
||||||
|
reader.readAsDataURL(file);
|
||||||
|
} else {
|
||||||
|
reader.readAsText(file);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
for(var f=0; f<files.length; f++) {
|
||||||
|
importFile(files[f]);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
ImportWidget.prototype.generateTitle = function(baseTitle) {
|
||||||
|
var c = 0;
|
||||||
|
do {
|
||||||
|
var title = baseTitle + (c ? " " + (c + 1) : "");
|
||||||
|
c++;
|
||||||
|
} while(this.renderer.renderTree.wiki.tiddlerExists(title));
|
||||||
|
return title;
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.import = ImportWidget;
|
||||||
|
|
||||||
|
})();
|
@ -150,11 +150,16 @@ a.tw-tiddlylink-missing {
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
-webkit-overflow-scrolling: touch;
|
-webkit-overflow-scrolling: touch;
|
||||||
padding: 4em 0em 2em 2em;
|
padding: 5em 0em 2em 2em;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
text-shadow: 0 1px 0 rgba(255,255,255, 0.8);
|
text-shadow: 0 1px 0 rgba(255,255,255, 0.8);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.sidebar-header p {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
@media print {
|
@media print {
|
||||||
.sidebar-header {
|
.sidebar-header {
|
||||||
display: none;
|
display: none;
|
||||||
@ -366,3 +371,13 @@ embed {
|
|||||||
height: 20em;
|
height: 20em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tw-dragover .tw-drag-target {
|
||||||
|
display: block;
|
||||||
|
background: red;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tw-drag-target {
|
||||||
|
display: none;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
@ -2,6 +2,12 @@ title: $:/templates/ControlPanel
|
|||||||
|
|
||||||
<$button message="tw-new-tiddler" class="btn btn-mini btn-success">New Tiddler</$button>
|
<$button message="tw-new-tiddler" class="btn btn-mini btn-success">New Tiddler</$button>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Import: <$import/>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
<$button message="tw-save-wiki" class="btn btn-mini btn-primary">Save Changes</$button>
|
<$button message="tw-save-wiki" class="btn btn-mini btn-primary">Save Changes</$button>
|
||||||
|
|
||||||
---
|
---
|
||||||
|
@ -3,6 +3,13 @@ title: $:/templates/PageTemplate
|
|||||||
<!-- The navigator catches navigation events and updates the story and history tiddlers -->
|
<!-- The navigator catches navigation events and updates the story and history tiddlers -->
|
||||||
<$navigator story="$:/StoryList" history="$:/HistoryList">
|
<$navigator story="$:/StoryList" history="$:/HistoryList">
|
||||||
|
|
||||||
|
|
||||||
|
<$import browse="no">
|
||||||
|
|
||||||
|
<div class="tw-drag-target">
|
||||||
|
Drop files here to import them as tiddlers
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- The sidebar header -->
|
<!-- The sidebar header -->
|
||||||
<header class="sidebar-header">
|
<header class="sidebar-header">
|
||||||
|
|
||||||
@ -21,5 +28,7 @@ title: $:/templates/PageTemplate
|
|||||||
<!-- End of story river -->
|
<!-- End of story river -->
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
</$import>
|
||||||
|
|
||||||
</$navigator>
|
</$navigator>
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
title: Improvements
|
|
||||||
modifier: JeremyRuston
|
|
||||||
tags: docs introduction
|
tags: docs introduction
|
||||||
|
modifier: JeremyRuston
|
||||||
|
title: Improvements
|
||||||
|
modified: 201304011801
|
||||||
|
|
||||||
TiddlyWiki5 is a complete rewrite of the original TiddlyWiki. It is now based on an elegant [[microkernel architecture|PluginMechanism]], that allows infinite customisation by replacing and augmenting the core modules. The new code is much easier to work with, and will hopefully prove an even more fertile ground for the community of open source developers around TiddlyWiki.
|
TiddlyWiki5 is a complete rewrite of the original TiddlyWiki. It is now based on an elegant [[microkernel architecture|PluginMechanism]], that allows infinite customisation by replacing and augmenting the core modules. The new code is much easier to work with, and will hopefully prove an even more fertile ground for the community of open source developers around TiddlyWiki.
|
||||||
|
|
||||||
@ -11,5 +12,6 @@ For end users, the important improvements include:
|
|||||||
* TiddlyWiki can now be run under [[node.js]] as well as in the browser, allowing it to be used as a personal web server
|
* TiddlyWiki can now be run under [[node.js]] as well as in the browser, allowing it to be used as a personal web server
|
||||||
* Tiddlers containing images are now supported just like WikiText tiddlers, including integrated editing of bitmap and SVG images
|
* Tiddlers containing images are now supported just like WikiText tiddlers, including integrated editing of bitmap and SVG images
|
||||||
* TiddlyWiki5 can directly build both itself and previous (2.x.x) versions of TiddlyWiki from their constituent separate files, without needing external tools
|
* TiddlyWiki5 can directly build both itself and previous (2.x.x) versions of TiddlyWiki from their constituent separate files, without needing external tools
|
||||||
|
* Easily [[import|ImportTiddlers]] content via drag and drop, copy and paste, or browsing for local files
|
||||||
|
|
||||||
The internal changes mean that TiddlyWiki5 is not compatible with previous versions, using different plugins, themes and so on. The changes to the WikiText have been carefully developed to try to maximise backwards compatibility but content prepared for previous versions of TiddlyWiki will need massaging to work properly in TiddlyWiki5.
|
The internal changes mean that TiddlyWiki5 is not compatible with previous versions, using different plugins, themes and so on. The changes to the WikiText have been carefully developed to try to maximise backwards compatibility but content prepared for previous versions of TiddlyWiki will need massaging to work properly in TiddlyWiki5.
|
||||||
|
11
editions/tw5.com/tiddlers/features/ImportTiddlers.tid
Normal file
11
editions/tw5.com/tiddlers/features/ImportTiddlers.tid
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
title: ImportTiddlers
|
||||||
|
tags: docs features
|
||||||
|
modified: 201304011804
|
||||||
|
|
||||||
|
You can import content into a TiddlyWiki file in several ways:
|
||||||
|
|
||||||
|
* Use the ''browse'' button to select one or more local files
|
||||||
|
* Drag and drop files from Windows Explorer or Mac OS X Finder into the TiddlyWiki5 browser window
|
||||||
|
* Paste content directly from the clipboard using the keyboard shortcut (control-V or command-V)
|
||||||
|
** Pasting is currently only supported in GoogleChrome
|
||||||
|
|
Loading…
Reference in New Issue
Block a user