Welcome to TiddlyWiki created by Jeremy Ruston; Copyright © 2004-2007 Jeremy Ruston, Copyright © 2007-2011 UnaMesa Association
Background: #fff
Foreground: #000
PrimaryPale: #8cf
PrimaryLight: #18f
PrimaryMid: #04b
PrimaryDark: #014
SecondaryPale: #ffc
SecondaryLight: #fe8
SecondaryMid: #db4
SecondaryDark: #841
TertiaryPale: #eee
TertiaryLight: #ccc
TertiaryMid: #999
TertiaryDark: #666
Error: #f88
<!--{{{-->
<div class='toolbar' macro='toolbar [[ToolbarCommands::EditToolbar]]'></div>
<div class='title' macro='view title'></div>
<div class='editor' macro='edit title'></div>
<div macro='annotations'></div>
<div class='editor' macro='edit text'></div>
<div class='editor' macro='edit tags'></div><div class='editorFooter'><span macro='message views.editor.tagPrompt'></span><span macro='tagChooser excludeLists'></span></div>
<!--}}}-->
When getting started, you may want to:
* Set your username for signing your edits: <<option txtUserName>>
* Change the page [[title|SiteTitle]] (now "<<tiddler SiteTitle>>") and [[subtitle|SiteSubtitle]] (now "<<tiddler SiteSubtitle>>"); they also set the browser tab title
* Create a tiddler where your content "starts"
** Use the button on the sidebar or [[link|My first tiddler]] it here, follow the link, edit, and click "done"
** It will be shown in the Timeline (usually on the right), but you may want to link it in the MainMenu (usually on the left)
** and/or make it open when the ~TiddlyWiki is opened by editing the list of [[DefaultTiddlers]] (separate links with spaces or linebreaks)
* Save your ~TiddlyWiki
** Although "download saving" works in any browser, it's not that convenient, so you'll probably want to use [[a dedicated saver|https://classic.tiddlywiki.com/#%5B%5BSetting up saving%5D%5D]]
<!--{{{-->
<link rel="icon" href="data:image/svg+xml,<svg xmlns=%22http://www.w3.org/2000/svg%22 viewBox=%220 0 100 100%22><text y=%22.9em%22 font-size=%2280%22>🖼️</text></svg>">
<link rel='alternate' type='application/rss+xml' title='RSS' href='index.xml' />
<!--}}}-->
These [[InterfaceOptions]] for customising [[TiddlyWiki]] are saved in your browser
Your username for signing your edits. Write it as a [[WikiWord]] (eg [[JoeBloggs]])
<<option txtUserName>>
<<option chkSaveBackups>> [[SaveBackups]]
<<option chkAutoSave>> [[AutoSave]]
<<option chkRegExpSearch>> [[RegExpSearch]]
<<option chkCaseSensitiveSearch>> [[CaseSensitiveSearch]]
<<option chkAnimate>> [[EnableAnimations]]
----
Also see [[AdvancedOptions]]
<!--{{{-->
<div class='header' role='banner'>
<div class='headerShadow'>
<span class='siteTitle' refresh='content' tiddler='SiteTitle'></span>
<span class='siteSubtitle' refresh='content' tiddler='SiteSubtitle'></span>
</div>
<div class='headerForeground'>
<span class='siteTitle' refresh='content' tiddler='SiteTitle'></span>
<span class='siteSubtitle' refresh='content' tiddler='SiteSubtitle'></span>
</div>
</div>
<div id='mainMenu' role='navigation' refresh='content' tiddler='MainMenu'></div>
<div id='sidebar'>
<div id='sidebarOptions' role='navigation' refresh='content' tiddler='SideBarOptions'></div>
<div id='sidebarTabs' role='complementary' refresh='content' force='true' tiddler='SideBarTabs'></div>
</div>
<div id='displayArea' role='main'>
<div id='messageArea'></div>
<div id='tiddlerDisplay'></div>
</div>
<!--}}}-->
/*{{{*/
body {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
a {color:[[ColorPalette::PrimaryMid]];}
a:hover {background-color:[[ColorPalette::PrimaryMid]]; color:[[ColorPalette::Background]];}
a img {border:0;}
h1, h2, h3, h4, h5, h6 { color: [[ColorPalette::SecondaryDark]]; }
h1 {border-bottom:2px solid [[ColorPalette::TertiaryLight]];}
h2,h3 {border-bottom:1px solid [[ColorPalette::TertiaryLight]];}
.txtOptionInput {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
.button {color:[[ColorPalette::PrimaryDark]]; border:1px solid [[ColorPalette::Background]];}
.button:hover {color:[[ColorPalette::PrimaryDark]]; background:[[ColorPalette::SecondaryLight]]; border-color:[[ColorPalette::SecondaryMid]];}
.button:active {color:[[ColorPalette::Background]]; background:[[ColorPalette::SecondaryMid]]; border:1px solid [[ColorPalette::SecondaryDark]];}
.header {
background: -moz-linear-gradient(to bottom, [[ColorPalette::PrimaryLight]], [[ColorPalette::PrimaryMid]]);
background: linear-gradient(to bottom, [[ColorPalette::PrimaryLight]], [[ColorPalette::PrimaryMid]]);
}
.header a:hover {background:transparent;}
.headerShadow {color:[[ColorPalette::Foreground]];}
.headerShadow a {font-weight:normal; color:[[ColorPalette::Foreground]];}
.headerForeground {color:[[ColorPalette::Background]];}
.headerForeground a {font-weight:normal; color:[[ColorPalette::PrimaryPale]];}
.tabSelected {
color:[[ColorPalette::Foreground]];
background:[[ColorPalette::Background]];
border-left:1px solid [[ColorPalette::TertiaryLight]];
border-top:1px solid [[ColorPalette::TertiaryLight]];
border-right:1px solid [[ColorPalette::TertiaryLight]];
}
.tabUnselected {color:[[ColorPalette::Background]]; background:[[ColorPalette::TertiaryMid]];}
.tabContents {border:1px solid [[ColorPalette::TertiaryLight]];}
.tabContents .button {border:0;}
#sidebar {}
#sidebarOptions input {border:1px solid [[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel {background:[[ColorPalette::PrimaryPale]];}
#sidebarOptions .sliderPanel a {border:none;color:[[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel a:hover {color:[[ColorPalette::Background]]; background:[[ColorPalette::PrimaryMid]];}
#sidebarOptions .sliderPanel a:active {color:[[ColorPalette::PrimaryMid]]; background:[[ColorPalette::Background]];}
.wizard { background:[[ColorPalette::PrimaryPale]]; }
.wizard__title { color:[[ColorPalette::PrimaryDark]]; border:none; }
.wizard__subtitle { color:[[ColorPalette::Foreground]]; border:none; }
.wizardStep { background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]]; }
.wizardStep.wizardStepDone {background:[[ColorPalette::TertiaryLight]];}
.wizardFooter {background:[[ColorPalette::PrimaryPale]];}
.wizardFooter .status {background:[[ColorPalette::PrimaryDark]]; color:[[ColorPalette::Background]];}
.wizardFooter .status a { color: [[ColorPalette::PrimaryPale]]; }
.wizard .button {
color:[[ColorPalette::Foreground]]; background:[[ColorPalette::SecondaryLight]]; border: 1px solid;
border-color:[[ColorPalette::SecondaryDark]];
}
.wizard .button:hover {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::Background]];}
.wizard .button:active {
color:[[ColorPalette::Background]]; background:[[ColorPalette::Foreground]]; border: 1px solid;
border-color:[[ColorPalette::PrimaryDark]] [[ColorPalette::PrimaryPale]] [[ColorPalette::PrimaryPale]] [[ColorPalette::PrimaryDark]];
}
.wizard .notChanged {background:transparent;}
.wizard .changedLocally {background:#80ff80;}
.wizard .changedServer {background:#8080ff;}
.wizard .changedBoth {background:#ff8080;}
.wizard .notFound {background:#ffff80;}
.wizard .putToServer {background:#ff80ff;}
.wizard .gotFromServer {background:#80ffff;}
#messageArea { background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; box-shadow: 1px 2px 5px [[ColorPalette::TertiaryMid]]; }
.messageToolbar__button { color:[[ColorPalette::PrimaryMid]]; background:[[ColorPalette::SecondaryPale]]; border:none; }
.messageToolbar__button_withIcon { background:inherit; }
.messageToolbar__button_withIcon:active { background:inherit; border:none; }
.tw-icon line { stroke: [[ColorPalette::TertiaryDark]]; }
.messageToolbar__button:hover .tw-icon line { stroke: [[ColorPalette::Foreground]]; }
.popup {
background: [[ColorPalette::Background]];
color: [[ColorPalette::TertiaryDark]];
box-shadow: 1px 2px 5px [[ColorPalette::TertiaryMid]];
}
.popup li a, .popup li a:visited, .popup li a:hover, .popup li a:active {
color:[[ColorPalette::Foreground]]; border: none;
}
.popup li a:hover { background:[[ColorPalette::SecondaryLight]]; }
.popup li a:active { background:[[ColorPalette::SecondaryPale]]; }
.popup li.disabled { color:[[ColorPalette::TertiaryMid]]; }
.popupHighlight {color:[[ColorPalette::Foreground]];}
.popup hr {color:[[ColorPalette::PrimaryDark]]; background:[[ColorPalette::PrimaryDark]]; border-bottom:1px;}
.listBreak div {border-bottom:1px solid [[ColorPalette::TertiaryDark]];}
.popupTiddler {background:[[ColorPalette::TertiaryPale]]; border:2px solid [[ColorPalette::TertiaryMid]];}
.tiddler .defaultCommand {font-weight:bold;}
.shadow .title {color:[[ColorPalette::TertiaryDark]];}
.title {color:[[ColorPalette::SecondaryDark]];}
.subtitle {color:[[ColorPalette::TertiaryDark]];}
.toolbar {color:[[ColorPalette::PrimaryMid]];}
.toolbar a {color:[[ColorPalette::TertiaryLight]];}
.selected .toolbar a {color:[[ColorPalette::TertiaryMid]];}
.selected .toolbar a:hover {color:[[ColorPalette::Foreground]];}
.tagging, .tagged { border: 2px solid [[ColorPalette::TertiaryPale]]; }
.selected .tagging, .selected .tagged { border: 2px solid [[ColorPalette::TertiaryLight]]; }
.tagging .listTitle, .tagged .listTitle {color:[[ColorPalette::PrimaryDark]];}
.tagging .button, .tagged .button { border:none; }
.footer {color:[[ColorPalette::TertiaryLight]];}
.selected .footer {color:[[ColorPalette::TertiaryMid]];}
.error, .errorButton {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::Error]];}
.warning {color:[[ColorPalette::Foreground]]; background:[[ColorPalette::SecondaryPale]];}
.lowlight {background:[[ColorPalette::TertiaryLight]];}
.zoomer {background:none; color:[[ColorPalette::TertiaryMid]]; border:3px solid [[ColorPalette::TertiaryMid]];}
.imageLink, #displayArea .imageLink {background:transparent;}
.annotation { background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; }
.viewer .listTitle {list-style-type:none; margin-left:-2em;}
.viewer .button {border:1px solid [[ColorPalette::SecondaryMid]];}
.viewer blockquote {border-left:3px solid [[ColorPalette::TertiaryDark]];}
.viewer th, .viewer thead td, .twtable th, .twtable thead td { background: [[ColorPalette::SecondaryMid]]; color: [[ColorPalette::Background]]; }
.viewer td, .viewer tr, .twtable td, .twtable tr { border: 1px solid [[ColorPalette::TertiaryLight]]; }
.twtable caption { color: [[ColorPalette::TertiaryMid]]; }
.viewer pre {background:[[ColorPalette::SecondaryPale]];}
.viewer code {color:[[ColorPalette::SecondaryDark]];}
.viewer hr {border:0; border-top:dashed 1px [[ColorPalette::TertiaryDark]]; color:[[ColorPalette::TertiaryDark]];}
.highlight, .marked {background:[[ColorPalette::SecondaryLight]];}
.editor input {border:1px solid [[ColorPalette::PrimaryMid]]; background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
.editor textarea {border:1px solid [[ColorPalette::PrimaryMid]]; width:100%; background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
.editorFooter {color:[[ColorPalette::TertiaryMid]];}
.readOnly {background:[[ColorPalette::TertiaryPale]];}
#backstageArea {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::TertiaryMid]];}
#backstageArea a {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::Background]]; border:none;}
#backstageArea a:hover {background:[[ColorPalette::SecondaryLight]]; color:[[ColorPalette::Foreground]]; }
#backstageArea a.backstageSelTab {background:[[ColorPalette::Background]]; color:[[ColorPalette::Foreground]];}
#backstageButton a {background:none; color:[[ColorPalette::Background]]; border:none;}
#backstageButton a:hover {background:[[ColorPalette::Foreground]]; color:[[ColorPalette::Background]]; border:none;}
#backstagePanel {background:[[ColorPalette::Background]]; border-color: [[ColorPalette::Background]] [[ColorPalette::TertiaryDark]] [[ColorPalette::TertiaryDark]] [[ColorPalette::TertiaryDark]];}
.backstagePanelFooter .button {border:none; color:[[ColorPalette::Background]];}
.backstagePanelFooter .button:hover {color:[[ColorPalette::Foreground]];}
#backstageCloak {background:[[ColorPalette::Foreground]]; opacity:0.6; filter:alpha(opacity=60);}
/*}}}*/
/*{{{*/
body { font-size:.75em; font-family:arial,helvetica,sans-serif; margin:0; padding:0; }
* html .tiddler {height:1%;}
h1,h2,h3,h4,h5,h6 {font-weight:bold; text-decoration:none;}
h1,h2,h3 {padding-bottom:1px; margin-top:1.2em;margin-bottom:0.3em;}
h4,h5,h6 {margin-top:1em;}
h1 {font-size:1.35em;}
h2 {font-size:1.25em;}
h3 {font-size:1.1em;}
h4 {font-size:1em;}
h5 {font-size:.9em;}
hr {height:1px;}
dt {font-weight:bold;}
ol {list-style-type:decimal;}
ol ol {list-style-type:lower-alpha;}
ol ol ol {list-style-type:lower-roman;}
ol ol ol ol {list-style-type:decimal;}
ol ol ol ol ol {list-style-type:lower-alpha;}
ol ol ol ol ol ol {list-style-type:lower-roman;}
ol ol ol ol ol ol ol {list-style-type:decimal;}
.txtOptionInput {width:11em; border-width: 1px; }
#contentWrapper .chkOptionInput {border:0;}
.indent {margin-left:3em;}
.outdent {margin-left:3em; text-indent:-3em;}
code.escaped {white-space:nowrap;}
a {text-decoration:none;}
.externalLink {text-decoration:underline;}
.tiddlyLinkExisting {font-weight:bold;}
.tiddlyLinkNonExisting {font-style:italic;}
/* the 'a' is required for IE, otherwise it renders the whole tiddler in bold */
a.tiddlyLinkNonExisting.shadow {font-weight:bold;}
#mainMenu .tiddlyLinkExisting,
#mainMenu .tiddlyLinkNonExisting,
#sidebarTabs .tiddlyLinkNonExisting {font-weight:normal; font-style:normal;}
#sidebarTabs .tiddlyLinkExisting {font-weight:bold; font-style:normal;}
.header {position:relative;}
.headerShadow {position:relative; padding:3em 0 1em 1em; left:-1px; top:-1px;}
.headerForeground {position:absolute; padding:3em 0 1em 1em; left:0; top:0;}
.siteTitle {font-size:3em;}
.siteSubtitle {font-size:1.2em;}
#mainMenu {position:absolute; left:0; width:10em; text-align:right; line-height:1.6em; padding:1.5em 0.5em 0.5em 0.5em; font-size:1.1em;}
#sidebar {position:absolute; right:3px; width:16em; font-size:.9em;}
#sidebarOptions {padding-top:0.3em;}
#sidebarOptions a {margin:0 0.2em; padding:0.2em 0.3em; display:block;}
#sidebarOptions input {margin:0.4em 0.5em;}
#sidebarOptions .sliderPanel {margin-left:1em; padding:0.5em; font-size:.85em;}
#sidebarOptions .sliderPanel a {font-weight:bold; display:inline; padding:0;}
#sidebarOptions .sliderPanel input {margin:0 0 0.3em 0;}
#sidebarTabs .tabContents {width:15em; overflow:hidden;}
.wizard { padding:0.1em 2em 0; }
.wizard__title { font-size:2em; }
.wizard__subtitle { font-size:1.2em; }
.wizard__title, .wizard__subtitle { font-weight:bold; background:none; padding:0; margin:0.4em 0 0.2em; }
.wizardStep { padding:1em; }
.wizardFooter { padding: 0.8em 0; }
.wizardFooter .status { display: inline-block; line-height: 1.5; padding: 0.3em 1em; }
.wizardFooter .button { margin:0.5em 0 0; font-size:1.2em; padding:0.2em 0.5em; }
#messageArea { position:fixed; top:2em; right:0; margin:0.5em; padding:0.7em 1em; z-index:2000; }
.messageToolbar { text-align:right; padding:0.2em 0; }
.messageToolbar__button { text-decoration:underline; }
.messageToolbar__button_withIcon { display: inline-block; }
.tw-icon { height: 1em; width: 1em; } /* width for IE */
.tw-icon line { stroke-width: 1; stroke-linecap: round; }
.messageArea__text a { text-decoration:underline; }
.popup {position:absolute; z-index:300; font-size:.9em; padding:0.3em 0; list-style:none; margin:0;}
.popup .popupMessage, .popup li.disabled, .popup li a { padding: 0.3em 0.7em; }
.popup li a {display:block; font-weight:normal; cursor:pointer;}
.popup hr {display:block; height:1px; width:auto; padding:0; margin:0.2em 0;}
.listBreak {font-size:1px; line-height:1px;}
.listBreak div {margin:2px 0;}
.tiddlerPopupButton {padding:0.2em;}
.popupTiddler {position: absolute; z-index:300; padding:1em; margin:0;}
.tabset {padding:1em 0 0 0.5em;}
.tab {display: inline-block; white-space: nowrap; position: relative; bottom: -0.7px; margin: 0 0.25em 0 0; padding:0.2em;}
.tabContents {padding:0.5em;}
.tabContents ul, .tabContents ol {margin:0; padding:0;}
.txtMainTab .tabContents li {list-style:none;}
.tabContents li.listLink { margin-left:.75em;}
#contentWrapper {display:block;}
#splashScreen {display:none;}
#displayArea {margin:1em 17em 0 14em;}
.toolbar {text-align:right; font-size:.9em;}
.tiddler { padding: 1em; }
.title { font-size: 1.6em; font-weight: bold; }
.subtitle { font-size: 1.1em; }
.missing .viewer, .missing .title { font-style: italic; }
.missing .subtitle { display: none; }
.tiddler .button {padding:0.2em 0.4em;}
.tagging {margin:0.5em 0.5em 0.5em 0; float:left; display:none;}
.isTag .tagging {display:block;}
.tagged {margin:0.5em; float:right;}
.tagging, .tagged {font-size:0.9em; padding:0.25em;}
.tagging ul, .tagged ul {list-style:none; margin:0.25em; padding:0;}
.tagged li, .tagging li { margin: 0.3em 0; }
.tagClear {clear:both;}
.footer {font-size:.9em;}
.footer li {display:inline;}
.annotation { padding: 0.5em 0.8em; margin: 0.5em 1px; }
.viewer {line-height:1.4em; padding-top:0.5em;}
.viewer .button {margin:0 0.25em; padding:0 0.25em;}
.viewer blockquote {line-height:1.5em; padding-left:0.8em;margin-left:2.5em;}
.viewer ul, .viewer ol {margin-left:0.5em; padding-left:1.5em;}
.viewer table, table.twtable { border-collapse: collapse; margin: 0.8em 0; }
.viewer th, .viewer td, .viewer tr, .viewer caption, .twtable th, .twtable td, .twtable tr, .twtable caption { padding: 0.2em 0.4em; }
.twtable caption { font-size: 0.9em; }
table.listView { margin: 0.8em 1.0em; }
table.listView th, table.listView td, table.listView tr { text-align: left; }
.listView > thead { position: sticky; top: 0; }
* html .viewer pre {width:99%; padding:0 0 1em 0;}
.viewer pre {padding:0.5em; overflow:auto;}
pre, code { font-family: monospace, monospace; font-size: 1em; }
.viewer pre, .viewer code { line-height: 1.4em; }
.editor {font-size:1.1em; line-height:1.4em;}
.editor input, .editor textarea {display:block; width:100%; box-sizing: border-box; font:inherit;}
.editorFooter {padding:0.25em 0; font-size:.9em;}
.editorFooter .button {padding-top:0; padding-bottom:0;}
.fieldsetFix {border:0; padding:0; margin:1px 0;}
.zoomer {font-size:1.1em; position:absolute; overflow:hidden;}
.zoomer div {padding:1em;}
* html #backstage {width:99%;}
* html #backstageArea {width:99%;}
#backstageArea {display:none; position:relative; overflow: hidden; z-index:150; padding:0.3em 0.5em;}
#backstageToolbar {position:relative;}
#backstageArea a {font-weight:bold; margin-left:0.5em; padding:0.3em 0.5em;}
#backstageButton {display:none; position:absolute; z-index:175; top:0; right:0;}
#backstageButton a {padding: 0.3em 0.5em; display: inline-block;}
#backstage {position:relative; width:100%; z-index:50;}
#backstagePanel { display:none; z-index:100; position:absolute; width:90%; margin:0 5%; }
.backstagePanelFooter {padding-top:0.2em; float:right;}
.backstagePanelFooter a {padding:0.2em 0.4em;}
#backstageCloak {display:none; z-index:20; position:absolute; width:100%; height:100px;}
.whenBackstage {display:none;}
.backstageVisible .whenBackstage {display:block;}
/*}}}*/
/***
StyleSheet for use when a translation requires any css style changes.
This StyleSheet can be used directly by languages such as Chinese, Japanese and Korean which need larger font sizes.
***/
/*{{{*/
body {font-size:0.8em;}
#sidebarOptions {font-size:1.05em;}
#sidebarOptions a {font-style:normal;}
#sidebarOptions .sliderPanel {font-size:0.95em;}
.subtitle {font-size:0.8em;}
.viewer table.listView {font-size:0.95em;}
/*}}}*/
/*{{{*/
@media print {
#mainMenu, #sidebar, #messageArea, .toolbar, #backstageButton, #backstageArea { display: none !important; }
#displayArea { margin: 1em 1em 0em; }
}
/*}}}*/
<!--{{{-->
<div class='toolbar' role='navigation' macro='toolbar [[ToolbarCommands::ViewToolbar]]'></div>
<div class='title' macro='view title'></div>
<div class='subtitle'><span macro='view modifier link'></span>, <span macro='view modified date'></span> (<span macro='message views.wikified.createdPrompt'></span> <span macro='view created date'></span>)</div>
<div class='tagging' macro='tagging'></div>
<div class='tagged' macro='tags'></div>
<div class='viewer' macro='view text wikified'></div>
<div class='tagClear'></div>
<!--}}}-->
Vanilla ~TiddlyWiki allows to embed static images with the {{{[img[url]]}}} syntax. But what if you need to create an image gallery with some nice navigation and hotkeys? Well, there are some cool js libs out there that can help. Here is an [[implementation|FancyBox2Plugin]] of a macro that creates galleries using the [[FancyBox library|http://fancyapps.com/fancybox/]]. To go further, see [[examples|FancyBox examples]] or [[documentation|FancyBox2PluginInfo]].
As usually, you can send your feedback to the Google group ([[main FB2P thread|https://groups.google.com/forum/#!topic/tiddlywiki/5K0-Znmf9us]]), but also you can leave comments right here (presumably, I will learn about them faster; I'm testing this now):
<<disqus thread:"ImageGalleries repo" forum:ylprojects>>
[[Creating image galleries in TiddlyWiki]]
/***
|Description |Add Disqus threads (comments) to your tiddlers|
|Source |https://github.com/YakovL/TiddlyWiki_DisqusPlugin/blob/master/DisqusPlugin.js|
|Author |Yakov Litvin|
|Version |0.3.1|
|License |[[MIT|https://github.com/YakovL/TiddlyWiki_YL_ExtensionsCollection/blob/master/Common%20License%20(MIT)]]|
***/
//{{{
config.macros.disqus = {
mountThread: function(container, shouldSubstituteContainer, forumName, threadId, threadUrl) {
// adapted from github.com/bimlas/tw5-disqus/blob/master/plugins/disqus/macro/comments.js
const loaderScriptId = "DISQUS-LOADER"
const currentLoaderScript = document.getElementById(loaderScriptId)
if(currentLoaderScript !== null) (document.head || document.body).removeChild(currentLoaderScript)
// D script updates the first element with this id on load and ignores others
const commentsElementId = "disqus_thread"
const threadDivClass = "disqus_thread" // not needed really
const $prevCommentsElement = jQuery("#" + commentsElementId)
if($prevCommentsElement[0]) { // maybe use while instead
const oldUrl = $prevCommentsElement.data("thread").url
const oldId = $prevCommentsElement.data("thread").id
const btn = jQuery("<a class=button>show comments</a>")
.data("thread", { url: oldUrl, id: oldId })
btn.on("click", () => {
// * forumName – pass via .data as well?
config.macros.disqus.mountThread(btn, true, forumName, oldId, oldUrl)
})
$prevCommentsElement.replaceWith(btn)
}
const $thread = jQuery("<div></div>").addClass(threadDivClass).attr("id", commentsElementId)
.data("thread", { url: threadUrl, id: threadId })
shouldSubstituteContainer ? jQuery(container).replaceWith($thread) : $thread.appendTo(container)
// basic code from https://ylprojects.disqus.com/admin/settings/universalcode/
window.disqus_config = function () {
// see more at https://help.disqus.com/customer/en/portal/articles/2158629
if(threadUrl) this.page.url = threadUrl
this.page.identifier = threadId
}
;(function() {
const loaderScript = document.createElement('script')
loaderScript.src = 'https://' + forumName + '.disqus.com/embed.js'
loaderScript.id = loaderScriptId
loaderScript.setAttribute('data-timestamp', +new Date())
;(document.head || document.body).appendChild(loaderScript)
})()
},
handler: function(place, macroName, params, wikifier, paramString, tiddler) {
const isLocal = window.isLocal || function() { return (document.location.protocol == "file:") }
if(isLocal()) return
const pParams = paramString.parseParams("anon", null, true, false, true)
// forum shortname, also known as disqus_shortname
// TODO: add a config for the default value instead of a hardcoded value for my personal usage
const forumShortname = getParam(pParams, "forum", "")
// also known as disqus_identifier
// TODO: check if this works with non-latin letters; .oO if just tiddler.title is enough, or we have to use window.location/SiteUrl as well (maybe permalink
const threadId = getParam(pParams, "thread", tiddler ? tiddler.title : "outside-content")
// used for permalinking purposes, also known as disqus_url
const url = getParam(pParams, "url", tiddler.title && story.getPermalink ? story.getPermalink(tiddler.title) : "")
this.mountThread(place, false, forumShortname, threadId, url)
}
}
//}}}
{{{
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
label:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
>>
}}}
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
label:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
>>
<<fancyBox
""
thumb:""
config:"{ openEffect: 'elastic' }"
align:center
>>
+++[Code]
{{{
<<fancyBox
""
thumb: ""
config:"{ openEffect: 'elastic' }"
align: center
>>
}}}
===
!!!SVG converted to base64
can be also used for thumbs, but are not supported as main images for now:
<<fancyBox
"">>
{{{
<<fancyBox
"">>
}}}
<<external [[FancyBox2Plugin]] file:"FancyBox2Plugin.js" plugin:true>>
!!!Single image
{{{
<<fancyBox "img/1.JPG" title:"Field of a cylindrical magnet" thumb:"img/1 thumb.png">>
}}}
<<fancyBox "img/1.JPG" title:"Field of a cylindrical magnet" thumb:"img/1 thumb.png">>
!!!Galleries
!!!!One-by-one syntax
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/1 thumb.png" "img/2 thumb.png" "img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
>>
+++[Show the macro code]
{{{
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/1 thumb.png" "img/2 thumb.png" "img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
>>
}}}
===
Note: when using in web, thumbs should be set: otherwise, the thumbs made of minified but not compressed images may cause slow gallery loading (on relatively slow connections): see [[how it works|Example of a gallery without thumbnails]].
!!!!Arbitrary order syntax
The macro can be used in +++[this manner]
{{{
<<fancyBox
"img/1.JPG" thumb:"img/1 thumb.png" title:"field of a cylindrical magnet"
image:"img/2.JPG" thumb:"img/2 thumb.png" title:"field of a cylindrical magnet on a larger scale"
image:"img/3.JPG" thumb:"img/3 thumb.png" title:"field of two parallel magnets"
image:"img/4.JPG" thumb:"img/4 thumb.png" title:"field of a couple of antiparallel magnets"
image:"img/5.JPG" thumb:"img/5 thumb.png" title:"field of two attracting magnets"
image:"img/6.JPG" thumb:"img/6 thumb.png" title:"field of two repelling magnets"
>>
}}}
===. This is useful when thumbs or titles are defined only for some images:
{{{
<<fancyBox
"img/1.JPG" title:"field of a cylindrical magnet"
image:"img/2.JPG" thumb:"img/2 thumb.png" title:"field of a cylindrical magnet on a larger scale"
image:"img/3.JPG" thumb:"img/3 thumb.png"
image:"img/4.JPG" title:"field of a couple of antiparallel magnets" thumb:"img/4 thumb.png"
image:"img/5.JPG" thumb:"img/5 thumb.png" title:"field of two attracting magnets"
image:"img/6.JPG" thumb:"img/6 thumb.png"
>>
}}}
resulting in:
<<fancyBox
"img/1.JPG" title:"field of a cylindrical magnet"
image:"img/2.JPG" thumb:"img/2 thumb.png" title:"field of a cylindrical magnet on a larger scale"
image:"img/3.JPG" thumb:"img/3 thumb.png"
image:"img/4.JPG" title:"field of a couple of antiparallel magnets" thumb:"img/4 thumb.png"
image:"img/5.JPG" thumb:"img/5 thumb.png" title:"field of two attracting magnets"
image:"img/6.JPG" thumb:"img/6 thumb.png"
>>
!!!For each image/wrapper params
!!!!Changing thumbnails size and styles
The following addition to the macro params can be used to set their size:
{{{
images-max-width:"15.6%"
}}}
note that there's the default value of {{{max-width}}} equal to {{{24%}}} and the default values of left and right margins equal to {{{0.5%}}} (set by the plugin) which can be overwritten like this. The used value 15.6% is a rough value of {{{100% / 6 - 1%}}} (6 pictures, 1% for margins). The result is:
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/1 thumb.png" "img/2 thumb.png" "img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
images-max-width:"15.6%"
>>
Any CSS property of each image can set using params with the {{{image-}}} prefix. For instance, here's a way to make all the images be of the same height (unfortunately, this is browser/screen-size/zoom dependent way as cm and %width can have different ratios; play with zooming to see how it works):
{{{
images-max-width:"17.5%" images-max-height:"2.7cm"
}}}
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/1 thumb.png" "img/2 thumb.png" "img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
images-max-width:"17.5%" images-max-height:"2.7cm"
>>
!!!!Using custom HTML instead of thumbnails
Starting the {{{thumb}}} param with {{{html:}}} allows to use a custom HTML instead:
{{{
<<fancyBox
"img/1.JPG" thumb:"html: custom <i>html</i> instead of a thumb"
image:"img/2.JPG" thumb:"img/2 thumb.png"
>>
}}}
<<fancyBox
"img/1.JPG" thumb:"html: custom <i>html</i> instead of a thumb"
image:"img/2.JPG" thumb:"img/2 thumb.png"
>>
!!!!Adding custom classes to images
Any class can be added to images like this:
{{{
images-class:"myBorder"
}}}
Here, CSS for {{{myBorder}}} is defined at StyleSheet which gives:
<<fancyBox
"img/1.JPG" "img/2.JPG" thumb:"img/1 thumb.png" "img/2 thumb.png" title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale"
images-class:"myBorder"
>>
!!!!Same for wrappers...
Likewise, styles and classes can be added to the {{{a}}} wrappers of the images.
{{{
images-wrapper:"myWrapper" images-wrapper:"just-another-class"
}}}
gives (with StyleSheet):
<<fancyBox
"img/1.JPG" thumb:"img/1 thumb.png" title:"field of a cylindrical magnet"
wrappers-class:"myWrapper" wrappers-class:"just-another-class"
>>
(ok, this is not a great example, in fact I'm not sure if styling of wrappers is needed at all, let me know what you think)
!!!Global (per macro) params
!!!!Align
{{{
<<fancyBox "img/1.JPG" title:"field of a cylindrical magnet" thumb:"img/1 thumb.png" align:center>>
}}}
(can be {{{left}}}, {{{center}}} or {{{right}}})
<<fancyBox "img/1.JPG" title:"field of a cylindrical magnet" thumb:"img/1 thumb.png" align:center>>
!!!!Inline
Sometimes, it's useful to show a thumb or custom html inline, for instance a clickable emoji: <<fancyBox
"img/1.JPG" thumb:"html: 👁️" inline
>>. This can be done via the {{{inline}}} param:
{{{
<<fancyBox "img/1.JPG" thumb:"html: 👁️" inline>>
}}}
!!!!Gallery id
Each gallery has and id which is by default generated so that different galleries have diffent ones. If two galleries have the same id, they are displayed as one, which can be useful. This can be made, usgin the {{{galleryId}}} param, here's a couple of "concept" examples:
* +++[Two-parts gallery]
<<fancyBox
"img/1.JPG" "img/2.JPG"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale"
images-max-width:"49%"
galleryId:"two-parts-gallery"
>><<fancyBox
"img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
images-max-width:"24%"
galleryId:"two-parts-gallery"
>>
{{{
<<fancyBox
"img/1.JPG" "img/2.JPG"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale"
images-max-width:"49%"
galleryId:"two-parts-gallery"
>><<fancyBox
"img/3.JPG" "img/4.JPG" "img/5.JPG" "img/6.JPG"
thumb:"img/3 thumb.png" "img/4 thumb.png" "img/5 thumb.png" "img/6 thumb.png"
title:"field of two parallel magnets" "field of a couple of antiparallel magnets" "field of two attracting magnets" "field of two repelling magnets"
images-max-width:"24%"
galleryId:"two-parts-gallery"
>>
}}}
===
* +++[Alignment using table]
|>|max-width:30em;<<fancyBox "img/1.JPG" title:"field of a cylindrical magnet" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/2.JPG" thumb:"img/2 thumb.png" title:"field of a cylindrical magnet on a larger scale" images-max-width:"100%" galleryId:"table-gallery">>|
|~|~|<<fancyBox "img/3.JPG" thumb:"img/3 thumb.png" title:"field of two parallel magnets" images-max-width:"100%" galleryId:"table-gallery">>|
|<<fancyBox "img/4.JPG" thumb:"img/4 thumb.png" title:"field of a couple of antiparallel magnets" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/5.JPG" thumb:"img/5 thumb.png" title:"field of two attracting magnets" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/6.JPG" thumb:"img/6 thumb.png" title:"field of two repelling magnets" images-max-width:"100%" galleryId:"table-gallery">>|
{{{
|>|max-width:30em;<<fancyBox "img/1.JPG" title:"field of a cylindrical magnet" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/2.JPG" thumb:"img/2 thumb.png" title:"field of a cylindrical magnet on a larger scale" images-max-width:"100%" galleryId:"table-gallery">>|
|~|~|<<fancyBox "img/3.JPG" thumb:"img/3 thumb.png" title:"field of two parallel magnets" images-max-width:"100%" galleryId:"table-gallery">>|
|<<fancyBox "img/4.JPG" thumb:"img/4 thumb.png" title:"field of a couple of antiparallel magnets" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/5.JPG" thumb:"img/5 thumb.png" title:"field of two attracting magnets" images-max-width:"100%" galleryId:"table-gallery">>|<<fancyBox "img/6.JPG" thumb:"img/6 thumb.png" title:"field of two repelling magnets" images-max-width:"100%" galleryId:"table-gallery">>|
}}}
===
!!!!Using fancyBox settings
In this example, the fancyBox options are used (via the {{{config}}} param) to change the opening effect:
{{{
config:"{ openEffect: 'elastic', openSpeed: 350 }"
}}}
<<fancyBox
"img/1.JPG" "img/2.JPG" "img/3.JPG" "img/4.JPG"
thumb:"img/1 thumb.png" "img/2 thumb.png" "img/3 thumb.png" "img/4 thumb.png"
title:"field of a cylindrical magnet" "field of a cylindrical magnet on a larger scale" "field of two parallel magnets" "field of a couple of antiparallel magnets"
config:"{ openEffect: 'elastic', openSpeed: 350 }"
>>
You can find more [[tips & tricks|http://fancyapps.com/fancybox/#useful]] and [[documentation|http://fancyapps.com/fancybox/#docs]] on the lib's page; the {{{config}}} param is evaluated and passed to the {{{.fancybox}}} function, so you can copy-paste codes from some examples directly to the macro.
!!!Using base64
It is in fact possible to attach images into ~TiddlyWiki in a text form using the [[base64 format|https://en.wikipedia.org/wiki/Base64]]. An image can be easily converted to base64 using [[this|http://www.askapache.com/online-tools/base64-image-converter/]] or [[this|http://www.base64-image.de/step-1.php]] or some other tool. There are several drawbacks: you have to use some external software, the byte-size of the image increases once it is converted to base64, and also it takes huge space in a tiddler and increases the TW size (not sure about loading time when used locally). Sometimes it can be useful nonetheless, especially for small images, icons; here's [[an exmaple of usage|Example of base64 usage]].
FancyBox2Plugin
<<include "node: ImageGalleriesRepo" filters:"[[FancyBox2Plugin installer and updater]] [[FancyBox2Plugin]]" import:4 noRefresh>>
|''Description''|Documentation for the [[FancyBox2Plugin|https://yakovl.github.io/TiddlyWiki_ImageGalleries/#FancyBox2Plugin]] v0.9.4|
|''Author''|Yakov Litvin|
|''Licence''|see [[the plugin|https://yakovl.github.io/TiddlyWiki_ImageGalleries/#FancyBox2Plugin]]|
|''Feedback''|see [[intro|https://yakovl.github.io/TiddlyWiki_ImageGalleries/#%5B%5BCreating%20image%20galleries%20in%20TiddlyWiki%5D%5D]]|
|''Credits''|The plugin was inspired by the earlier Mario Pietsch's [[implementation|http://fancybox.tiddlyspace.com/]] which seems to get broken at some point|
The plugin defines the {{{fancyBox}}} macro which creates an image gallery, like in the [[examples|https://yakovl.github.io/TiddlyWiki_ImageGalleries/#%5B%5BFancyBox%20examples%5D%5D]].
!!!Plugin installation
is the simplest one: just import the plugin tiddler, or copy-paste and add the {{{systemConfig}}} tag. No external icons are required as they are [[converted|http://www.askapache.com/online-tools/base64-image-converter/]] to base64 and inserted directly into StyleSheetFancyBox.
!!!Macro syntax
The simplest usage is rather self-explanatory:
{{{
<<fancyBox image1url image2url ...>>
}}}
!!!!Per image params
For each image, a title (shown in the fancyBox) and a "thumbnail" (an alternative image, usually of a smaller size, shown in the text, when the fancyBox is not opened) can be defined. For one image, they are defined as follows (the {{{[}}}...{{{]}}} wrapper means that the param is optional):
{{{
<<fancyBox imageUrl [thumb:thumbnailUrl] [title:imageTitle]>>
}}}
Instead of the {{{title}}} param, {{{label}}} can be used, but with multiple images they shouldn't be mixed: use either {{{title}}}s or {{{label}}}s in the same macro (this is done for partial backward compability with Mario's plugin).
For multiple images, there's a couple of ways to set the images/titles:
# put images' urls one by one -- then other params are applied starting from the first image and then sequentially:{{jD{
{{{
<<fancyBox
image1url image2url ... imageNurl
[thumb:image1thumb image2thumb ... imageMthumb]
[title:image1title image2title ... imageLtitle]
>>
}}}
}}} The first thumb corresponds to the first image, the second -- to the second one, ..., same for titles (here M ≤ N, L ≤ N).
# or, put the additional params after images' url -- in this case it's possible to add those to several images at arbitrary number:{{jD{
{{{
<<fancyBox
[image:]image1url [thumb:image1thumb] [title:image1title]
image:image2url [thumb:image2thumb] [title:image2title]
...
image:imageNurl [thumb:imageNthumb] [title:imageNtitle]
>>
}}}
}}} If "thumb" and "title" are not used, the next image url doesn't require the "image:" part.
Keep in mind that using thumbnails can speed up tiddler loading considerably, especially on the web.
!!!!For each image/wrapper params
Aside changing styles for the whole document (see below), it is possible to set them for a single macro. For that, one can use the {{{images-}}} and {{{wrappers-}}} prefixed params. These params are to be used like CSS properties: for instance, if you'd like to increase the margins around the images and add red borders, you can write:
{{{
<<fancyBox ...
images-margin:"1%"
images-border:"thin solid red"
>>
}}}
and the {{{margin: 1%; border: thin solid red;}}} styling is applied to each picture. Any CSS property name can be used, but this doesn't allow to use pseudo-selectors like {{{:hover}}} or {{{:before}}}.
As each image is wrapped into an {{{a}}} element, there's also a possibility to style the wrappers -- this is done in the same way using the {{{wrappers-}}} params but it doesn't seem like a thing to use frequently.
Alternatively, you can use the {{{images-class}}} and {{{wrappers-class}}} params to add DOM classes to them and use external styling (StyleSheet and such):
{{{
<<fancyBox ...
wrappers-class:"myWrapperClass" wrappers-class:"my-other-wrapper_class"
>>
}}}
!!!!"Global" params
The plugin supports fancyBox configuration via the "global" ''{{{config}}} param'': the whole param is evaluated and passed to the {{{.fancybox}}} function. This example is a adoption of the "Remove white border around content" [[Fiddle|http://jsfiddle.net/nJfAZ/]]:
{{{
<<fancyBox
image1url image2url ... imageNurl
config:"{ padding: 0 }"
>>
}}}
For further tweaking, see fancyBox [[documentation|http://fancyapps.com/fancybox/#docs]], tips & tricks and examples (on the same page).
Using the ''{{{align}}} param'', one can move the block with images to {{{center}}} or {{{right}}}. Presumably, this is mostly useful for moving single images to center:
{{{
<<fancyBox someImage align:center>>
}}}
Each gallery gets an auto-generated id. You can also set the id yourself by using the {{{galleryId}}} param: if two or more galleries has the same id, the are concatenated in fancyBox:
{{{
<<fancyBox ... galleryId:"my-id">>
...
<<fancyBox ... galleryId:"my-id">>
}}}
If you have multiple images in a single folder, you may want to use the ''{{{folder}}} and {{{thumbFolder}}} params'': the former makes all pathes of images relative to it ("" if omitted):
{{{
<<fancyBox "some path/subpath1/image1.jpg"
"some path/subpath2/image2.png"
...
>>
}}}
is the same as
{{{
<<fancyBox "subpath1/image1.jpg"
"subpath2/image2.png"
...
folder:"some path"
>>
}}}
This also allows to move large numbers of images without changing each url in each macro. {{{thumbFolder}}} does the same to images' thumbnails, the only difference is: by default it's equal to the {{{folder}}} param value.
!!!Styles definition and tweaking
~FancyBox styling is contained in the StyleSheetFancyBox shadow. It can be changed to meet your needs; deleting it brings it back to its initial (shadow) state.
!!!What is not implemented (yet)
~FancyBox allows to work with content other than images, which is to be implemented in further versions. Some other features may be implemented in the future, like settings for better representation of thumbnails etc.
/***
|''Name''|InnerExternalLinkPlugin|
|''Version''|0.9.1|
|''Author''|Yakov Litvin|
***/
//{{{
config.extensions.InnerExternalLinkPlugin_orig_createExternalLink = createExternalLink;
createExternalLink = function(place,url,label) {
var permaLinkRegExp = /([^#]+)#(?:(?:([^%\[]+))|(?:\[\[([^\]]+)\]\])|(?:%5B%5B([^\]]+)%5D%5D))/,
// better to use decoding
match = permaLinkRegExp.exec(url),
siteUrlText = store.getTiddlerText("SiteUrl"),
pageLocation = window.location,
urlRegExp = /^(([^:\/\?#]+):)?(\/\/([^\/\?#]*))?([^\?#]*)(\?([^#]*))?(#(.*))?/,
matched = false;
if(match) {
var bareUrl = match[1],
tiddlerName = match[4]? match[4] : (match[3]? match[3] : match[2]);
tiddlerName = decodeURIComponent(tiddlerName);
} else
return config.extensions.InnerExternalLinkPlugin_orig_createExternalLink(place,url,label);
if(siteUrlText == bareUrl)
matched = true;
match = urlRegExp.exec(pageLocation.toString());
if(match[1]+match[3]+match[5] == bareUrl)
matched = true;
// may be additional normalizing (stripping the protocol etc, encoding/decoding) should be added
if(matched)
return createTiddlyLink(place, tiddlerName, label);
else
return config.extensions.InnerExternalLinkPlugin_orig_createExternalLink(place,url,label);
};
//}}}
[[Intro|Creating image galleries in TiddlyWiki]]
[[Plugin|FancyBox2Plugin]]
[[Examples|FancyBox examples]]
[[Docs|FancyBox2PluginInfo]]
[[Licence|Yakov Litvin Public Licence]]
<!--{{{-->
<div style="text-align: center; margin: 2%"><!-- my wrapper -->
<!-- Yandex.Metrika informer -->
<a href="https://metrika.yandex.ru/stat/?id=22899424&from=informer"
target="_blank" rel="nofollow"><img src="//bs.yandex.ru/informer/22899424/3_1_FFFFFFFF_EFEFEFFF_0_pageviews"
style="width:88px; height:31px; border:0;" alt="Яндекс.Метрика" title="Яндекс.Метрика: данные за сегодня (просмотры, визиты и уникальные посетители)" onclick="try{Ya.Metrika.informer({i:this,id:22899424,lang:'ru'});return false}catch(e){}"/></a>
<!-- /Yandex.Metrika informer -->
<!-- Yandex.Metrika counter -->
<script type="text/javascript">
(function (d, w, c) {
(w[c] = w[c] || []).push(function() {
try {
w.yaCounter22899424 = new Ya.Metrika({id:22899424,
clickmap:true,
trackLinks:true,
accurateTrackBounce:true,
trackHash:true});
} catch(e) { }
});
var n = d.getElementsByTagName("script")[0],
s = d.createElement("script"),
f = function () { n.parentNode.insertBefore(s, n); };
s.type = "text/javascript";
s.async = true;
s.src = (d.location.protocol == "https:" ? "https:" : "http:") + "//mc.yandex.ru/metrika/watch.js";
if (w.opera == "[object Opera]") {
d.addEventListener("DOMContentLoaded", f, false);
} else { f(); }
})(document, window, "yandex_metrika_callbacks");
</script>
<noscript><div><img src="//mc.yandex.ru/watch/22899424" style="position:absolute; left:-9999px;" alt="" /></div></noscript>
<!-- /Yandex.Metrika counter -->
</div><!-- end of my wrapper -->
<!--}}}-->
/***
|Name|NestedSlidersPlugin|
|Source|http://www.TiddlyTools.com/#NestedSlidersPlugin|
|Documentation|http://www.TiddlyTools.com/#NestedSlidersPluginInfo|
|Version|2.4.9|
|Author|Eric Shulman|
|License|http://www.TiddlyTools.com/#LegalStatements|
|~CoreVersion|2.1|
|Type|plugin|
|Description|show content in nest-able sliding/floating panels, without creating separate tiddlers for each panel's content|
!!!!!Configuration
<<<
<<option chkFloatingSlidersAnimate>> allow floating sliders to animate when opening/closing
Note: for floating slider animation to occur you must also allow animation in general (see [[AdvancedOptions]]).
<<<
!!!!!Code
***/
//{{{
version.extensions.NestedSlidersPlugin= {major: 2, minor: 4, revision: 9, date: new Date(2008,11,15)};
// options for deferred rendering of sliders that are not initially displayed
if (config.options.chkFloatingSlidersAnimate===undefined)
config.options.chkFloatingSlidersAnimate=false; // avoid clipping problems in IE
// default styles for 'floating' class
setStylesheet(".floatingPanel { position:absolute; z-index:10; padding:0.5em; margin:0em; \
background-color:#eee; color:#000; border:1px solid #000; text-align:left; }","floatingPanelStylesheet");
// if removeCookie() function is not defined by TW core, define it here.
if (window.removeCookie===undefined) {
window.removeCookie=function(name) {
document.cookie = name+'=; expires=Thu, 01-Jan-1970 00:00:01 UTC; path=/;';
}
}
config.formatters.push( {
name: "nestedSliders",
match: "\\n?\\+{3}",
terminator: "\\s*\\={3}\\n?",
lookahead: "\\n?\\+{3}(\\+)?(\\([^\\)]*\\))?(\\!*)?(\\^(?:[^\\^\\*\\@\\[\\>]*\\^)?)?(\\*)?(\\@)?(?:\\{\\{([\\w]+[\\s\\w]*)\\{)?(\\[[^\\]]*\\])?(\\[[^\\]]*\\])?(?:\\}{3})?(\\#[^:]*\\:)?(\\>)?(\\.\\.\\.)?\\s*",
handler: function(w)
{
lookaheadRegExp = new RegExp(this.lookahead,"mg");
lookaheadRegExp.lastIndex = w.matchStart;
var lookaheadMatch = lookaheadRegExp.exec(w.source)
if(lookaheadMatch && lookaheadMatch.index == w.matchStart)
{
var defopen=lookaheadMatch[1];
var cookiename=lookaheadMatch[2];
var header=lookaheadMatch[3];
var panelwidth=lookaheadMatch[4];
var transient=lookaheadMatch[5];
var hover=lookaheadMatch[6];
var buttonClass=lookaheadMatch[7];
var label=lookaheadMatch[8];
var openlabel=lookaheadMatch[9];
var panelID=lookaheadMatch[10];
var blockquote=lookaheadMatch[11];
var deferred=lookaheadMatch[12];
// location for rendering button and panel
var place=w.output;
// default to closed, no cookie, no accesskey, no alternate text/tip
var show="none"; var cookie=""; var key="";
var closedtext=">"; var closedtip="";
var openedtext="<"; var openedtip="";
// extra "+", default to open
if (defopen) show="block";
// cookie, use saved open/closed state
if (cookiename) {
cookie=cookiename.trim().slice(1,-1);
cookie="chkSlider"+cookie;
if (config.options[cookie]==undefined)
{ config.options[cookie] = (show=="block") }
show=config.options[cookie]?"block":"none";
}
// parse label/tooltip/accesskey: [label=X|tooltip]
if (label) {
var parts=label.trim().slice(1,-1).split("|");
closedtext=parts.shift();
if (closedtext.substr(closedtext.length-2,1)=="=")
{ key=closedtext.substr(closedtext.length-1,1); closedtext=closedtext.slice(0,-2); }
openedtext=closedtext;
if (parts.length) closedtip=openedtip=parts.join("|");
else { closedtip="show "+closedtext; openedtip="hide "+closedtext; }
}
// parse alternate label/tooltip: [label|tooltip]
if (openlabel) {
var parts=openlabel.trim().slice(1,-1).split("|");
openedtext=parts.shift();
if (parts.length) openedtip=parts.join("|");
else openedtip="hide "+openedtext;
}
var title=show=='block'?openedtext:closedtext;
var tooltip=show=='block'?openedtip:closedtip;
// create the button
if (header) { // use "Hn" header format instead of button/link
var lvl=(header.length>5)?5:header.length;
var btn = createTiddlyElement(createTiddlyElement(place,"h"+lvl,null,null,null),"a",null,buttonClass,title);
btn.onclick=onClickNestedSlider;
btn.setAttribute("href","javascript:;");
btn.setAttribute("title",tooltip);
}
else
var btn = createTiddlyButton(place,title,tooltip,onClickNestedSlider,buttonClass);
btn.innerHTML=title; // enables use of HTML entities in label
// set extra button attributes
btn.setAttribute("closedtext",closedtext);
btn.setAttribute("closedtip",closedtip);
btn.setAttribute("openedtext",openedtext);
btn.setAttribute("openedtip",openedtip);
btn.sliderCookie = cookie; // save the cookiename (if any) in the button object
btn.defOpen=defopen!=null; // save default open/closed state (boolean)
btn.keyparam=key; // save the access key letter ("" if none)
if (key.length) {
btn.setAttribute("accessKey",key); // init access key
btn.onfocus=function(){this.setAttribute("accessKey",this.keyparam);}; // **reclaim** access key on focus
}
btn.setAttribute("hover",hover?"true":"false");
btn.onmouseover=function(ev) {
// optional 'open on hover' handling
if (this.getAttribute("hover")=="true" && this.sliderPanel.style.display=='none') {
document.onclick.call(document,ev); // close transients
onClickNestedSlider(ev); // open this slider
}
// mouseover on button aligns floater position with button
if (window.adjustSliderPos) window.adjustSliderPos(this.parentNode,this,this.sliderPanel);
}
// create slider panel
var panelClass=panelwidth?"floatingPanel":"sliderPanel";
if (panelID) panelID=panelID.slice(1,-1); // trim off delimiters
var panel=createTiddlyElement(place,"div",panelID,panelClass,null);
panel.button = btn; // so the slider panel know which button it belongs to
btn.sliderPanel=panel; // so the button knows which slider panel it belongs to
panel.defaultPanelWidth=(panelwidth && panelwidth.length>2)?panelwidth.slice(1,-1):"";
panel.setAttribute("transient",transient=="*"?"true":"false");
panel.style.display = show;
panel.style.width=panel.defaultPanelWidth;
panel.onmouseover=function(event) // mouseover on panel aligns floater position with button
{ if (window.adjustSliderPos) window.adjustSliderPos(this.parentNode,this.button,this); }
// render slider (or defer until shown)
w.nextMatch = lookaheadMatch.index + lookaheadMatch[0].length;
if ((show=="block")||!deferred) {
// render now if panel is supposed to be shown or NOT deferred rendering
w.subWikify(blockquote?createTiddlyElement(panel,"blockquote"):panel,this.terminator);
// align floater position with button
if (window.adjustSliderPos) window.adjustSliderPos(place,btn,panel);
}
else {
var src = w.source.substr(w.nextMatch);
var endpos=findMatchingDelimiter(src,"+++","===");
panel.setAttribute("raw",src.substr(0,endpos));
panel.setAttribute("blockquote",blockquote?"true":"false");
panel.setAttribute("rendered","false");
w.nextMatch += endpos+3;
if (w.source.substr(w.nextMatch,1)=="\n") w.nextMatch++;
}
}
}
}
)
function findMatchingDelimiter(src,starttext,endtext) {
var startpos = 0;
var endpos = src.indexOf(endtext);
// check for nested delimiters
while (src.substring(startpos,endpos-1).indexOf(starttext)!=-1) {
// count number of nested 'starts'
var startcount=0;
var temp = src.substring(startpos,endpos-1);
var pos=temp.indexOf(starttext);
while (pos!=-1) { startcount++; pos=temp.indexOf(starttext,pos+starttext.length); }
// set up to check for additional 'starts' after adjusting endpos
startpos=endpos+endtext.length;
// find endpos for corresponding number of matching 'ends'
while (startcount && endpos!=-1) {
endpos = src.indexOf(endtext,endpos+endtext.length);
startcount--;
}
}
return (endpos==-1)?src.length:endpos;
}
//}}}
//{{{
window.onClickNestedSlider=function(e)
{
if (!e) var e = window.event;
var theTarget = resolveTarget(e);
while (theTarget && theTarget.sliderPanel==undefined) theTarget=theTarget.parentNode;
if (!theTarget) return false;
var theSlider = theTarget.sliderPanel;
var isOpen = theSlider.style.display!="none";
// if SHIFT-CLICK, dock panel first (see [[MoveablePanelPlugin]])
if (e.shiftKey && config.macros.moveablePanel) config.macros.moveablePanel.dock(theSlider,e);
// toggle label
theTarget.innerHTML=isOpen?theTarget.getAttribute("closedText"):theTarget.getAttribute("openedText");
// toggle tooltip
theTarget.setAttribute("title",isOpen?theTarget.getAttribute("closedTip"):theTarget.getAttribute("openedTip"));
// deferred rendering (if needed)
if (theSlider.getAttribute("rendered")=="false") {
var place=theSlider;
if (theSlider.getAttribute("blockquote")=="true")
place=createTiddlyElement(place,"blockquote");
wikify(theSlider.getAttribute("raw"),place);
theSlider.setAttribute("rendered","true");
}
// show/hide the slider
if(config.options.chkAnimate && (!hasClass(theSlider,'floatingPanel') || config.options.chkFloatingSlidersAnimate))
anim.startAnimating(new Slider(theSlider,!isOpen,e.shiftKey || e.altKey,"none"));
else
theSlider.style.display = isOpen ? "none" : "block";
// reset to default width (might have been changed via plugin code)
theSlider.style.width=theSlider.defaultPanelWidth;
// align floater panel position with target button
if (!isOpen && window.adjustSliderPos) window.adjustSliderPos(theSlider.parentNode,theTarget,theSlider);
// if showing panel, set focus to first 'focus-able' element in panel
if (theSlider.style.display!="none") {
var ctrls=theSlider.getElementsByTagName("*");
for (var c=0; c<ctrls.length; c++) {
var t=ctrls[c].tagName.toLowerCase();
if ((t=="input" && ctrls[c].type!="hidden") || t=="textarea" || t=="select")
{ try{ ctrls[c].focus(); } catch(err){;} break; }
}
}
var cookie=theTarget.sliderCookie;
if (cookie && cookie.length) {
config.options[cookie]=!isOpen;
if (config.options[cookie]!=theTarget.defOpen) window.saveOptionCookie(cookie);
else window.removeCookie(cookie); // remove cookie if slider is in default display state
}
// prevent SHIFT-CLICK from being processed by browser (opens blank window... yuck!)
// prevent clicks *within* a slider button from being processed by browser
// but allow plain click to bubble up to page background (to close transients, if any)
if (e.shiftKey || theTarget!=resolveTarget(e))
{ e.cancelBubble=true; if (e.stopPropagation) e.stopPropagation(); }
Popup.remove(); // close open popup (if any)
return false;
}
//}}}
//{{{
// click in document background closes transient panels
document.nestedSliders_savedOnClick=document.onclick;
document.onclick=function(ev) { if (!ev) var ev=window.event; var target=resolveTarget(ev);
if (document.nestedSliders_savedOnClick)
var retval=document.nestedSliders_savedOnClick.apply(this,arguments);
// if click was inside a popup... leave transient panels alone
var p=target; while (p) if (hasClass(p,"popup")) break; else p=p.parentNode;
if (p) return retval;
// if click was inside transient panel (or something contained by a transient panel), leave it alone
var p=target; while (p) {
if ((hasClass(p,"floatingPanel")||hasClass(p,"sliderPanel"))&&p.getAttribute("transient")=="true") break;
p=p.parentNode;
}
if (p) return retval;
// otherwise, find and close all transient panels...
var all=document.all?document.all:document.getElementsByTagName("DIV");
for (var i=0; i<all.length; i++) {
// if it is not a transient panel, or the click was on the button that opened this panel, don't close it.
if (all[i].getAttribute("transient")!="true" || all[i].button==target) continue;
// otherwise, if the panel is currently visible, close it by clicking it's button
if (all[i].style.display!="none") window.onClickNestedSlider({target:all[i].button})
if (!hasClass(all[i],"floatingPanel")&&!hasClass(all[i],"sliderPanel")) all[i].style.display="none";
}
return retval;
};
//}}}
//{{{
// adjust floating panel position based on button position
if (window.adjustSliderPos==undefined) window.adjustSliderPos=function(place,btn,panel) {
if (hasClass(panel,"floatingPanel") && !hasClass(panel,"undocked")) {
// see [[MoveablePanelPlugin]] for use of 'undocked'
var rightEdge=document.body.offsetWidth-1;
var panelWidth=panel.offsetWidth;
var left=0;
var top=btn.offsetHeight;
if (place.style.position=="relative" && findPosX(btn)+panelWidth>rightEdge) {
left-=findPosX(btn)+panelWidth-rightEdge; // shift panel relative to button
if (findPosX(btn)+left<0) left=-findPosX(btn); // stay within left edge
}
if (place.style.position!="relative") {
var left=findPosX(btn);
var top=findPosY(btn)+btn.offsetHeight;
var p=place; while (p && !hasClass(p,'floatingPanel')) p=p.parentNode;
if (p) { left-=findPosX(p); top-=findPosY(p); }
if (left+panelWidth>rightEdge) left=rightEdge-panelWidth;
if (left<0) left=0;
}
panel.style.left=left+"px"; panel.style.top=top+"px";
}
}
//}}}
//{{{
// TW2.1 and earlier:
// hijack Slider stop handler so overflow is visible after animation has completed
Slider.prototype.coreStop = Slider.prototype.stop;
Slider.prototype.stop = function()
{ this.coreStop.apply(this,arguments); this.element.style.overflow = "visible"; }
// TW2.2+
// hijack Morpher stop handler so sliderPanel/floatingPanel overflow is visible after animation has completed
if (version.major+.1*version.minor+.01*version.revision>=2.2) {
Morpher.prototype.coreStop = Morpher.prototype.stop;
Morpher.prototype.stop = function() {
this.coreStop.apply(this,arguments);
var e=this.element;
if (hasClass(e,"sliderPanel")||hasClass(e,"floatingPanel")) {
// adjust panel overflow and position after animation
e.style.overflow = "visible";
if (window.adjustSliderPos) window.adjustSliderPos(e.parentNode,e.button,e);
}
};
}
//}}}
<<search>><<closeAll>><<permaview>><<saveChanges>><<slider chkSliderOptionsPanel OptionsPanel "options »" "Change TiddlyWiki advanced options">>
/%<<slider chkSliSideBarTabs SideBarTabs##Tabs "Aggregated navigation lists" "">>
!Tabs
%/<<tabs txtMainTab
"Timeline" "Timeline" SideBarTabs##TabTimeline
"All" "All tiddlers" SideBarTabs##TabAll
"Tags" "All tags" SideBarTabs##TabTags
"More" "More lists" SideBarTabs##TabsMore
>>/%
!TabTimeline
<<timeline>>
!TabAll
<<list all>>
!TabTags
<<allTags excludeLists>>
!TabsMore
<<tabs txtMoreTab
"Missing" "Missing tiddlers" SideBarTabs##TabMoreMissing
"Orphans" "Orphaned tiddlers" SideBarTabs##TabMoreOrphans
"Shadowed" "Shadowed tiddlers" SideBarTabs##TabMoreShadowed
>>
!TabMoreMissing
<<list missing>>
!TabMoreOrphans
<<list orphans>>
!TabMoreShadowed
<<list shadowed>>
!%/
a tool to create image galleries in ~TiddlyWiki
https://yakovl.github.io/TiddlyWiki_ImageGalleries/
/*{{{*/
div[tags=""] .tagged { display: none; }
.myBorder { border: thin solid red; }
.myWrapper:before { content:"\\o/"; }
.myWrapper:after { content:"\\o/"; }
/*}}}*/
/***
|Description |Allows to store any number of tiddlers as external files and more|
|Source |https://github.com/YakovL/TiddlyWiki_TiddlerInFilePlugin/blob/master/TiddlerInFilePlugin.js|
|Author |Yakov Litvin|
|Version |1.1.3*|
|License |[[MIT|https://github.com/YakovL/TiddlyWiki_YL_ExtensionsCollection/blob/master/Common%20License%20(MIT)]]|
!!!Usage
Once the plugin is installed (copy - tag {{{systemConfig}}} - reload) storing tiddlers in files is done via 2 steps:
# list (describe) those in [[ExternalTiddlersList]] by writing {{{<<external>>}}} macros there
# if the file exists and the tiddler doesn't, reload TW (external tiddler will be loaded on startup);<br>if the tiddler exists and the file doesn't, just save your TW
Here's how the macro is used:
{{{
<<external [[MyTestTiddler]]>>
}}}
will store the tiddler's text in {{{MyTestTiddler.txt}}} in the same folder as TW. There's a number of other options:
{{{
<<external [[tiddler name]]
[file:<relative path with or without filename and extension>]
[format:{externalized | text}]
[keepInternal:true]
[plugin:true]
>>
}}}
Examples of the {{{file}}} param usage:
* {{{<<external [[MyTestTiddler]] file:"other name">>}}} makes file name different from tiddler name
* {{{<<external [[MyPlugin]] file:"MyPlugin.js" plugin:true>>}}} sets a custom file extension (see also the {{{plugin}}} option below)
* {{{<<external [[MyLog]] file:"../logs/">>}}} will store the file in another folder (note that omitted filename after {{{/}}} means "use tiddler name as filename and default extension")
* the plugin doesn't take care of forbidden characters yet ({{{*}}}, {{{?}}}, {{{"}}} etc), so be careful about those
Supported formats are
* {{{text}}} (default) – only tiddler text is stored in the file with {{{.txt}}} extension; and
* {{{externalized}}} – whole tiddler is stored in the same format as in TW store, file has {{{.tid.html}}} extension and is displayed is monospace tiddler text when opened in browser
Formats can be added by extending {{{config.macros.external.fileFormats}}}.
The {{{keepInternal}}} option makes TW save the tiddler in both external file and TW itself. This, for instance, affects tiddlers stored in {{{text}}} format: without it, all fields like creator, modified etc are destroyed on reload (since are not saved), but with it they are preserved.
The {{{plugin}}} option makes TW evaluate the tiddler like it does with plugins. Note that it doesn't handle plugin dependencies yet. @@color:red;Warning@@: TIFP doesn't currently take care of installing only once, so use {{{plugin}}} with {{{keepInternal}}} ''only'' if you understand the outcome (for instance, some plugins may create infinite loops; also remember that internal version will be always installed first).
***/
//{{{
config.macros.external = {
fileFormats: {
text: {
extension: 'txt',
externalize: function(tiddler) { return tiddler.text },
// changes tiddler as a side-effect not to remove existing fields
internalize: function(tiddler, source) {
tiddler.text = source
}
},
externalized: {
extension: 'tid.html',
externalize: function(tiddler) {
return store.getSaver().externalizeTiddler(store, tiddler)
},
// like for 'text', extends tiddler, doesn't create from scratch
internalize: function(tiddler, source) {
var div = createTiddlyElement(document.body, 'div')
div.setAttribute('style','display:none;')
div.innerHTML = source
store.getLoader().internalizeTiddler(store, tiddler,
tiddler.title, div.firstChild)
div.remove()
}
}/*,
tid: { extension: 'tid' },
json: { extension: 'tid.json' }
externalizedWithFormatter? sane to implement?
*/
},
// here and below "meta" means "info about registered external tiddler,
// be it loaded or not"
getExtension: function(meta) {
const format = this.fileFormats[meta.fileFormat]
if(!format) return //# ok??
return format.extension
},
externalizeTiddler: function(meta) {
const format = this.fileFormats[meta.fileFormat]
if(!format) return //# ok??
return format.externalize(meta.tiddler)
},
internalizeTiddler: function(meta, source) {
const format = this.fileFormats[meta.fileFormat]
if(!format) return //# ok??
const tiddler = store.fetchTiddler(meta.tiddlerName) ||
new Tiddler(meta.tiddlerName)
format.internalize(tiddler, source) //# pass meta to tiddler?
tiddler.doNotSave = function() { return !meta.keepInTW }
meta.tiddler = tiddler
return tiddler
},
listName: "ExternalTiddlersList",
// read files list, load
init: function() {
const listTiddler = store.fetchTiddler(this.listName)
if(!listTiddler || !listTiddler.text) return
wikify(listTiddler.text, createTiddlyElement(null, 'div'))
for(let meta of this.tiddlersMeta) this.loadExternal(meta)
},
handler: function(place, macroName, params, wikifier, paramString, tiddler) {
// parse params, register
const defaultParam = 'tiddler'
const pParams = paramString.parseParams(defaultParam, null, true)
const meta = {}
meta.tiddlerName = getParam(pParams, defaultParam)
if(!meta.tiddlerName) return
// although called .fileName, it actually can contain relative part of a path
// fallback to meta.tiddlerName is set when calculating the full path
meta.fileName = getParam(pParams, 'file', '')
//# check if contains "bad" characters (like " or * ..for local paths only)
meta.fileFormat = getParam(pParams, 'format', 'text')
meta.isPlugin = getFlag(pParams, 'plugin') //# allow just "plugin" instead of "plugin:true"?
const keepInternal = getParam(pParams, 'keepInternal')
meta.keepInTW = !!keepInternal && keepInternal !== 'false' //# ~
this.registerExternal(meta)
// visual feedback
const macroText = wikifier.source.substring(wikifier.matchStart, wikifier.nextMatch)
createTiddlyText(place, 'external ')
createTiddlyLink(place, meta.tiddlerName, true)
createTiddlyText(place, ' (')
createTiddlyElement(place, 'code', '', '', macroText.slice(2 + macroName.length + 1, -2))
createTiddlyText(place, ')')
},
// describes tiddlers registered as external, not necessarily loaded
tiddlersMeta: [],
registerExternal: function(meta) {
//# check if already registered, don't register twice
this.tiddlersMeta.push(meta)
},
getMetaFor: function(tiddlerOrTitle) {
var isTitle = typeof tiddlerOrTitle == "string"
for(meta of this.tiddlersMeta)
if(isTitle && meta.tiddlerName == tiddlerOrTitle ||
!isTitle && meta.tiddler == tiddlerOrTitle)
return meta
},
loadExternal: function(meta) {
// sync loading fails on startup because TF injects new mozillaLoadFile too late
// var tiddlerText = loadFile(getLocalPath(getFullPath(meta.fileName)));
// onExternalTiddlerLoad(tiddlerText !== null, meta, tiddlerText);
// so we use async instead:
const callback = this.onExternalTiddlerLoad
const path = getFullPath(meta.fileName, meta.tiddlerName, this.getExtension(meta))
// httpReq("GET", path, callback, meta) uses default dataType,
// which causes js to get evaluated on load. To avoid this, we customize the ajax call:
jQuery.ajax({
type: "GET",
url: path,
dataType: "text",
processData: false,
cache: false,
complete: function(xhr, textStatus) {
if((!xhr.status && location.protocol === "file:") || (xhr.status >= 200 && xhr.status < 300) || xhr.status === 304)
callback(true, meta, xhr.responseText, path, xhr)
else
callback(false, meta, null, path, xhr)
}
})
//# rename onExternalTiddlerLoad into internalizeAndRegister?
},
onExternalTiddlerLoad: function(success, meta, responseText) {
if(!success) return //# notify somehow? may fail because file is not created yet or ...
const tiddler = config.macros.external.internalizeTiddler(meta, responseText)
store.addTiddler(tiddler)
//# what if tiddler already exists?
if(meta.isPlugin) {
// make it look normally
tiddler.tags.pushUnique('systemConfig')
const author = store.getTiddlerText(tiddler.title + "::Author")
if(author) {
tiddler.creator = tiddler.creator || author
tiddler.modifier = tiddler.modifier || tiddler.creator
}
eval(tiddler.text)
// for plugins introducing macros, formatters etc (may be adjusted in the future)
story.refreshAllTiddlers()
refreshAll()
// apply CSS
store.notifyAll()
}
//meta.lastLoaded = responseText
},
saveExternal: function(meta, callback) {
const fullPath = getFullPath(meta.fileName, meta.tiddlerName, this.getExtension(meta))
// we don't try to save remote files (yet)
if(!isLocalAbsolutePath(fullPath)) {
//# if(callback) callback(.., '[saving remote is not supported]')
return
}
const localPath = getLocalPath(fullPath)
const contentToSave = this.externalizeTiddler(meta)
// save only if have something to save
//if(contentToSave != meta.lastLoaded) {
saveFile(localPath, contentToSave)
//# get result of saving, return it
//# or move externalizing into a separate helper?
// meta.lastLoaded = contentToSave
//# this assumes saving didn't fail, which may be wrong
//}
//# if(callback) callback(.., '[...]')
},
saveAll: function() {
let overallSuccess = true
for(let meta of this.tiddlersMeta) {
// a tiddler may got created after registration
if(!meta.tiddler) {
let tiddlerInStore = store.fetchTiddler(meta.tiddlerName)
if(tiddlerInStore) {
meta.tiddler = tiddlerInStore
} else
// tiddler doesn't exist and we do nothing
continue
//# based on config, we can show a warning instead
}
//# if(meta.tiddler.title != meta.tiddlerName)
// means tiddler got renamed → change meta.tiddlerName &
// update this.listName . If store contains another tiddler
// with that name, still keep the registered one?
overallSuccess = this.saveExternal(meta) && overallSuccess
//# if saving failed, do something! (.oO dirty, notifying)
}
return overallSuccess
}
}
//# see implementations in STP (share to the core?)
function isAbsolutePath(path) {
// covers http:, https:, file:, other schemas, windows paths (D:\...)
if(/^\w+\:/.exec(path)) return true
// unix absolute paths, starting with /
if(/^\//.exec(path)) return true
return false
}
function isLocalAbsolutePath(path) {
//# rename? we're going to check whether an absolute path is local, not path is absolute local
return /^\w\:/.exec(path) || /^\//.exec(path) || /^file\:/.exec(path)
}
function getFullPath(subPath, nameFallback, extension) {
const fileNamePosition = subPath.lastIndexOf('/') + 1
const fileName = subPath.substr(fileNamePosition)
if(fileName && fileName.indexOf('.') == -1)
subPath += '.' + extension
if(!fileName)
subPath += nameFallback + '.' + extension
if(isAbsolutePath(subPath)) return subPath
const url = window.location.toString()
const base = url.substr(0, url.lastIndexOf('/') + 1)
return base + subPath
}
//# ideally, don't save main store if it were not changed
if(!config.macros.external.orig_saveChanges) {
config.macros.external.orig_saveChanges = saveChanges
saveChanges = function(onlyIfDirty, tiddlers) {
config.macros.external.saveAll()
//# should we do smth about setDirty (if saving of a tiddler failed)?
return config.macros.external.orig_saveChanges.apply(this, arguments)
}
}
// hijack method of store (since not present in TiddlyWiki.prototype)
if(!config.macros.external.orig_deleteTiddler) {
config.macros.external.orig_deleteTiddler = store.deleteTiddler
store.deleteTiddler = function(title) {
var registeredMeta = config.macros.external.getMetaFor(title)
if(registeredMeta) registeredMeta.tiddler = null
return config.macros.external.orig_deleteTiddler.apply(this, arguments)
}
}
//}}}
|~ViewToolbar|closeTiddler closeOthers +editTiddler jump permalink > fields references deleteTiddler|
|~EditToolbar|+saveTiddler -cancelTiddler references deleteTiddler|
Yakov Litvin Public Licence Version 0.4
----
Copyright (c) 2014, Yakov Litvin
Redistribution and use in source and compressed source/binary forms, with or without modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
* Redistributions in compressed source/binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.