wiki 0.0.1 → 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (60) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +2 -0
  3. data/README.md +10 -2
  4. data/bin/wiki +4 -0
  5. data/default-data/pages/how-to-wiki +1069 -0
  6. data/default-data/pages/welcome-visitors +336 -0
  7. data/default-data/status/favicon.png +0 -0
  8. data/default-data/status/local-identity +7 -0
  9. data/lib/wiki/favicon.rb +0 -1
  10. data/lib/wiki/page.rb +3 -6
  11. data/lib/wiki/server.rb +7 -3
  12. data/lib/wiki/version.rb +1 -1
  13. data/package.json +70 -0
  14. data/wiki.gemspec +20 -13
  15. metadata +205 -58
  16. data/lib/wiki/views/client/Gruntfile.js +0 -50
  17. data/lib/wiki/views/client/ReadMe.md +0 -67
  18. data/lib/wiki/views/client/build-test.bat +0 -10
  19. data/lib/wiki/views/client/build.bat +0 -8
  20. data/lib/wiki/views/client/builder.pl +0 -41
  21. data/lib/wiki/views/client/client.coffee +0 -3
  22. data/lib/wiki/views/client/client.js +0 -3607
  23. data/lib/wiki/views/client/crosses.png +0 -0
  24. data/lib/wiki/views/client/images/external-link-ltr-icon.png +0 -0
  25. data/lib/wiki/views/client/images/noise.png +0 -0
  26. data/lib/wiki/views/client/images/oops.jpg +0 -0
  27. data/lib/wiki/views/client/js/d3/d3.behavior.js +0 -198
  28. data/lib/wiki/views/client/js/d3/d3.chart.js +0 -984
  29. data/lib/wiki/views/client/js/d3/d3.csv.js +0 -92
  30. data/lib/wiki/views/client/js/d3/d3.geo.js +0 -566
  31. data/lib/wiki/views/client/js/d3/d3.geom.js +0 -825
  32. data/lib/wiki/views/client/js/d3/d3.js +0 -3597
  33. data/lib/wiki/views/client/js/d3/d3.layout.js +0 -1923
  34. data/lib/wiki/views/client/js/d3/d3.time.js +0 -660
  35. data/lib/wiki/views/client/js/images/ui-bg_glass_65_ffffff_1x400.png +0 -0
  36. data/lib/wiki/views/client/js/images/ui-icons_222222_256x240.png +0 -0
  37. data/lib/wiki/views/client/js/jquery-1.6.2.min.js +0 -18
  38. data/lib/wiki/views/client/js/jquery-1.7.1.min.js +0 -4
  39. data/lib/wiki/views/client/js/jquery-1.9.1.min.js +0 -5
  40. data/lib/wiki/views/client/js/jquery-migrate-1.1.1.min.js +0 -3
  41. data/lib/wiki/views/client/js/jquery-ui-1.10.1.custom.min.css +0 -5
  42. data/lib/wiki/views/client/js/jquery-ui-1.10.1.custom.min.js +0 -6
  43. data/lib/wiki/views/client/js/jquery-ui-1.8.16.custom.css +0 -339
  44. data/lib/wiki/views/client/js/jquery-ui-1.8.16.custom.min.js +0 -315
  45. data/lib/wiki/views/client/js/jquery.ie.cors.js +0 -310
  46. data/lib/wiki/views/client/js/jquery.ui.touch-punch.min.js +0 -11
  47. data/lib/wiki/views/client/js/modernizr.custom.63710.js +0 -824
  48. data/lib/wiki/views/client/js/sockjs-0.3.min.js +0 -27
  49. data/lib/wiki/views/client/js/underscore-min.js +0 -30
  50. data/lib/wiki/views/client/mkplugin.sh +0 -97
  51. data/lib/wiki/views/client/package.json +0 -36
  52. data/lib/wiki/views/client/runtests.html +0 -26
  53. data/lib/wiki/views/client/style.css +0 -339
  54. data/lib/wiki/views/client/test/mocha.css +0 -231
  55. data/lib/wiki/views/client/test/mocha.js +0 -5340
  56. data/lib/wiki/views/client/test/testclient.js +0 -17133
  57. data/lib/wiki/views/client/testclient.coffee +0 -18
  58. data/lib/wiki/views/client/theme/granite.css +0 -59
  59. data/lib/wiki/views/client/theme/stoneSeamless.jpg +0 -0
  60. data/lib/wiki/views/client/twitter-maintainance.jpg +0 -0
@@ -1,10 +0,0 @@
1
- @ECHO OFF
2
- ::
3
- :: Used on Windows to build testclient.js as npm start and test don't work!
4
- ::
5
-
6
- :: Build testclient.js - need to expand .\plugins\*\test.coffee as wildcard does not work on Windows
7
-
8
- echo "Building test\testclient.js"
9
-
10
- .\node_modules\.bin\browserify.cmd testclient.coffee .\plugins\calendar\test.coffee .\plugins\changes\test.coffee .\plugins\efficiency\test.coffee .\plugins\report\test.coffee .\plugins\txtzyme\test.coffee -o test\testclient.js
@@ -1,8 +0,0 @@
1
- @ECHO OFF
2
- ::
3
- :: Used on Windows to build client.js as npm start and test don't work!
4
- ::
5
-
6
- echo "Building client.js"
7
-
8
- .\node_modules\.bin\browserify.cmd client.coffee -o client.js
@@ -1,41 +0,0 @@
1
- #!/usr/bin/perl
2
-
3
- #this script exists because the browserify -w option does not appear to work?
4
-
5
- my ($old, $new);
6
- my $OSXsay = ($^O eq 'darwin');
7
-
8
- sub say {
9
- my $msg = shift;
10
- if ($OSXsay) {
11
- `say $msg&`;
12
- }
13
- print $msg."\n";
14
- }
15
- sub run {
16
- $trouble = `($_[0] || echo 'failed to run') 2>&1`;
17
- return unless $trouble;
18
- if ($trouble =~ /( on line \d+)/) {
19
- say("having trouble $1.");
20
- print "\n$_[0]\n$trouble";
21
- }elsif ($trouble =~ /failed to run/) {
22
- say('failed to run');
23
- print("\n$_[0]\n$trouble");
24
- }
25
-
26
- }
27
-
28
- while (sleep 1) {
29
- $new = `ls -lt *.coffee lib/*.coffee test/*.coffee plugins/*/*.coffee`;
30
- next if $old eq $new;
31
- $old = $new;
32
- print `clear; date`;
33
- say('client.');
34
- run('./node_modules/.bin/browserify -t coffeeify client.coffee --debug > client.js');
35
- say('test.');
36
- run('./node_modules/.bin/browserify -t coffeeify testclient.coffee ./plugins/*/test.coffee --debug > test/testclient.js');
37
- say('plugins.');
38
- run('./node_modules/.bin/coffee -c ./plugins/*.coffee');
39
- run('./node_modules/.bin/coffee -c ./plugins/*/*.coffee');
40
- say('done.');
41
- }
@@ -1,3 +0,0 @@
1
- window.wiki = require('wiki-client/lib/wiki')
2
- require('wiki-client/lib/legacy')
3
-
@@ -1,3607 +0,0 @@
1
- ;(function(e,t,n){function i(n,s){if(!t[n]){if(!e[n]){var o=typeof require=="function"&&require;if(!s&&o)return o(n,!0);if(r)return r(n,!0);throw new Error("Cannot find module '"+n+"'")}var u=t[n]={exports:{}};e[n][0].call(u.exports,function(t){var r=e[n][1][t];return i(r?r:t)},u,u.exports)}return t[n].exports}var r=typeof require=="function"&&require;for(var s=0;s<n.length;s++)i(n[s]);return i})({1:[function(require,module,exports){
2
- window.wiki = require('wiki-client/lib/wiki');
3
-
4
- require('wiki-client/lib/legacy');
5
-
6
-
7
- },{"wiki-client/lib/legacy":3,"wiki-client/lib/wiki":2}],2:[function(require,module,exports){
8
- (function() {
9
- var createSynopsis, wiki,
10
- __slice = [].slice;
11
-
12
- createSynopsis = require('./synopsis');
13
-
14
- wiki = {
15
- createSynopsis: createSynopsis
16
- };
17
-
18
- wiki.persona = require('./persona');
19
-
20
- wiki.log = function() {
21
- var things;
22
- things = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
23
- if ((typeof console !== "undefined" && console !== null ? console.log : void 0) != null) {
24
- return console.log.apply(console, things);
25
- }
26
- };
27
-
28
- wiki.asSlug = function(name) {
29
- return name.replace(/\s/g, '-').replace(/[^A-Za-z0-9-]/g, '').toLowerCase();
30
- };
31
-
32
- wiki.useLocalStorage = function() {
33
- return $(".login").length > 0;
34
- };
35
-
36
- wiki.resolutionContext = [];
37
-
38
- wiki.resolveFrom = function(addition, callback) {
39
- wiki.resolutionContext.push(addition);
40
- try {
41
- return callback();
42
- } finally {
43
- wiki.resolutionContext.pop();
44
- }
45
- };
46
-
47
- wiki.getData = function(vis) {
48
- var idx, who;
49
- if (vis) {
50
- idx = $('.item').index(vis);
51
- who = $(".item:lt(" + idx + ")").filter('.chart,.data,.calculator').last();
52
- if (who != null) {
53
- return who.data('item').data;
54
- } else {
55
- return {};
56
- }
57
- } else {
58
- who = $('.chart,.data,.calculator').last();
59
- if (who != null) {
60
- return who.data('item').data;
61
- } else {
62
- return {};
63
- }
64
- }
65
- };
66
-
67
- wiki.getDataNodes = function(vis) {
68
- var idx, who;
69
- if (vis) {
70
- idx = $('.item').index(vis);
71
- who = $(".item:lt(" + idx + ")").filter('.chart,.data,.calculator').toArray().reverse();
72
- return $(who);
73
- } else {
74
- who = $('.chart,.data,.calculator').toArray().reverse();
75
- return $(who);
76
- }
77
- };
78
-
79
- wiki.createPage = function(name, loc) {
80
- var $page, site;
81
- if (loc && loc !== 'view') {
82
- site = loc;
83
- }
84
- $page = $("<div class=\"page\" id=\"" + name + "\">\n <div class=\"twins\"> <p> </p> </div>\n <div class=\"header\">\n <h1> <img class=\"favicon\" src=\"" + (site ? "//" + site : "") + "/favicon.png\" height=\"32px\"> " + name + " </h1>\n </div>\n</div>");
85
- if (site) {
86
- $page.data('site', site);
87
- }
88
- return $page;
89
- };
90
-
91
- wiki.getItem = function(element) {
92
- if ($(element).length > 0) {
93
- return $(element).data("item") || $(element).data('staticItem');
94
- }
95
- };
96
-
97
- wiki.resolveLinks = function(string) {
98
- var renderInternalLink;
99
- renderInternalLink = function(match, name) {
100
- var slug;
101
- slug = wiki.asSlug(name);
102
- return "<a class=\"internal\" href=\"/" + slug + ".html\" data-page-name=\"" + slug + "\" title=\"" + (wiki.resolutionContext.join(' => ')) + "\">" + name + "</a>";
103
- };
104
- return string.replace(/\[\[([^\]]+)\]\]/gi, renderInternalLink).replace(/\[(http.*?) (.*?)\]/gi, "<a class=\"external\" target=\"_blank\" href=\"$1\" title=\"$1\" rel=\"nofollow\">$2 <img src=\"/images/external-link-ltr-icon.png\"></a>");
105
- };
106
-
107
- module.exports = wiki;
108
-
109
- }).call(this);
110
-
111
- },{"./persona":5,"./synopsis":4}],3:[function(require,module,exports){
112
- (function() {
113
- var active, newPage, pageHandler, plugin, refresh, state, util, wiki;
114
-
115
- wiki = require('./wiki');
116
-
117
- util = require('./util');
118
-
119
- pageHandler = wiki.pageHandler = require('./pageHandler');
120
-
121
- plugin = require('./plugin');
122
-
123
- state = require('./state');
124
-
125
- active = require('./active');
126
-
127
- refresh = require('./refresh');
128
-
129
- newPage = require('./page').newPage;
130
-
131
- Array.prototype.last = function() {
132
- return this[this.length - 1];
133
- };
134
-
135
- $(function() {
136
- var LEFTARROW, RIGHTARROW, createTextElement, doInternalLink, finishClick, getTemplate, sleep, textEditor;
137
- window.dialog = $('<div></div>').html('This dialog will show every time!').dialog({
138
- autoOpen: false,
139
- title: 'Basic Dialog',
140
- height: 600,
141
- width: 800
142
- });
143
- wiki.dialog = function(title, html) {
144
- window.dialog.html(html);
145
- window.dialog.dialog("option", "title", wiki.resolveLinks(title));
146
- return window.dialog.dialog('open');
147
- };
148
- sleep = function(time, done) {
149
- return setTimeout(done, time);
150
- };
151
- wiki.removeItem = function($item, item) {
152
- pageHandler.put($item.parents('.page:first'), {
153
- type: 'remove',
154
- id: item.id
155
- });
156
- return $item.remove();
157
- };
158
- wiki.createItem = function($page, $before, item) {
159
- var $item, before;
160
- if ($page == null) {
161
- $page = $before.parents('.page');
162
- }
163
- item.id = util.randomBytes(8);
164
- $item = $("<div class=\"item " + item.type + "\" data-id=\"" + "\"</div>");
165
- $item.data('item', item).data('pageElement', $page);
166
- if ($before != null) {
167
- $before.after($item);
168
- } else {
169
- $page.find('.story').append($item);
170
- }
171
- plugin["do"]($item, item);
172
- before = wiki.getItem($before);
173
- sleep(500, function() {
174
- return pageHandler.put($page, {
175
- item: item,
176
- id: item.id,
177
- type: 'add',
178
- after: before != null ? before.id : void 0
179
- });
180
- });
181
- return $item;
182
- };
183
- createTextElement = function(pageElement, beforeElement, initialText) {
184
- var item, itemBefore, itemElement;
185
- item = {
186
- type: 'paragraph',
187
- id: util.randomBytes(8),
188
- text: initialText
189
- };
190
- itemElement = $("<div class=\"item paragraph\" data-id=" + item.id + "></div>");
191
- itemElement.data('item', item).data('pageElement', pageElement);
192
- beforeElement.after(itemElement);
193
- plugin["do"](itemElement, item);
194
- itemBefore = wiki.getItem(beforeElement);
195
- wiki.textEditor(itemElement, item);
196
- return sleep(500, function() {
197
- return pageHandler.put(pageElement, {
198
- item: item,
199
- id: item.id,
200
- type: 'add',
201
- after: itemBefore != null ? itemBefore.id : void 0
202
- });
203
- });
204
- };
205
- textEditor = wiki.textEditor = function(div, item, caretPos, doubleClicked) {
206
- var original, textarea, _ref;
207
- if (div.hasClass('textEditing')) {
208
- return;
209
- }
210
- div.addClass('textEditing');
211
- textarea = $("<textarea>" + (original = (_ref = item.text) != null ? _ref : '') + "</textarea>").focusout(function() {
212
- div.removeClass('textEditing');
213
- if (item.text = textarea.val()) {
214
- plugin["do"](div.empty(), item);
215
- if (item.text === original) {
216
- return;
217
- }
218
- pageHandler.put(div.parents('.page:first'), {
219
- type: 'edit',
220
- id: item.id,
221
- item: item
222
- });
223
- } else {
224
- pageHandler.put(div.parents('.page:first'), {
225
- type: 'remove',
226
- id: item.id
227
- });
228
- div.remove();
229
- }
230
- return null;
231
- }).bind('keydown', function(e) {
232
- var middle, page, pageElement, prefix, prevItem, prevTextLen, sel, suffix, text;
233
- if ((e.altKey || e.ctlKey || e.metaKey) && e.which === 83) {
234
- textarea.focusout();
235
- return false;
236
- }
237
- if ((e.altKey || e.ctlKey || e.metaKey) && e.which === 73) {
238
- e.preventDefault();
239
- if (!e.shiftKey) {
240
- page = $(e.target).parents('.page');
241
- }
242
- doInternalLink("about " + item.type + " plugin", page);
243
- return false;
244
- }
245
- if (item.type === 'paragraph') {
246
- sel = util.getSelectionPos(textarea);
247
- if (e.which === $.ui.keyCode.BACKSPACE && sel.start === 0 && sel.start === sel.end) {
248
- prevItem = wiki.getItem(div.prev());
249
- if (prevItem.type !== 'paragraph') {
250
- return false;
251
- }
252
- prevTextLen = prevItem.text.length;
253
- prevItem.text += textarea.val();
254
- textarea.val('');
255
- textEditor(div.prev(), prevItem, prevTextLen);
256
- return false;
257
- } else if (e.which === $.ui.keyCode.ENTER && item.type === 'paragraph') {
258
- if (!sel) {
259
- return false;
260
- }
261
- text = textarea.val();
262
- prefix = text.substring(0, sel.start);
263
- if (sel.start !== sel.end) {
264
- middle = text.substring(sel.start, sel.end);
265
- }
266
- suffix = text.substring(sel.end);
267
- if (prefix === '') {
268
- textarea.val(' ');
269
- } else {
270
- textarea.val(prefix);
271
- }
272
- textarea.focusout();
273
- pageElement = div.parent().parent();
274
- createTextElement(pageElement, div, suffix);
275
- if (middle != null) {
276
- createTextElement(pageElement, div, middle);
277
- }
278
- if (prefix === '') {
279
- createTextElement(pageElement, div, '');
280
- }
281
- return false;
282
- }
283
- }
284
- });
285
- div.html(textarea);
286
- if (caretPos != null) {
287
- return util.setCaretPosition(textarea, caretPos);
288
- } else if (doubleClicked) {
289
- util.setCaretPosition(textarea, textarea.val().length);
290
- return textarea.scrollTop(textarea[0].scrollHeight - textarea.height());
291
- } else {
292
- return textarea.focus();
293
- }
294
- };
295
- doInternalLink = wiki.doInternalLink = function(name, page, site) {
296
- if (site == null) {
297
- site = null;
298
- }
299
- name = wiki.asSlug(name);
300
- if (page != null) {
301
- $(page).nextAll().remove();
302
- }
303
- wiki.createPage(name, site).appendTo($('.main')).each(refresh);
304
- return active.set($('.page').last());
305
- };
306
- LEFTARROW = 37;
307
- RIGHTARROW = 39;
308
- $(document).keydown(function(event) {
309
- var direction, newIndex, pages;
310
- direction = (function() {
311
- switch (event.which) {
312
- case LEFTARROW:
313
- return -1;
314
- case RIGHTARROW:
315
- return +1;
316
- }
317
- })();
318
- if (direction && !(event.target.tagName === "TEXTAREA")) {
319
- pages = $('.page');
320
- newIndex = pages.index($('.active')) + direction;
321
- if ((0 <= newIndex && newIndex < pages.length)) {
322
- return active.set(pages.eq(newIndex));
323
- }
324
- }
325
- });
326
- $(window).on('popstate', state.show);
327
- $(document).ajaxError(function(event, request, settings) {
328
- if (request.status === 0 || request.status === 404) {
329
- return;
330
- }
331
- wiki.log('ajax error', event, request, settings);
332
- return $('.main').prepend("<li class='error'>\n Error on " + settings.url + ": " + request.responseText + "\n</li>");
333
- });
334
- getTemplate = function(slug, done) {
335
- if (!slug) {
336
- return done(null);
337
- }
338
- wiki.log('getTemplate', slug);
339
- return pageHandler.get({
340
- whenGotten: function(data, siteFound) {
341
- return done(data.story);
342
- },
343
- whenNotGotten: function() {
344
- return done(null);
345
- },
346
- pageInformation: {
347
- slug: slug
348
- }
349
- });
350
- };
351
- finishClick = function(e, name) {
352
- var page;
353
- e.preventDefault();
354
- if (!e.shiftKey) {
355
- page = $(e.target).parents('.page');
356
- }
357
- doInternalLink(name, page, $(e.target).data('site'));
358
- return false;
359
- };
360
- $('.main').delegate('.show-page-source', 'click', function(e) {
361
- var json, pageElement;
362
- e.preventDefault();
363
- pageElement = $(this).parent().parent();
364
- json = pageElement.data('data');
365
- return wiki.dialog("JSON for " + json.title, $('<pre/>').text(JSON.stringify(json, null, 2)));
366
- }).delegate('.page', 'click', function(e) {
367
- if (!$(e.target).is("a")) {
368
- return active.set(this);
369
- }
370
- }).delegate('.internal', 'click', function(e) {
371
- var name;
372
- name = $(e.target).data('pageName');
373
- pageHandler.context = $(e.target).attr('title').split(' => ');
374
- return finishClick(e, name);
375
- }).delegate('img.remote', 'click', function(e) {
376
- var name;
377
- name = $(e.target).data('slug');
378
- pageHandler.context = [$(e.target).data('site')];
379
- return finishClick(e, name);
380
- }).delegate('.revision', 'dblclick', function(e) {
381
- var $page, action, json, page, rev;
382
- e.preventDefault();
383
- $page = $(this).parents('.page');
384
- page = $page.data('data');
385
- rev = page.journal.length - 1;
386
- action = page.journal[rev];
387
- json = JSON.stringify(action, null, 2);
388
- return wiki.dialog("Revision " + rev + ", " + action.type + " action", $('<pre/>').text(json));
389
- }).delegate('.action', 'click', function(e) {
390
- var $action, $page, name, rev, slug;
391
- e.preventDefault();
392
- $action = $(e.target);
393
- if ($action.is('.fork') && ((name = $action.data('slug')) != null)) {
394
- pageHandler.context = [$action.data('site')];
395
- return finishClick(e, (name.split('_'))[0]);
396
- } else {
397
- $page = $(this).parents('.page');
398
- slug = wiki.asSlug($page.data('data').title);
399
- rev = $(this).parent().children().index($action);
400
- if (!e.shiftKey) {
401
- $page.nextAll().remove();
402
- }
403
- wiki.createPage("" + slug + "_rev" + rev, $page.data('site')).appendTo($('.main')).each(refresh);
404
- return active.set($('.page').last());
405
- }
406
- }).delegate('.fork-page', 'click', function(e) {
407
- var item, pageElement, remoteSite;
408
- pageElement = $(e.target).parents('.page');
409
- if (pageElement.hasClass('local')) {
410
- if (!wiki.useLocalStorage()) {
411
- item = pageElement.data('data');
412
- pageElement.removeClass('local');
413
- return pageHandler.put(pageElement, {
414
- type: 'fork',
415
- item: item
416
- });
417
- }
418
- } else {
419
- if ((remoteSite = pageElement.data('site')) != null) {
420
- return pageHandler.put(pageElement, {
421
- type: 'fork',
422
- site: remoteSite
423
- });
424
- }
425
- }
426
- }).delegate('.action', 'hover', function() {
427
- var id;
428
- id = $(this).attr('data-id');
429
- $("[data-id=" + id + "]").toggleClass('target');
430
- return $('.main').trigger('rev');
431
- }).delegate('.item', 'hover', function() {
432
- var id;
433
- id = $(this).attr('data-id');
434
- return $(".action[data-id=" + id + "]").toggleClass('target');
435
- }).delegate('button.create', 'click', function(e) {
436
- return getTemplate($(e.target).data('slug'), function(story) {
437
- var $page, page, pageObject;
438
- $page = $(e.target).parents('.page:first');
439
- $page.removeClass('ghost');
440
- page = $page.data('data');
441
- page.story = story || [];
442
- pageObject = newPage(page, null);
443
- page = pageObject.getRawPage();
444
- pageHandler.put($page, {
445
- type: 'create',
446
- id: page.id,
447
- item: {
448
- title: page.title,
449
- story: page.story
450
- }
451
- });
452
- return wiki.buildPage(pageObject, $page.empty());
453
- });
454
- }).delegate('.ghost', 'rev', function(e) {
455
- var $item, $page, position;
456
- wiki.log('rev', e);
457
- $page = $(e.target).parents('.page:first');
458
- $item = $page.find('.target');
459
- position = $item.offset().top + $page.scrollTop() - $page.height() / 2;
460
- wiki.log('scroll', $page, $item, position);
461
- return $page.stop().animate({
462
- scrollTop: postion
463
- }, 'slow');
464
- }).delegate('.score', 'hover', function(e) {
465
- return $('.main').trigger('thumb', $(e.target).data('thumb'));
466
- });
467
- $(".provider input").click(function() {
468
- $("footer input:first").val($(this).attr('data-provider'));
469
- return $("footer form").submit();
470
- });
471
- $('body').on('new-neighbor-done', function(e, neighbor) {
472
- return $('.page').each(function(index, element) {
473
- return wiki.emitTwins($(element));
474
- });
475
- });
476
- return $(function() {
477
- state.first();
478
- $('.page').each(refresh);
479
- return active.set($('.page').last());
480
- });
481
- });
482
-
483
- }).call(this);
484
-
485
- },{"./active":10,"./page":12,"./pageHandler":7,"./plugin":8,"./refresh":11,"./state":9,"./util":6,"./wiki":2}],4:[function(require,module,exports){
486
- (function() {
487
- module.exports = function(page) {
488
- var p1, p2, synopsis;
489
- synopsis = page.synopsis;
490
- if ((page != null) && (page.story != null)) {
491
- p1 = page.story[0];
492
- p2 = page.story[1];
493
- if (p1 && p1.type === 'paragraph') {
494
- synopsis || (synopsis = p1.text);
495
- }
496
- if (p2 && p2.type === 'paragraph') {
497
- synopsis || (synopsis = p2.text);
498
- }
499
- if (p1 && (p1.text != null)) {
500
- synopsis || (synopsis = p1.text);
501
- }
502
- if (p2 && (p2.text != null)) {
503
- synopsis || (synopsis = p2.text);
504
- }
505
- synopsis || (synopsis = (page.story != null) && ("A page with " + page.story.length + " items."));
506
- } else {
507
- synopsis = 'A page with no story.';
508
- }
509
- return synopsis;
510
- };
511
-
512
- }).call(this);
513
-
514
- },{}],5:[function(require,module,exports){
515
- (function() {
516
- module.exports = function(owner) {
517
- $("#user-email").hide();
518
- $("#persona-login-btn").hide();
519
- $("#persona-logout-btn").hide();
520
- navigator.id.watch({
521
- loggedInUser: owner,
522
- onlogin: function(assertion) {
523
- return $.post("/persona_login", {
524
- assertion: assertion
525
- }, function(verified) {
526
- verified = JSON.parse(verified);
527
- if ("okay" === verified.status) {
528
- return window.location = "/";
529
- } else {
530
- navigator.id.logout();
531
- if ("wrong-address" === verified.status) {
532
- return window.location = "/oops";
533
- }
534
- }
535
- });
536
- },
537
- onlogout: function() {
538
- return $.post("/persona_logout", function() {
539
- return window.location = "/";
540
- });
541
- },
542
- onready: function() {
543
- if (owner) {
544
- $("#persona-login-btn").hide();
545
- return $("#persona-logout-btn").show();
546
- } else {
547
- $("#persona-login-btn").show();
548
- return $("#persona-logout-btn").hide();
549
- }
550
- }
551
- });
552
- $("#persona-login-btn").click(function(e) {
553
- e.preventDefault();
554
- return navigator.id.request({});
555
- });
556
- return $("#persona-logout-btn").click(function(e) {
557
- e.preventDefault();
558
- return navigator.id.logout();
559
- });
560
- };
561
-
562
- }).call(this);
563
-
564
- },{}],10:[function(require,module,exports){
565
- (function() {
566
- var active, findScrollContainer, scrollTo;
567
-
568
- module.exports = active = {};
569
-
570
- active.scrollContainer = void 0;
571
-
572
- findScrollContainer = function() {
573
- var scrolled;
574
- scrolled = $("body, html").filter(function() {
575
- return $(this).scrollLeft() > 0;
576
- });
577
- if (scrolled.length > 0) {
578
- return scrolled;
579
- } else {
580
- return $("body, html").scrollLeft(12).filter(function() {
581
- return $(this).scrollLeft() > 0;
582
- }).scrollTop(0);
583
- }
584
- };
585
-
586
- scrollTo = function(el) {
587
- var bodyWidth, contentWidth, maxX, minX, target, width;
588
- if (active.scrollContainer == null) {
589
- active.scrollContainer = findScrollContainer();
590
- }
591
- bodyWidth = $("body").width();
592
- minX = active.scrollContainer.scrollLeft();
593
- maxX = minX + bodyWidth;
594
- target = el.position().left;
595
- width = el.outerWidth(true);
596
- contentWidth = $(".page").outerWidth(true) * $(".page").size();
597
- if (target < minX) {
598
- return active.scrollContainer.animate({
599
- scrollLeft: target
600
- });
601
- } else if (target + width > maxX) {
602
- return active.scrollContainer.animate({
603
- scrollLeft: target - (bodyWidth - width)
604
- });
605
- } else if (maxX > $(".pages").outerWidth()) {
606
- return active.scrollContainer.animate({
607
- scrollLeft: Math.min(target, contentWidth - bodyWidth)
608
- });
609
- }
610
- };
611
-
612
- active.set = function(el) {
613
- el = $(el);
614
- $(".active").removeClass("active");
615
- return scrollTo(el.addClass("active"));
616
- };
617
-
618
- }).call(this);
619
-
620
- },{}],6:[function(require,module,exports){
621
- (function() {
622
- var util, wiki;
623
-
624
- wiki = require('./wiki');
625
-
626
- module.exports = wiki.util = util = {};
627
-
628
- util.symbols = {
629
- create: '☼',
630
- add: '+',
631
- edit: '✎',
632
- fork: '⚑',
633
- move: '↕',
634
- remove: '✕'
635
- };
636
-
637
- util.randomByte = function() {
638
- return (((1 + Math.random()) * 0x100) | 0).toString(16).substring(1);
639
- };
640
-
641
- util.randomBytes = function(n) {
642
- return ((function() {
643
- var _i, _results;
644
- _results = [];
645
- for (_i = 1; 1 <= n ? _i <= n : _i >= n; 1 <= n ? _i++ : _i--) {
646
- _results.push(util.randomByte());
647
- }
648
- return _results;
649
- })()).join('');
650
- };
651
-
652
- util.formatTime = function(time) {
653
- var am, d, h, mi, mo;
654
- d = new Date((time > 10000000000 ? time : time * 1000));
655
- mo = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'][d.getMonth()];
656
- h = d.getHours();
657
- am = h < 12 ? 'AM' : 'PM';
658
- h = h === 0 ? 12 : h > 12 ? h - 12 : h;
659
- mi = (d.getMinutes() < 10 ? "0" : "") + d.getMinutes();
660
- return "" + h + ":" + mi + " " + am + "<br>" + (d.getDate()) + " " + mo + " " + (d.getFullYear());
661
- };
662
-
663
- util.formatDate = function(msSinceEpoch) {
664
- var am, d, day, h, mi, mo, sec, wk, yr;
665
- d = new Date(msSinceEpoch);
666
- wk = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'][d.getDay()];
667
- mo = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'][d.getMonth()];
668
- day = d.getDate();
669
- yr = d.getFullYear();
670
- h = d.getHours();
671
- am = h < 12 ? 'AM' : 'PM';
672
- h = h === 0 ? 12 : h > 12 ? h - 12 : h;
673
- mi = (d.getMinutes() < 10 ? "0" : "") + d.getMinutes();
674
- sec = (d.getSeconds() < 10 ? "0" : "") + d.getSeconds();
675
- return "" + wk + " " + mo + " " + day + ", " + yr + "<br>" + h + ":" + mi + ":" + sec + " " + am;
676
- };
677
-
678
- util.formatElapsedTime = function(msSinceEpoch) {
679
- var days, hrs, mins, months, msecs, secs, weeks, years;
680
- msecs = new Date().getTime() - msSinceEpoch;
681
- if ((secs = msecs / 1000) < 2) {
682
- return "" + (Math.floor(msecs)) + " milliseconds ago";
683
- }
684
- if ((mins = secs / 60) < 2) {
685
- return "" + (Math.floor(secs)) + " seconds ago";
686
- }
687
- if ((hrs = mins / 60) < 2) {
688
- return "" + (Math.floor(mins)) + " minutes ago";
689
- }
690
- if ((days = hrs / 24) < 2) {
691
- return "" + (Math.floor(hrs)) + " hours ago";
692
- }
693
- if ((weeks = days / 7) < 2) {
694
- return "" + (Math.floor(days)) + " days ago";
695
- }
696
- if ((months = days / 31) < 2) {
697
- return "" + (Math.floor(weeks)) + " weeks ago";
698
- }
699
- if ((years = days / 365) < 2) {
700
- return "" + (Math.floor(months)) + " months ago";
701
- }
702
- return "" + (Math.floor(years)) + " years ago";
703
- };
704
-
705
- util.emptyPage = function() {
706
- return {
707
- title: 'empty',
708
- story: [],
709
- journal: []
710
- };
711
- };
712
-
713
- util.getSelectionPos = function(jQueryElement) {
714
- var el, iePos, sel;
715
- el = jQueryElement.get(0);
716
- if (document.selection) {
717
- el.focus();
718
- sel = document.selection.createRange();
719
- sel.moveStart('character', -el.value.length);
720
- iePos = sel.text.length;
721
- return {
722
- start: iePos,
723
- end: iePos
724
- };
725
- } else {
726
- return {
727
- start: el.selectionStart,
728
- end: el.selectionEnd
729
- };
730
- }
731
- };
732
-
733
- util.setCaretPosition = function(jQueryElement, caretPos) {
734
- var el, range;
735
- el = jQueryElement.get(0);
736
- if (el != null) {
737
- if (el.createTextRange) {
738
- range = el.createTextRange();
739
- range.move("character", caretPos);
740
- range.select();
741
- } else {
742
- el.setSelectionRange(caretPos, caretPos);
743
- }
744
- return el.focus();
745
- }
746
- };
747
-
748
- }).call(this);
749
-
750
- },{"./wiki":2}],8:[function(require,module,exports){
751
- (function() {
752
- var getScript, plugin, scripts, util, wiki;
753
-
754
- util = require('./util');
755
-
756
- wiki = require('./wiki');
757
-
758
- module.exports = plugin = {};
759
-
760
- scripts = {};
761
-
762
- getScript = wiki.getScript = function(url, callback) {
763
- if (callback == null) {
764
- callback = function() {};
765
- }
766
- if (scripts[url] != null) {
767
- return callback();
768
- } else {
769
- return $.getScript(url).done(function() {
770
- scripts[url] = true;
771
- return callback();
772
- }).fail(function() {
773
- return callback();
774
- });
775
- }
776
- };
777
-
778
- plugin.get = wiki.getPlugin = function(name, callback) {
779
- if (window.plugins[name]) {
780
- return callback(window.plugins[name]);
781
- }
782
- return getScript("/plugins/" + name + "/" + name + ".js", function() {
783
- if (window.plugins[name]) {
784
- return callback(window.plugins[name]);
785
- }
786
- return getScript("/plugins/" + name + ".js", function() {
787
- return callback(window.plugins[name]);
788
- });
789
- });
790
- };
791
-
792
- plugin["do"] = wiki.doPlugin = function(div, item, done) {
793
- var error;
794
- if (done == null) {
795
- done = function() {};
796
- }
797
- error = function(ex) {
798
- var errorElement;
799
- errorElement = $("<div />").addClass('error');
800
- errorElement.text(ex.toString());
801
- return div.append(errorElement);
802
- };
803
- div.data('pageElement', div.parents(".page"));
804
- div.data('item', item);
805
- return plugin.get(item.type, function(script) {
806
- var err;
807
- try {
808
- if (script == null) {
809
- throw TypeError("Can't find plugin for '" + item.type + "'");
810
- }
811
- if (script.emit.length > 2) {
812
- return script.emit(div, item, function() {
813
- script.bind(div, item);
814
- return done();
815
- });
816
- } else {
817
- script.emit(div, item);
818
- script.bind(div, item);
819
- return done();
820
- }
821
- } catch (_error) {
822
- err = _error;
823
- wiki.log('plugin error', err);
824
- error(err);
825
- return done();
826
- }
827
- });
828
- };
829
-
830
- wiki.registerPlugin = function(pluginName, pluginFn) {
831
- return window.plugins[pluginName] = pluginFn($);
832
- };
833
-
834
- window.plugins = {
835
- reference: require('./reference'),
836
- factory: require('./factory'),
837
- paragraph: {
838
- emit: function(div, item) {
839
- var text, _i, _len, _ref, _results;
840
- _ref = item.text.split(/\n\n+/);
841
- _results = [];
842
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
843
- text = _ref[_i];
844
- if (text.match(/\S/)) {
845
- _results.push(div.append("<p>" + (wiki.resolveLinks(text)) + "</p>"));
846
- } else {
847
- _results.push(void 0);
848
- }
849
- }
850
- return _results;
851
- },
852
- bind: function(div, item) {
853
- return div.dblclick(function() {
854
- return wiki.textEditor(div, item, null, true);
855
- });
856
- }
857
- },
858
- image: {
859
- emit: function(div, item) {
860
- item.text || (item.text = item.caption);
861
- return div.append("<img class=thumbnail src=\"" + item.url + "\"> <p>" + (wiki.resolveLinks(item.text)) + "</p>");
862
- },
863
- bind: function(div, item) {
864
- div.dblclick(function() {
865
- return wiki.textEditor(div, item);
866
- });
867
- return div.find('img').dblclick(function() {
868
- return wiki.dialog(item.text, this);
869
- });
870
- }
871
- },
872
- future: {
873
- emit: function(div, item) {
874
- var info, _i, _len, _ref, _results;
875
- div.append("" + item.text + "<br><br><button class=\"create\">create</button> new blank page");
876
- if (((info = wiki.neighborhood[location.host]) != null) && (info.sitemap != null)) {
877
- _ref = info.sitemap;
878
- _results = [];
879
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
880
- item = _ref[_i];
881
- if (item.slug.match(/-template$/)) {
882
- _results.push(div.append("<br><button class=\"create\" data-slug=" + item.slug + ">create</button> from " + (wiki.resolveLinks("[[" + item.title + "]]"))));
883
- } else {
884
- _results.push(void 0);
885
- }
886
- }
887
- return _results;
888
- }
889
- },
890
- bind: function(div, item) {}
891
- }
892
- };
893
-
894
- }).call(this);
895
-
896
- },{"./factory":14,"./reference":13,"./util":6,"./wiki":2}],9:[function(require,module,exports){
897
- (function() {
898
- var active, state, wiki,
899
- __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };
900
-
901
- wiki = require('./wiki');
902
-
903
- active = require('./active');
904
-
905
- module.exports = state = {};
906
-
907
- state.pagesInDom = function() {
908
- return $.makeArray($(".page").map(function(_, el) {
909
- return el.id;
910
- }));
911
- };
912
-
913
- state.urlPages = function() {
914
- var i;
915
- return ((function() {
916
- var _i, _len, _ref, _results;
917
- _ref = $(location).attr('pathname').split('/');
918
- _results = [];
919
- for (_i = 0, _len = _ref.length; _i < _len; _i += 2) {
920
- i = _ref[_i];
921
- _results.push(i);
922
- }
923
- return _results;
924
- })()).slice(1);
925
- };
926
-
927
- state.locsInDom = function() {
928
- return $.makeArray($(".page").map(function(_, el) {
929
- return $(el).data('site') || 'view';
930
- }));
931
- };
932
-
933
- state.urlLocs = function() {
934
- var j, _i, _len, _ref, _results;
935
- _ref = $(location).attr('pathname').split('/').slice(1);
936
- _results = [];
937
- for (_i = 0, _len = _ref.length; _i < _len; _i += 2) {
938
- j = _ref[_i];
939
- _results.push(j);
940
- }
941
- return _results;
942
- };
943
-
944
- state.setUrl = function() {
945
- var idx, locs, page, pages, url, _ref;
946
- document.title = (_ref = $('.page:last').data('data')) != null ? _ref.title : void 0;
947
- if (history && history.pushState) {
948
- locs = state.locsInDom();
949
- pages = state.pagesInDom();
950
- url = ((function() {
951
- var _i, _len, _results;
952
- _results = [];
953
- for (idx = _i = 0, _len = pages.length; _i < _len; idx = ++_i) {
954
- page = pages[idx];
955
- _results.push("/" + ((locs != null ? locs[idx] : void 0) || 'view') + "/" + page);
956
- }
957
- return _results;
958
- })()).join('');
959
- if (url !== $(location).attr('pathname')) {
960
- return history.pushState(null, null, url);
961
- }
962
- }
963
- };
964
-
965
- state.show = function(e) {
966
- var idx, name, newLocs, newPages, old, oldLocs, oldPages, previous, _i, _len, _ref;
967
- oldPages = state.pagesInDom();
968
- newPages = state.urlPages();
969
- oldLocs = state.locsInDom();
970
- newLocs = state.urlLocs();
971
- if (!location.pathname || location.pathname === '/') {
972
- return;
973
- }
974
- previous = $('.page').eq(0);
975
- for (idx = _i = 0, _len = newPages.length; _i < _len; idx = ++_i) {
976
- name = newPages[idx];
977
- if (name !== oldPages[idx]) {
978
- old = $('.page').eq(idx);
979
- if (old) {
980
- old.remove();
981
- }
982
- wiki.createPage(name, newLocs[idx]).insertAfter(previous).each(wiki.refresh);
983
- }
984
- previous = $('.page').eq(idx);
985
- }
986
- previous.nextAll().remove();
987
- active.set($('.page').last());
988
- return document.title = (_ref = $('.page:last').data('data')) != null ? _ref.title : void 0;
989
- };
990
-
991
- state.first = function() {
992
- var firstUrlLocs, firstUrlPages, idx, oldPages, urlPage, _i, _len, _results;
993
- state.setUrl();
994
- firstUrlPages = state.urlPages();
995
- firstUrlLocs = state.urlLocs();
996
- oldPages = state.pagesInDom();
997
- _results = [];
998
- for (idx = _i = 0, _len = firstUrlPages.length; _i < _len; idx = ++_i) {
999
- urlPage = firstUrlPages[idx];
1000
- if (__indexOf.call(oldPages, urlPage) < 0) {
1001
- if (urlPage !== '') {
1002
- _results.push(wiki.createPage(urlPage, firstUrlLocs[idx]).appendTo('.main'));
1003
- } else {
1004
- _results.push(void 0);
1005
- }
1006
- }
1007
- }
1008
- return _results;
1009
- };
1010
-
1011
- }).call(this);
1012
-
1013
- },{"./active":10,"./wiki":2}],15:[function(require,module,exports){
1014
- (function() {
1015
- var create;
1016
-
1017
- create = function(revIndex, data) {
1018
- var afterIndex, editIndex, itemId, items, journal, journalEntry, removeIndex, revJournal, revStory, revStoryIds, revTitle, storyItem, _i, _j, _k, _len, _len1, _len2, _ref;
1019
- journal = data.journal;
1020
- revTitle = data.title;
1021
- revStory = [];
1022
- revJournal = journal.slice(0, +(+revIndex) + 1 || 9e9);
1023
- for (_i = 0, _len = revJournal.length; _i < _len; _i++) {
1024
- journalEntry = revJournal[_i];
1025
- revStoryIds = revStory.map(function(storyItem) {
1026
- return storyItem.id;
1027
- });
1028
- switch (journalEntry.type) {
1029
- case 'create':
1030
- if (journalEntry.item.title != null) {
1031
- revTitle = journalEntry.item.title;
1032
- revStory = journalEntry.item.story || [];
1033
- }
1034
- break;
1035
- case 'add':
1036
- if ((afterIndex = revStoryIds.indexOf(journalEntry.after)) !== -1) {
1037
- revStory.splice(afterIndex + 1, 0, journalEntry.item);
1038
- } else {
1039
- revStory.push(journalEntry.item);
1040
- }
1041
- break;
1042
- case 'edit':
1043
- if ((editIndex = revStoryIds.indexOf(journalEntry.id)) !== -1) {
1044
- revStory.splice(editIndex, 1, journalEntry.item);
1045
- } else {
1046
- revStory.push(journalEntry.item);
1047
- }
1048
- break;
1049
- case 'move':
1050
- items = {};
1051
- for (_j = 0, _len1 = revStory.length; _j < _len1; _j++) {
1052
- storyItem = revStory[_j];
1053
- items[storyItem.id] = storyItem;
1054
- }
1055
- revStory = [];
1056
- _ref = journalEntry.order;
1057
- for (_k = 0, _len2 = _ref.length; _k < _len2; _k++) {
1058
- itemId = _ref[_k];
1059
- if (items[itemId] != null) {
1060
- revStory.push(items[itemId]);
1061
- }
1062
- }
1063
- break;
1064
- case 'remove':
1065
- if ((removeIndex = revStoryIds.indexOf(journalEntry.id)) !== -1) {
1066
- revStory.splice(removeIndex, 1);
1067
- }
1068
- }
1069
- }
1070
- return {
1071
- story: revStory,
1072
- journal: revJournal,
1073
- title: revTitle
1074
- };
1075
- };
1076
-
1077
- exports.create = create;
1078
-
1079
- }).call(this);
1080
-
1081
- },{}],13:[function(require,module,exports){
1082
- (function() {
1083
- var bind, emit;
1084
-
1085
- emit = function($item, item) {
1086
- var site, slug;
1087
- slug = item.slug || 'welcome-visitors';
1088
- site = item.site;
1089
- return wiki.resolveFrom(site, function() {
1090
- return $item.append("<p style='margin-bottom:3px;'>\n <img class='remote'\n src='//" + site + "/favicon.png'\n title='" + site + "'\n data-site=\"" + site + "\"\n data-slug=\"" + slug + "\"\n >\n " + (wiki.resolveLinks("[[" + (item.title || slug) + "]]")) + "\n</p>\n<div>\n " + (wiki.resolveLinks(item.text)) + "\n</div>");
1091
- });
1092
- };
1093
-
1094
- bind = function($item, item) {
1095
- return $item.dblclick(function() {
1096
- return wiki.textEditor($item, item);
1097
- });
1098
- };
1099
-
1100
- module.exports = {
1101
- emit: emit,
1102
- bind: bind
1103
- };
1104
-
1105
- }).call(this);
1106
-
1107
- },{}],14:[function(require,module,exports){
1108
- (function() {
1109
- var arrayToJson, bind, csvToArray, emit,
1110
- __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };
1111
-
1112
- emit = function(div, item) {
1113
- var showMenu, showPrompt;
1114
- div.append('<p>Double-Click to Edit<br>Drop Text or Image to Insert</p>');
1115
- showMenu = function() {
1116
- var info, menu, menuItem, name, _i, _len, _ref, _ref1;
1117
- menu = div.find('p').append("<br>Or Choose a Plugin");
1118
- menuItem = function(title, name) {
1119
- return menu.append("<li><a class=\"menu\" href=\"#\" title=\"" + title + "\">" + name + "</a></li>");
1120
- };
1121
- if (Array.isArray(window.catalog)) {
1122
- _ref = window.catalog;
1123
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
1124
- info = _ref[_i];
1125
- menuItem(info.title, info.name);
1126
- }
1127
- } else {
1128
- _ref1 = window.catalog;
1129
- for (name in _ref1) {
1130
- info = _ref1[name];
1131
- menuItem(info.menu, name);
1132
- }
1133
- }
1134
- return menu.find('a.menu').click(function(evt) {
1135
- div.removeClass('factory').addClass(item.type = evt.target.text.toLowerCase());
1136
- div.unbind();
1137
- return wiki.textEditor(div, item);
1138
- });
1139
- };
1140
- showPrompt = function() {
1141
- return div.append("<p>" + (wiki.resolveLinks(item.prompt)) + "</b>");
1142
- };
1143
- if (item.prompt) {
1144
- return showPrompt();
1145
- } else if (window.catalog != null) {
1146
- return showMenu();
1147
- } else {
1148
- return $.getJSON('/system/factories.json', function(data) {
1149
- window.catalog = data;
1150
- return showMenu();
1151
- });
1152
- }
1153
- };
1154
-
1155
- bind = function(div, item) {
1156
- var syncEditAction;
1157
- syncEditAction = function() {
1158
- var err, pageElement;
1159
- wiki.log('factory item', item);
1160
- div.empty().unbind();
1161
- div.removeClass("factory").addClass(item.type);
1162
- pageElement = div.parents('.page:first');
1163
- try {
1164
- div.data('pageElement', pageElement);
1165
- div.data('item', item);
1166
- wiki.getPlugin(item.type, function(plugin) {
1167
- plugin.emit(div, item);
1168
- return plugin.bind(div, item);
1169
- });
1170
- } catch (_error) {
1171
- err = _error;
1172
- div.append("<p class='error'>" + err + "</p>");
1173
- }
1174
- return wiki.pageHandler.put(pageElement, {
1175
- type: 'edit',
1176
- id: item.id,
1177
- item: item
1178
- });
1179
- };
1180
- div.dblclick(function() {
1181
- div.removeClass('factory').addClass(item.type = 'paragraph');
1182
- div.unbind();
1183
- return wiki.textEditor(div, item);
1184
- });
1185
- div.bind('dragenter', function(evt) {
1186
- return evt.preventDefault();
1187
- });
1188
- div.bind('dragover', function(evt) {
1189
- return evt.preventDefault();
1190
- });
1191
- return div.bind("drop", function(dropEvent) {
1192
- var dt, found, ignore, origin, punt, readFile, url;
1193
- punt = function(data) {
1194
- item.prompt = "<b>Unexpected Item</b><br>We can't make sense of the drop.<br>" + (JSON.stringify(data)) + "<br>Try something else or see [[About Factory Plugin]].";
1195
- data.userAgent = navigator.userAgent;
1196
- item.punt = data;
1197
- wiki.log('factory punt', dropEvent);
1198
- return syncEditAction();
1199
- };
1200
- readFile = function(file) {
1201
- var majorType, minorType, reader, _ref;
1202
- if (file != null) {
1203
- _ref = file.type.split("/"), majorType = _ref[0], minorType = _ref[1];
1204
- reader = new FileReader();
1205
- if (majorType === "image") {
1206
- reader.onload = function(loadEvent) {
1207
- item.type = 'image';
1208
- item.url = loadEvent.target.result;
1209
- item.caption || (item.caption = "Uploaded image");
1210
- return syncEditAction();
1211
- };
1212
- return reader.readAsDataURL(file);
1213
- } else if (majorType === "text") {
1214
- reader.onload = function(loadEvent) {
1215
- var array, result;
1216
- result = loadEvent.target.result;
1217
- if (minorType === 'csv') {
1218
- item.type = 'data';
1219
- item.columns = (array = csvToArray(result))[0];
1220
- item.data = arrayToJson(array);
1221
- item.text = file.fileName;
1222
- } else {
1223
- item.type = 'paragraph';
1224
- item.text = result;
1225
- }
1226
- return syncEditAction();
1227
- };
1228
- return reader.readAsText(file);
1229
- } else {
1230
- return punt({
1231
- number: 1,
1232
- name: file.fileName,
1233
- type: file.type
1234
- });
1235
- }
1236
- } else {
1237
- return punt({
1238
- number: 2,
1239
- types: dropEvent.originalEvent.dataTransfer.types
1240
- });
1241
- }
1242
- };
1243
- dropEvent.preventDefault();
1244
- if ((dt = dropEvent.originalEvent.dataTransfer) != null) {
1245
- if ((dt.types != null) && (__indexOf.call(dt.types, 'text/uri-list') >= 0 || __indexOf.call(dt.types, 'text/x-moz-url') >= 0) && !(__indexOf.call(dt.types, 'Files') >= 0)) {
1246
- url = dt.getData('URL');
1247
- if (found = url.match(/^http:\/\/([a-zA-Z0-9:.-]+)(\/([a-zA-Z0-9:.-]+)\/([a-z0-9-]+(_rev\d+)?))+$/)) {
1248
- wiki.log('factory drop url', found);
1249
- ignore = found[0], origin = found[1], ignore = found[2], item.site = found[3], item.slug = found[4], ignore = found[5];
1250
- if ($.inArray(item.site, ['view', 'local', 'origin']) >= 0) {
1251
- item.site = origin;
1252
- }
1253
- return $.getJSON("http://" + item.site + "/" + item.slug + ".json", function(remote) {
1254
- wiki.log('factory remote', remote);
1255
- item.type = 'reference';
1256
- item.title = remote.title || item.slug;
1257
- item.text = wiki.createSynopsis(remote);
1258
- syncEditAction();
1259
- if (item.site != null) {
1260
- return wiki.registerNeighbor(item.site);
1261
- }
1262
- });
1263
- } else {
1264
- return punt({
1265
- number: 4,
1266
- url: url,
1267
- types: dt.types
1268
- });
1269
- }
1270
- } else if (__indexOf.call(dt.types, 'Files') >= 0) {
1271
- return readFile(dt.files[0]);
1272
- } else {
1273
- return punt({
1274
- number: 5,
1275
- types: dt.types
1276
- });
1277
- }
1278
- } else {
1279
- return punt({
1280
- number: 6,
1281
- trouble: "no data transfer object"
1282
- });
1283
- }
1284
- });
1285
- };
1286
-
1287
- csvToArray = function(strData, strDelimiter) {
1288
- var arrData, arrMatches, objPattern, strMatchedDelimiter, strMatchedValue;
1289
- strDelimiter = strDelimiter || ",";
1290
- objPattern = new RegExp("(\\" + strDelimiter + "|\\r?\\n|\\r|^)" + "(?:\"([^\"]*(?:\"\"[^\"]*)*)\"|" + "([^\"\\" + strDelimiter + "\\r\\n]*))", "gi");
1291
- arrData = [[]];
1292
- arrMatches = null;
1293
- while (arrMatches = objPattern.exec(strData)) {
1294
- strMatchedDelimiter = arrMatches[1];
1295
- if (strMatchedDelimiter.length && (strMatchedDelimiter !== strDelimiter)) {
1296
- arrData.push([]);
1297
- }
1298
- if (arrMatches[2]) {
1299
- strMatchedValue = arrMatches[2].replace(new RegExp("\"\"", "g"), "\"");
1300
- } else {
1301
- strMatchedValue = arrMatches[3];
1302
- }
1303
- arrData[arrData.length - 1].push(strMatchedValue);
1304
- }
1305
- return arrData;
1306
- };
1307
-
1308
- arrayToJson = function(array) {
1309
- var cols, row, rowToObject, _i, _len, _results;
1310
- cols = array.shift();
1311
- rowToObject = function(row) {
1312
- var k, obj, v, _i, _len, _ref, _ref1;
1313
- obj = {};
1314
- _ref = _.zip(cols, row);
1315
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
1316
- _ref1 = _ref[_i], k = _ref1[0], v = _ref1[1];
1317
- if ((v != null) && (v.match(/\S/)) && v !== 'NULL') {
1318
- obj[k] = v;
1319
- }
1320
- }
1321
- return obj;
1322
- };
1323
- _results = [];
1324
- for (_i = 0, _len = array.length; _i < _len; _i++) {
1325
- row = array[_i];
1326
- _results.push(rowToObject(row));
1327
- }
1328
- return _results;
1329
- };
1330
-
1331
- module.exports = {
1332
- emit: emit,
1333
- bind: bind
1334
- };
1335
-
1336
- }).call(this);
1337
-
1338
- },{}],16:[function(require,module,exports){
1339
- (function() {
1340
- var util;
1341
-
1342
- util = require('./util');
1343
-
1344
- module.exports = function(journalElement, action) {
1345
- var actionElement, actionTitle, controls, pageElement;
1346
- pageElement = journalElement.parents('.page:first');
1347
- actionTitle = action.type || 'separator';
1348
- if (action.date != null) {
1349
- actionTitle += " " + (util.formatElapsedTime(action.date));
1350
- }
1351
- actionElement = $("<a href=\"#\" /> ").addClass("action").addClass(action.type || 'separator').text(action.symbol || util.symbols[action.type]).attr('title', actionTitle).attr('data-id', action.id || "0").data('action', action);
1352
- controls = journalElement.children('.control-buttons');
1353
- if (controls.length > 0) {
1354
- actionElement.insertBefore(controls);
1355
- } else {
1356
- actionElement.appendTo(journalElement);
1357
- }
1358
- if (action.type === 'fork' && (action.site != null)) {
1359
- return actionElement.css("background-image", "url(//" + action.site + "/favicon.png)").attr("href", "//" + action.site + "/" + (pageElement.attr('id')) + ".html").data("site", action.site).data("slug", pageElement.attr('id'));
1360
- }
1361
- };
1362
-
1363
- }).call(this);
1364
-
1365
- },{"./util":6}],7:[function(require,module,exports){
1366
- (function() {
1367
- var addToJournal, newPage, pageFromLocalStorage, pageHandler, pushToLocal, pushToServer, recursiveGet, revision, state, util, wiki, _;
1368
-
1369
- _ = require('underscore');
1370
-
1371
- wiki = require('./wiki');
1372
-
1373
- util = require('./util');
1374
-
1375
- state = require('./state');
1376
-
1377
- revision = require('./revision');
1378
-
1379
- addToJournal = require('./addToJournal');
1380
-
1381
- newPage = require('./page').newPage;
1382
-
1383
- module.exports = pageHandler = {};
1384
-
1385
- pageFromLocalStorage = function(slug) {
1386
- var json;
1387
- if (json = localStorage[slug]) {
1388
- return JSON.parse(json);
1389
- } else {
1390
- return void 0;
1391
- }
1392
- };
1393
-
1394
- recursiveGet = function(_arg) {
1395
- var localContext, localPage, pageInformation, rev, site, slug, url, whenGotten, whenNotGotten;
1396
- pageInformation = _arg.pageInformation, whenGotten = _arg.whenGotten, whenNotGotten = _arg.whenNotGotten, localContext = _arg.localContext;
1397
- slug = pageInformation.slug, rev = pageInformation.rev, site = pageInformation.site;
1398
- if (site) {
1399
- localContext = [];
1400
- } else {
1401
- site = localContext.shift();
1402
- }
1403
- if (site === window.location.host) {
1404
- site = 'origin';
1405
- }
1406
- if (site === 'view') {
1407
- site = null;
1408
- }
1409
- if (site != null) {
1410
- if (site === 'local') {
1411
- if (localPage = pageFromLocalStorage(pageInformation.slug)) {
1412
- return whenGotten(newPage(localPage, 'local'));
1413
- } else {
1414
- return whenNotGotten();
1415
- }
1416
- } else {
1417
- if (site === 'origin') {
1418
- url = "/" + slug + ".json";
1419
- } else {
1420
- url = "http://" + site + "/" + slug + ".json";
1421
- }
1422
- }
1423
- } else {
1424
- url = "/" + slug + ".json";
1425
- }
1426
- return $.ajax({
1427
- type: 'GET',
1428
- dataType: 'json',
1429
- url: url + ("?random=" + (util.randomBytes(4))),
1430
- success: function(page) {
1431
- if (rev) {
1432
- page = revision.create(rev, page);
1433
- }
1434
- return whenGotten(newPage(page, site));
1435
- },
1436
- error: function(xhr, type, msg) {
1437
- var report;
1438
- if ((xhr.status !== 404) && (xhr.status !== 0)) {
1439
- wiki.log('pageHandler.get error', xhr, xhr.status, type, msg);
1440
- report = {
1441
- 'title': "" + xhr.status + " " + msg,
1442
- 'story': [
1443
- {
1444
- 'type': 'paragraph',
1445
- 'id': '928739187243',
1446
- 'text': "<pre>" + xhr.responseText
1447
- }
1448
- ]
1449
- };
1450
- return whenGotten(report, 'local');
1451
- }
1452
- if (localContext.length > 0) {
1453
- return recursiveGet({
1454
- pageInformation: pageInformation,
1455
- whenGotten: whenGotten,
1456
- whenNotGotten: whenNotGotten,
1457
- localContext: localContext
1458
- });
1459
- } else {
1460
- return whenNotGotten();
1461
- }
1462
- }
1463
- });
1464
- };
1465
-
1466
- pageHandler.get = function(_arg) {
1467
- var localPage, pageInformation, whenGotten, whenNotGotten;
1468
- whenGotten = _arg.whenGotten, whenNotGotten = _arg.whenNotGotten, pageInformation = _arg.pageInformation;
1469
- if (!pageInformation.site) {
1470
- if (localPage = pageFromLocalStorage(pageInformation.slug)) {
1471
- if (pageInformation.rev) {
1472
- localPage = revision.create(pageInformation.rev, localPage);
1473
- }
1474
- return whenGotten(newPage(localPage, 'local'));
1475
- }
1476
- }
1477
- if (!pageHandler.context.length) {
1478
- pageHandler.context = ['view'];
1479
- }
1480
- return recursiveGet({
1481
- pageInformation: pageInformation,
1482
- whenGotten: whenGotten,
1483
- whenNotGotten: whenNotGotten,
1484
- localContext: _.clone(pageHandler.context)
1485
- });
1486
- };
1487
-
1488
- pageHandler.context = [];
1489
-
1490
- pushToLocal = function(pageElement, pagePutInfo, action) {
1491
- var page, site;
1492
- if (action.type === 'create') {
1493
- page = {
1494
- title: action.item.title,
1495
- story: [],
1496
- journal: []
1497
- };
1498
- } else {
1499
- page = pageFromLocalStorage(pagePutInfo.slug);
1500
- page || (page = pageElement.data("data"));
1501
- if (page.journal == null) {
1502
- page.journal = [];
1503
- }
1504
- if ((site = action['fork']) != null) {
1505
- page.journal = page.journal.concat({
1506
- 'type': 'fork',
1507
- 'site': site
1508
- });
1509
- delete action['fork'];
1510
- }
1511
- page.story = $(pageElement).find(".item").map(function() {
1512
- return $(this).data("item");
1513
- }).get();
1514
- }
1515
- page.journal = page.journal.concat(action);
1516
- localStorage[pagePutInfo.slug] = JSON.stringify(page);
1517
- return addToJournal(pageElement.find('.journal'), action);
1518
- };
1519
-
1520
- pushToServer = function(pageElement, pagePutInfo, action) {
1521
- return $.ajax({
1522
- type: 'PUT',
1523
- url: "/page/" + pagePutInfo.slug + "/action",
1524
- data: {
1525
- 'action': JSON.stringify(action)
1526
- },
1527
- success: function() {
1528
- addToJournal(pageElement.find('.journal'), action);
1529
- if (action.type === 'fork') {
1530
- localStorage.removeItem(pageElement.attr('id'));
1531
- return state.setUrl;
1532
- }
1533
- },
1534
- error: function(xhr, type, msg) {
1535
- return wiki.log("pageHandler.put ajax error callback", type, msg);
1536
- }
1537
- });
1538
- };
1539
-
1540
- pageHandler.put = function(pageElement, action) {
1541
- var checkedSite, forkFrom, pagePutInfo;
1542
- checkedSite = function() {
1543
- var site;
1544
- switch (site = pageElement.data('site')) {
1545
- case 'origin':
1546
- case 'local':
1547
- case 'view':
1548
- return null;
1549
- case location.host:
1550
- return null;
1551
- default:
1552
- return site;
1553
- }
1554
- };
1555
- pagePutInfo = {
1556
- slug: pageElement.attr('id').split('_rev')[0],
1557
- rev: pageElement.attr('id').split('_rev')[1],
1558
- site: checkedSite(),
1559
- local: pageElement.hasClass('local')
1560
- };
1561
- forkFrom = pagePutInfo.site;
1562
- wiki.log('pageHandler.put', action, pagePutInfo);
1563
- if (wiki.useLocalStorage()) {
1564
- if (pagePutInfo.site != null) {
1565
- wiki.log('remote => local');
1566
- } else if (!pagePutInfo.local) {
1567
- wiki.log('origin => local');
1568
- action.site = forkFrom = location.host;
1569
- }
1570
- }
1571
- action.date = (new Date()).getTime();
1572
- if (action.site === 'origin') {
1573
- delete action.site;
1574
- }
1575
- if (forkFrom) {
1576
- pageElement.find('h1 img').attr('src', '/favicon.png');
1577
- pageElement.find('h1 a').attr('href', '/');
1578
- pageElement.data('site', null);
1579
- pageElement.removeClass('remote');
1580
- state.setUrl();
1581
- if (action.type !== 'fork') {
1582
- action.fork = forkFrom;
1583
- addToJournal(pageElement.find('.journal'), {
1584
- type: 'fork',
1585
- site: forkFrom,
1586
- date: action.date
1587
- });
1588
- }
1589
- }
1590
- if (wiki.useLocalStorage() || pagePutInfo.site === 'local') {
1591
- pushToLocal(pageElement, pagePutInfo, action);
1592
- return pageElement.addClass("local");
1593
- } else {
1594
- return pushToServer(pageElement, pagePutInfo, action);
1595
- }
1596
- };
1597
-
1598
- }).call(this);
1599
-
1600
- },{"./addToJournal":16,"./page":12,"./revision":15,"./state":9,"./util":6,"./wiki":2,"underscore":17}],11:[function(require,module,exports){
1601
- (function() {
1602
- var addToJournal, buildPageHeader, createFactory, emitHeader, emitTwins, handleDragging, initAddButton, initDragging, neighborhood, pageHandler, plugin, refresh, renderPageIntoPageElement, state, util, wiki, _;
1603
-
1604
- _ = require('underscore');
1605
-
1606
- util = require('./util');
1607
-
1608
- pageHandler = require('./pageHandler');
1609
-
1610
- plugin = require('./plugin');
1611
-
1612
- state = require('./state');
1613
-
1614
- neighborhood = require('./neighborhood');
1615
-
1616
- addToJournal = require('./addToJournal');
1617
-
1618
- wiki = require('./wiki');
1619
-
1620
- handleDragging = function(evt, ui) {
1621
- var action, before, beforeElement, destinationPageElement, equals, item, itemElement, moveFromPage, moveToPage, moveWithinPage, order, sourcePageElement, sourceSite, thisPageElement;
1622
- itemElement = ui.item;
1623
- item = wiki.getItem(itemElement);
1624
- thisPageElement = $(this).parents('.page:first');
1625
- sourcePageElement = itemElement.data('pageElement');
1626
- sourceSite = sourcePageElement.data('site');
1627
- destinationPageElement = itemElement.parents('.page:first');
1628
- equals = function(a, b) {
1629
- return a && b && a.get(0) === b.get(0);
1630
- };
1631
- moveWithinPage = !sourcePageElement || equals(sourcePageElement, destinationPageElement);
1632
- moveFromPage = !moveWithinPage && equals(thisPageElement, sourcePageElement);
1633
- moveToPage = !moveWithinPage && equals(thisPageElement, destinationPageElement);
1634
- if (moveFromPage) {
1635
- if (sourcePageElement.hasClass('ghost') || sourcePageElement.attr('id') === destinationPageElement.attr('id')) {
1636
- return;
1637
- }
1638
- }
1639
- action = moveWithinPage ? (order = $(this).children().map(function(_, value) {
1640
- return $(value).attr('data-id');
1641
- }).get(), {
1642
- type: 'move',
1643
- order: order
1644
- }) : moveFromPage ? (wiki.log('drag from', sourcePageElement.find('h1').text()), {
1645
- type: 'remove'
1646
- }) : moveToPage ? (itemElement.data('pageElement', thisPageElement), beforeElement = itemElement.prev('.item'), before = wiki.getItem(beforeElement), {
1647
- type: 'add',
1648
- item: item,
1649
- after: before != null ? before.id : void 0
1650
- }) : void 0;
1651
- action.id = item.id;
1652
- return pageHandler.put(thisPageElement, action);
1653
- };
1654
-
1655
- initDragging = function($page) {
1656
- var $story;
1657
- $story = $page.find('.story');
1658
- return $story.sortable({
1659
- connectWith: '.page .story'
1660
- }).on("sortupdate", handleDragging);
1661
- };
1662
-
1663
- initAddButton = function($page) {
1664
- return $page.find(".add-factory").live("click", function(evt) {
1665
- if ($page.hasClass('ghost')) {
1666
- return;
1667
- }
1668
- evt.preventDefault();
1669
- return createFactory($page);
1670
- });
1671
- };
1672
-
1673
- createFactory = function($page) {
1674
- var before, beforeElement, item, itemElement;
1675
- item = {
1676
- type: "factory",
1677
- id: util.randomBytes(8)
1678
- };
1679
- itemElement = $("<div />", {
1680
- "class": "item factory"
1681
- }).data('item', item).attr('data-id', item.id);
1682
- itemElement.data('pageElement', $page);
1683
- $page.find(".story").append(itemElement);
1684
- plugin["do"](itemElement, item);
1685
- beforeElement = itemElement.prev('.item');
1686
- before = wiki.getItem(beforeElement);
1687
- return pageHandler.put($page, {
1688
- item: item,
1689
- id: item.id,
1690
- type: "add",
1691
- after: before != null ? before.id : void 0
1692
- });
1693
- };
1694
-
1695
- buildPageHeader = function(_arg) {
1696
- var favicon_src, header_href, page, tooltip;
1697
- page = _arg.page, tooltip = _arg.tooltip, header_href = _arg.header_href, favicon_src = _arg.favicon_src;
1698
- if (page.plugin) {
1699
- tooltip += "\n" + page.plugin + " plugin";
1700
- }
1701
- return "<h1 title=\"" + tooltip + "\"><a href=\"" + header_href + "\"><img src=\"" + favicon_src + "\" height=\"32px\" class=\"favicon\"></a> " + page.title + "</h1>";
1702
- };
1703
-
1704
- emitHeader = function($header, $page, pageObject) {
1705
- var date, header, isRemotePage, page, pageHeader, rev, viewHere;
1706
- page = pageObject.getRawPage();
1707
- isRemotePage = pageObject.isRemote();
1708
- header = '';
1709
- viewHere = wiki.asSlug(page.title) === 'welcome-visitors' ? "" : "/view/" + (pageObject.getSlug());
1710
- pageHeader = isRemotePage ? buildPageHeader({
1711
- tooltip: pageObject.getRemoteSite(),
1712
- header_href: "//" + (pageObject.getRemoteSite()) + "/view/welcome-visitors" + viewHere,
1713
- favicon_src: "http://" + (pageObject.getRemoteSite()) + "/favicon.png",
1714
- page: page
1715
- }) : buildPageHeader({
1716
- tooltip: location.host,
1717
- header_href: "/view/welcome-visitors" + viewHere,
1718
- favicon_src: "/favicon.png",
1719
- page: page
1720
- });
1721
- $header.append(pageHeader);
1722
- if (!isRemotePage) {
1723
- $('img.favicon', $page).error(function(e) {
1724
- return plugin.get('favicon', function(favicon) {
1725
- return favicon.create();
1726
- });
1727
- });
1728
- }
1729
- if ($page.attr('id').match(/_rev/)) {
1730
- rev = page.journal.length - 1;
1731
- date = page.journal[rev].date;
1732
- $page.addClass('ghost').data('rev', rev);
1733
- return $header.append($("<h2 class=\"revision\">\n <span>\n " + (date != null ? util.formatDate(date) : "Revision " + rev) + "\n </span>\n</h2>"));
1734
- }
1735
- };
1736
-
1737
- emitTwins = wiki.emitTwins = function($page) {
1738
- var actions, bin, bins, flags, i, info, item, legend, page, remoteSite, site, slug, twins, viewing, _i, _len, _ref, _ref1, _ref2, _ref3;
1739
- page = $page.data('data');
1740
- site = $page.data('site') || window.location.host;
1741
- if (site === 'view' || site === 'origin') {
1742
- site = window.location.host;
1743
- }
1744
- slug = wiki.asSlug(page.title);
1745
- if (((actions = (_ref = page.journal) != null ? _ref.length : void 0) != null) && ((viewing = (_ref1 = page.journal[actions - 1]) != null ? _ref1.date : void 0) != null)) {
1746
- viewing = Math.floor(viewing / 1000) * 1000;
1747
- bins = {
1748
- newer: [],
1749
- same: [],
1750
- older: []
1751
- };
1752
- _ref2 = wiki.neighborhood;
1753
- for (remoteSite in _ref2) {
1754
- info = _ref2[remoteSite];
1755
- if (remoteSite !== site && (info.sitemap != null)) {
1756
- _ref3 = info.sitemap;
1757
- for (_i = 0, _len = _ref3.length; _i < _len; _i++) {
1758
- item = _ref3[_i];
1759
- if (item.slug === slug) {
1760
- bin = item.date > viewing ? bins.newer : item.date < viewing ? bins.older : bins.same;
1761
- bin.push({
1762
- remoteSite: remoteSite,
1763
- item: item
1764
- });
1765
- }
1766
- }
1767
- }
1768
- }
1769
- twins = [];
1770
- for (legend in bins) {
1771
- bin = bins[legend];
1772
- if (!bin.length) {
1773
- continue;
1774
- }
1775
- bin.sort(function(a, b) {
1776
- return a.item.date < b.item.date;
1777
- });
1778
- flags = (function() {
1779
- var _j, _len1, _ref4, _results;
1780
- _results = [];
1781
- for (i = _j = 0, _len1 = bin.length; _j < _len1; i = ++_j) {
1782
- _ref4 = bin[i], remoteSite = _ref4.remoteSite, item = _ref4.item;
1783
- if (i >= 8) {
1784
- break;
1785
- }
1786
- _results.push("<img class=\"remote\"\nsrc=\"http://" + remoteSite + "/favicon.png\"\ndata-slug=\"" + slug + "\"\ndata-site=\"" + remoteSite + "\"\ntitle=\"" + remoteSite + "\">");
1787
- }
1788
- return _results;
1789
- })();
1790
- twins.push("" + (flags.join('&nbsp;')) + " " + legend);
1791
- }
1792
- if (twins) {
1793
- return $page.find('.twins').html("<p>" + (twins.join(", ")) + "</p>");
1794
- }
1795
- }
1796
- };
1797
-
1798
- renderPageIntoPageElement = function(pageObject, $page) {
1799
- var $footer, $header, $journal, $story, $twins, host, slug, _ref;
1800
- $page.data("data", pageObject.getRawPage());
1801
- if (pageObject.isRemote()) {
1802
- $page.data("site", pageObject.getRemoteSite());
1803
- }
1804
- slug = $page.attr('id');
1805
- wiki.resolutionContext = pageObject.getContext();
1806
- $page.empty();
1807
- _ref = ['twins', 'header', 'story', 'journal', 'footer'].map(function(className) {
1808
- return $("<div />").addClass(className).appendTo($page);
1809
- }), $twins = _ref[0], $header = _ref[1], $story = _ref[2], $journal = _ref[3], $footer = _ref[4];
1810
- emitHeader($header, $page, pageObject);
1811
- pageObject.seqItems(function(item, done) {
1812
- var $item;
1813
- if ((item != null ? item.type : void 0) && (item != null ? item.id : void 0)) {
1814
- $item = $("<div class=\"item " + item.type + "\" data-id=\"" + item.id + "\">");
1815
- $story.append($item);
1816
- return plugin["do"]($item, item, done);
1817
- } else {
1818
- $story.append($("<div><p class=\"error\">Can't make sense of story[" + i + "]</p></div>"));
1819
- return done();
1820
- }
1821
- });
1822
- pageObject.seqActions(function(each, done) {
1823
- if (each.separator) {
1824
- addToJournal($journal, each.separator);
1825
- }
1826
- addToJournal($journal, each.action);
1827
- return done();
1828
- });
1829
- emitTwins($page);
1830
- $journal.append("<div class=\"control-buttons\">\n <a href=\"#\" class=\"button fork-page\" title=\"fork this page\">" + util.symbols['fork'] + "</a>\n <a href=\"#\" class=\"button add-factory\" title=\"add paragraph\">" + util.symbols['add'] + "</a>\n</div>");
1831
- host = pageObject.getRemoteSite() || location.host;
1832
- return $footer.append("<a id=\"license\" href=\"http://creativecommons.org/licenses/by-sa/3.0/\">CC BY-SA 3.0</a> .\n<a class=\"show-page-source\" href=\"/" + slug + ".json?random=" + (util.randomBytes(4)) + "\" title=\"source\">JSON</a> .\n<a href= \"//" + host + "/" + slug + ".html\">" + host + "</a>");
1833
- };
1834
-
1835
- wiki.buildPage = function(pageObject, $page) {
1836
- if (pageObject.isLocal()) {
1837
- $page.addClass('local');
1838
- }
1839
- if (pageObject.isRemote()) {
1840
- $page.addClass('remote');
1841
- }
1842
- if (pageObject.isPlugin()) {
1843
- $page.addClass('plugin');
1844
- }
1845
- renderPageIntoPageElement(pageObject, $page);
1846
- state.setUrl();
1847
- initDragging($page);
1848
- initAddButton($page);
1849
- return $page;
1850
- };
1851
-
1852
- module.exports = refresh = wiki.refresh = function() {
1853
- var $page, createGhostPage, emptyPage, pageInformation, rev, slug, whenGotten, _ref;
1854
- $page = $(this);
1855
- _ref = $page.attr('id').split('_rev'), slug = _ref[0], rev = _ref[1];
1856
- pageInformation = {
1857
- slug: slug,
1858
- rev: rev,
1859
- site: $page.data('site')
1860
- };
1861
- emptyPage = require('./page').emptyPage;
1862
- createGhostPage = function() {
1863
- var hit, hits, info, pageObject, result, site, title, _i, _len, _ref1;
1864
- title = $("a[href=\"/" + slug + ".html\"]:last").text() || slug;
1865
- pageObject = emptyPage();
1866
- pageObject.setTitle(title);
1867
- hits = [];
1868
- _ref1 = wiki.neighborhood;
1869
- for (site in _ref1) {
1870
- info = _ref1[site];
1871
- if (info.sitemap != null) {
1872
- result = _.find(info.sitemap, function(each) {
1873
- return each.slug === slug;
1874
- });
1875
- if (result != null) {
1876
- hits.push({
1877
- "type": "reference",
1878
- "site": site,
1879
- "slug": slug,
1880
- "title": result.title || slug,
1881
- "text": result.synopsis || ''
1882
- });
1883
- }
1884
- }
1885
- }
1886
- if (hits.length > 0) {
1887
- pageObject.addItem({
1888
- 'type': 'future',
1889
- 'text': 'We could not find this page in the expected context.',
1890
- 'title': title
1891
- });
1892
- pageObject.addItem({
1893
- 'type': 'paragraph',
1894
- 'text': "We did find the page in your current neighborhood."
1895
- });
1896
- for (_i = 0, _len = hits.length; _i < _len; _i++) {
1897
- hit = hits[_i];
1898
- pageObject.addItem(hit);
1899
- }
1900
- } else {
1901
- pageObject.addItem({
1902
- 'type': 'future',
1903
- 'text': 'We could not find this page.',
1904
- 'title': title
1905
- });
1906
- }
1907
- return wiki.buildPage(pageObject, $page).addClass('ghost');
1908
- };
1909
- whenGotten = function(pageObject) {
1910
- var site, _i, _len, _ref1, _results;
1911
- wiki.buildPage(pageObject, $page);
1912
- _ref1 = pageObject.getNeighbors(location.host);
1913
- _results = [];
1914
- for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
1915
- site = _ref1[_i];
1916
- _results.push(neighborhood.registerNeighbor(site));
1917
- }
1918
- return _results;
1919
- };
1920
- return pageHandler.get({
1921
- whenGotten: whenGotten,
1922
- whenNotGotten: createGhostPage,
1923
- pageInformation: pageInformation
1924
- });
1925
- };
1926
-
1927
- }).call(this);
1928
-
1929
- },{"./addToJournal":16,"./neighborhood":18,"./page":12,"./pageHandler":7,"./plugin":8,"./state":9,"./util":6,"./wiki":2,"underscore":17}],12:[function(require,module,exports){
1930
- (function() {
1931
- var asSlug, emptyPage, newPage, nowSections, util, _;
1932
-
1933
- util = require('./util');
1934
-
1935
- _ = require('underscore');
1936
-
1937
- asSlug = function(name) {
1938
- return name.replace(/\s/g, '-').replace(/[^A-Za-z0-9-]/g, '').toLowerCase();
1939
- };
1940
-
1941
- emptyPage = function() {
1942
- return newPage({}, null);
1943
- };
1944
-
1945
- nowSections = function(now) {
1946
- return [
1947
- {
1948
- symbol: '❄',
1949
- date: now - 1000 * 60 * 60 * 24 * 366,
1950
- period: 'a Year'
1951
- }, {
1952
- symbol: '⚘',
1953
- date: now - 1000 * 60 * 60 * 24 * 31 * 3,
1954
- period: 'a Season'
1955
- }, {
1956
- symbol: '⚪',
1957
- date: now - 1000 * 60 * 60 * 24 * 31,
1958
- period: 'a Month'
1959
- }, {
1960
- symbol: '☽',
1961
- date: now - 1000 * 60 * 60 * 24 * 7,
1962
- period: 'a Week'
1963
- }, {
1964
- symbol: '☀',
1965
- date: now - 1000 * 60 * 60 * 24,
1966
- period: 'a Day'
1967
- }, {
1968
- symbol: '⌚',
1969
- date: now - 1000 * 60 * 60,
1970
- period: 'an Hour'
1971
- }
1972
- ];
1973
- };
1974
-
1975
- newPage = function(json, site) {
1976
- var addItem, addParagraph, getContext, getNeighbors, getRawPage, getRemoteSite, getSlug, isLocal, isPlugin, isRemote, page, seqActions, seqItems, setTitle;
1977
- page = _.extend({}, util.emptyPage(), json);
1978
- page.story || (page.story = []);
1979
- page.journal || (page.journal = []);
1980
- getRawPage = function() {
1981
- return page;
1982
- };
1983
- getContext = function() {
1984
- var action, addContext, context, _i, _len, _ref;
1985
- context = ['view'];
1986
- if (isRemote()) {
1987
- context.push(site);
1988
- }
1989
- addContext = function(site) {
1990
- if ((site != null) && !_.include(context, site)) {
1991
- return context.push(site);
1992
- }
1993
- };
1994
- _ref = page.journal.slice(0).reverse();
1995
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
1996
- action = _ref[_i];
1997
- addContext(action.site);
1998
- }
1999
- return context;
2000
- };
2001
- isPlugin = function() {
2002
- return page.plugin != null;
2003
- };
2004
- isRemote = function() {
2005
- return !(site === (void 0) || site === null || site === 'view' || site === 'origin' || site === 'local');
2006
- };
2007
- isLocal = function() {
2008
- return site === 'local';
2009
- };
2010
- getRemoteSite = function() {
2011
- if (isRemote()) {
2012
- return site;
2013
- }
2014
- return null;
2015
- };
2016
- getSlug = function() {
2017
- return asSlug(page.title);
2018
- };
2019
- getNeighbors = function(host) {
2020
- var action, item, neighbors, _i, _j, _len, _len1, _ref, _ref1;
2021
- neighbors = [];
2022
- if (isRemote()) {
2023
- neighbors.push(site);
2024
- } else {
2025
- if (host != null) {
2026
- neighbors.push(host);
2027
- }
2028
- }
2029
- _ref = page.story;
2030
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
2031
- item = _ref[_i];
2032
- if (item.site != null) {
2033
- neighbors.push(item.site);
2034
- }
2035
- }
2036
- _ref1 = page.journal;
2037
- for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
2038
- action = _ref1[_j];
2039
- if (action.site != null) {
2040
- neighbors.push(action.site);
2041
- }
2042
- }
2043
- return _.uniq(neighbors);
2044
- };
2045
- setTitle = function(title) {
2046
- return page.title = title;
2047
- };
2048
- addItem = function(item) {
2049
- item = _.extend({}, {
2050
- id: util.randomBytes(8)
2051
- }, item);
2052
- return page.story.push(item);
2053
- };
2054
- seqItems = function(each) {
2055
- var emitItem;
2056
- emitItem = function(i) {
2057
- if (i >= page.story.length) {
2058
- return;
2059
- }
2060
- return each(page.story[i], function() {
2061
- return emitItem(i + 1);
2062
- });
2063
- };
2064
- return emitItem(0);
2065
- };
2066
- addParagraph = function(text) {
2067
- var type;
2068
- type = "paragraph";
2069
- return addItem({
2070
- type: type,
2071
- text: text
2072
- });
2073
- };
2074
- seqActions = function(each) {
2075
- var emitAction, sections, smaller;
2076
- smaller = 0;
2077
- sections = nowSections((new Date).getTime());
2078
- emitAction = function(i) {
2079
- var action, bigger, section, separator, _i, _len;
2080
- if (i >= page.journal.length) {
2081
- return;
2082
- }
2083
- action = page.journal[i];
2084
- bigger = action.date || 0;
2085
- separator = null;
2086
- for (_i = 0, _len = sections.length; _i < _len; _i++) {
2087
- section = sections[_i];
2088
- if (section.date > smaller && section.date < bigger) {
2089
- separator = section;
2090
- }
2091
- }
2092
- smaller = bigger;
2093
- return each({
2094
- action: action,
2095
- separator: separator
2096
- }, function() {
2097
- return emitAction(i + 1);
2098
- });
2099
- };
2100
- return emitAction(0);
2101
- };
2102
- return {
2103
- getRawPage: getRawPage,
2104
- getContext: getContext,
2105
- isPlugin: isPlugin,
2106
- isRemote: isRemote,
2107
- isLocal: isLocal,
2108
- getRemoteSite: getRemoteSite,
2109
- getSlug: getSlug,
2110
- getNeighbors: getNeighbors,
2111
- setTitle: setTitle,
2112
- addItem: addItem,
2113
- addParagraph: addParagraph,
2114
- seqItems: seqItems,
2115
- seqActions: seqActions
2116
- };
2117
- };
2118
-
2119
- module.exports = {
2120
- newPage: newPage,
2121
- emptyPage: emptyPage
2122
- };
2123
-
2124
- }).call(this);
2125
-
2126
- },{"./util":6,"underscore":17}],17:[function(require,module,exports){
2127
- (function(){// Underscore.js 1.5.2
2128
- // http://underscorejs.org
2129
- // (c) 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
2130
- // Underscore may be freely distributed under the MIT license.
2131
-
2132
- (function() {
2133
-
2134
- // Baseline setup
2135
- // --------------
2136
-
2137
- // Establish the root object, `window` in the browser, or `exports` on the server.
2138
- var root = this;
2139
-
2140
- // Save the previous value of the `_` variable.
2141
- var previousUnderscore = root._;
2142
-
2143
- // Establish the object that gets returned to break out of a loop iteration.
2144
- var breaker = {};
2145
-
2146
- // Save bytes in the minified (but not gzipped) version:
2147
- var ArrayProto = Array.prototype, ObjProto = Object.prototype, FuncProto = Function.prototype;
2148
-
2149
- // Create quick reference variables for speed access to core prototypes.
2150
- var
2151
- push = ArrayProto.push,
2152
- slice = ArrayProto.slice,
2153
- concat = ArrayProto.concat,
2154
- toString = ObjProto.toString,
2155
- hasOwnProperty = ObjProto.hasOwnProperty;
2156
-
2157
- // All **ECMAScript 5** native function implementations that we hope to use
2158
- // are declared here.
2159
- var
2160
- nativeForEach = ArrayProto.forEach,
2161
- nativeMap = ArrayProto.map,
2162
- nativeReduce = ArrayProto.reduce,
2163
- nativeReduceRight = ArrayProto.reduceRight,
2164
- nativeFilter = ArrayProto.filter,
2165
- nativeEvery = ArrayProto.every,
2166
- nativeSome = ArrayProto.some,
2167
- nativeIndexOf = ArrayProto.indexOf,
2168
- nativeLastIndexOf = ArrayProto.lastIndexOf,
2169
- nativeIsArray = Array.isArray,
2170
- nativeKeys = Object.keys,
2171
- nativeBind = FuncProto.bind;
2172
-
2173
- // Create a safe reference to the Underscore object for use below.
2174
- var _ = function(obj) {
2175
- if (obj instanceof _) return obj;
2176
- if (!(this instanceof _)) return new _(obj);
2177
- this._wrapped = obj;
2178
- };
2179
-
2180
- // Export the Underscore object for **Node.js**, with
2181
- // backwards-compatibility for the old `require()` API. If we're in
2182
- // the browser, add `_` as a global object via a string identifier,
2183
- // for Closure Compiler "advanced" mode.
2184
- if (typeof exports !== 'undefined') {
2185
- if (typeof module !== 'undefined' && module.exports) {
2186
- exports = module.exports = _;
2187
- }
2188
- exports._ = _;
2189
- } else {
2190
- root._ = _;
2191
- }
2192
-
2193
- // Current version.
2194
- _.VERSION = '1.5.2';
2195
-
2196
- // Collection Functions
2197
- // --------------------
2198
-
2199
- // The cornerstone, an `each` implementation, aka `forEach`.
2200
- // Handles objects with the built-in `forEach`, arrays, and raw objects.
2201
- // Delegates to **ECMAScript 5**'s native `forEach` if available.
2202
- var each = _.each = _.forEach = function(obj, iterator, context) {
2203
- if (obj == null) return;
2204
- if (nativeForEach && obj.forEach === nativeForEach) {
2205
- obj.forEach(iterator, context);
2206
- } else if (obj.length === +obj.length) {
2207
- for (var i = 0, length = obj.length; i < length; i++) {
2208
- if (iterator.call(context, obj[i], i, obj) === breaker) return;
2209
- }
2210
- } else {
2211
- var keys = _.keys(obj);
2212
- for (var i = 0, length = keys.length; i < length; i++) {
2213
- if (iterator.call(context, obj[keys[i]], keys[i], obj) === breaker) return;
2214
- }
2215
- }
2216
- };
2217
-
2218
- // Return the results of applying the iterator to each element.
2219
- // Delegates to **ECMAScript 5**'s native `map` if available.
2220
- _.map = _.collect = function(obj, iterator, context) {
2221
- var results = [];
2222
- if (obj == null) return results;
2223
- if (nativeMap && obj.map === nativeMap) return obj.map(iterator, context);
2224
- each(obj, function(value, index, list) {
2225
- results.push(iterator.call(context, value, index, list));
2226
- });
2227
- return results;
2228
- };
2229
-
2230
- var reduceError = 'Reduce of empty array with no initial value';
2231
-
2232
- // **Reduce** builds up a single result from a list of values, aka `inject`,
2233
- // or `foldl`. Delegates to **ECMAScript 5**'s native `reduce` if available.
2234
- _.reduce = _.foldl = _.inject = function(obj, iterator, memo, context) {
2235
- var initial = arguments.length > 2;
2236
- if (obj == null) obj = [];
2237
- if (nativeReduce && obj.reduce === nativeReduce) {
2238
- if (context) iterator = _.bind(iterator, context);
2239
- return initial ? obj.reduce(iterator, memo) : obj.reduce(iterator);
2240
- }
2241
- each(obj, function(value, index, list) {
2242
- if (!initial) {
2243
- memo = value;
2244
- initial = true;
2245
- } else {
2246
- memo = iterator.call(context, memo, value, index, list);
2247
- }
2248
- });
2249
- if (!initial) throw new TypeError(reduceError);
2250
- return memo;
2251
- };
2252
-
2253
- // The right-associative version of reduce, also known as `foldr`.
2254
- // Delegates to **ECMAScript 5**'s native `reduceRight` if available.
2255
- _.reduceRight = _.foldr = function(obj, iterator, memo, context) {
2256
- var initial = arguments.length > 2;
2257
- if (obj == null) obj = [];
2258
- if (nativeReduceRight && obj.reduceRight === nativeReduceRight) {
2259
- if (context) iterator = _.bind(iterator, context);
2260
- return initial ? obj.reduceRight(iterator, memo) : obj.reduceRight(iterator);
2261
- }
2262
- var length = obj.length;
2263
- if (length !== +length) {
2264
- var keys = _.keys(obj);
2265
- length = keys.length;
2266
- }
2267
- each(obj, function(value, index, list) {
2268
- index = keys ? keys[--length] : --length;
2269
- if (!initial) {
2270
- memo = obj[index];
2271
- initial = true;
2272
- } else {
2273
- memo = iterator.call(context, memo, obj[index], index, list);
2274
- }
2275
- });
2276
- if (!initial) throw new TypeError(reduceError);
2277
- return memo;
2278
- };
2279
-
2280
- // Return the first value which passes a truth test. Aliased as `detect`.
2281
- _.find = _.detect = function(obj, iterator, context) {
2282
- var result;
2283
- any(obj, function(value, index, list) {
2284
- if (iterator.call(context, value, index, list)) {
2285
- result = value;
2286
- return true;
2287
- }
2288
- });
2289
- return result;
2290
- };
2291
-
2292
- // Return all the elements that pass a truth test.
2293
- // Delegates to **ECMAScript 5**'s native `filter` if available.
2294
- // Aliased as `select`.
2295
- _.filter = _.select = function(obj, iterator, context) {
2296
- var results = [];
2297
- if (obj == null) return results;
2298
- if (nativeFilter && obj.filter === nativeFilter) return obj.filter(iterator, context);
2299
- each(obj, function(value, index, list) {
2300
- if (iterator.call(context, value, index, list)) results.push(value);
2301
- });
2302
- return results;
2303
- };
2304
-
2305
- // Return all the elements for which a truth test fails.
2306
- _.reject = function(obj, iterator, context) {
2307
- return _.filter(obj, function(value, index, list) {
2308
- return !iterator.call(context, value, index, list);
2309
- }, context);
2310
- };
2311
-
2312
- // Determine whether all of the elements match a truth test.
2313
- // Delegates to **ECMAScript 5**'s native `every` if available.
2314
- // Aliased as `all`.
2315
- _.every = _.all = function(obj, iterator, context) {
2316
- iterator || (iterator = _.identity);
2317
- var result = true;
2318
- if (obj == null) return result;
2319
- if (nativeEvery && obj.every === nativeEvery) return obj.every(iterator, context);
2320
- each(obj, function(value, index, list) {
2321
- if (!(result = result && iterator.call(context, value, index, list))) return breaker;
2322
- });
2323
- return !!result;
2324
- };
2325
-
2326
- // Determine if at least one element in the object matches a truth test.
2327
- // Delegates to **ECMAScript 5**'s native `some` if available.
2328
- // Aliased as `any`.
2329
- var any = _.some = _.any = function(obj, iterator, context) {
2330
- iterator || (iterator = _.identity);
2331
- var result = false;
2332
- if (obj == null) return result;
2333
- if (nativeSome && obj.some === nativeSome) return obj.some(iterator, context);
2334
- each(obj, function(value, index, list) {
2335
- if (result || (result = iterator.call(context, value, index, list))) return breaker;
2336
- });
2337
- return !!result;
2338
- };
2339
-
2340
- // Determine if the array or object contains a given value (using `===`).
2341
- // Aliased as `include`.
2342
- _.contains = _.include = function(obj, target) {
2343
- if (obj == null) return false;
2344
- if (nativeIndexOf && obj.indexOf === nativeIndexOf) return obj.indexOf(target) != -1;
2345
- return any(obj, function(value) {
2346
- return value === target;
2347
- });
2348
- };
2349
-
2350
- // Invoke a method (with arguments) on every item in a collection.
2351
- _.invoke = function(obj, method) {
2352
- var args = slice.call(arguments, 2);
2353
- var isFunc = _.isFunction(method);
2354
- return _.map(obj, function(value) {
2355
- return (isFunc ? method : value[method]).apply(value, args);
2356
- });
2357
- };
2358
-
2359
- // Convenience version of a common use case of `map`: fetching a property.
2360
- _.pluck = function(obj, key) {
2361
- return _.map(obj, function(value){ return value[key]; });
2362
- };
2363
-
2364
- // Convenience version of a common use case of `filter`: selecting only objects
2365
- // containing specific `key:value` pairs.
2366
- _.where = function(obj, attrs, first) {
2367
- if (_.isEmpty(attrs)) return first ? void 0 : [];
2368
- return _[first ? 'find' : 'filter'](obj, function(value) {
2369
- for (var key in attrs) {
2370
- if (attrs[key] !== value[key]) return false;
2371
- }
2372
- return true;
2373
- });
2374
- };
2375
-
2376
- // Convenience version of a common use case of `find`: getting the first object
2377
- // containing specific `key:value` pairs.
2378
- _.findWhere = function(obj, attrs) {
2379
- return _.where(obj, attrs, true);
2380
- };
2381
-
2382
- // Return the maximum element or (element-based computation).
2383
- // Can't optimize arrays of integers longer than 65,535 elements.
2384
- // See [WebKit Bug 80797](https://bugs.webkit.org/show_bug.cgi?id=80797)
2385
- _.max = function(obj, iterator, context) {
2386
- if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {
2387
- return Math.max.apply(Math, obj);
2388
- }
2389
- if (!iterator && _.isEmpty(obj)) return -Infinity;
2390
- var result = {computed : -Infinity, value: -Infinity};
2391
- each(obj, function(value, index, list) {
2392
- var computed = iterator ? iterator.call(context, value, index, list) : value;
2393
- computed > result.computed && (result = {value : value, computed : computed});
2394
- });
2395
- return result.value;
2396
- };
2397
-
2398
- // Return the minimum element (or element-based computation).
2399
- _.min = function(obj, iterator, context) {
2400
- if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {
2401
- return Math.min.apply(Math, obj);
2402
- }
2403
- if (!iterator && _.isEmpty(obj)) return Infinity;
2404
- var result = {computed : Infinity, value: Infinity};
2405
- each(obj, function(value, index, list) {
2406
- var computed = iterator ? iterator.call(context, value, index, list) : value;
2407
- computed < result.computed && (result = {value : value, computed : computed});
2408
- });
2409
- return result.value;
2410
- };
2411
-
2412
- // Shuffle an array, using the modern version of the
2413
- // [Fisher-Yates shuffle](http://en.wikipedia.org/wiki/Fisher–Yates_shuffle).
2414
- _.shuffle = function(obj) {
2415
- var rand;
2416
- var index = 0;
2417
- var shuffled = [];
2418
- each(obj, function(value) {
2419
- rand = _.random(index++);
2420
- shuffled[index - 1] = shuffled[rand];
2421
- shuffled[rand] = value;
2422
- });
2423
- return shuffled;
2424
- };
2425
-
2426
- // Sample **n** random values from an array.
2427
- // If **n** is not specified, returns a single random element from the array.
2428
- // The internal `guard` argument allows it to work with `map`.
2429
- _.sample = function(obj, n, guard) {
2430
- if (arguments.length < 2 || guard) {
2431
- return obj[_.random(obj.length - 1)];
2432
- }
2433
- return _.shuffle(obj).slice(0, Math.max(0, n));
2434
- };
2435
-
2436
- // An internal function to generate lookup iterators.
2437
- var lookupIterator = function(value) {
2438
- return _.isFunction(value) ? value : function(obj){ return obj[value]; };
2439
- };
2440
-
2441
- // Sort the object's values by a criterion produced by an iterator.
2442
- _.sortBy = function(obj, value, context) {
2443
- var iterator = lookupIterator(value);
2444
- return _.pluck(_.map(obj, function(value, index, list) {
2445
- return {
2446
- value: value,
2447
- index: index,
2448
- criteria: iterator.call(context, value, index, list)
2449
- };
2450
- }).sort(function(left, right) {
2451
- var a = left.criteria;
2452
- var b = right.criteria;
2453
- if (a !== b) {
2454
- if (a > b || a === void 0) return 1;
2455
- if (a < b || b === void 0) return -1;
2456
- }
2457
- return left.index - right.index;
2458
- }), 'value');
2459
- };
2460
-
2461
- // An internal function used for aggregate "group by" operations.
2462
- var group = function(behavior) {
2463
- return function(obj, value, context) {
2464
- var result = {};
2465
- var iterator = value == null ? _.identity : lookupIterator(value);
2466
- each(obj, function(value, index) {
2467
- var key = iterator.call(context, value, index, obj);
2468
- behavior(result, key, value);
2469
- });
2470
- return result;
2471
- };
2472
- };
2473
-
2474
- // Groups the object's values by a criterion. Pass either a string attribute
2475
- // to group by, or a function that returns the criterion.
2476
- _.groupBy = group(function(result, key, value) {
2477
- (_.has(result, key) ? result[key] : (result[key] = [])).push(value);
2478
- });
2479
-
2480
- // Indexes the object's values by a criterion, similar to `groupBy`, but for
2481
- // when you know that your index values will be unique.
2482
- _.indexBy = group(function(result, key, value) {
2483
- result[key] = value;
2484
- });
2485
-
2486
- // Counts instances of an object that group by a certain criterion. Pass
2487
- // either a string attribute to count by, or a function that returns the
2488
- // criterion.
2489
- _.countBy = group(function(result, key) {
2490
- _.has(result, key) ? result[key]++ : result[key] = 1;
2491
- });
2492
-
2493
- // Use a comparator function to figure out the smallest index at which
2494
- // an object should be inserted so as to maintain order. Uses binary search.
2495
- _.sortedIndex = function(array, obj, iterator, context) {
2496
- iterator = iterator == null ? _.identity : lookupIterator(iterator);
2497
- var value = iterator.call(context, obj);
2498
- var low = 0, high = array.length;
2499
- while (low < high) {
2500
- var mid = (low + high) >>> 1;
2501
- iterator.call(context, array[mid]) < value ? low = mid + 1 : high = mid;
2502
- }
2503
- return low;
2504
- };
2505
-
2506
- // Safely create a real, live array from anything iterable.
2507
- _.toArray = function(obj) {
2508
- if (!obj) return [];
2509
- if (_.isArray(obj)) return slice.call(obj);
2510
- if (obj.length === +obj.length) return _.map(obj, _.identity);
2511
- return _.values(obj);
2512
- };
2513
-
2514
- // Return the number of elements in an object.
2515
- _.size = function(obj) {
2516
- if (obj == null) return 0;
2517
- return (obj.length === +obj.length) ? obj.length : _.keys(obj).length;
2518
- };
2519
-
2520
- // Array Functions
2521
- // ---------------
2522
-
2523
- // Get the first element of an array. Passing **n** will return the first N
2524
- // values in the array. Aliased as `head` and `take`. The **guard** check
2525
- // allows it to work with `_.map`.
2526
- _.first = _.head = _.take = function(array, n, guard) {
2527
- if (array == null) return void 0;
2528
- return (n == null) || guard ? array[0] : slice.call(array, 0, n);
2529
- };
2530
-
2531
- // Returns everything but the last entry of the array. Especially useful on
2532
- // the arguments object. Passing **n** will return all the values in
2533
- // the array, excluding the last N. The **guard** check allows it to work with
2534
- // `_.map`.
2535
- _.initial = function(array, n, guard) {
2536
- return slice.call(array, 0, array.length - ((n == null) || guard ? 1 : n));
2537
- };
2538
-
2539
- // Get the last element of an array. Passing **n** will return the last N
2540
- // values in the array. The **guard** check allows it to work with `_.map`.
2541
- _.last = function(array, n, guard) {
2542
- if (array == null) return void 0;
2543
- if ((n == null) || guard) {
2544
- return array[array.length - 1];
2545
- } else {
2546
- return slice.call(array, Math.max(array.length - n, 0));
2547
- }
2548
- };
2549
-
2550
- // Returns everything but the first entry of the array. Aliased as `tail` and `drop`.
2551
- // Especially useful on the arguments object. Passing an **n** will return
2552
- // the rest N values in the array. The **guard**
2553
- // check allows it to work with `_.map`.
2554
- _.rest = _.tail = _.drop = function(array, n, guard) {
2555
- return slice.call(array, (n == null) || guard ? 1 : n);
2556
- };
2557
-
2558
- // Trim out all falsy values from an array.
2559
- _.compact = function(array) {
2560
- return _.filter(array, _.identity);
2561
- };
2562
-
2563
- // Internal implementation of a recursive `flatten` function.
2564
- var flatten = function(input, shallow, output) {
2565
- if (shallow && _.every(input, _.isArray)) {
2566
- return concat.apply(output, input);
2567
- }
2568
- each(input, function(value) {
2569
- if (_.isArray(value) || _.isArguments(value)) {
2570
- shallow ? push.apply(output, value) : flatten(value, shallow, output);
2571
- } else {
2572
- output.push(value);
2573
- }
2574
- });
2575
- return output;
2576
- };
2577
-
2578
- // Flatten out an array, either recursively (by default), or just one level.
2579
- _.flatten = function(array, shallow) {
2580
- return flatten(array, shallow, []);
2581
- };
2582
-
2583
- // Return a version of the array that does not contain the specified value(s).
2584
- _.without = function(array) {
2585
- return _.difference(array, slice.call(arguments, 1));
2586
- };
2587
-
2588
- // Produce a duplicate-free version of the array. If the array has already
2589
- // been sorted, you have the option of using a faster algorithm.
2590
- // Aliased as `unique`.
2591
- _.uniq = _.unique = function(array, isSorted, iterator, context) {
2592
- if (_.isFunction(isSorted)) {
2593
- context = iterator;
2594
- iterator = isSorted;
2595
- isSorted = false;
2596
- }
2597
- var initial = iterator ? _.map(array, iterator, context) : array;
2598
- var results = [];
2599
- var seen = [];
2600
- each(initial, function(value, index) {
2601
- if (isSorted ? (!index || seen[seen.length - 1] !== value) : !_.contains(seen, value)) {
2602
- seen.push(value);
2603
- results.push(array[index]);
2604
- }
2605
- });
2606
- return results;
2607
- };
2608
-
2609
- // Produce an array that contains the union: each distinct element from all of
2610
- // the passed-in arrays.
2611
- _.union = function() {
2612
- return _.uniq(_.flatten(arguments, true));
2613
- };
2614
-
2615
- // Produce an array that contains every item shared between all the
2616
- // passed-in arrays.
2617
- _.intersection = function(array) {
2618
- var rest = slice.call(arguments, 1);
2619
- return _.filter(_.uniq(array), function(item) {
2620
- return _.every(rest, function(other) {
2621
- return _.indexOf(other, item) >= 0;
2622
- });
2623
- });
2624
- };
2625
-
2626
- // Take the difference between one array and a number of other arrays.
2627
- // Only the elements present in just the first array will remain.
2628
- _.difference = function(array) {
2629
- var rest = concat.apply(ArrayProto, slice.call(arguments, 1));
2630
- return _.filter(array, function(value){ return !_.contains(rest, value); });
2631
- };
2632
-
2633
- // Zip together multiple lists into a single array -- elements that share
2634
- // an index go together.
2635
- _.zip = function() {
2636
- var length = _.max(_.pluck(arguments, "length").concat(0));
2637
- var results = new Array(length);
2638
- for (var i = 0; i < length; i++) {
2639
- results[i] = _.pluck(arguments, '' + i);
2640
- }
2641
- return results;
2642
- };
2643
-
2644
- // Converts lists into objects. Pass either a single array of `[key, value]`
2645
- // pairs, or two parallel arrays of the same length -- one of keys, and one of
2646
- // the corresponding values.
2647
- _.object = function(list, values) {
2648
- if (list == null) return {};
2649
- var result = {};
2650
- for (var i = 0, length = list.length; i < length; i++) {
2651
- if (values) {
2652
- result[list[i]] = values[i];
2653
- } else {
2654
- result[list[i][0]] = list[i][1];
2655
- }
2656
- }
2657
- return result;
2658
- };
2659
-
2660
- // If the browser doesn't supply us with indexOf (I'm looking at you, **MSIE**),
2661
- // we need this function. Return the position of the first occurrence of an
2662
- // item in an array, or -1 if the item is not included in the array.
2663
- // Delegates to **ECMAScript 5**'s native `indexOf` if available.
2664
- // If the array is large and already in sort order, pass `true`
2665
- // for **isSorted** to use binary search.
2666
- _.indexOf = function(array, item, isSorted) {
2667
- if (array == null) return -1;
2668
- var i = 0, length = array.length;
2669
- if (isSorted) {
2670
- if (typeof isSorted == 'number') {
2671
- i = (isSorted < 0 ? Math.max(0, length + isSorted) : isSorted);
2672
- } else {
2673
- i = _.sortedIndex(array, item);
2674
- return array[i] === item ? i : -1;
2675
- }
2676
- }
2677
- if (nativeIndexOf && array.indexOf === nativeIndexOf) return array.indexOf(item, isSorted);
2678
- for (; i < length; i++) if (array[i] === item) return i;
2679
- return -1;
2680
- };
2681
-
2682
- // Delegates to **ECMAScript 5**'s native `lastIndexOf` if available.
2683
- _.lastIndexOf = function(array, item, from) {
2684
- if (array == null) return -1;
2685
- var hasIndex = from != null;
2686
- if (nativeLastIndexOf && array.lastIndexOf === nativeLastIndexOf) {
2687
- return hasIndex ? array.lastIndexOf(item, from) : array.lastIndexOf(item);
2688
- }
2689
- var i = (hasIndex ? from : array.length);
2690
- while (i--) if (array[i] === item) return i;
2691
- return -1;
2692
- };
2693
-
2694
- // Generate an integer Array containing an arithmetic progression. A port of
2695
- // the native Python `range()` function. See
2696
- // [the Python documentation](http://docs.python.org/library/functions.html#range).
2697
- _.range = function(start, stop, step) {
2698
- if (arguments.length <= 1) {
2699
- stop = start || 0;
2700
- start = 0;
2701
- }
2702
- step = arguments[2] || 1;
2703
-
2704
- var length = Math.max(Math.ceil((stop - start) / step), 0);
2705
- var idx = 0;
2706
- var range = new Array(length);
2707
-
2708
- while(idx < length) {
2709
- range[idx++] = start;
2710
- start += step;
2711
- }
2712
-
2713
- return range;
2714
- };
2715
-
2716
- // Function (ahem) Functions
2717
- // ------------------
2718
-
2719
- // Reusable constructor function for prototype setting.
2720
- var ctor = function(){};
2721
-
2722
- // Create a function bound to a given object (assigning `this`, and arguments,
2723
- // optionally). Delegates to **ECMAScript 5**'s native `Function.bind` if
2724
- // available.
2725
- _.bind = function(func, context) {
2726
- var args, bound;
2727
- if (nativeBind && func.bind === nativeBind) return nativeBind.apply(func, slice.call(arguments, 1));
2728
- if (!_.isFunction(func)) throw new TypeError;
2729
- args = slice.call(arguments, 2);
2730
- return bound = function() {
2731
- if (!(this instanceof bound)) return func.apply(context, args.concat(slice.call(arguments)));
2732
- ctor.prototype = func.prototype;
2733
- var self = new ctor;
2734
- ctor.prototype = null;
2735
- var result = func.apply(self, args.concat(slice.call(arguments)));
2736
- if (Object(result) === result) return result;
2737
- return self;
2738
- };
2739
- };
2740
-
2741
- // Partially apply a function by creating a version that has had some of its
2742
- // arguments pre-filled, without changing its dynamic `this` context.
2743
- _.partial = function(func) {
2744
- var args = slice.call(arguments, 1);
2745
- return function() {
2746
- return func.apply(this, args.concat(slice.call(arguments)));
2747
- };
2748
- };
2749
-
2750
- // Bind all of an object's methods to that object. Useful for ensuring that
2751
- // all callbacks defined on an object belong to it.
2752
- _.bindAll = function(obj) {
2753
- var funcs = slice.call(arguments, 1);
2754
- if (funcs.length === 0) throw new Error("bindAll must be passed function names");
2755
- each(funcs, function(f) { obj[f] = _.bind(obj[f], obj); });
2756
- return obj;
2757
- };
2758
-
2759
- // Memoize an expensive function by storing its results.
2760
- _.memoize = function(func, hasher) {
2761
- var memo = {};
2762
- hasher || (hasher = _.identity);
2763
- return function() {
2764
- var key = hasher.apply(this, arguments);
2765
- return _.has(memo, key) ? memo[key] : (memo[key] = func.apply(this, arguments));
2766
- };
2767
- };
2768
-
2769
- // Delays a function for the given number of milliseconds, and then calls
2770
- // it with the arguments supplied.
2771
- _.delay = function(func, wait) {
2772
- var args = slice.call(arguments, 2);
2773
- return setTimeout(function(){ return func.apply(null, args); }, wait);
2774
- };
2775
-
2776
- // Defers a function, scheduling it to run after the current call stack has
2777
- // cleared.
2778
- _.defer = function(func) {
2779
- return _.delay.apply(_, [func, 1].concat(slice.call(arguments, 1)));
2780
- };
2781
-
2782
- // Returns a function, that, when invoked, will only be triggered at most once
2783
- // during a given window of time. Normally, the throttled function will run
2784
- // as much as it can, without ever going more than once per `wait` duration;
2785
- // but if you'd like to disable the execution on the leading edge, pass
2786
- // `{leading: false}`. To disable execution on the trailing edge, ditto.
2787
- _.throttle = function(func, wait, options) {
2788
- var context, args, result;
2789
- var timeout = null;
2790
- var previous = 0;
2791
- options || (options = {});
2792
- var later = function() {
2793
- previous = options.leading === false ? 0 : new Date;
2794
- timeout = null;
2795
- result = func.apply(context, args);
2796
- };
2797
- return function() {
2798
- var now = new Date;
2799
- if (!previous && options.leading === false) previous = now;
2800
- var remaining = wait - (now - previous);
2801
- context = this;
2802
- args = arguments;
2803
- if (remaining <= 0) {
2804
- clearTimeout(timeout);
2805
- timeout = null;
2806
- previous = now;
2807
- result = func.apply(context, args);
2808
- } else if (!timeout && options.trailing !== false) {
2809
- timeout = setTimeout(later, remaining);
2810
- }
2811
- return result;
2812
- };
2813
- };
2814
-
2815
- // Returns a function, that, as long as it continues to be invoked, will not
2816
- // be triggered. The function will be called after it stops being called for
2817
- // N milliseconds. If `immediate` is passed, trigger the function on the
2818
- // leading edge, instead of the trailing.
2819
- _.debounce = function(func, wait, immediate) {
2820
- var timeout, args, context, timestamp, result;
2821
- return function() {
2822
- context = this;
2823
- args = arguments;
2824
- timestamp = new Date();
2825
- var later = function() {
2826
- var last = (new Date()) - timestamp;
2827
- if (last < wait) {
2828
- timeout = setTimeout(later, wait - last);
2829
- } else {
2830
- timeout = null;
2831
- if (!immediate) result = func.apply(context, args);
2832
- }
2833
- };
2834
- var callNow = immediate && !timeout;
2835
- if (!timeout) {
2836
- timeout = setTimeout(later, wait);
2837
- }
2838
- if (callNow) result = func.apply(context, args);
2839
- return result;
2840
- };
2841
- };
2842
-
2843
- // Returns a function that will be executed at most one time, no matter how
2844
- // often you call it. Useful for lazy initialization.
2845
- _.once = function(func) {
2846
- var ran = false, memo;
2847
- return function() {
2848
- if (ran) return memo;
2849
- ran = true;
2850
- memo = func.apply(this, arguments);
2851
- func = null;
2852
- return memo;
2853
- };
2854
- };
2855
-
2856
- // Returns the first function passed as an argument to the second,
2857
- // allowing you to adjust arguments, run code before and after, and
2858
- // conditionally execute the original function.
2859
- _.wrap = function(func, wrapper) {
2860
- return function() {
2861
- var args = [func];
2862
- push.apply(args, arguments);
2863
- return wrapper.apply(this, args);
2864
- };
2865
- };
2866
-
2867
- // Returns a function that is the composition of a list of functions, each
2868
- // consuming the return value of the function that follows.
2869
- _.compose = function() {
2870
- var funcs = arguments;
2871
- return function() {
2872
- var args = arguments;
2873
- for (var i = funcs.length - 1; i >= 0; i--) {
2874
- args = [funcs[i].apply(this, args)];
2875
- }
2876
- return args[0];
2877
- };
2878
- };
2879
-
2880
- // Returns a function that will only be executed after being called N times.
2881
- _.after = function(times, func) {
2882
- return function() {
2883
- if (--times < 1) {
2884
- return func.apply(this, arguments);
2885
- }
2886
- };
2887
- };
2888
-
2889
- // Object Functions
2890
- // ----------------
2891
-
2892
- // Retrieve the names of an object's properties.
2893
- // Delegates to **ECMAScript 5**'s native `Object.keys`
2894
- _.keys = nativeKeys || function(obj) {
2895
- if (obj !== Object(obj)) throw new TypeError('Invalid object');
2896
- var keys = [];
2897
- for (var key in obj) if (_.has(obj, key)) keys.push(key);
2898
- return keys;
2899
- };
2900
-
2901
- // Retrieve the values of an object's properties.
2902
- _.values = function(obj) {
2903
- var keys = _.keys(obj);
2904
- var length = keys.length;
2905
- var values = new Array(length);
2906
- for (var i = 0; i < length; i++) {
2907
- values[i] = obj[keys[i]];
2908
- }
2909
- return values;
2910
- };
2911
-
2912
- // Convert an object into a list of `[key, value]` pairs.
2913
- _.pairs = function(obj) {
2914
- var keys = _.keys(obj);
2915
- var length = keys.length;
2916
- var pairs = new Array(length);
2917
- for (var i = 0; i < length; i++) {
2918
- pairs[i] = [keys[i], obj[keys[i]]];
2919
- }
2920
- return pairs;
2921
- };
2922
-
2923
- // Invert the keys and values of an object. The values must be serializable.
2924
- _.invert = function(obj) {
2925
- var result = {};
2926
- var keys = _.keys(obj);
2927
- for (var i = 0, length = keys.length; i < length; i++) {
2928
- result[obj[keys[i]]] = keys[i];
2929
- }
2930
- return result;
2931
- };
2932
-
2933
- // Return a sorted list of the function names available on the object.
2934
- // Aliased as `methods`
2935
- _.functions = _.methods = function(obj) {
2936
- var names = [];
2937
- for (var key in obj) {
2938
- if (_.isFunction(obj[key])) names.push(key);
2939
- }
2940
- return names.sort();
2941
- };
2942
-
2943
- // Extend a given object with all the properties in passed-in object(s).
2944
- _.extend = function(obj) {
2945
- each(slice.call(arguments, 1), function(source) {
2946
- if (source) {
2947
- for (var prop in source) {
2948
- obj[prop] = source[prop];
2949
- }
2950
- }
2951
- });
2952
- return obj;
2953
- };
2954
-
2955
- // Return a copy of the object only containing the whitelisted properties.
2956
- _.pick = function(obj) {
2957
- var copy = {};
2958
- var keys = concat.apply(ArrayProto, slice.call(arguments, 1));
2959
- each(keys, function(key) {
2960
- if (key in obj) copy[key] = obj[key];
2961
- });
2962
- return copy;
2963
- };
2964
-
2965
- // Return a copy of the object without the blacklisted properties.
2966
- _.omit = function(obj) {
2967
- var copy = {};
2968
- var keys = concat.apply(ArrayProto, slice.call(arguments, 1));
2969
- for (var key in obj) {
2970
- if (!_.contains(keys, key)) copy[key] = obj[key];
2971
- }
2972
- return copy;
2973
- };
2974
-
2975
- // Fill in a given object with default properties.
2976
- _.defaults = function(obj) {
2977
- each(slice.call(arguments, 1), function(source) {
2978
- if (source) {
2979
- for (var prop in source) {
2980
- if (obj[prop] === void 0) obj[prop] = source[prop];
2981
- }
2982
- }
2983
- });
2984
- return obj;
2985
- };
2986
-
2987
- // Create a (shallow-cloned) duplicate of an object.
2988
- _.clone = function(obj) {
2989
- if (!_.isObject(obj)) return obj;
2990
- return _.isArray(obj) ? obj.slice() : _.extend({}, obj);
2991
- };
2992
-
2993
- // Invokes interceptor with the obj, and then returns obj.
2994
- // The primary purpose of this method is to "tap into" a method chain, in
2995
- // order to perform operations on intermediate results within the chain.
2996
- _.tap = function(obj, interceptor) {
2997
- interceptor(obj);
2998
- return obj;
2999
- };
3000
-
3001
- // Internal recursive comparison function for `isEqual`.
3002
- var eq = function(a, b, aStack, bStack) {
3003
- // Identical objects are equal. `0 === -0`, but they aren't identical.
3004
- // See the [Harmony `egal` proposal](http://wiki.ecmascript.org/doku.php?id=harmony:egal).
3005
- if (a === b) return a !== 0 || 1 / a == 1 / b;
3006
- // A strict comparison is necessary because `null == undefined`.
3007
- if (a == null || b == null) return a === b;
3008
- // Unwrap any wrapped objects.
3009
- if (a instanceof _) a = a._wrapped;
3010
- if (b instanceof _) b = b._wrapped;
3011
- // Compare `[[Class]]` names.
3012
- var className = toString.call(a);
3013
- if (className != toString.call(b)) return false;
3014
- switch (className) {
3015
- // Strings, numbers, dates, and booleans are compared by value.
3016
- case '[object String]':
3017
- // Primitives and their corresponding object wrappers are equivalent; thus, `"5"` is
3018
- // equivalent to `new String("5")`.
3019
- return a == String(b);
3020
- case '[object Number]':
3021
- // `NaN`s are equivalent, but non-reflexive. An `egal` comparison is performed for
3022
- // other numeric values.
3023
- return a != +a ? b != +b : (a == 0 ? 1 / a == 1 / b : a == +b);
3024
- case '[object Date]':
3025
- case '[object Boolean]':
3026
- // Coerce dates and booleans to numeric primitive values. Dates are compared by their
3027
- // millisecond representations. Note that invalid dates with millisecond representations
3028
- // of `NaN` are not equivalent.
3029
- return +a == +b;
3030
- // RegExps are compared by their source patterns and flags.
3031
- case '[object RegExp]':
3032
- return a.source == b.source &&
3033
- a.global == b.global &&
3034
- a.multiline == b.multiline &&
3035
- a.ignoreCase == b.ignoreCase;
3036
- }
3037
- if (typeof a != 'object' || typeof b != 'object') return false;
3038
- // Assume equality for cyclic structures. The algorithm for detecting cyclic
3039
- // structures is adapted from ES 5.1 section 15.12.3, abstract operation `JO`.
3040
- var length = aStack.length;
3041
- while (length--) {
3042
- // Linear search. Performance is inversely proportional to the number of
3043
- // unique nested structures.
3044
- if (aStack[length] == a) return bStack[length] == b;
3045
- }
3046
- // Objects with different constructors are not equivalent, but `Object`s
3047
- // from different frames are.
3048
- var aCtor = a.constructor, bCtor = b.constructor;
3049
- if (aCtor !== bCtor && !(_.isFunction(aCtor) && (aCtor instanceof aCtor) &&
3050
- _.isFunction(bCtor) && (bCtor instanceof bCtor))) {
3051
- return false;
3052
- }
3053
- // Add the first object to the stack of traversed objects.
3054
- aStack.push(a);
3055
- bStack.push(b);
3056
- var size = 0, result = true;
3057
- // Recursively compare objects and arrays.
3058
- if (className == '[object Array]') {
3059
- // Compare array lengths to determine if a deep comparison is necessary.
3060
- size = a.length;
3061
- result = size == b.length;
3062
- if (result) {
3063
- // Deep compare the contents, ignoring non-numeric properties.
3064
- while (size--) {
3065
- if (!(result = eq(a[size], b[size], aStack, bStack))) break;
3066
- }
3067
- }
3068
- } else {
3069
- // Deep compare objects.
3070
- for (var key in a) {
3071
- if (_.has(a, key)) {
3072
- // Count the expected number of properties.
3073
- size++;
3074
- // Deep compare each member.
3075
- if (!(result = _.has(b, key) && eq(a[key], b[key], aStack, bStack))) break;
3076
- }
3077
- }
3078
- // Ensure that both objects contain the same number of properties.
3079
- if (result) {
3080
- for (key in b) {
3081
- if (_.has(b, key) && !(size--)) break;
3082
- }
3083
- result = !size;
3084
- }
3085
- }
3086
- // Remove the first object from the stack of traversed objects.
3087
- aStack.pop();
3088
- bStack.pop();
3089
- return result;
3090
- };
3091
-
3092
- // Perform a deep comparison to check if two objects are equal.
3093
- _.isEqual = function(a, b) {
3094
- return eq(a, b, [], []);
3095
- };
3096
-
3097
- // Is a given array, string, or object empty?
3098
- // An "empty" object has no enumerable own-properties.
3099
- _.isEmpty = function(obj) {
3100
- if (obj == null) return true;
3101
- if (_.isArray(obj) || _.isString(obj)) return obj.length === 0;
3102
- for (var key in obj) if (_.has(obj, key)) return false;
3103
- return true;
3104
- };
3105
-
3106
- // Is a given value a DOM element?
3107
- _.isElement = function(obj) {
3108
- return !!(obj && obj.nodeType === 1);
3109
- };
3110
-
3111
- // Is a given value an array?
3112
- // Delegates to ECMA5's native Array.isArray
3113
- _.isArray = nativeIsArray || function(obj) {
3114
- return toString.call(obj) == '[object Array]';
3115
- };
3116
-
3117
- // Is a given variable an object?
3118
- _.isObject = function(obj) {
3119
- return obj === Object(obj);
3120
- };
3121
-
3122
- // Add some isType methods: isArguments, isFunction, isString, isNumber, isDate, isRegExp.
3123
- each(['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp'], function(name) {
3124
- _['is' + name] = function(obj) {
3125
- return toString.call(obj) == '[object ' + name + ']';
3126
- };
3127
- });
3128
-
3129
- // Define a fallback version of the method in browsers (ahem, IE), where
3130
- // there isn't any inspectable "Arguments" type.
3131
- if (!_.isArguments(arguments)) {
3132
- _.isArguments = function(obj) {
3133
- return !!(obj && _.has(obj, 'callee'));
3134
- };
3135
- }
3136
-
3137
- // Optimize `isFunction` if appropriate.
3138
- if (typeof (/./) !== 'function') {
3139
- _.isFunction = function(obj) {
3140
- return typeof obj === 'function';
3141
- };
3142
- }
3143
-
3144
- // Is a given object a finite number?
3145
- _.isFinite = function(obj) {
3146
- return isFinite(obj) && !isNaN(parseFloat(obj));
3147
- };
3148
-
3149
- // Is the given value `NaN`? (NaN is the only number which does not equal itself).
3150
- _.isNaN = function(obj) {
3151
- return _.isNumber(obj) && obj != +obj;
3152
- };
3153
-
3154
- // Is a given value a boolean?
3155
- _.isBoolean = function(obj) {
3156
- return obj === true || obj === false || toString.call(obj) == '[object Boolean]';
3157
- };
3158
-
3159
- // Is a given value equal to null?
3160
- _.isNull = function(obj) {
3161
- return obj === null;
3162
- };
3163
-
3164
- // Is a given variable undefined?
3165
- _.isUndefined = function(obj) {
3166
- return obj === void 0;
3167
- };
3168
-
3169
- // Shortcut function for checking if an object has a given property directly
3170
- // on itself (in other words, not on a prototype).
3171
- _.has = function(obj, key) {
3172
- return hasOwnProperty.call(obj, key);
3173
- };
3174
-
3175
- // Utility Functions
3176
- // -----------------
3177
-
3178
- // Run Underscore.js in *noConflict* mode, returning the `_` variable to its
3179
- // previous owner. Returns a reference to the Underscore object.
3180
- _.noConflict = function() {
3181
- root._ = previousUnderscore;
3182
- return this;
3183
- };
3184
-
3185
- // Keep the identity function around for default iterators.
3186
- _.identity = function(value) {
3187
- return value;
3188
- };
3189
-
3190
- // Run a function **n** times.
3191
- _.times = function(n, iterator, context) {
3192
- var accum = Array(Math.max(0, n));
3193
- for (var i = 0; i < n; i++) accum[i] = iterator.call(context, i);
3194
- return accum;
3195
- };
3196
-
3197
- // Return a random integer between min and max (inclusive).
3198
- _.random = function(min, max) {
3199
- if (max == null) {
3200
- max = min;
3201
- min = 0;
3202
- }
3203
- return min + Math.floor(Math.random() * (max - min + 1));
3204
- };
3205
-
3206
- // List of HTML entities for escaping.
3207
- var entityMap = {
3208
- escape: {
3209
- '&': '&amp;',
3210
- '<': '&lt;',
3211
- '>': '&gt;',
3212
- '"': '&quot;',
3213
- "'": '&#x27;'
3214
- }
3215
- };
3216
- entityMap.unescape = _.invert(entityMap.escape);
3217
-
3218
- // Regexes containing the keys and values listed immediately above.
3219
- var entityRegexes = {
3220
- escape: new RegExp('[' + _.keys(entityMap.escape).join('') + ']', 'g'),
3221
- unescape: new RegExp('(' + _.keys(entityMap.unescape).join('|') + ')', 'g')
3222
- };
3223
-
3224
- // Functions for escaping and unescaping strings to/from HTML interpolation.
3225
- _.each(['escape', 'unescape'], function(method) {
3226
- _[method] = function(string) {
3227
- if (string == null) return '';
3228
- return ('' + string).replace(entityRegexes[method], function(match) {
3229
- return entityMap[method][match];
3230
- });
3231
- };
3232
- });
3233
-
3234
- // If the value of the named `property` is a function then invoke it with the
3235
- // `object` as context; otherwise, return it.
3236
- _.result = function(object, property) {
3237
- if (object == null) return void 0;
3238
- var value = object[property];
3239
- return _.isFunction(value) ? value.call(object) : value;
3240
- };
3241
-
3242
- // Add your own custom functions to the Underscore object.
3243
- _.mixin = function(obj) {
3244
- each(_.functions(obj), function(name) {
3245
- var func = _[name] = obj[name];
3246
- _.prototype[name] = function() {
3247
- var args = [this._wrapped];
3248
- push.apply(args, arguments);
3249
- return result.call(this, func.apply(_, args));
3250
- };
3251
- });
3252
- };
3253
-
3254
- // Generate a unique integer id (unique within the entire client session).
3255
- // Useful for temporary DOM ids.
3256
- var idCounter = 0;
3257
- _.uniqueId = function(prefix) {
3258
- var id = ++idCounter + '';
3259
- return prefix ? prefix + id : id;
3260
- };
3261
-
3262
- // By default, Underscore uses ERB-style template delimiters, change the
3263
- // following template settings to use alternative delimiters.
3264
- _.templateSettings = {
3265
- evaluate : /<%([\s\S]+?)%>/g,
3266
- interpolate : /<%=([\s\S]+?)%>/g,
3267
- escape : /<%-([\s\S]+?)%>/g
3268
- };
3269
-
3270
- // When customizing `templateSettings`, if you don't want to define an
3271
- // interpolation, evaluation or escaping regex, we need one that is
3272
- // guaranteed not to match.
3273
- var noMatch = /(.)^/;
3274
-
3275
- // Certain characters need to be escaped so that they can be put into a
3276
- // string literal.
3277
- var escapes = {
3278
- "'": "'",
3279
- '\\': '\\',
3280
- '\r': 'r',
3281
- '\n': 'n',
3282
- '\t': 't',
3283
- '\u2028': 'u2028',
3284
- '\u2029': 'u2029'
3285
- };
3286
-
3287
- var escaper = /\\|'|\r|\n|\t|\u2028|\u2029/g;
3288
-
3289
- // JavaScript micro-templating, similar to John Resig's implementation.
3290
- // Underscore templating handles arbitrary delimiters, preserves whitespace,
3291
- // and correctly escapes quotes within interpolated code.
3292
- _.template = function(text, data, settings) {
3293
- var render;
3294
- settings = _.defaults({}, settings, _.templateSettings);
3295
-
3296
- // Combine delimiters into one regular expression via alternation.
3297
- var matcher = new RegExp([
3298
- (settings.escape || noMatch).source,
3299
- (settings.interpolate || noMatch).source,
3300
- (settings.evaluate || noMatch).source
3301
- ].join('|') + '|$', 'g');
3302
-
3303
- // Compile the template source, escaping string literals appropriately.
3304
- var index = 0;
3305
- var source = "__p+='";
3306
- text.replace(matcher, function(match, escape, interpolate, evaluate, offset) {
3307
- source += text.slice(index, offset)
3308
- .replace(escaper, function(match) { return '\\' + escapes[match]; });
3309
-
3310
- if (escape) {
3311
- source += "'+\n((__t=(" + escape + "))==null?'':_.escape(__t))+\n'";
3312
- }
3313
- if (interpolate) {
3314
- source += "'+\n((__t=(" + interpolate + "))==null?'':__t)+\n'";
3315
- }
3316
- if (evaluate) {
3317
- source += "';\n" + evaluate + "\n__p+='";
3318
- }
3319
- index = offset + match.length;
3320
- return match;
3321
- });
3322
- source += "';\n";
3323
-
3324
- // If a variable is not specified, place data values in local scope.
3325
- if (!settings.variable) source = 'with(obj||{}){\n' + source + '}\n';
3326
-
3327
- source = "var __t,__p='',__j=Array.prototype.join," +
3328
- "print=function(){__p+=__j.call(arguments,'');};\n" +
3329
- source + "return __p;\n";
3330
-
3331
- try {
3332
- render = new Function(settings.variable || 'obj', '_', source);
3333
- } catch (e) {
3334
- e.source = source;
3335
- throw e;
3336
- }
3337
-
3338
- if (data) return render(data, _);
3339
- var template = function(data) {
3340
- return render.call(this, data, _);
3341
- };
3342
-
3343
- // Provide the compiled function source as a convenience for precompilation.
3344
- template.source = 'function(' + (settings.variable || 'obj') + '){\n' + source + '}';
3345
-
3346
- return template;
3347
- };
3348
-
3349
- // Add a "chain" function, which will delegate to the wrapper.
3350
- _.chain = function(obj) {
3351
- return _(obj).chain();
3352
- };
3353
-
3354
- // OOP
3355
- // ---------------
3356
- // If Underscore is called as a function, it returns a wrapped object that
3357
- // can be used OO-style. This wrapper holds altered versions of all the
3358
- // underscore functions. Wrapped objects may be chained.
3359
-
3360
- // Helper function to continue chaining intermediate results.
3361
- var result = function(obj) {
3362
- return this._chain ? _(obj).chain() : obj;
3363
- };
3364
-
3365
- // Add all of the Underscore functions to the wrapper object.
3366
- _.mixin(_);
3367
-
3368
- // Add all mutator Array functions to the wrapper.
3369
- each(['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], function(name) {
3370
- var method = ArrayProto[name];
3371
- _.prototype[name] = function() {
3372
- var obj = this._wrapped;
3373
- method.apply(obj, arguments);
3374
- if ((name == 'shift' || name == 'splice') && obj.length === 0) delete obj[0];
3375
- return result.call(this, obj);
3376
- };
3377
- });
3378
-
3379
- // Add all accessor Array functions to the wrapper.
3380
- each(['concat', 'join', 'slice'], function(name) {
3381
- var method = ArrayProto[name];
3382
- _.prototype[name] = function() {
3383
- return result.call(this, method.apply(this._wrapped, arguments));
3384
- };
3385
- });
3386
-
3387
- _.extend(_.prototype, {
3388
-
3389
- // Start chaining a wrapped Underscore object.
3390
- chain: function() {
3391
- this._chain = true;
3392
- return this;
3393
- },
3394
-
3395
- // Extracts the result from a wrapped and chained object.
3396
- value: function() {
3397
- return this._wrapped;
3398
- }
3399
-
3400
- });
3401
-
3402
- }).call(this);
3403
-
3404
- })()
3405
- },{}],19:[function(require,module,exports){
3406
- (function() {
3407
- var active, createSearch, emptyPage, util, wiki;
3408
-
3409
- wiki = require('./wiki');
3410
-
3411
- util = require('./util');
3412
-
3413
- active = require('./active');
3414
-
3415
- emptyPage = require('./page').emptyPage;
3416
-
3417
- createSearch = function(_arg) {
3418
- var neighborhood, performSearch;
3419
- neighborhood = _arg.neighborhood;
3420
- performSearch = function(searchQuery) {
3421
- var $resultPage, result, resultPage, searchResults, tally, _i, _len, _ref;
3422
- searchResults = neighborhood.search(searchQuery);
3423
- tally = searchResults.tally;
3424
- resultPage = emptyPage();
3425
- resultPage.setTitle("Search for '" + searchQuery + "'");
3426
- resultPage.addParagraph("String '" + searchQuery + "' found on " + (tally.finds || 'none') + " of " + (tally.pages || 'no') + " pages from " + (tally.sites || 'no') + " sites.\nText matched on " + (tally.title || 'no') + " titles, " + (tally.text || 'no') + " paragraphs, and " + (tally.slug || 'no') + " slugs.\nElapsed time " + tally.msec + " milliseconds.");
3427
- _ref = searchResults.finds;
3428
- for (_i = 0, _len = _ref.length; _i < _len; _i++) {
3429
- result = _ref[_i];
3430
- resultPage.addItem({
3431
- "type": "reference",
3432
- "site": result.site,
3433
- "slug": result.page.slug,
3434
- "title": result.page.title,
3435
- "text": result.page.synopsis || ''
3436
- });
3437
- }
3438
- $resultPage = wiki.createPage(resultPage.getSlug()).addClass('ghost');
3439
- $resultPage.appendTo($('.main'));
3440
- wiki.buildPage(resultPage, $resultPage);
3441
- return active.set($('.page').last());
3442
- };
3443
- return {
3444
- performSearch: performSearch
3445
- };
3446
- };
3447
-
3448
- module.exports = createSearch;
3449
-
3450
- }).call(this);
3451
-
3452
- },{"./active":10,"./page":12,"./util":6,"./wiki":2}],18:[function(require,module,exports){
3453
- (function() {
3454
- var active, createSearch, neighborhood, nextAvailableFetch, nextFetchInterval, populateSiteInfoFor, util, wiki, _,
3455
- __hasProp = {}.hasOwnProperty;
3456
-
3457
- _ = require('underscore');
3458
-
3459
- wiki = require('./wiki');
3460
-
3461
- active = require('./active');
3462
-
3463
- util = require('./util');
3464
-
3465
- createSearch = require('./search');
3466
-
3467
- module.exports = neighborhood = {};
3468
-
3469
- if (wiki.neighborhood == null) {
3470
- wiki.neighborhood = {};
3471
- }
3472
-
3473
- nextAvailableFetch = 0;
3474
-
3475
- nextFetchInterval = 2000;
3476
-
3477
- populateSiteInfoFor = function(site, neighborInfo) {
3478
- var fetchMap, now, transition;
3479
- if (neighborInfo.sitemapRequestInflight) {
3480
- return;
3481
- }
3482
- neighborInfo.sitemapRequestInflight = true;
3483
- transition = function(site, from, to) {
3484
- return $(".neighbor[data-site=\"" + site + "\"]").find('div').removeClass(from).addClass(to);
3485
- };
3486
- fetchMap = function() {
3487
- var request, sitemapUrl;
3488
- sitemapUrl = "http://" + site + "/system/sitemap.json";
3489
- transition(site, 'wait', 'fetch');
3490
- request = $.ajax({
3491
- type: 'GET',
3492
- dataType: 'json',
3493
- url: sitemapUrl
3494
- });
3495
- return request.always(function() {
3496
- return neighborInfo.sitemapRequestInflight = false;
3497
- }).done(function(data) {
3498
- neighborInfo.sitemap = data;
3499
- transition(site, 'fetch', 'done');
3500
- return $('body').trigger('new-neighbor-done', site);
3501
- }).fail(function(data) {
3502
- return transition(site, 'fetch', 'fail');
3503
- });
3504
- };
3505
- now = Date.now();
3506
- if (now > nextAvailableFetch) {
3507
- nextAvailableFetch = now + nextFetchInterval;
3508
- return setTimeout(fetchMap, 100);
3509
- } else {
3510
- setTimeout(fetchMap, nextAvailableFetch - now);
3511
- return nextAvailableFetch += nextFetchInterval;
3512
- }
3513
- };
3514
-
3515
- wiki.registerNeighbor = neighborhood.registerNeighbor = function(site) {
3516
- var neighborInfo;
3517
- if (wiki.neighborhood[site] != null) {
3518
- return;
3519
- }
3520
- neighborInfo = {};
3521
- wiki.neighborhood[site] = neighborInfo;
3522
- populateSiteInfoFor(site, neighborInfo);
3523
- return $('body').trigger('new-neighbor', site);
3524
- };
3525
-
3526
- neighborhood.listNeighbors = function() {
3527
- return _.keys(wiki.neighborhood);
3528
- };
3529
-
3530
- neighborhood.search = function(searchQuery) {
3531
- var finds, match, matchingPages, neighborInfo, neighborSite, sitemap, start, tally, tick, _ref;
3532
- finds = [];
3533
- tally = {};
3534
- tick = function(key) {
3535
- if (tally[key] != null) {
3536
- return tally[key]++;
3537
- } else {
3538
- return tally[key] = 1;
3539
- }
3540
- };
3541
- match = function(key, text) {
3542
- var hit;
3543
- hit = (text != null) && text.toLowerCase().indexOf(searchQuery.toLowerCase()) >= 0;
3544
- if (hit) {
3545
- tick(key);
3546
- }
3547
- return hit;
3548
- };
3549
- start = Date.now();
3550
- _ref = wiki.neighborhood;
3551
- for (neighborSite in _ref) {
3552
- if (!__hasProp.call(_ref, neighborSite)) continue;
3553
- neighborInfo = _ref[neighborSite];
3554
- sitemap = neighborInfo.sitemap;
3555
- if (sitemap != null) {
3556
- tick('sites');
3557
- }
3558
- matchingPages = _.each(sitemap, function(page) {
3559
- tick('pages');
3560
- if (!(match('title', page.title) || match('text', page.synopsis) || match('slug', page.slug))) {
3561
- return;
3562
- }
3563
- tick('finds');
3564
- return finds.push({
3565
- page: page,
3566
- site: neighborSite,
3567
- rank: 1
3568
- });
3569
- });
3570
- }
3571
- tally['msec'] = Date.now() - start;
3572
- return {
3573
- finds: finds,
3574
- tally: tally
3575
- };
3576
- };
3577
-
3578
- $(function() {
3579
- var $neighborhood, flag, search;
3580
- $neighborhood = $('.neighborhood');
3581
- flag = function(site) {
3582
- return "<span class=\"neighbor\" data-site=\"" + site + "\">\n <div class=\"wait\">\n <img src=\"http://" + site + "/favicon.png\" title=\"" + site + "\">\n </div>\n</span>";
3583
- };
3584
- $('body').on('new-neighbor', function(e, site) {
3585
- return $neighborhood.append(flag(site));
3586
- }).delegate('.neighbor img', 'click', function(e) {
3587
- return wiki.doInternalLink('welcome-visitors', null, this.title);
3588
- });
3589
- search = createSearch({
3590
- neighborhood: neighborhood
3591
- });
3592
- return $('input.search').on('keypress', function(e) {
3593
- var searchQuery;
3594
- if (e.keyCode !== 13) {
3595
- return;
3596
- }
3597
- searchQuery = $(this).val();
3598
- search.performSearch(searchQuery);
3599
- return $(this).val("");
3600
- });
3601
- });
3602
-
3603
- }).call(this);
3604
-
3605
- },{"./active":10,"./search":19,"./util":6,"./wiki":2,"underscore":17}]},{},[1])
3606
- //@ sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/client.coffee","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/wiki.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/legacy.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/synopsis.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/persona.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/active.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/util.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/plugin.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/state.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/revision.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/reference.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/factory.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/addToJournal.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/pageHandler.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/refresh.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/page.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/underscore/underscore.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/search.js","/Users/wcunningham/FedWiki/Smallest-Federated-Wiki/client/node_modules/wiki-client/lib/neighborhood.js"],"names":[],"mappings":";AAAA,CAAO,EAAO,CAAd,EAAM,CAAQ,eAAA;;AACd,CADA,MACA,iBAAA;;;;ACDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7vCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["window.wiki = require('wiki-client/lib/wiki')\nrequire('wiki-client/lib/legacy')\n\n","(function() {\n  var createSynopsis, wiki,\n    __slice = [].slice;\n\n  createSynopsis = require('./synopsis');\n\n  wiki = {\n    createSynopsis: createSynopsis\n  };\n\n  wiki.persona = require('./persona');\n\n  wiki.log = function() {\n    var things;\n    things = 1 <= arguments.length ? __slice.call(arguments, 0) : [];\n    if ((typeof console !== \"undefined\" && console !== null ? console.log : void 0) != null) {\n      return console.log.apply(console, things);\n    }\n  };\n\n  wiki.asSlug = function(name) {\n    return name.replace(/\\s/g, '-').replace(/[^A-Za-z0-9-]/g, '').toLowerCase();\n  };\n\n  wiki.useLocalStorage = function() {\n    return $(\".login\").length > 0;\n  };\n\n  wiki.resolutionContext = [];\n\n  wiki.resolveFrom = function(addition, callback) {\n    wiki.resolutionContext.push(addition);\n    try {\n      return callback();\n    } finally {\n      wiki.resolutionContext.pop();\n    }\n  };\n\n  wiki.getData = function(vis) {\n    var idx, who;\n    if (vis) {\n      idx = $('.item').index(vis);\n      who = $(\".item:lt(\" + idx + \")\").filter('.chart,.data,.calculator').last();\n      if (who != null) {\n        return who.data('item').data;\n      } else {\n        return {};\n      }\n    } else {\n      who = $('.chart,.data,.calculator').last();\n      if (who != null) {\n        return who.data('item').data;\n      } else {\n        return {};\n      }\n    }\n  };\n\n  wiki.getDataNodes = function(vis) {\n    var idx, who;\n    if (vis) {\n      idx = $('.item').index(vis);\n      who = $(\".item:lt(\" + idx + \")\").filter('.chart,.data,.calculator').toArray().reverse();\n      return $(who);\n    } else {\n      who = $('.chart,.data,.calculator').toArray().reverse();\n      return $(who);\n    }\n  };\n\n  wiki.createPage = function(name, loc) {\n    var $page, site;\n    if (loc && loc !== 'view') {\n      site = loc;\n    }\n    $page = $(\"<div class=\\\"page\\\" id=\\\"\" + name + \"\\\">\\n  <div class=\\\"twins\\\"> <p> </p> </div>\\n  <div class=\\\"header\\\">\\n    <h1> <img class=\\\"favicon\\\" src=\\\"\" + (site ? \"//\" + site : \"\") + \"/favicon.png\\\" height=\\\"32px\\\"> \" + name + \" </h1>\\n  </div>\\n</div>\");\n    if (site) {\n      $page.data('site', site);\n    }\n    return $page;\n  };\n\n  wiki.getItem = function(element) {\n    if ($(element).length > 0) {\n      return $(element).data(\"item\") || $(element).data('staticItem');\n    }\n  };\n\n  wiki.resolveLinks = function(string) {\n    var renderInternalLink;\n    renderInternalLink = function(match, name) {\n      var slug;\n      slug = wiki.asSlug(name);\n      return \"<a class=\\\"internal\\\" href=\\\"/\" + slug + \".html\\\" data-page-name=\\\"\" + slug + \"\\\" title=\\\"\" + (wiki.resolutionContext.join(' => ')) + \"\\\">\" + name + \"</a>\";\n    };\n    return string.replace(/\\[\\[([^\\]]+)\\]\\]/gi, renderInternalLink).replace(/\\[(http.*?) (.*?)\\]/gi, \"<a class=\\\"external\\\" target=\\\"_blank\\\" href=\\\"$1\\\" title=\\\"$1\\\" rel=\\\"nofollow\\\">$2 <img src=\\\"/images/external-link-ltr-icon.png\\\"></a>\");\n  };\n\n  module.exports = wiki;\n\n}).call(this);\n","(function() {\n  var active, newPage, pageHandler, plugin, refresh, state, util, wiki;\n\n  wiki = require('./wiki');\n\n  util = require('./util');\n\n  pageHandler = wiki.pageHandler = require('./pageHandler');\n\n  plugin = require('./plugin');\n\n  state = require('./state');\n\n  active = require('./active');\n\n  refresh = require('./refresh');\n\n  newPage = require('./page').newPage;\n\n  Array.prototype.last = function() {\n    return this[this.length - 1];\n  };\n\n  $(function() {\n    var LEFTARROW, RIGHTARROW, createTextElement, doInternalLink, finishClick, getTemplate, sleep, textEditor;\n    window.dialog = $('<div></div>').html('This dialog will show every time!').dialog({\n      autoOpen: false,\n      title: 'Basic Dialog',\n      height: 600,\n      width: 800\n    });\n    wiki.dialog = function(title, html) {\n      window.dialog.html(html);\n      window.dialog.dialog(\"option\", \"title\", wiki.resolveLinks(title));\n      return window.dialog.dialog('open');\n    };\n    sleep = function(time, done) {\n      return setTimeout(done, time);\n    };\n    wiki.removeItem = function($item, item) {\n      pageHandler.put($item.parents('.page:first'), {\n        type: 'remove',\n        id: item.id\n      });\n      return $item.remove();\n    };\n    wiki.createItem = function($page, $before, item) {\n      var $item, before;\n      if ($page == null) {\n        $page = $before.parents('.page');\n      }\n      item.id = util.randomBytes(8);\n      $item = $(\"<div class=\\\"item \" + item.type + \"\\\" data-id=\\\"\" + \"\\\"</div>\");\n      $item.data('item', item).data('pageElement', $page);\n      if ($before != null) {\n        $before.after($item);\n      } else {\n        $page.find('.story').append($item);\n      }\n      plugin[\"do\"]($item, item);\n      before = wiki.getItem($before);\n      sleep(500, function() {\n        return pageHandler.put($page, {\n          item: item,\n          id: item.id,\n          type: 'add',\n          after: before != null ? before.id : void 0\n        });\n      });\n      return $item;\n    };\n    createTextElement = function(pageElement, beforeElement, initialText) {\n      var item, itemBefore, itemElement;\n      item = {\n        type: 'paragraph',\n        id: util.randomBytes(8),\n        text: initialText\n      };\n      itemElement = $(\"<div class=\\\"item paragraph\\\" data-id=\" + item.id + \"></div>\");\n      itemElement.data('item', item).data('pageElement', pageElement);\n      beforeElement.after(itemElement);\n      plugin[\"do\"](itemElement, item);\n      itemBefore = wiki.getItem(beforeElement);\n      wiki.textEditor(itemElement, item);\n      return sleep(500, function() {\n        return pageHandler.put(pageElement, {\n          item: item,\n          id: item.id,\n          type: 'add',\n          after: itemBefore != null ? itemBefore.id : void 0\n        });\n      });\n    };\n    textEditor = wiki.textEditor = function(div, item, caretPos, doubleClicked) {\n      var original, textarea, _ref;\n      if (div.hasClass('textEditing')) {\n        return;\n      }\n      div.addClass('textEditing');\n      textarea = $(\"<textarea>\" + (original = (_ref = item.text) != null ? _ref : '') + \"</textarea>\").focusout(function() {\n        div.removeClass('textEditing');\n        if (item.text = textarea.val()) {\n          plugin[\"do\"](div.empty(), item);\n          if (item.text === original) {\n            return;\n          }\n          pageHandler.put(div.parents('.page:first'), {\n            type: 'edit',\n            id: item.id,\n            item: item\n          });\n        } else {\n          pageHandler.put(div.parents('.page:first'), {\n            type: 'remove',\n            id: item.id\n          });\n          div.remove();\n        }\n        return null;\n      }).bind('keydown', function(e) {\n        var middle, page, pageElement, prefix, prevItem, prevTextLen, sel, suffix, text;\n        if ((e.altKey || e.ctlKey || e.metaKey) && e.which === 83) {\n          textarea.focusout();\n          return false;\n        }\n        if ((e.altKey || e.ctlKey || e.metaKey) && e.which === 73) {\n          e.preventDefault();\n          if (!e.shiftKey) {\n            page = $(e.target).parents('.page');\n          }\n          doInternalLink(\"about \" + item.type + \" plugin\", page);\n          return false;\n        }\n        if (item.type === 'paragraph') {\n          sel = util.getSelectionPos(textarea);\n          if (e.which === $.ui.keyCode.BACKSPACE && sel.start === 0 && sel.start === sel.end) {\n            prevItem = wiki.getItem(div.prev());\n            if (prevItem.type !== 'paragraph') {\n              return false;\n            }\n            prevTextLen = prevItem.text.length;\n            prevItem.text += textarea.val();\n            textarea.val('');\n            textEditor(div.prev(), prevItem, prevTextLen);\n            return false;\n          } else if (e.which === $.ui.keyCode.ENTER && item.type === 'paragraph') {\n            if (!sel) {\n              return false;\n            }\n            text = textarea.val();\n            prefix = text.substring(0, sel.start);\n            if (sel.start !== sel.end) {\n              middle = text.substring(sel.start, sel.end);\n            }\n            suffix = text.substring(sel.end);\n            if (prefix === '') {\n              textarea.val(' ');\n            } else {\n              textarea.val(prefix);\n            }\n            textarea.focusout();\n            pageElement = div.parent().parent();\n            createTextElement(pageElement, div, suffix);\n            if (middle != null) {\n              createTextElement(pageElement, div, middle);\n            }\n            if (prefix === '') {\n              createTextElement(pageElement, div, '');\n            }\n            return false;\n          }\n        }\n      });\n      div.html(textarea);\n      if (caretPos != null) {\n        return util.setCaretPosition(textarea, caretPos);\n      } else if (doubleClicked) {\n        util.setCaretPosition(textarea, textarea.val().length);\n        return textarea.scrollTop(textarea[0].scrollHeight - textarea.height());\n      } else {\n        return textarea.focus();\n      }\n    };\n    doInternalLink = wiki.doInternalLink = function(name, page, site) {\n      if (site == null) {\n        site = null;\n      }\n      name = wiki.asSlug(name);\n      if (page != null) {\n        $(page).nextAll().remove();\n      }\n      wiki.createPage(name, site).appendTo($('.main')).each(refresh);\n      return active.set($('.page').last());\n    };\n    LEFTARROW = 37;\n    RIGHTARROW = 39;\n    $(document).keydown(function(event) {\n      var direction, newIndex, pages;\n      direction = (function() {\n        switch (event.which) {\n          case LEFTARROW:\n            return -1;\n          case RIGHTARROW:\n            return +1;\n        }\n      })();\n      if (direction && !(event.target.tagName === \"TEXTAREA\")) {\n        pages = $('.page');\n        newIndex = pages.index($('.active')) + direction;\n        if ((0 <= newIndex && newIndex < pages.length)) {\n          return active.set(pages.eq(newIndex));\n        }\n      }\n    });\n    $(window).on('popstate', state.show);\n    $(document).ajaxError(function(event, request, settings) {\n      if (request.status === 0 || request.status === 404) {\n        return;\n      }\n      wiki.log('ajax error', event, request, settings);\n      return $('.main').prepend(\"<li class='error'>\\n  Error on \" + settings.url + \": \" + request.responseText + \"\\n</li>\");\n    });\n    getTemplate = function(slug, done) {\n      if (!slug) {\n        return done(null);\n      }\n      wiki.log('getTemplate', slug);\n      return pageHandler.get({\n        whenGotten: function(data, siteFound) {\n          return done(data.story);\n        },\n        whenNotGotten: function() {\n          return done(null);\n        },\n        pageInformation: {\n          slug: slug\n        }\n      });\n    };\n    finishClick = function(e, name) {\n      var page;\n      e.preventDefault();\n      if (!e.shiftKey) {\n        page = $(e.target).parents('.page');\n      }\n      doInternalLink(name, page, $(e.target).data('site'));\n      return false;\n    };\n    $('.main').delegate('.show-page-source', 'click', function(e) {\n      var json, pageElement;\n      e.preventDefault();\n      pageElement = $(this).parent().parent();\n      json = pageElement.data('data');\n      return wiki.dialog(\"JSON for \" + json.title, $('<pre/>').text(JSON.stringify(json, null, 2)));\n    }).delegate('.page', 'click', function(e) {\n      if (!$(e.target).is(\"a\")) {\n        return active.set(this);\n      }\n    }).delegate('.internal', 'click', function(e) {\n      var name;\n      name = $(e.target).data('pageName');\n      pageHandler.context = $(e.target).attr('title').split(' => ');\n      return finishClick(e, name);\n    }).delegate('img.remote', 'click', function(e) {\n      var name;\n      name = $(e.target).data('slug');\n      pageHandler.context = [$(e.target).data('site')];\n      return finishClick(e, name);\n    }).delegate('.revision', 'dblclick', function(e) {\n      var $page, action, json, page, rev;\n      e.preventDefault();\n      $page = $(this).parents('.page');\n      page = $page.data('data');\n      rev = page.journal.length - 1;\n      action = page.journal[rev];\n      json = JSON.stringify(action, null, 2);\n      return wiki.dialog(\"Revision \" + rev + \", \" + action.type + \" action\", $('<pre/>').text(json));\n    }).delegate('.action', 'click', function(e) {\n      var $action, $page, name, rev, slug;\n      e.preventDefault();\n      $action = $(e.target);\n      if ($action.is('.fork') && ((name = $action.data('slug')) != null)) {\n        pageHandler.context = [$action.data('site')];\n        return finishClick(e, (name.split('_'))[0]);\n      } else {\n        $page = $(this).parents('.page');\n        slug = wiki.asSlug($page.data('data').title);\n        rev = $(this).parent().children().index($action);\n        if (!e.shiftKey) {\n          $page.nextAll().remove();\n        }\n        wiki.createPage(\"\" + slug + \"_rev\" + rev, $page.data('site')).appendTo($('.main')).each(refresh);\n        return active.set($('.page').last());\n      }\n    }).delegate('.fork-page', 'click', function(e) {\n      var item, pageElement, remoteSite;\n      pageElement = $(e.target).parents('.page');\n      if (pageElement.hasClass('local')) {\n        if (!wiki.useLocalStorage()) {\n          item = pageElement.data('data');\n          pageElement.removeClass('local');\n          return pageHandler.put(pageElement, {\n            type: 'fork',\n            item: item\n          });\n        }\n      } else {\n        if ((remoteSite = pageElement.data('site')) != null) {\n          return pageHandler.put(pageElement, {\n            type: 'fork',\n            site: remoteSite\n          });\n        }\n      }\n    }).delegate('.action', 'hover', function() {\n      var id;\n      id = $(this).attr('data-id');\n      $(\"[data-id=\" + id + \"]\").toggleClass('target');\n      return $('.main').trigger('rev');\n    }).delegate('.item', 'hover', function() {\n      var id;\n      id = $(this).attr('data-id');\n      return $(\".action[data-id=\" + id + \"]\").toggleClass('target');\n    }).delegate('button.create', 'click', function(e) {\n      return getTemplate($(e.target).data('slug'), function(story) {\n        var $page, page, pageObject;\n        $page = $(e.target).parents('.page:first');\n        $page.removeClass('ghost');\n        page = $page.data('data');\n        page.story = story || [];\n        pageObject = newPage(page, null);\n        page = pageObject.getRawPage();\n        pageHandler.put($page, {\n          type: 'create',\n          id: page.id,\n          item: {\n            title: page.title,\n            story: page.story\n          }\n        });\n        return wiki.buildPage(pageObject, $page.empty());\n      });\n    }).delegate('.ghost', 'rev', function(e) {\n      var $item, $page, position;\n      wiki.log('rev', e);\n      $page = $(e.target).parents('.page:first');\n      $item = $page.find('.target');\n      position = $item.offset().top + $page.scrollTop() - $page.height() / 2;\n      wiki.log('scroll', $page, $item, position);\n      return $page.stop().animate({\n        scrollTop: postion\n      }, 'slow');\n    }).delegate('.score', 'hover', function(e) {\n      return $('.main').trigger('thumb', $(e.target).data('thumb'));\n    });\n    $(\".provider input\").click(function() {\n      $(\"footer input:first\").val($(this).attr('data-provider'));\n      return $(\"footer form\").submit();\n    });\n    $('body').on('new-neighbor-done', function(e, neighbor) {\n      return $('.page').each(function(index, element) {\n        return wiki.emitTwins($(element));\n      });\n    });\n    return $(function() {\n      state.first();\n      $('.page').each(refresh);\n      return active.set($('.page').last());\n    });\n  });\n\n}).call(this);\n","(function() {\n  module.exports = function(page) {\n    var p1, p2, synopsis;\n    synopsis = page.synopsis;\n    if ((page != null) && (page.story != null)) {\n      p1 = page.story[0];\n      p2 = page.story[1];\n      if (p1 && p1.type === 'paragraph') {\n        synopsis || (synopsis = p1.text);\n      }\n      if (p2 && p2.type === 'paragraph') {\n        synopsis || (synopsis = p2.text);\n      }\n      if (p1 && (p1.text != null)) {\n        synopsis || (synopsis = p1.text);\n      }\n      if (p2 && (p2.text != null)) {\n        synopsis || (synopsis = p2.text);\n      }\n      synopsis || (synopsis = (page.story != null) && (\"A page with \" + page.story.length + \" items.\"));\n    } else {\n      synopsis = 'A page with no story.';\n    }\n    return synopsis;\n  };\n\n}).call(this);\n","(function() {\n  module.exports = function(owner) {\n    $(\"#user-email\").hide();\n    $(\"#persona-login-btn\").hide();\n    $(\"#persona-logout-btn\").hide();\n    navigator.id.watch({\n      loggedInUser: owner,\n      onlogin: function(assertion) {\n        return $.post(\"/persona_login\", {\n          assertion: assertion\n        }, function(verified) {\n          verified = JSON.parse(verified);\n          if (\"okay\" === verified.status) {\n            return window.location = \"/\";\n          } else {\n            navigator.id.logout();\n            if (\"wrong-address\" === verified.status) {\n              return window.location = \"/oops\";\n            }\n          }\n        });\n      },\n      onlogout: function() {\n        return $.post(\"/persona_logout\", function() {\n          return window.location = \"/\";\n        });\n      },\n      onready: function() {\n        if (owner) {\n          $(\"#persona-login-btn\").hide();\n          return $(\"#persona-logout-btn\").show();\n        } else {\n          $(\"#persona-login-btn\").show();\n          return $(\"#persona-logout-btn\").hide();\n        }\n      }\n    });\n    $(\"#persona-login-btn\").click(function(e) {\n      e.preventDefault();\n      return navigator.id.request({});\n    });\n    return $(\"#persona-logout-btn\").click(function(e) {\n      e.preventDefault();\n      return navigator.id.logout();\n    });\n  };\n\n}).call(this);\n","(function() {\n  var active, findScrollContainer, scrollTo;\n\n  module.exports = active = {};\n\n  active.scrollContainer = void 0;\n\n  findScrollContainer = function() {\n    var scrolled;\n    scrolled = $(\"body, html\").filter(function() {\n      return $(this).scrollLeft() > 0;\n    });\n    if (scrolled.length > 0) {\n      return scrolled;\n    } else {\n      return $(\"body, html\").scrollLeft(12).filter(function() {\n        return $(this).scrollLeft() > 0;\n      }).scrollTop(0);\n    }\n  };\n\n  scrollTo = function(el) {\n    var bodyWidth, contentWidth, maxX, minX, target, width;\n    if (active.scrollContainer == null) {\n      active.scrollContainer = findScrollContainer();\n    }\n    bodyWidth = $(\"body\").width();\n    minX = active.scrollContainer.scrollLeft();\n    maxX = minX + bodyWidth;\n    target = el.position().left;\n    width = el.outerWidth(true);\n    contentWidth = $(\".page\").outerWidth(true) * $(\".page\").size();\n    if (target < minX) {\n      return active.scrollContainer.animate({\n        scrollLeft: target\n      });\n    } else if (target + width > maxX) {\n      return active.scrollContainer.animate({\n        scrollLeft: target - (bodyWidth - width)\n      });\n    } else if (maxX > $(\".pages\").outerWidth()) {\n      return active.scrollContainer.animate({\n        scrollLeft: Math.min(target, contentWidth - bodyWidth)\n      });\n    }\n  };\n\n  active.set = function(el) {\n    el = $(el);\n    $(\".active\").removeClass(\"active\");\n    return scrollTo(el.addClass(\"active\"));\n  };\n\n}).call(this);\n","(function() {\n  var util, wiki;\n\n  wiki = require('./wiki');\n\n  module.exports = wiki.util = util = {};\n\n  util.symbols = {\n    create: '☼',\n    add: '+',\n    edit: '✎',\n    fork: '⚑',\n    move: '↕',\n    remove: '✕'\n  };\n\n  util.randomByte = function() {\n    return (((1 + Math.random()) * 0x100) | 0).toString(16).substring(1);\n  };\n\n  util.randomBytes = function(n) {\n    return ((function() {\n      var _i, _results;\n      _results = [];\n      for (_i = 1; 1 <= n ? _i <= n : _i >= n; 1 <= n ? _i++ : _i--) {\n        _results.push(util.randomByte());\n      }\n      return _results;\n    })()).join('');\n  };\n\n  util.formatTime = function(time) {\n    var am, d, h, mi, mo;\n    d = new Date((time > 10000000000 ? time : time * 1000));\n    mo = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'][d.getMonth()];\n    h = d.getHours();\n    am = h < 12 ? 'AM' : 'PM';\n    h = h === 0 ? 12 : h > 12 ? h - 12 : h;\n    mi = (d.getMinutes() < 10 ? \"0\" : \"\") + d.getMinutes();\n    return \"\" + h + \":\" + mi + \" \" + am + \"<br>\" + (d.getDate()) + \" \" + mo + \" \" + (d.getFullYear());\n  };\n\n  util.formatDate = function(msSinceEpoch) {\n    var am, d, day, h, mi, mo, sec, wk, yr;\n    d = new Date(msSinceEpoch);\n    wk = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'][d.getDay()];\n    mo = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'][d.getMonth()];\n    day = d.getDate();\n    yr = d.getFullYear();\n    h = d.getHours();\n    am = h < 12 ? 'AM' : 'PM';\n    h = h === 0 ? 12 : h > 12 ? h - 12 : h;\n    mi = (d.getMinutes() < 10 ? \"0\" : \"\") + d.getMinutes();\n    sec = (d.getSeconds() < 10 ? \"0\" : \"\") + d.getSeconds();\n    return \"\" + wk + \" \" + mo + \" \" + day + \", \" + yr + \"<br>\" + h + \":\" + mi + \":\" + sec + \" \" + am;\n  };\n\n  util.formatElapsedTime = function(msSinceEpoch) {\n    var days, hrs, mins, months, msecs, secs, weeks, years;\n    msecs = new Date().getTime() - msSinceEpoch;\n    if ((secs = msecs / 1000) < 2) {\n      return \"\" + (Math.floor(msecs)) + \" milliseconds ago\";\n    }\n    if ((mins = secs / 60) < 2) {\n      return \"\" + (Math.floor(secs)) + \" seconds ago\";\n    }\n    if ((hrs = mins / 60) < 2) {\n      return \"\" + (Math.floor(mins)) + \" minutes ago\";\n    }\n    if ((days = hrs / 24) < 2) {\n      return \"\" + (Math.floor(hrs)) + \" hours ago\";\n    }\n    if ((weeks = days / 7) < 2) {\n      return \"\" + (Math.floor(days)) + \" days ago\";\n    }\n    if ((months = days / 31) < 2) {\n      return \"\" + (Math.floor(weeks)) + \" weeks ago\";\n    }\n    if ((years = days / 365) < 2) {\n      return \"\" + (Math.floor(months)) + \" months ago\";\n    }\n    return \"\" + (Math.floor(years)) + \" years ago\";\n  };\n\n  util.emptyPage = function() {\n    return {\n      title: 'empty',\n      story: [],\n      journal: []\n    };\n  };\n\n  util.getSelectionPos = function(jQueryElement) {\n    var el, iePos, sel;\n    el = jQueryElement.get(0);\n    if (document.selection) {\n      el.focus();\n      sel = document.selection.createRange();\n      sel.moveStart('character', -el.value.length);\n      iePos = sel.text.length;\n      return {\n        start: iePos,\n        end: iePos\n      };\n    } else {\n      return {\n        start: el.selectionStart,\n        end: el.selectionEnd\n      };\n    }\n  };\n\n  util.setCaretPosition = function(jQueryElement, caretPos) {\n    var el, range;\n    el = jQueryElement.get(0);\n    if (el != null) {\n      if (el.createTextRange) {\n        range = el.createTextRange();\n        range.move(\"character\", caretPos);\n        range.select();\n      } else {\n        el.setSelectionRange(caretPos, caretPos);\n      }\n      return el.focus();\n    }\n  };\n\n}).call(this);\n","(function() {\n  var getScript, plugin, scripts, util, wiki;\n\n  util = require('./util');\n\n  wiki = require('./wiki');\n\n  module.exports = plugin = {};\n\n  scripts = {};\n\n  getScript = wiki.getScript = function(url, callback) {\n    if (callback == null) {\n      callback = function() {};\n    }\n    if (scripts[url] != null) {\n      return callback();\n    } else {\n      return $.getScript(url).done(function() {\n        scripts[url] = true;\n        return callback();\n      }).fail(function() {\n        return callback();\n      });\n    }\n  };\n\n  plugin.get = wiki.getPlugin = function(name, callback) {\n    if (window.plugins[name]) {\n      return callback(window.plugins[name]);\n    }\n    return getScript(\"/plugins/\" + name + \"/\" + name + \".js\", function() {\n      if (window.plugins[name]) {\n        return callback(window.plugins[name]);\n      }\n      return getScript(\"/plugins/\" + name + \".js\", function() {\n        return callback(window.plugins[name]);\n      });\n    });\n  };\n\n  plugin[\"do\"] = wiki.doPlugin = function(div, item, done) {\n    var error;\n    if (done == null) {\n      done = function() {};\n    }\n    error = function(ex) {\n      var errorElement;\n      errorElement = $(\"<div />\").addClass('error');\n      errorElement.text(ex.toString());\n      return div.append(errorElement);\n    };\n    div.data('pageElement', div.parents(\".page\"));\n    div.data('item', item);\n    return plugin.get(item.type, function(script) {\n      var err;\n      try {\n        if (script == null) {\n          throw TypeError(\"Can't find plugin for '\" + item.type + \"'\");\n        }\n        if (script.emit.length > 2) {\n          return script.emit(div, item, function() {\n            script.bind(div, item);\n            return done();\n          });\n        } else {\n          script.emit(div, item);\n          script.bind(div, item);\n          return done();\n        }\n      } catch (_error) {\n        err = _error;\n        wiki.log('plugin error', err);\n        error(err);\n        return done();\n      }\n    });\n  };\n\n  wiki.registerPlugin = function(pluginName, pluginFn) {\n    return window.plugins[pluginName] = pluginFn($);\n  };\n\n  window.plugins = {\n    reference: require('./reference'),\n    factory: require('./factory'),\n    paragraph: {\n      emit: function(div, item) {\n        var text, _i, _len, _ref, _results;\n        _ref = item.text.split(/\\n\\n+/);\n        _results = [];\n        for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n          text = _ref[_i];\n          if (text.match(/\\S/)) {\n            _results.push(div.append(\"<p>\" + (wiki.resolveLinks(text)) + \"</p>\"));\n          } else {\n            _results.push(void 0);\n          }\n        }\n        return _results;\n      },\n      bind: function(div, item) {\n        return div.dblclick(function() {\n          return wiki.textEditor(div, item, null, true);\n        });\n      }\n    },\n    image: {\n      emit: function(div, item) {\n        item.text || (item.text = item.caption);\n        return div.append(\"<img class=thumbnail src=\\\"\" + item.url + \"\\\"> <p>\" + (wiki.resolveLinks(item.text)) + \"</p>\");\n      },\n      bind: function(div, item) {\n        div.dblclick(function() {\n          return wiki.textEditor(div, item);\n        });\n        return div.find('img').dblclick(function() {\n          return wiki.dialog(item.text, this);\n        });\n      }\n    },\n    future: {\n      emit: function(div, item) {\n        var info, _i, _len, _ref, _results;\n        div.append(\"\" + item.text + \"<br><br><button class=\\\"create\\\">create</button> new blank page\");\n        if (((info = wiki.neighborhood[location.host]) != null) && (info.sitemap != null)) {\n          _ref = info.sitemap;\n          _results = [];\n          for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n            item = _ref[_i];\n            if (item.slug.match(/-template$/)) {\n              _results.push(div.append(\"<br><button class=\\\"create\\\" data-slug=\" + item.slug + \">create</button> from \" + (wiki.resolveLinks(\"[[\" + item.title + \"]]\"))));\n            } else {\n              _results.push(void 0);\n            }\n          }\n          return _results;\n        }\n      },\n      bind: function(div, item) {}\n    }\n  };\n\n}).call(this);\n","(function() {\n  var active, state, wiki,\n    __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };\n\n  wiki = require('./wiki');\n\n  active = require('./active');\n\n  module.exports = state = {};\n\n  state.pagesInDom = function() {\n    return $.makeArray($(\".page\").map(function(_, el) {\n      return el.id;\n    }));\n  };\n\n  state.urlPages = function() {\n    var i;\n    return ((function() {\n      var _i, _len, _ref, _results;\n      _ref = $(location).attr('pathname').split('/');\n      _results = [];\n      for (_i = 0, _len = _ref.length; _i < _len; _i += 2) {\n        i = _ref[_i];\n        _results.push(i);\n      }\n      return _results;\n    })()).slice(1);\n  };\n\n  state.locsInDom = function() {\n    return $.makeArray($(\".page\").map(function(_, el) {\n      return $(el).data('site') || 'view';\n    }));\n  };\n\n  state.urlLocs = function() {\n    var j, _i, _len, _ref, _results;\n    _ref = $(location).attr('pathname').split('/').slice(1);\n    _results = [];\n    for (_i = 0, _len = _ref.length; _i < _len; _i += 2) {\n      j = _ref[_i];\n      _results.push(j);\n    }\n    return _results;\n  };\n\n  state.setUrl = function() {\n    var idx, locs, page, pages, url, _ref;\n    document.title = (_ref = $('.page:last').data('data')) != null ? _ref.title : void 0;\n    if (history && history.pushState) {\n      locs = state.locsInDom();\n      pages = state.pagesInDom();\n      url = ((function() {\n        var _i, _len, _results;\n        _results = [];\n        for (idx = _i = 0, _len = pages.length; _i < _len; idx = ++_i) {\n          page = pages[idx];\n          _results.push(\"/\" + ((locs != null ? locs[idx] : void 0) || 'view') + \"/\" + page);\n        }\n        return _results;\n      })()).join('');\n      if (url !== $(location).attr('pathname')) {\n        return history.pushState(null, null, url);\n      }\n    }\n  };\n\n  state.show = function(e) {\n    var idx, name, newLocs, newPages, old, oldLocs, oldPages, previous, _i, _len, _ref;\n    oldPages = state.pagesInDom();\n    newPages = state.urlPages();\n    oldLocs = state.locsInDom();\n    newLocs = state.urlLocs();\n    if (!location.pathname || location.pathname === '/') {\n      return;\n    }\n    previous = $('.page').eq(0);\n    for (idx = _i = 0, _len = newPages.length; _i < _len; idx = ++_i) {\n      name = newPages[idx];\n      if (name !== oldPages[idx]) {\n        old = $('.page').eq(idx);\n        if (old) {\n          old.remove();\n        }\n        wiki.createPage(name, newLocs[idx]).insertAfter(previous).each(wiki.refresh);\n      }\n      previous = $('.page').eq(idx);\n    }\n    previous.nextAll().remove();\n    active.set($('.page').last());\n    return document.title = (_ref = $('.page:last').data('data')) != null ? _ref.title : void 0;\n  };\n\n  state.first = function() {\n    var firstUrlLocs, firstUrlPages, idx, oldPages, urlPage, _i, _len, _results;\n    state.setUrl();\n    firstUrlPages = state.urlPages();\n    firstUrlLocs = state.urlLocs();\n    oldPages = state.pagesInDom();\n    _results = [];\n    for (idx = _i = 0, _len = firstUrlPages.length; _i < _len; idx = ++_i) {\n      urlPage = firstUrlPages[idx];\n      if (__indexOf.call(oldPages, urlPage) < 0) {\n        if (urlPage !== '') {\n          _results.push(wiki.createPage(urlPage, firstUrlLocs[idx]).appendTo('.main'));\n        } else {\n          _results.push(void 0);\n        }\n      }\n    }\n    return _results;\n  };\n\n}).call(this);\n","(function() {\n  var create;\n\n  create = function(revIndex, data) {\n    var afterIndex, editIndex, itemId, items, journal, journalEntry, removeIndex, revJournal, revStory, revStoryIds, revTitle, storyItem, _i, _j, _k, _len, _len1, _len2, _ref;\n    journal = data.journal;\n    revTitle = data.title;\n    revStory = [];\n    revJournal = journal.slice(0, +(+revIndex) + 1 || 9e9);\n    for (_i = 0, _len = revJournal.length; _i < _len; _i++) {\n      journalEntry = revJournal[_i];\n      revStoryIds = revStory.map(function(storyItem) {\n        return storyItem.id;\n      });\n      switch (journalEntry.type) {\n        case 'create':\n          if (journalEntry.item.title != null) {\n            revTitle = journalEntry.item.title;\n            revStory = journalEntry.item.story || [];\n          }\n          break;\n        case 'add':\n          if ((afterIndex = revStoryIds.indexOf(journalEntry.after)) !== -1) {\n            revStory.splice(afterIndex + 1, 0, journalEntry.item);\n          } else {\n            revStory.push(journalEntry.item);\n          }\n          break;\n        case 'edit':\n          if ((editIndex = revStoryIds.indexOf(journalEntry.id)) !== -1) {\n            revStory.splice(editIndex, 1, journalEntry.item);\n          } else {\n            revStory.push(journalEntry.item);\n          }\n          break;\n        case 'move':\n          items = {};\n          for (_j = 0, _len1 = revStory.length; _j < _len1; _j++) {\n            storyItem = revStory[_j];\n            items[storyItem.id] = storyItem;\n          }\n          revStory = [];\n          _ref = journalEntry.order;\n          for (_k = 0, _len2 = _ref.length; _k < _len2; _k++) {\n            itemId = _ref[_k];\n            if (items[itemId] != null) {\n              revStory.push(items[itemId]);\n            }\n          }\n          break;\n        case 'remove':\n          if ((removeIndex = revStoryIds.indexOf(journalEntry.id)) !== -1) {\n            revStory.splice(removeIndex, 1);\n          }\n      }\n    }\n    return {\n      story: revStory,\n      journal: revJournal,\n      title: revTitle\n    };\n  };\n\n  exports.create = create;\n\n}).call(this);\n","(function() {\n  var bind, emit;\n\n  emit = function($item, item) {\n    var site, slug;\n    slug = item.slug || 'welcome-visitors';\n    site = item.site;\n    return wiki.resolveFrom(site, function() {\n      return $item.append(\"<p style='margin-bottom:3px;'>\\n  <img class='remote'\\n    src='//\" + site + \"/favicon.png'\\n    title='\" + site + \"'\\n    data-site=\\\"\" + site + \"\\\"\\n    data-slug=\\\"\" + slug + \"\\\"\\n  >\\n  \" + (wiki.resolveLinks(\"[[\" + (item.title || slug) + \"]]\")) + \"\\n</p>\\n<div>\\n  \" + (wiki.resolveLinks(item.text)) + \"\\n</div>\");\n    });\n  };\n\n  bind = function($item, item) {\n    return $item.dblclick(function() {\n      return wiki.textEditor($item, item);\n    });\n  };\n\n  module.exports = {\n    emit: emit,\n    bind: bind\n  };\n\n}).call(this);\n","(function() {\n  var arrayToJson, bind, csvToArray, emit,\n    __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };\n\n  emit = function(div, item) {\n    var showMenu, showPrompt;\n    div.append('<p>Double-Click to Edit<br>Drop Text or Image to Insert</p>');\n    showMenu = function() {\n      var info, menu, menuItem, name, _i, _len, _ref, _ref1;\n      menu = div.find('p').append(\"<br>Or Choose a Plugin\");\n      menuItem = function(title, name) {\n        return menu.append(\"<li><a class=\\\"menu\\\" href=\\\"#\\\" title=\\\"\" + title + \"\\\">\" + name + \"</a></li>\");\n      };\n      if (Array.isArray(window.catalog)) {\n        _ref = window.catalog;\n        for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n          info = _ref[_i];\n          menuItem(info.title, info.name);\n        }\n      } else {\n        _ref1 = window.catalog;\n        for (name in _ref1) {\n          info = _ref1[name];\n          menuItem(info.menu, name);\n        }\n      }\n      return menu.find('a.menu').click(function(evt) {\n        div.removeClass('factory').addClass(item.type = evt.target.text.toLowerCase());\n        div.unbind();\n        return wiki.textEditor(div, item);\n      });\n    };\n    showPrompt = function() {\n      return div.append(\"<p>\" + (wiki.resolveLinks(item.prompt)) + \"</b>\");\n    };\n    if (item.prompt) {\n      return showPrompt();\n    } else if (window.catalog != null) {\n      return showMenu();\n    } else {\n      return $.getJSON('/system/factories.json', function(data) {\n        window.catalog = data;\n        return showMenu();\n      });\n    }\n  };\n\n  bind = function(div, item) {\n    var syncEditAction;\n    syncEditAction = function() {\n      var err, pageElement;\n      wiki.log('factory item', item);\n      div.empty().unbind();\n      div.removeClass(\"factory\").addClass(item.type);\n      pageElement = div.parents('.page:first');\n      try {\n        div.data('pageElement', pageElement);\n        div.data('item', item);\n        wiki.getPlugin(item.type, function(plugin) {\n          plugin.emit(div, item);\n          return plugin.bind(div, item);\n        });\n      } catch (_error) {\n        err = _error;\n        div.append(\"<p class='error'>\" + err + \"</p>\");\n      }\n      return wiki.pageHandler.put(pageElement, {\n        type: 'edit',\n        id: item.id,\n        item: item\n      });\n    };\n    div.dblclick(function() {\n      div.removeClass('factory').addClass(item.type = 'paragraph');\n      div.unbind();\n      return wiki.textEditor(div, item);\n    });\n    div.bind('dragenter', function(evt) {\n      return evt.preventDefault();\n    });\n    div.bind('dragover', function(evt) {\n      return evt.preventDefault();\n    });\n    return div.bind(\"drop\", function(dropEvent) {\n      var dt, found, ignore, origin, punt, readFile, url;\n      punt = function(data) {\n        item.prompt = \"<b>Unexpected Item</b><br>We can't make sense of the drop.<br>\" + (JSON.stringify(data)) + \"<br>Try something else or see [[About Factory Plugin]].\";\n        data.userAgent = navigator.userAgent;\n        item.punt = data;\n        wiki.log('factory punt', dropEvent);\n        return syncEditAction();\n      };\n      readFile = function(file) {\n        var majorType, minorType, reader, _ref;\n        if (file != null) {\n          _ref = file.type.split(\"/\"), majorType = _ref[0], minorType = _ref[1];\n          reader = new FileReader();\n          if (majorType === \"image\") {\n            reader.onload = function(loadEvent) {\n              item.type = 'image';\n              item.url = loadEvent.target.result;\n              item.caption || (item.caption = \"Uploaded image\");\n              return syncEditAction();\n            };\n            return reader.readAsDataURL(file);\n          } else if (majorType === \"text\") {\n            reader.onload = function(loadEvent) {\n              var array, result;\n              result = loadEvent.target.result;\n              if (minorType === 'csv') {\n                item.type = 'data';\n                item.columns = (array = csvToArray(result))[0];\n                item.data = arrayToJson(array);\n                item.text = file.fileName;\n              } else {\n                item.type = 'paragraph';\n                item.text = result;\n              }\n              return syncEditAction();\n            };\n            return reader.readAsText(file);\n          } else {\n            return punt({\n              number: 1,\n              name: file.fileName,\n              type: file.type\n            });\n          }\n        } else {\n          return punt({\n            number: 2,\n            types: dropEvent.originalEvent.dataTransfer.types\n          });\n        }\n      };\n      dropEvent.preventDefault();\n      if ((dt = dropEvent.originalEvent.dataTransfer) != null) {\n        if ((dt.types != null) && (__indexOf.call(dt.types, 'text/uri-list') >= 0 || __indexOf.call(dt.types, 'text/x-moz-url') >= 0) && !(__indexOf.call(dt.types, 'Files') >= 0)) {\n          url = dt.getData('URL');\n          if (found = url.match(/^http:\\/\\/([a-zA-Z0-9:.-]+)(\\/([a-zA-Z0-9:.-]+)\\/([a-z0-9-]+(_rev\\d+)?))+$/)) {\n            wiki.log('factory drop url', found);\n            ignore = found[0], origin = found[1], ignore = found[2], item.site = found[3], item.slug = found[4], ignore = found[5];\n            if ($.inArray(item.site, ['view', 'local', 'origin']) >= 0) {\n              item.site = origin;\n            }\n            return $.getJSON(\"http://\" + item.site + \"/\" + item.slug + \".json\", function(remote) {\n              wiki.log('factory remote', remote);\n              item.type = 'reference';\n              item.title = remote.title || item.slug;\n              item.text = wiki.createSynopsis(remote);\n              syncEditAction();\n              if (item.site != null) {\n                return wiki.registerNeighbor(item.site);\n              }\n            });\n          } else {\n            return punt({\n              number: 4,\n              url: url,\n              types: dt.types\n            });\n          }\n        } else if (__indexOf.call(dt.types, 'Files') >= 0) {\n          return readFile(dt.files[0]);\n        } else {\n          return punt({\n            number: 5,\n            types: dt.types\n          });\n        }\n      } else {\n        return punt({\n          number: 6,\n          trouble: \"no data transfer object\"\n        });\n      }\n    });\n  };\n\n  csvToArray = function(strData, strDelimiter) {\n    var arrData, arrMatches, objPattern, strMatchedDelimiter, strMatchedValue;\n    strDelimiter = strDelimiter || \",\";\n    objPattern = new RegExp(\"(\\\\\" + strDelimiter + \"|\\\\r?\\\\n|\\\\r|^)\" + \"(?:\\\"([^\\\"]*(?:\\\"\\\"[^\\\"]*)*)\\\"|\" + \"([^\\\"\\\\\" + strDelimiter + \"\\\\r\\\\n]*))\", \"gi\");\n    arrData = [[]];\n    arrMatches = null;\n    while (arrMatches = objPattern.exec(strData)) {\n      strMatchedDelimiter = arrMatches[1];\n      if (strMatchedDelimiter.length && (strMatchedDelimiter !== strDelimiter)) {\n        arrData.push([]);\n      }\n      if (arrMatches[2]) {\n        strMatchedValue = arrMatches[2].replace(new RegExp(\"\\\"\\\"\", \"g\"), \"\\\"\");\n      } else {\n        strMatchedValue = arrMatches[3];\n      }\n      arrData[arrData.length - 1].push(strMatchedValue);\n    }\n    return arrData;\n  };\n\n  arrayToJson = function(array) {\n    var cols, row, rowToObject, _i, _len, _results;\n    cols = array.shift();\n    rowToObject = function(row) {\n      var k, obj, v, _i, _len, _ref, _ref1;\n      obj = {};\n      _ref = _.zip(cols, row);\n      for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n        _ref1 = _ref[_i], k = _ref1[0], v = _ref1[1];\n        if ((v != null) && (v.match(/\\S/)) && v !== 'NULL') {\n          obj[k] = v;\n        }\n      }\n      return obj;\n    };\n    _results = [];\n    for (_i = 0, _len = array.length; _i < _len; _i++) {\n      row = array[_i];\n      _results.push(rowToObject(row));\n    }\n    return _results;\n  };\n\n  module.exports = {\n    emit: emit,\n    bind: bind\n  };\n\n}).call(this);\n","(function() {\n  var util;\n\n  util = require('./util');\n\n  module.exports = function(journalElement, action) {\n    var actionElement, actionTitle, controls, pageElement;\n    pageElement = journalElement.parents('.page:first');\n    actionTitle = action.type || 'separator';\n    if (action.date != null) {\n      actionTitle += \" \" + (util.formatElapsedTime(action.date));\n    }\n    actionElement = $(\"<a href=\\\"#\\\" /> \").addClass(\"action\").addClass(action.type || 'separator').text(action.symbol || util.symbols[action.type]).attr('title', actionTitle).attr('data-id', action.id || \"0\").data('action', action);\n    controls = journalElement.children('.control-buttons');\n    if (controls.length > 0) {\n      actionElement.insertBefore(controls);\n    } else {\n      actionElement.appendTo(journalElement);\n    }\n    if (action.type === 'fork' && (action.site != null)) {\n      return actionElement.css(\"background-image\", \"url(//\" + action.site + \"/favicon.png)\").attr(\"href\", \"//\" + action.site + \"/\" + (pageElement.attr('id')) + \".html\").data(\"site\", action.site).data(\"slug\", pageElement.attr('id'));\n    }\n  };\n\n}).call(this);\n","(function() {\n  var addToJournal, newPage, pageFromLocalStorage, pageHandler, pushToLocal, pushToServer, recursiveGet, revision, state, util, wiki, _;\n\n  _ = require('underscore');\n\n  wiki = require('./wiki');\n\n  util = require('./util');\n\n  state = require('./state');\n\n  revision = require('./revision');\n\n  addToJournal = require('./addToJournal');\n\n  newPage = require('./page').newPage;\n\n  module.exports = pageHandler = {};\n\n  pageFromLocalStorage = function(slug) {\n    var json;\n    if (json = localStorage[slug]) {\n      return JSON.parse(json);\n    } else {\n      return void 0;\n    }\n  };\n\n  recursiveGet = function(_arg) {\n    var localContext, localPage, pageInformation, rev, site, slug, url, whenGotten, whenNotGotten;\n    pageInformation = _arg.pageInformation, whenGotten = _arg.whenGotten, whenNotGotten = _arg.whenNotGotten, localContext = _arg.localContext;\n    slug = pageInformation.slug, rev = pageInformation.rev, site = pageInformation.site;\n    if (site) {\n      localContext = [];\n    } else {\n      site = localContext.shift();\n    }\n    if (site === window.location.host) {\n      site = 'origin';\n    }\n    if (site === 'view') {\n      site = null;\n    }\n    if (site != null) {\n      if (site === 'local') {\n        if (localPage = pageFromLocalStorage(pageInformation.slug)) {\n          return whenGotten(newPage(localPage, 'local'));\n        } else {\n          return whenNotGotten();\n        }\n      } else {\n        if (site === 'origin') {\n          url = \"/\" + slug + \".json\";\n        } else {\n          url = \"http://\" + site + \"/\" + slug + \".json\";\n        }\n      }\n    } else {\n      url = \"/\" + slug + \".json\";\n    }\n    return $.ajax({\n      type: 'GET',\n      dataType: 'json',\n      url: url + (\"?random=\" + (util.randomBytes(4))),\n      success: function(page) {\n        if (rev) {\n          page = revision.create(rev, page);\n        }\n        return whenGotten(newPage(page, site));\n      },\n      error: function(xhr, type, msg) {\n        var report;\n        if ((xhr.status !== 404) && (xhr.status !== 0)) {\n          wiki.log('pageHandler.get error', xhr, xhr.status, type, msg);\n          report = {\n            'title': \"\" + xhr.status + \" \" + msg,\n            'story': [\n              {\n                'type': 'paragraph',\n                'id': '928739187243',\n                'text': \"<pre>\" + xhr.responseText\n              }\n            ]\n          };\n          return whenGotten(report, 'local');\n        }\n        if (localContext.length > 0) {\n          return recursiveGet({\n            pageInformation: pageInformation,\n            whenGotten: whenGotten,\n            whenNotGotten: whenNotGotten,\n            localContext: localContext\n          });\n        } else {\n          return whenNotGotten();\n        }\n      }\n    });\n  };\n\n  pageHandler.get = function(_arg) {\n    var localPage, pageInformation, whenGotten, whenNotGotten;\n    whenGotten = _arg.whenGotten, whenNotGotten = _arg.whenNotGotten, pageInformation = _arg.pageInformation;\n    if (!pageInformation.site) {\n      if (localPage = pageFromLocalStorage(pageInformation.slug)) {\n        if (pageInformation.rev) {\n          localPage = revision.create(pageInformation.rev, localPage);\n        }\n        return whenGotten(newPage(localPage, 'local'));\n      }\n    }\n    if (!pageHandler.context.length) {\n      pageHandler.context = ['view'];\n    }\n    return recursiveGet({\n      pageInformation: pageInformation,\n      whenGotten: whenGotten,\n      whenNotGotten: whenNotGotten,\n      localContext: _.clone(pageHandler.context)\n    });\n  };\n\n  pageHandler.context = [];\n\n  pushToLocal = function(pageElement, pagePutInfo, action) {\n    var page, site;\n    if (action.type === 'create') {\n      page = {\n        title: action.item.title,\n        story: [],\n        journal: []\n      };\n    } else {\n      page = pageFromLocalStorage(pagePutInfo.slug);\n      page || (page = pageElement.data(\"data\"));\n      if (page.journal == null) {\n        page.journal = [];\n      }\n      if ((site = action['fork']) != null) {\n        page.journal = page.journal.concat({\n          'type': 'fork',\n          'site': site\n        });\n        delete action['fork'];\n      }\n      page.story = $(pageElement).find(\".item\").map(function() {\n        return $(this).data(\"item\");\n      }).get();\n    }\n    page.journal = page.journal.concat(action);\n    localStorage[pagePutInfo.slug] = JSON.stringify(page);\n    return addToJournal(pageElement.find('.journal'), action);\n  };\n\n  pushToServer = function(pageElement, pagePutInfo, action) {\n    return $.ajax({\n      type: 'PUT',\n      url: \"/page/\" + pagePutInfo.slug + \"/action\",\n      data: {\n        'action': JSON.stringify(action)\n      },\n      success: function() {\n        addToJournal(pageElement.find('.journal'), action);\n        if (action.type === 'fork') {\n          localStorage.removeItem(pageElement.attr('id'));\n          return state.setUrl;\n        }\n      },\n      error: function(xhr, type, msg) {\n        return wiki.log(\"pageHandler.put ajax error callback\", type, msg);\n      }\n    });\n  };\n\n  pageHandler.put = function(pageElement, action) {\n    var checkedSite, forkFrom, pagePutInfo;\n    checkedSite = function() {\n      var site;\n      switch (site = pageElement.data('site')) {\n        case 'origin':\n        case 'local':\n        case 'view':\n          return null;\n        case location.host:\n          return null;\n        default:\n          return site;\n      }\n    };\n    pagePutInfo = {\n      slug: pageElement.attr('id').split('_rev')[0],\n      rev: pageElement.attr('id').split('_rev')[1],\n      site: checkedSite(),\n      local: pageElement.hasClass('local')\n    };\n    forkFrom = pagePutInfo.site;\n    wiki.log('pageHandler.put', action, pagePutInfo);\n    if (wiki.useLocalStorage()) {\n      if (pagePutInfo.site != null) {\n        wiki.log('remote => local');\n      } else if (!pagePutInfo.local) {\n        wiki.log('origin => local');\n        action.site = forkFrom = location.host;\n      }\n    }\n    action.date = (new Date()).getTime();\n    if (action.site === 'origin') {\n      delete action.site;\n    }\n    if (forkFrom) {\n      pageElement.find('h1 img').attr('src', '/favicon.png');\n      pageElement.find('h1 a').attr('href', '/');\n      pageElement.data('site', null);\n      pageElement.removeClass('remote');\n      state.setUrl();\n      if (action.type !== 'fork') {\n        action.fork = forkFrom;\n        addToJournal(pageElement.find('.journal'), {\n          type: 'fork',\n          site: forkFrom,\n          date: action.date\n        });\n      }\n    }\n    if (wiki.useLocalStorage() || pagePutInfo.site === 'local') {\n      pushToLocal(pageElement, pagePutInfo, action);\n      return pageElement.addClass(\"local\");\n    } else {\n      return pushToServer(pageElement, pagePutInfo, action);\n    }\n  };\n\n}).call(this);\n","(function() {\n  var addToJournal, buildPageHeader, createFactory, emitHeader, emitTwins, handleDragging, initAddButton, initDragging, neighborhood, pageHandler, plugin, refresh, renderPageIntoPageElement, state, util, wiki, _;\n\n  _ = require('underscore');\n\n  util = require('./util');\n\n  pageHandler = require('./pageHandler');\n\n  plugin = require('./plugin');\n\n  state = require('./state');\n\n  neighborhood = require('./neighborhood');\n\n  addToJournal = require('./addToJournal');\n\n  wiki = require('./wiki');\n\n  handleDragging = function(evt, ui) {\n    var action, before, beforeElement, destinationPageElement, equals, item, itemElement, moveFromPage, moveToPage, moveWithinPage, order, sourcePageElement, sourceSite, thisPageElement;\n    itemElement = ui.item;\n    item = wiki.getItem(itemElement);\n    thisPageElement = $(this).parents('.page:first');\n    sourcePageElement = itemElement.data('pageElement');\n    sourceSite = sourcePageElement.data('site');\n    destinationPageElement = itemElement.parents('.page:first');\n    equals = function(a, b) {\n      return a && b && a.get(0) === b.get(0);\n    };\n    moveWithinPage = !sourcePageElement || equals(sourcePageElement, destinationPageElement);\n    moveFromPage = !moveWithinPage && equals(thisPageElement, sourcePageElement);\n    moveToPage = !moveWithinPage && equals(thisPageElement, destinationPageElement);\n    if (moveFromPage) {\n      if (sourcePageElement.hasClass('ghost') || sourcePageElement.attr('id') === destinationPageElement.attr('id')) {\n        return;\n      }\n    }\n    action = moveWithinPage ? (order = $(this).children().map(function(_, value) {\n      return $(value).attr('data-id');\n    }).get(), {\n      type: 'move',\n      order: order\n    }) : moveFromPage ? (wiki.log('drag from', sourcePageElement.find('h1').text()), {\n      type: 'remove'\n    }) : moveToPage ? (itemElement.data('pageElement', thisPageElement), beforeElement = itemElement.prev('.item'), before = wiki.getItem(beforeElement), {\n      type: 'add',\n      item: item,\n      after: before != null ? before.id : void 0\n    }) : void 0;\n    action.id = item.id;\n    return pageHandler.put(thisPageElement, action);\n  };\n\n  initDragging = function($page) {\n    var $story;\n    $story = $page.find('.story');\n    return $story.sortable({\n      connectWith: '.page .story'\n    }).on(\"sortupdate\", handleDragging);\n  };\n\n  initAddButton = function($page) {\n    return $page.find(\".add-factory\").live(\"click\", function(evt) {\n      if ($page.hasClass('ghost')) {\n        return;\n      }\n      evt.preventDefault();\n      return createFactory($page);\n    });\n  };\n\n  createFactory = function($page) {\n    var before, beforeElement, item, itemElement;\n    item = {\n      type: \"factory\",\n      id: util.randomBytes(8)\n    };\n    itemElement = $(\"<div />\", {\n      \"class\": \"item factory\"\n    }).data('item', item).attr('data-id', item.id);\n    itemElement.data('pageElement', $page);\n    $page.find(\".story\").append(itemElement);\n    plugin[\"do\"](itemElement, item);\n    beforeElement = itemElement.prev('.item');\n    before = wiki.getItem(beforeElement);\n    return pageHandler.put($page, {\n      item: item,\n      id: item.id,\n      type: \"add\",\n      after: before != null ? before.id : void 0\n    });\n  };\n\n  buildPageHeader = function(_arg) {\n    var favicon_src, header_href, page, tooltip;\n    page = _arg.page, tooltip = _arg.tooltip, header_href = _arg.header_href, favicon_src = _arg.favicon_src;\n    if (page.plugin) {\n      tooltip += \"\\n\" + page.plugin + \" plugin\";\n    }\n    return \"<h1 title=\\\"\" + tooltip + \"\\\"><a href=\\\"\" + header_href + \"\\\"><img src=\\\"\" + favicon_src + \"\\\" height=\\\"32px\\\" class=\\\"favicon\\\"></a> \" + page.title + \"</h1>\";\n  };\n\n  emitHeader = function($header, $page, pageObject) {\n    var date, header, isRemotePage, page, pageHeader, rev, viewHere;\n    page = pageObject.getRawPage();\n    isRemotePage = pageObject.isRemote();\n    header = '';\n    viewHere = wiki.asSlug(page.title) === 'welcome-visitors' ? \"\" : \"/view/\" + (pageObject.getSlug());\n    pageHeader = isRemotePage ? buildPageHeader({\n      tooltip: pageObject.getRemoteSite(),\n      header_href: \"//\" + (pageObject.getRemoteSite()) + \"/view/welcome-visitors\" + viewHere,\n      favicon_src: \"http://\" + (pageObject.getRemoteSite()) + \"/favicon.png\",\n      page: page\n    }) : buildPageHeader({\n      tooltip: location.host,\n      header_href: \"/view/welcome-visitors\" + viewHere,\n      favicon_src: \"/favicon.png\",\n      page: page\n    });\n    $header.append(pageHeader);\n    if (!isRemotePage) {\n      $('img.favicon', $page).error(function(e) {\n        return plugin.get('favicon', function(favicon) {\n          return favicon.create();\n        });\n      });\n    }\n    if ($page.attr('id').match(/_rev/)) {\n      rev = page.journal.length - 1;\n      date = page.journal[rev].date;\n      $page.addClass('ghost').data('rev', rev);\n      return $header.append($(\"<h2 class=\\\"revision\\\">\\n  <span>\\n    \" + (date != null ? util.formatDate(date) : \"Revision \" + rev) + \"\\n  </span>\\n</h2>\"));\n    }\n  };\n\n  emitTwins = wiki.emitTwins = function($page) {\n    var actions, bin, bins, flags, i, info, item, legend, page, remoteSite, site, slug, twins, viewing, _i, _len, _ref, _ref1, _ref2, _ref3;\n    page = $page.data('data');\n    site = $page.data('site') || window.location.host;\n    if (site === 'view' || site === 'origin') {\n      site = window.location.host;\n    }\n    slug = wiki.asSlug(page.title);\n    if (((actions = (_ref = page.journal) != null ? _ref.length : void 0) != null) && ((viewing = (_ref1 = page.journal[actions - 1]) != null ? _ref1.date : void 0) != null)) {\n      viewing = Math.floor(viewing / 1000) * 1000;\n      bins = {\n        newer: [],\n        same: [],\n        older: []\n      };\n      _ref2 = wiki.neighborhood;\n      for (remoteSite in _ref2) {\n        info = _ref2[remoteSite];\n        if (remoteSite !== site && (info.sitemap != null)) {\n          _ref3 = info.sitemap;\n          for (_i = 0, _len = _ref3.length; _i < _len; _i++) {\n            item = _ref3[_i];\n            if (item.slug === slug) {\n              bin = item.date > viewing ? bins.newer : item.date < viewing ? bins.older : bins.same;\n              bin.push({\n                remoteSite: remoteSite,\n                item: item\n              });\n            }\n          }\n        }\n      }\n      twins = [];\n      for (legend in bins) {\n        bin = bins[legend];\n        if (!bin.length) {\n          continue;\n        }\n        bin.sort(function(a, b) {\n          return a.item.date < b.item.date;\n        });\n        flags = (function() {\n          var _j, _len1, _ref4, _results;\n          _results = [];\n          for (i = _j = 0, _len1 = bin.length; _j < _len1; i = ++_j) {\n            _ref4 = bin[i], remoteSite = _ref4.remoteSite, item = _ref4.item;\n            if (i >= 8) {\n              break;\n            }\n            _results.push(\"<img class=\\\"remote\\\"\\nsrc=\\\"http://\" + remoteSite + \"/favicon.png\\\"\\ndata-slug=\\\"\" + slug + \"\\\"\\ndata-site=\\\"\" + remoteSite + \"\\\"\\ntitle=\\\"\" + remoteSite + \"\\\">\");\n          }\n          return _results;\n        })();\n        twins.push(\"\" + (flags.join('&nbsp;')) + \" \" + legend);\n      }\n      if (twins) {\n        return $page.find('.twins').html(\"<p>\" + (twins.join(\", \")) + \"</p>\");\n      }\n    }\n  };\n\n  renderPageIntoPageElement = function(pageObject, $page) {\n    var $footer, $header, $journal, $story, $twins, host, slug, _ref;\n    $page.data(\"data\", pageObject.getRawPage());\n    if (pageObject.isRemote()) {\n      $page.data(\"site\", pageObject.getRemoteSite());\n    }\n    slug = $page.attr('id');\n    wiki.resolutionContext = pageObject.getContext();\n    $page.empty();\n    _ref = ['twins', 'header', 'story', 'journal', 'footer'].map(function(className) {\n      return $(\"<div />\").addClass(className).appendTo($page);\n    }), $twins = _ref[0], $header = _ref[1], $story = _ref[2], $journal = _ref[3], $footer = _ref[4];\n    emitHeader($header, $page, pageObject);\n    pageObject.seqItems(function(item, done) {\n      var $item;\n      if ((item != null ? item.type : void 0) && (item != null ? item.id : void 0)) {\n        $item = $(\"<div class=\\\"item \" + item.type + \"\\\" data-id=\\\"\" + item.id + \"\\\">\");\n        $story.append($item);\n        return plugin[\"do\"]($item, item, done);\n      } else {\n        $story.append($(\"<div><p class=\\\"error\\\">Can't make sense of story[\" + i + \"]</p></div>\"));\n        return done();\n      }\n    });\n    pageObject.seqActions(function(each, done) {\n      if (each.separator) {\n        addToJournal($journal, each.separator);\n      }\n      addToJournal($journal, each.action);\n      return done();\n    });\n    emitTwins($page);\n    $journal.append(\"<div class=\\\"control-buttons\\\">\\n  <a href=\\\"#\\\" class=\\\"button fork-page\\\" title=\\\"fork this page\\\">\" + util.symbols['fork'] + \"</a>\\n  <a href=\\\"#\\\" class=\\\"button add-factory\\\" title=\\\"add paragraph\\\">\" + util.symbols['add'] + \"</a>\\n</div>\");\n    host = pageObject.getRemoteSite() || location.host;\n    return $footer.append(\"<a id=\\\"license\\\" href=\\\"http://creativecommons.org/licenses/by-sa/3.0/\\\">CC BY-SA 3.0</a> .\\n<a class=\\\"show-page-source\\\" href=\\\"/\" + slug + \".json?random=\" + (util.randomBytes(4)) + \"\\\" title=\\\"source\\\">JSON</a> .\\n<a href= \\\"//\" + host + \"/\" + slug + \".html\\\">\" + host + \"</a>\");\n  };\n\n  wiki.buildPage = function(pageObject, $page) {\n    if (pageObject.isLocal()) {\n      $page.addClass('local');\n    }\n    if (pageObject.isRemote()) {\n      $page.addClass('remote');\n    }\n    if (pageObject.isPlugin()) {\n      $page.addClass('plugin');\n    }\n    renderPageIntoPageElement(pageObject, $page);\n    state.setUrl();\n    initDragging($page);\n    initAddButton($page);\n    return $page;\n  };\n\n  module.exports = refresh = wiki.refresh = function() {\n    var $page, createGhostPage, emptyPage, pageInformation, rev, slug, whenGotten, _ref;\n    $page = $(this);\n    _ref = $page.attr('id').split('_rev'), slug = _ref[0], rev = _ref[1];\n    pageInformation = {\n      slug: slug,\n      rev: rev,\n      site: $page.data('site')\n    };\n    emptyPage = require('./page').emptyPage;\n    createGhostPage = function() {\n      var hit, hits, info, pageObject, result, site, title, _i, _len, _ref1;\n      title = $(\"a[href=\\\"/\" + slug + \".html\\\"]:last\").text() || slug;\n      pageObject = emptyPage();\n      pageObject.setTitle(title);\n      hits = [];\n      _ref1 = wiki.neighborhood;\n      for (site in _ref1) {\n        info = _ref1[site];\n        if (info.sitemap != null) {\n          result = _.find(info.sitemap, function(each) {\n            return each.slug === slug;\n          });\n          if (result != null) {\n            hits.push({\n              \"type\": \"reference\",\n              \"site\": site,\n              \"slug\": slug,\n              \"title\": result.title || slug,\n              \"text\": result.synopsis || ''\n            });\n          }\n        }\n      }\n      if (hits.length > 0) {\n        pageObject.addItem({\n          'type': 'future',\n          'text': 'We could not find this page in the expected context.',\n          'title': title\n        });\n        pageObject.addItem({\n          'type': 'paragraph',\n          'text': \"We did find the page in your current neighborhood.\"\n        });\n        for (_i = 0, _len = hits.length; _i < _len; _i++) {\n          hit = hits[_i];\n          pageObject.addItem(hit);\n        }\n      } else {\n        pageObject.addItem({\n          'type': 'future',\n          'text': 'We could not find this page.',\n          'title': title\n        });\n      }\n      return wiki.buildPage(pageObject, $page).addClass('ghost');\n    };\n    whenGotten = function(pageObject) {\n      var site, _i, _len, _ref1, _results;\n      wiki.buildPage(pageObject, $page);\n      _ref1 = pageObject.getNeighbors(location.host);\n      _results = [];\n      for (_i = 0, _len = _ref1.length; _i < _len; _i++) {\n        site = _ref1[_i];\n        _results.push(neighborhood.registerNeighbor(site));\n      }\n      return _results;\n    };\n    return pageHandler.get({\n      whenGotten: whenGotten,\n      whenNotGotten: createGhostPage,\n      pageInformation: pageInformation\n    });\n  };\n\n}).call(this);\n","(function() {\n  var asSlug, emptyPage, newPage, nowSections, util, _;\n\n  util = require('./util');\n\n  _ = require('underscore');\n\n  asSlug = function(name) {\n    return name.replace(/\\s/g, '-').replace(/[^A-Za-z0-9-]/g, '').toLowerCase();\n  };\n\n  emptyPage = function() {\n    return newPage({}, null);\n  };\n\n  nowSections = function(now) {\n    return [\n      {\n        symbol: '❄',\n        date: now - 1000 * 60 * 60 * 24 * 366,\n        period: 'a Year'\n      }, {\n        symbol: '⚘',\n        date: now - 1000 * 60 * 60 * 24 * 31 * 3,\n        period: 'a Season'\n      }, {\n        symbol: '⚪',\n        date: now - 1000 * 60 * 60 * 24 * 31,\n        period: 'a Month'\n      }, {\n        symbol: '☽',\n        date: now - 1000 * 60 * 60 * 24 * 7,\n        period: 'a Week'\n      }, {\n        symbol: '☀',\n        date: now - 1000 * 60 * 60 * 24,\n        period: 'a Day'\n      }, {\n        symbol: '⌚',\n        date: now - 1000 * 60 * 60,\n        period: 'an Hour'\n      }\n    ];\n  };\n\n  newPage = function(json, site) {\n    var addItem, addParagraph, getContext, getNeighbors, getRawPage, getRemoteSite, getSlug, isLocal, isPlugin, isRemote, page, seqActions, seqItems, setTitle;\n    page = _.extend({}, util.emptyPage(), json);\n    page.story || (page.story = []);\n    page.journal || (page.journal = []);\n    getRawPage = function() {\n      return page;\n    };\n    getContext = function() {\n      var action, addContext, context, _i, _len, _ref;\n      context = ['view'];\n      if (isRemote()) {\n        context.push(site);\n      }\n      addContext = function(site) {\n        if ((site != null) && !_.include(context, site)) {\n          return context.push(site);\n        }\n      };\n      _ref = page.journal.slice(0).reverse();\n      for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n        action = _ref[_i];\n        addContext(action.site);\n      }\n      return context;\n    };\n    isPlugin = function() {\n      return page.plugin != null;\n    };\n    isRemote = function() {\n      return !(site === (void 0) || site === null || site === 'view' || site === 'origin' || site === 'local');\n    };\n    isLocal = function() {\n      return site === 'local';\n    };\n    getRemoteSite = function() {\n      if (isRemote()) {\n        return site;\n      }\n      return null;\n    };\n    getSlug = function() {\n      return asSlug(page.title);\n    };\n    getNeighbors = function(host) {\n      var action, item, neighbors, _i, _j, _len, _len1, _ref, _ref1;\n      neighbors = [];\n      if (isRemote()) {\n        neighbors.push(site);\n      } else {\n        if (host != null) {\n          neighbors.push(host);\n        }\n      }\n      _ref = page.story;\n      for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n        item = _ref[_i];\n        if (item.site != null) {\n          neighbors.push(item.site);\n        }\n      }\n      _ref1 = page.journal;\n      for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {\n        action = _ref1[_j];\n        if (action.site != null) {\n          neighbors.push(action.site);\n        }\n      }\n      return _.uniq(neighbors);\n    };\n    setTitle = function(title) {\n      return page.title = title;\n    };\n    addItem = function(item) {\n      item = _.extend({}, {\n        id: util.randomBytes(8)\n      }, item);\n      return page.story.push(item);\n    };\n    seqItems = function(each) {\n      var emitItem;\n      emitItem = function(i) {\n        if (i >= page.story.length) {\n          return;\n        }\n        return each(page.story[i], function() {\n          return emitItem(i + 1);\n        });\n      };\n      return emitItem(0);\n    };\n    addParagraph = function(text) {\n      var type;\n      type = \"paragraph\";\n      return addItem({\n        type: type,\n        text: text\n      });\n    };\n    seqActions = function(each) {\n      var emitAction, sections, smaller;\n      smaller = 0;\n      sections = nowSections((new Date).getTime());\n      emitAction = function(i) {\n        var action, bigger, section, separator, _i, _len;\n        if (i >= page.journal.length) {\n          return;\n        }\n        action = page.journal[i];\n        bigger = action.date || 0;\n        separator = null;\n        for (_i = 0, _len = sections.length; _i < _len; _i++) {\n          section = sections[_i];\n          if (section.date > smaller && section.date < bigger) {\n            separator = section;\n          }\n        }\n        smaller = bigger;\n        return each({\n          action: action,\n          separator: separator\n        }, function() {\n          return emitAction(i + 1);\n        });\n      };\n      return emitAction(0);\n    };\n    return {\n      getRawPage: getRawPage,\n      getContext: getContext,\n      isPlugin: isPlugin,\n      isRemote: isRemote,\n      isLocal: isLocal,\n      getRemoteSite: getRemoteSite,\n      getSlug: getSlug,\n      getNeighbors: getNeighbors,\n      setTitle: setTitle,\n      addItem: addItem,\n      addParagraph: addParagraph,\n      seqItems: seqItems,\n      seqActions: seqActions\n    };\n  };\n\n  module.exports = {\n    newPage: newPage,\n    emptyPage: emptyPage\n  };\n\n}).call(this);\n","(function(){//     Underscore.js 1.5.2\n//     http://underscorejs.org\n//     (c) 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors\n//     Underscore may be freely distributed under the MIT license.\n\n(function() {\n\n  // Baseline setup\n  // --------------\n\n  // Establish the root object, `window` in the browser, or `exports` on the server.\n  var root = this;\n\n  // Save the previous value of the `_` variable.\n  var previousUnderscore = root._;\n\n  // Establish the object that gets returned to break out of a loop iteration.\n  var breaker = {};\n\n  // Save bytes in the minified (but not gzipped) version:\n  var ArrayProto = Array.prototype, ObjProto = Object.prototype, FuncProto = Function.prototype;\n\n  // Create quick reference variables for speed access to core prototypes.\n  var\n    push             = ArrayProto.push,\n    slice            = ArrayProto.slice,\n    concat           = ArrayProto.concat,\n    toString         = ObjProto.toString,\n    hasOwnProperty   = ObjProto.hasOwnProperty;\n\n  // All **ECMAScript 5** native function implementations that we hope to use\n  // are declared here.\n  var\n    nativeForEach      = ArrayProto.forEach,\n    nativeMap          = ArrayProto.map,\n    nativeReduce       = ArrayProto.reduce,\n    nativeReduceRight  = ArrayProto.reduceRight,\n    nativeFilter       = ArrayProto.filter,\n    nativeEvery        = ArrayProto.every,\n    nativeSome         = ArrayProto.some,\n    nativeIndexOf      = ArrayProto.indexOf,\n    nativeLastIndexOf  = ArrayProto.lastIndexOf,\n    nativeIsArray      = Array.isArray,\n    nativeKeys         = Object.keys,\n    nativeBind         = FuncProto.bind;\n\n  // Create a safe reference to the Underscore object for use below.\n  var _ = function(obj) {\n    if (obj instanceof _) return obj;\n    if (!(this instanceof _)) return new _(obj);\n    this._wrapped = obj;\n  };\n\n  // Export the Underscore object for **Node.js**, with\n  // backwards-compatibility for the old `require()` API. If we're in\n  // the browser, add `_` as a global object via a string identifier,\n  // for Closure Compiler \"advanced\" mode.\n  if (typeof exports !== 'undefined') {\n    if (typeof module !== 'undefined' && module.exports) {\n      exports = module.exports = _;\n    }\n    exports._ = _;\n  } else {\n    root._ = _;\n  }\n\n  // Current version.\n  _.VERSION = '1.5.2';\n\n  // Collection Functions\n  // --------------------\n\n  // The cornerstone, an `each` implementation, aka `forEach`.\n  // Handles objects with the built-in `forEach`, arrays, and raw objects.\n  // Delegates to **ECMAScript 5**'s native `forEach` if available.\n  var each = _.each = _.forEach = function(obj, iterator, context) {\n    if (obj == null) return;\n    if (nativeForEach && obj.forEach === nativeForEach) {\n      obj.forEach(iterator, context);\n    } else if (obj.length === +obj.length) {\n      for (var i = 0, length = obj.length; i < length; i++) {\n        if (iterator.call(context, obj[i], i, obj) === breaker) return;\n      }\n    } else {\n      var keys = _.keys(obj);\n      for (var i = 0, length = keys.length; i < length; i++) {\n        if (iterator.call(context, obj[keys[i]], keys[i], obj) === breaker) return;\n      }\n    }\n  };\n\n  // Return the results of applying the iterator to each element.\n  // Delegates to **ECMAScript 5**'s native `map` if available.\n  _.map = _.collect = function(obj, iterator, context) {\n    var results = [];\n    if (obj == null) return results;\n    if (nativeMap && obj.map === nativeMap) return obj.map(iterator, context);\n    each(obj, function(value, index, list) {\n      results.push(iterator.call(context, value, index, list));\n    });\n    return results;\n  };\n\n  var reduceError = 'Reduce of empty array with no initial value';\n\n  // **Reduce** builds up a single result from a list of values, aka `inject`,\n  // or `foldl`. Delegates to **ECMAScript 5**'s native `reduce` if available.\n  _.reduce = _.foldl = _.inject = function(obj, iterator, memo, context) {\n    var initial = arguments.length > 2;\n    if (obj == null) obj = [];\n    if (nativeReduce && obj.reduce === nativeReduce) {\n      if (context) iterator = _.bind(iterator, context);\n      return initial ? obj.reduce(iterator, memo) : obj.reduce(iterator);\n    }\n    each(obj, function(value, index, list) {\n      if (!initial) {\n        memo = value;\n        initial = true;\n      } else {\n        memo = iterator.call(context, memo, value, index, list);\n      }\n    });\n    if (!initial) throw new TypeError(reduceError);\n    return memo;\n  };\n\n  // The right-associative version of reduce, also known as `foldr`.\n  // Delegates to **ECMAScript 5**'s native `reduceRight` if available.\n  _.reduceRight = _.foldr = function(obj, iterator, memo, context) {\n    var initial = arguments.length > 2;\n    if (obj == null) obj = [];\n    if (nativeReduceRight && obj.reduceRight === nativeReduceRight) {\n      if (context) iterator = _.bind(iterator, context);\n      return initial ? obj.reduceRight(iterator, memo) : obj.reduceRight(iterator);\n    }\n    var length = obj.length;\n    if (length !== +length) {\n      var keys = _.keys(obj);\n      length = keys.length;\n    }\n    each(obj, function(value, index, list) {\n      index = keys ? keys[--length] : --length;\n      if (!initial) {\n        memo = obj[index];\n        initial = true;\n      } else {\n        memo = iterator.call(context, memo, obj[index], index, list);\n      }\n    });\n    if (!initial) throw new TypeError(reduceError);\n    return memo;\n  };\n\n  // Return the first value which passes a truth test. Aliased as `detect`.\n  _.find = _.detect = function(obj, iterator, context) {\n    var result;\n    any(obj, function(value, index, list) {\n      if (iterator.call(context, value, index, list)) {\n        result = value;\n        return true;\n      }\n    });\n    return result;\n  };\n\n  // Return all the elements that pass a truth test.\n  // Delegates to **ECMAScript 5**'s native `filter` if available.\n  // Aliased as `select`.\n  _.filter = _.select = function(obj, iterator, context) {\n    var results = [];\n    if (obj == null) return results;\n    if (nativeFilter && obj.filter === nativeFilter) return obj.filter(iterator, context);\n    each(obj, function(value, index, list) {\n      if (iterator.call(context, value, index, list)) results.push(value);\n    });\n    return results;\n  };\n\n  // Return all the elements for which a truth test fails.\n  _.reject = function(obj, iterator, context) {\n    return _.filter(obj, function(value, index, list) {\n      return !iterator.call(context, value, index, list);\n    }, context);\n  };\n\n  // Determine whether all of the elements match a truth test.\n  // Delegates to **ECMAScript 5**'s native `every` if available.\n  // Aliased as `all`.\n  _.every = _.all = function(obj, iterator, context) {\n    iterator || (iterator = _.identity);\n    var result = true;\n    if (obj == null) return result;\n    if (nativeEvery && obj.every === nativeEvery) return obj.every(iterator, context);\n    each(obj, function(value, index, list) {\n      if (!(result = result && iterator.call(context, value, index, list))) return breaker;\n    });\n    return !!result;\n  };\n\n  // Determine if at least one element in the object matches a truth test.\n  // Delegates to **ECMAScript 5**'s native `some` if available.\n  // Aliased as `any`.\n  var any = _.some = _.any = function(obj, iterator, context) {\n    iterator || (iterator = _.identity);\n    var result = false;\n    if (obj == null) return result;\n    if (nativeSome && obj.some === nativeSome) return obj.some(iterator, context);\n    each(obj, function(value, index, list) {\n      if (result || (result = iterator.call(context, value, index, list))) return breaker;\n    });\n    return !!result;\n  };\n\n  // Determine if the array or object contains a given value (using `===`).\n  // Aliased as `include`.\n  _.contains = _.include = function(obj, target) {\n    if (obj == null) return false;\n    if (nativeIndexOf && obj.indexOf === nativeIndexOf) return obj.indexOf(target) != -1;\n    return any(obj, function(value) {\n      return value === target;\n    });\n  };\n\n  // Invoke a method (with arguments) on every item in a collection.\n  _.invoke = function(obj, method) {\n    var args = slice.call(arguments, 2);\n    var isFunc = _.isFunction(method);\n    return _.map(obj, function(value) {\n      return (isFunc ? method : value[method]).apply(value, args);\n    });\n  };\n\n  // Convenience version of a common use case of `map`: fetching a property.\n  _.pluck = function(obj, key) {\n    return _.map(obj, function(value){ return value[key]; });\n  };\n\n  // Convenience version of a common use case of `filter`: selecting only objects\n  // containing specific `key:value` pairs.\n  _.where = function(obj, attrs, first) {\n    if (_.isEmpty(attrs)) return first ? void 0 : [];\n    return _[first ? 'find' : 'filter'](obj, function(value) {\n      for (var key in attrs) {\n        if (attrs[key] !== value[key]) return false;\n      }\n      return true;\n    });\n  };\n\n  // Convenience version of a common use case of `find`: getting the first object\n  // containing specific `key:value` pairs.\n  _.findWhere = function(obj, attrs) {\n    return _.where(obj, attrs, true);\n  };\n\n  // Return the maximum element or (element-based computation).\n  // Can't optimize arrays of integers longer than 65,535 elements.\n  // See [WebKit Bug 80797](https://bugs.webkit.org/show_bug.cgi?id=80797)\n  _.max = function(obj, iterator, context) {\n    if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {\n      return Math.max.apply(Math, obj);\n    }\n    if (!iterator && _.isEmpty(obj)) return -Infinity;\n    var result = {computed : -Infinity, value: -Infinity};\n    each(obj, function(value, index, list) {\n      var computed = iterator ? iterator.call(context, value, index, list) : value;\n      computed > result.computed && (result = {value : value, computed : computed});\n    });\n    return result.value;\n  };\n\n  // Return the minimum element (or element-based computation).\n  _.min = function(obj, iterator, context) {\n    if (!iterator && _.isArray(obj) && obj[0] === +obj[0] && obj.length < 65535) {\n      return Math.min.apply(Math, obj);\n    }\n    if (!iterator && _.isEmpty(obj)) return Infinity;\n    var result = {computed : Infinity, value: Infinity};\n    each(obj, function(value, index, list) {\n      var computed = iterator ? iterator.call(context, value, index, list) : value;\n      computed < result.computed && (result = {value : value, computed : computed});\n    });\n    return result.value;\n  };\n\n  // Shuffle an array, using the modern version of the \n  // [Fisher-Yates shuffle](http://en.wikipedia.org/wiki/Fisher–Yates_shuffle).\n  _.shuffle = function(obj) {\n    var rand;\n    var index = 0;\n    var shuffled = [];\n    each(obj, function(value) {\n      rand = _.random(index++);\n      shuffled[index - 1] = shuffled[rand];\n      shuffled[rand] = value;\n    });\n    return shuffled;\n  };\n\n  // Sample **n** random values from an array.\n  // If **n** is not specified, returns a single random element from the array.\n  // The internal `guard` argument allows it to work with `map`.\n  _.sample = function(obj, n, guard) {\n    if (arguments.length < 2 || guard) {\n      return obj[_.random(obj.length - 1)];\n    }\n    return _.shuffle(obj).slice(0, Math.max(0, n));\n  };\n\n  // An internal function to generate lookup iterators.\n  var lookupIterator = function(value) {\n    return _.isFunction(value) ? value : function(obj){ return obj[value]; };\n  };\n\n  // Sort the object's values by a criterion produced by an iterator.\n  _.sortBy = function(obj, value, context) {\n    var iterator = lookupIterator(value);\n    return _.pluck(_.map(obj, function(value, index, list) {\n      return {\n        value: value,\n        index: index,\n        criteria: iterator.call(context, value, index, list)\n      };\n    }).sort(function(left, right) {\n      var a = left.criteria;\n      var b = right.criteria;\n      if (a !== b) {\n        if (a > b || a === void 0) return 1;\n        if (a < b || b === void 0) return -1;\n      }\n      return left.index - right.index;\n    }), 'value');\n  };\n\n  // An internal function used for aggregate \"group by\" operations.\n  var group = function(behavior) {\n    return function(obj, value, context) {\n      var result = {};\n      var iterator = value == null ? _.identity : lookupIterator(value);\n      each(obj, function(value, index) {\n        var key = iterator.call(context, value, index, obj);\n        behavior(result, key, value);\n      });\n      return result;\n    };\n  };\n\n  // Groups the object's values by a criterion. Pass either a string attribute\n  // to group by, or a function that returns the criterion.\n  _.groupBy = group(function(result, key, value) {\n    (_.has(result, key) ? result[key] : (result[key] = [])).push(value);\n  });\n\n  // Indexes the object's values by a criterion, similar to `groupBy`, but for\n  // when you know that your index values will be unique.\n  _.indexBy = group(function(result, key, value) {\n    result[key] = value;\n  });\n\n  // Counts instances of an object that group by a certain criterion. Pass\n  // either a string attribute to count by, or a function that returns the\n  // criterion.\n  _.countBy = group(function(result, key) {\n    _.has(result, key) ? result[key]++ : result[key] = 1;\n  });\n\n  // Use a comparator function to figure out the smallest index at which\n  // an object should be inserted so as to maintain order. Uses binary search.\n  _.sortedIndex = function(array, obj, iterator, context) {\n    iterator = iterator == null ? _.identity : lookupIterator(iterator);\n    var value = iterator.call(context, obj);\n    var low = 0, high = array.length;\n    while (low < high) {\n      var mid = (low + high) >>> 1;\n      iterator.call(context, array[mid]) < value ? low = mid + 1 : high = mid;\n    }\n    return low;\n  };\n\n  // Safely create a real, live array from anything iterable.\n  _.toArray = function(obj) {\n    if (!obj) return [];\n    if (_.isArray(obj)) return slice.call(obj);\n    if (obj.length === +obj.length) return _.map(obj, _.identity);\n    return _.values(obj);\n  };\n\n  // Return the number of elements in an object.\n  _.size = function(obj) {\n    if (obj == null) return 0;\n    return (obj.length === +obj.length) ? obj.length : _.keys(obj).length;\n  };\n\n  // Array Functions\n  // ---------------\n\n  // Get the first element of an array. Passing **n** will return the first N\n  // values in the array. Aliased as `head` and `take`. The **guard** check\n  // allows it to work with `_.map`.\n  _.first = _.head = _.take = function(array, n, guard) {\n    if (array == null) return void 0;\n    return (n == null) || guard ? array[0] : slice.call(array, 0, n);\n  };\n\n  // Returns everything but the last entry of the array. Especially useful on\n  // the arguments object. Passing **n** will return all the values in\n  // the array, excluding the last N. The **guard** check allows it to work with\n  // `_.map`.\n  _.initial = function(array, n, guard) {\n    return slice.call(array, 0, array.length - ((n == null) || guard ? 1 : n));\n  };\n\n  // Get the last element of an array. Passing **n** will return the last N\n  // values in the array. The **guard** check allows it to work with `_.map`.\n  _.last = function(array, n, guard) {\n    if (array == null) return void 0;\n    if ((n == null) || guard) {\n      return array[array.length - 1];\n    } else {\n      return slice.call(array, Math.max(array.length - n, 0));\n    }\n  };\n\n  // Returns everything but the first entry of the array. Aliased as `tail` and `drop`.\n  // Especially useful on the arguments object. Passing an **n** will return\n  // the rest N values in the array. The **guard**\n  // check allows it to work with `_.map`.\n  _.rest = _.tail = _.drop = function(array, n, guard) {\n    return slice.call(array, (n == null) || guard ? 1 : n);\n  };\n\n  // Trim out all falsy values from an array.\n  _.compact = function(array) {\n    return _.filter(array, _.identity);\n  };\n\n  // Internal implementation of a recursive `flatten` function.\n  var flatten = function(input, shallow, output) {\n    if (shallow && _.every(input, _.isArray)) {\n      return concat.apply(output, input);\n    }\n    each(input, function(value) {\n      if (_.isArray(value) || _.isArguments(value)) {\n        shallow ? push.apply(output, value) : flatten(value, shallow, output);\n      } else {\n        output.push(value);\n      }\n    });\n    return output;\n  };\n\n  // Flatten out an array, either recursively (by default), or just one level.\n  _.flatten = function(array, shallow) {\n    return flatten(array, shallow, []);\n  };\n\n  // Return a version of the array that does not contain the specified value(s).\n  _.without = function(array) {\n    return _.difference(array, slice.call(arguments, 1));\n  };\n\n  // Produce a duplicate-free version of the array. If the array has already\n  // been sorted, you have the option of using a faster algorithm.\n  // Aliased as `unique`.\n  _.uniq = _.unique = function(array, isSorted, iterator, context) {\n    if (_.isFunction(isSorted)) {\n      context = iterator;\n      iterator = isSorted;\n      isSorted = false;\n    }\n    var initial = iterator ? _.map(array, iterator, context) : array;\n    var results = [];\n    var seen = [];\n    each(initial, function(value, index) {\n      if (isSorted ? (!index || seen[seen.length - 1] !== value) : !_.contains(seen, value)) {\n        seen.push(value);\n        results.push(array[index]);\n      }\n    });\n    return results;\n  };\n\n  // Produce an array that contains the union: each distinct element from all of\n  // the passed-in arrays.\n  _.union = function() {\n    return _.uniq(_.flatten(arguments, true));\n  };\n\n  // Produce an array that contains every item shared between all the\n  // passed-in arrays.\n  _.intersection = function(array) {\n    var rest = slice.call(arguments, 1);\n    return _.filter(_.uniq(array), function(item) {\n      return _.every(rest, function(other) {\n        return _.indexOf(other, item) >= 0;\n      });\n    });\n  };\n\n  // Take the difference between one array and a number of other arrays.\n  // Only the elements present in just the first array will remain.\n  _.difference = function(array) {\n    var rest = concat.apply(ArrayProto, slice.call(arguments, 1));\n    return _.filter(array, function(value){ return !_.contains(rest, value); });\n  };\n\n  // Zip together multiple lists into a single array -- elements that share\n  // an index go together.\n  _.zip = function() {\n    var length = _.max(_.pluck(arguments, \"length\").concat(0));\n    var results = new Array(length);\n    for (var i = 0; i < length; i++) {\n      results[i] = _.pluck(arguments, '' + i);\n    }\n    return results;\n  };\n\n  // Converts lists into objects. Pass either a single array of `[key, value]`\n  // pairs, or two parallel arrays of the same length -- one of keys, and one of\n  // the corresponding values.\n  _.object = function(list, values) {\n    if (list == null) return {};\n    var result = {};\n    for (var i = 0, length = list.length; i < length; i++) {\n      if (values) {\n        result[list[i]] = values[i];\n      } else {\n        result[list[i][0]] = list[i][1];\n      }\n    }\n    return result;\n  };\n\n  // If the browser doesn't supply us with indexOf (I'm looking at you, **MSIE**),\n  // we need this function. Return the position of the first occurrence of an\n  // item in an array, or -1 if the item is not included in the array.\n  // Delegates to **ECMAScript 5**'s native `indexOf` if available.\n  // If the array is large and already in sort order, pass `true`\n  // for **isSorted** to use binary search.\n  _.indexOf = function(array, item, isSorted) {\n    if (array == null) return -1;\n    var i = 0, length = array.length;\n    if (isSorted) {\n      if (typeof isSorted == 'number') {\n        i = (isSorted < 0 ? Math.max(0, length + isSorted) : isSorted);\n      } else {\n        i = _.sortedIndex(array, item);\n        return array[i] === item ? i : -1;\n      }\n    }\n    if (nativeIndexOf && array.indexOf === nativeIndexOf) return array.indexOf(item, isSorted);\n    for (; i < length; i++) if (array[i] === item) return i;\n    return -1;\n  };\n\n  // Delegates to **ECMAScript 5**'s native `lastIndexOf` if available.\n  _.lastIndexOf = function(array, item, from) {\n    if (array == null) return -1;\n    var hasIndex = from != null;\n    if (nativeLastIndexOf && array.lastIndexOf === nativeLastIndexOf) {\n      return hasIndex ? array.lastIndexOf(item, from) : array.lastIndexOf(item);\n    }\n    var i = (hasIndex ? from : array.length);\n    while (i--) if (array[i] === item) return i;\n    return -1;\n  };\n\n  // Generate an integer Array containing an arithmetic progression. A port of\n  // the native Python `range()` function. See\n  // [the Python documentation](http://docs.python.org/library/functions.html#range).\n  _.range = function(start, stop, step) {\n    if (arguments.length <= 1) {\n      stop = start || 0;\n      start = 0;\n    }\n    step = arguments[2] || 1;\n\n    var length = Math.max(Math.ceil((stop - start) / step), 0);\n    var idx = 0;\n    var range = new Array(length);\n\n    while(idx < length) {\n      range[idx++] = start;\n      start += step;\n    }\n\n    return range;\n  };\n\n  // Function (ahem) Functions\n  // ------------------\n\n  // Reusable constructor function for prototype setting.\n  var ctor = function(){};\n\n  // Create a function bound to a given object (assigning `this`, and arguments,\n  // optionally). Delegates to **ECMAScript 5**'s native `Function.bind` if\n  // available.\n  _.bind = function(func, context) {\n    var args, bound;\n    if (nativeBind && func.bind === nativeBind) return nativeBind.apply(func, slice.call(arguments, 1));\n    if (!_.isFunction(func)) throw new TypeError;\n    args = slice.call(arguments, 2);\n    return bound = function() {\n      if (!(this instanceof bound)) return func.apply(context, args.concat(slice.call(arguments)));\n      ctor.prototype = func.prototype;\n      var self = new ctor;\n      ctor.prototype = null;\n      var result = func.apply(self, args.concat(slice.call(arguments)));\n      if (Object(result) === result) return result;\n      return self;\n    };\n  };\n\n  // Partially apply a function by creating a version that has had some of its\n  // arguments pre-filled, without changing its dynamic `this` context.\n  _.partial = function(func) {\n    var args = slice.call(arguments, 1);\n    return function() {\n      return func.apply(this, args.concat(slice.call(arguments)));\n    };\n  };\n\n  // Bind all of an object's methods to that object. Useful for ensuring that\n  // all callbacks defined on an object belong to it.\n  _.bindAll = function(obj) {\n    var funcs = slice.call(arguments, 1);\n    if (funcs.length === 0) throw new Error(\"bindAll must be passed function names\");\n    each(funcs, function(f) { obj[f] = _.bind(obj[f], obj); });\n    return obj;\n  };\n\n  // Memoize an expensive function by storing its results.\n  _.memoize = function(func, hasher) {\n    var memo = {};\n    hasher || (hasher = _.identity);\n    return function() {\n      var key = hasher.apply(this, arguments);\n      return _.has(memo, key) ? memo[key] : (memo[key] = func.apply(this, arguments));\n    };\n  };\n\n  // Delays a function for the given number of milliseconds, and then calls\n  // it with the arguments supplied.\n  _.delay = function(func, wait) {\n    var args = slice.call(arguments, 2);\n    return setTimeout(function(){ return func.apply(null, args); }, wait);\n  };\n\n  // Defers a function, scheduling it to run after the current call stack has\n  // cleared.\n  _.defer = function(func) {\n    return _.delay.apply(_, [func, 1].concat(slice.call(arguments, 1)));\n  };\n\n  // Returns a function, that, when invoked, will only be triggered at most once\n  // during a given window of time. Normally, the throttled function will run\n  // as much as it can, without ever going more than once per `wait` duration;\n  // but if you'd like to disable the execution on the leading edge, pass\n  // `{leading: false}`. To disable execution on the trailing edge, ditto.\n  _.throttle = function(func, wait, options) {\n    var context, args, result;\n    var timeout = null;\n    var previous = 0;\n    options || (options = {});\n    var later = function() {\n      previous = options.leading === false ? 0 : new Date;\n      timeout = null;\n      result = func.apply(context, args);\n    };\n    return function() {\n      var now = new Date;\n      if (!previous && options.leading === false) previous = now;\n      var remaining = wait - (now - previous);\n      context = this;\n      args = arguments;\n      if (remaining <= 0) {\n        clearTimeout(timeout);\n        timeout = null;\n        previous = now;\n        result = func.apply(context, args);\n      } else if (!timeout && options.trailing !== false) {\n        timeout = setTimeout(later, remaining);\n      }\n      return result;\n    };\n  };\n\n  // Returns a function, that, as long as it continues to be invoked, will not\n  // be triggered. The function will be called after it stops being called for\n  // N milliseconds. If `immediate` is passed, trigger the function on the\n  // leading edge, instead of the trailing.\n  _.debounce = function(func, wait, immediate) {\n    var timeout, args, context, timestamp, result;\n    return function() {\n      context = this;\n      args = arguments;\n      timestamp = new Date();\n      var later = function() {\n        var last = (new Date()) - timestamp;\n        if (last < wait) {\n          timeout = setTimeout(later, wait - last);\n        } else {\n          timeout = null;\n          if (!immediate) result = func.apply(context, args);\n        }\n      };\n      var callNow = immediate && !timeout;\n      if (!timeout) {\n        timeout = setTimeout(later, wait);\n      }\n      if (callNow) result = func.apply(context, args);\n      return result;\n    };\n  };\n\n  // Returns a function that will be executed at most one time, no matter how\n  // often you call it. Useful for lazy initialization.\n  _.once = function(func) {\n    var ran = false, memo;\n    return function() {\n      if (ran) return memo;\n      ran = true;\n      memo = func.apply(this, arguments);\n      func = null;\n      return memo;\n    };\n  };\n\n  // Returns the first function passed as an argument to the second,\n  // allowing you to adjust arguments, run code before and after, and\n  // conditionally execute the original function.\n  _.wrap = function(func, wrapper) {\n    return function() {\n      var args = [func];\n      push.apply(args, arguments);\n      return wrapper.apply(this, args);\n    };\n  };\n\n  // Returns a function that is the composition of a list of functions, each\n  // consuming the return value of the function that follows.\n  _.compose = function() {\n    var funcs = arguments;\n    return function() {\n      var args = arguments;\n      for (var i = funcs.length - 1; i >= 0; i--) {\n        args = [funcs[i].apply(this, args)];\n      }\n      return args[0];\n    };\n  };\n\n  // Returns a function that will only be executed after being called N times.\n  _.after = function(times, func) {\n    return function() {\n      if (--times < 1) {\n        return func.apply(this, arguments);\n      }\n    };\n  };\n\n  // Object Functions\n  // ----------------\n\n  // Retrieve the names of an object's properties.\n  // Delegates to **ECMAScript 5**'s native `Object.keys`\n  _.keys = nativeKeys || function(obj) {\n    if (obj !== Object(obj)) throw new TypeError('Invalid object');\n    var keys = [];\n    for (var key in obj) if (_.has(obj, key)) keys.push(key);\n    return keys;\n  };\n\n  // Retrieve the values of an object's properties.\n  _.values = function(obj) {\n    var keys = _.keys(obj);\n    var length = keys.length;\n    var values = new Array(length);\n    for (var i = 0; i < length; i++) {\n      values[i] = obj[keys[i]];\n    }\n    return values;\n  };\n\n  // Convert an object into a list of `[key, value]` pairs.\n  _.pairs = function(obj) {\n    var keys = _.keys(obj);\n    var length = keys.length;\n    var pairs = new Array(length);\n    for (var i = 0; i < length; i++) {\n      pairs[i] = [keys[i], obj[keys[i]]];\n    }\n    return pairs;\n  };\n\n  // Invert the keys and values of an object. The values must be serializable.\n  _.invert = function(obj) {\n    var result = {};\n    var keys = _.keys(obj);\n    for (var i = 0, length = keys.length; i < length; i++) {\n      result[obj[keys[i]]] = keys[i];\n    }\n    return result;\n  };\n\n  // Return a sorted list of the function names available on the object.\n  // Aliased as `methods`\n  _.functions = _.methods = function(obj) {\n    var names = [];\n    for (var key in obj) {\n      if (_.isFunction(obj[key])) names.push(key);\n    }\n    return names.sort();\n  };\n\n  // Extend a given object with all the properties in passed-in object(s).\n  _.extend = function(obj) {\n    each(slice.call(arguments, 1), function(source) {\n      if (source) {\n        for (var prop in source) {\n          obj[prop] = source[prop];\n        }\n      }\n    });\n    return obj;\n  };\n\n  // Return a copy of the object only containing the whitelisted properties.\n  _.pick = function(obj) {\n    var copy = {};\n    var keys = concat.apply(ArrayProto, slice.call(arguments, 1));\n    each(keys, function(key) {\n      if (key in obj) copy[key] = obj[key];\n    });\n    return copy;\n  };\n\n   // Return a copy of the object without the blacklisted properties.\n  _.omit = function(obj) {\n    var copy = {};\n    var keys = concat.apply(ArrayProto, slice.call(arguments, 1));\n    for (var key in obj) {\n      if (!_.contains(keys, key)) copy[key] = obj[key];\n    }\n    return copy;\n  };\n\n  // Fill in a given object with default properties.\n  _.defaults = function(obj) {\n    each(slice.call(arguments, 1), function(source) {\n      if (source) {\n        for (var prop in source) {\n          if (obj[prop] === void 0) obj[prop] = source[prop];\n        }\n      }\n    });\n    return obj;\n  };\n\n  // Create a (shallow-cloned) duplicate of an object.\n  _.clone = function(obj) {\n    if (!_.isObject(obj)) return obj;\n    return _.isArray(obj) ? obj.slice() : _.extend({}, obj);\n  };\n\n  // Invokes interceptor with the obj, and then returns obj.\n  // The primary purpose of this method is to \"tap into\" a method chain, in\n  // order to perform operations on intermediate results within the chain.\n  _.tap = function(obj, interceptor) {\n    interceptor(obj);\n    return obj;\n  };\n\n  // Internal recursive comparison function for `isEqual`.\n  var eq = function(a, b, aStack, bStack) {\n    // Identical objects are equal. `0 === -0`, but they aren't identical.\n    // See the [Harmony `egal` proposal](http://wiki.ecmascript.org/doku.php?id=harmony:egal).\n    if (a === b) return a !== 0 || 1 / a == 1 / b;\n    // A strict comparison is necessary because `null == undefined`.\n    if (a == null || b == null) return a === b;\n    // Unwrap any wrapped objects.\n    if (a instanceof _) a = a._wrapped;\n    if (b instanceof _) b = b._wrapped;\n    // Compare `[[Class]]` names.\n    var className = toString.call(a);\n    if (className != toString.call(b)) return false;\n    switch (className) {\n      // Strings, numbers, dates, and booleans are compared by value.\n      case '[object String]':\n        // Primitives and their corresponding object wrappers are equivalent; thus, `\"5\"` is\n        // equivalent to `new String(\"5\")`.\n        return a == String(b);\n      case '[object Number]':\n        // `NaN`s are equivalent, but non-reflexive. An `egal` comparison is performed for\n        // other numeric values.\n        return a != +a ? b != +b : (a == 0 ? 1 / a == 1 / b : a == +b);\n      case '[object Date]':\n      case '[object Boolean]':\n        // Coerce dates and booleans to numeric primitive values. Dates are compared by their\n        // millisecond representations. Note that invalid dates with millisecond representations\n        // of `NaN` are not equivalent.\n        return +a == +b;\n      // RegExps are compared by their source patterns and flags.\n      case '[object RegExp]':\n        return a.source == b.source &&\n               a.global == b.global &&\n               a.multiline == b.multiline &&\n               a.ignoreCase == b.ignoreCase;\n    }\n    if (typeof a != 'object' || typeof b != 'object') return false;\n    // Assume equality for cyclic structures. The algorithm for detecting cyclic\n    // structures is adapted from ES 5.1 section 15.12.3, abstract operation `JO`.\n    var length = aStack.length;\n    while (length--) {\n      // Linear search. Performance is inversely proportional to the number of\n      // unique nested structures.\n      if (aStack[length] == a) return bStack[length] == b;\n    }\n    // Objects with different constructors are not equivalent, but `Object`s\n    // from different frames are.\n    var aCtor = a.constructor, bCtor = b.constructor;\n    if (aCtor !== bCtor && !(_.isFunction(aCtor) && (aCtor instanceof aCtor) &&\n                             _.isFunction(bCtor) && (bCtor instanceof bCtor))) {\n      return false;\n    }\n    // Add the first object to the stack of traversed objects.\n    aStack.push(a);\n    bStack.push(b);\n    var size = 0, result = true;\n    // Recursively compare objects and arrays.\n    if (className == '[object Array]') {\n      // Compare array lengths to determine if a deep comparison is necessary.\n      size = a.length;\n      result = size == b.length;\n      if (result) {\n        // Deep compare the contents, ignoring non-numeric properties.\n        while (size--) {\n          if (!(result = eq(a[size], b[size], aStack, bStack))) break;\n        }\n      }\n    } else {\n      // Deep compare objects.\n      for (var key in a) {\n        if (_.has(a, key)) {\n          // Count the expected number of properties.\n          size++;\n          // Deep compare each member.\n          if (!(result = _.has(b, key) && eq(a[key], b[key], aStack, bStack))) break;\n        }\n      }\n      // Ensure that both objects contain the same number of properties.\n      if (result) {\n        for (key in b) {\n          if (_.has(b, key) && !(size--)) break;\n        }\n        result = !size;\n      }\n    }\n    // Remove the first object from the stack of traversed objects.\n    aStack.pop();\n    bStack.pop();\n    return result;\n  };\n\n  // Perform a deep comparison to check if two objects are equal.\n  _.isEqual = function(a, b) {\n    return eq(a, b, [], []);\n  };\n\n  // Is a given array, string, or object empty?\n  // An \"empty\" object has no enumerable own-properties.\n  _.isEmpty = function(obj) {\n    if (obj == null) return true;\n    if (_.isArray(obj) || _.isString(obj)) return obj.length === 0;\n    for (var key in obj) if (_.has(obj, key)) return false;\n    return true;\n  };\n\n  // Is a given value a DOM element?\n  _.isElement = function(obj) {\n    return !!(obj && obj.nodeType === 1);\n  };\n\n  // Is a given value an array?\n  // Delegates to ECMA5's native Array.isArray\n  _.isArray = nativeIsArray || function(obj) {\n    return toString.call(obj) == '[object Array]';\n  };\n\n  // Is a given variable an object?\n  _.isObject = function(obj) {\n    return obj === Object(obj);\n  };\n\n  // Add some isType methods: isArguments, isFunction, isString, isNumber, isDate, isRegExp.\n  each(['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp'], function(name) {\n    _['is' + name] = function(obj) {\n      return toString.call(obj) == '[object ' + name + ']';\n    };\n  });\n\n  // Define a fallback version of the method in browsers (ahem, IE), where\n  // there isn't any inspectable \"Arguments\" type.\n  if (!_.isArguments(arguments)) {\n    _.isArguments = function(obj) {\n      return !!(obj && _.has(obj, 'callee'));\n    };\n  }\n\n  // Optimize `isFunction` if appropriate.\n  if (typeof (/./) !== 'function') {\n    _.isFunction = function(obj) {\n      return typeof obj === 'function';\n    };\n  }\n\n  // Is a given object a finite number?\n  _.isFinite = function(obj) {\n    return isFinite(obj) && !isNaN(parseFloat(obj));\n  };\n\n  // Is the given value `NaN`? (NaN is the only number which does not equal itself).\n  _.isNaN = function(obj) {\n    return _.isNumber(obj) && obj != +obj;\n  };\n\n  // Is a given value a boolean?\n  _.isBoolean = function(obj) {\n    return obj === true || obj === false || toString.call(obj) == '[object Boolean]';\n  };\n\n  // Is a given value equal to null?\n  _.isNull = function(obj) {\n    return obj === null;\n  };\n\n  // Is a given variable undefined?\n  _.isUndefined = function(obj) {\n    return obj === void 0;\n  };\n\n  // Shortcut function for checking if an object has a given property directly\n  // on itself (in other words, not on a prototype).\n  _.has = function(obj, key) {\n    return hasOwnProperty.call(obj, key);\n  };\n\n  // Utility Functions\n  // -----------------\n\n  // Run Underscore.js in *noConflict* mode, returning the `_` variable to its\n  // previous owner. Returns a reference to the Underscore object.\n  _.noConflict = function() {\n    root._ = previousUnderscore;\n    return this;\n  };\n\n  // Keep the identity function around for default iterators.\n  _.identity = function(value) {\n    return value;\n  };\n\n  // Run a function **n** times.\n  _.times = function(n, iterator, context) {\n    var accum = Array(Math.max(0, n));\n    for (var i = 0; i < n; i++) accum[i] = iterator.call(context, i);\n    return accum;\n  };\n\n  // Return a random integer between min and max (inclusive).\n  _.random = function(min, max) {\n    if (max == null) {\n      max = min;\n      min = 0;\n    }\n    return min + Math.floor(Math.random() * (max - min + 1));\n  };\n\n  // List of HTML entities for escaping.\n  var entityMap = {\n    escape: {\n      '&': '&amp;',\n      '<': '&lt;',\n      '>': '&gt;',\n      '\"': '&quot;',\n      \"'\": '&#x27;'\n    }\n  };\n  entityMap.unescape = _.invert(entityMap.escape);\n\n  // Regexes containing the keys and values listed immediately above.\n  var entityRegexes = {\n    escape:   new RegExp('[' + _.keys(entityMap.escape).join('') + ']', 'g'),\n    unescape: new RegExp('(' + _.keys(entityMap.unescape).join('|') + ')', 'g')\n  };\n\n  // Functions for escaping and unescaping strings to/from HTML interpolation.\n  _.each(['escape', 'unescape'], function(method) {\n    _[method] = function(string) {\n      if (string == null) return '';\n      return ('' + string).replace(entityRegexes[method], function(match) {\n        return entityMap[method][match];\n      });\n    };\n  });\n\n  // If the value of the named `property` is a function then invoke it with the\n  // `object` as context; otherwise, return it.\n  _.result = function(object, property) {\n    if (object == null) return void 0;\n    var value = object[property];\n    return _.isFunction(value) ? value.call(object) : value;\n  };\n\n  // Add your own custom functions to the Underscore object.\n  _.mixin = function(obj) {\n    each(_.functions(obj), function(name) {\n      var func = _[name] = obj[name];\n      _.prototype[name] = function() {\n        var args = [this._wrapped];\n        push.apply(args, arguments);\n        return result.call(this, func.apply(_, args));\n      };\n    });\n  };\n\n  // Generate a unique integer id (unique within the entire client session).\n  // Useful for temporary DOM ids.\n  var idCounter = 0;\n  _.uniqueId = function(prefix) {\n    var id = ++idCounter + '';\n    return prefix ? prefix + id : id;\n  };\n\n  // By default, Underscore uses ERB-style template delimiters, change the\n  // following template settings to use alternative delimiters.\n  _.templateSettings = {\n    evaluate    : /<%([\\s\\S]+?)%>/g,\n    interpolate : /<%=([\\s\\S]+?)%>/g,\n    escape      : /<%-([\\s\\S]+?)%>/g\n  };\n\n  // When customizing `templateSettings`, if you don't want to define an\n  // interpolation, evaluation or escaping regex, we need one that is\n  // guaranteed not to match.\n  var noMatch = /(.)^/;\n\n  // Certain characters need to be escaped so that they can be put into a\n  // string literal.\n  var escapes = {\n    \"'\":      \"'\",\n    '\\\\':     '\\\\',\n    '\\r':     'r',\n    '\\n':     'n',\n    '\\t':     't',\n    '\\u2028': 'u2028',\n    '\\u2029': 'u2029'\n  };\n\n  var escaper = /\\\\|'|\\r|\\n|\\t|\\u2028|\\u2029/g;\n\n  // JavaScript micro-templating, similar to John Resig's implementation.\n  // Underscore templating handles arbitrary delimiters, preserves whitespace,\n  // and correctly escapes quotes within interpolated code.\n  _.template = function(text, data, settings) {\n    var render;\n    settings = _.defaults({}, settings, _.templateSettings);\n\n    // Combine delimiters into one regular expression via alternation.\n    var matcher = new RegExp([\n      (settings.escape || noMatch).source,\n      (settings.interpolate || noMatch).source,\n      (settings.evaluate || noMatch).source\n    ].join('|') + '|$', 'g');\n\n    // Compile the template source, escaping string literals appropriately.\n    var index = 0;\n    var source = \"__p+='\";\n    text.replace(matcher, function(match, escape, interpolate, evaluate, offset) {\n      source += text.slice(index, offset)\n        .replace(escaper, function(match) { return '\\\\' + escapes[match]; });\n\n      if (escape) {\n        source += \"'+\\n((__t=(\" + escape + \"))==null?'':_.escape(__t))+\\n'\";\n      }\n      if (interpolate) {\n        source += \"'+\\n((__t=(\" + interpolate + \"))==null?'':__t)+\\n'\";\n      }\n      if (evaluate) {\n        source += \"';\\n\" + evaluate + \"\\n__p+='\";\n      }\n      index = offset + match.length;\n      return match;\n    });\n    source += \"';\\n\";\n\n    // If a variable is not specified, place data values in local scope.\n    if (!settings.variable) source = 'with(obj||{}){\\n' + source + '}\\n';\n\n    source = \"var __t,__p='',__j=Array.prototype.join,\" +\n      \"print=function(){__p+=__j.call(arguments,'');};\\n\" +\n      source + \"return __p;\\n\";\n\n    try {\n      render = new Function(settings.variable || 'obj', '_', source);\n    } catch (e) {\n      e.source = source;\n      throw e;\n    }\n\n    if (data) return render(data, _);\n    var template = function(data) {\n      return render.call(this, data, _);\n    };\n\n    // Provide the compiled function source as a convenience for precompilation.\n    template.source = 'function(' + (settings.variable || 'obj') + '){\\n' + source + '}';\n\n    return template;\n  };\n\n  // Add a \"chain\" function, which will delegate to the wrapper.\n  _.chain = function(obj) {\n    return _(obj).chain();\n  };\n\n  // OOP\n  // ---------------\n  // If Underscore is called as a function, it returns a wrapped object that\n  // can be used OO-style. This wrapper holds altered versions of all the\n  // underscore functions. Wrapped objects may be chained.\n\n  // Helper function to continue chaining intermediate results.\n  var result = function(obj) {\n    return this._chain ? _(obj).chain() : obj;\n  };\n\n  // Add all of the Underscore functions to the wrapper object.\n  _.mixin(_);\n\n  // Add all mutator Array functions to the wrapper.\n  each(['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], function(name) {\n    var method = ArrayProto[name];\n    _.prototype[name] = function() {\n      var obj = this._wrapped;\n      method.apply(obj, arguments);\n      if ((name == 'shift' || name == 'splice') && obj.length === 0) delete obj[0];\n      return result.call(this, obj);\n    };\n  });\n\n  // Add all accessor Array functions to the wrapper.\n  each(['concat', 'join', 'slice'], function(name) {\n    var method = ArrayProto[name];\n    _.prototype[name] = function() {\n      return result.call(this, method.apply(this._wrapped, arguments));\n    };\n  });\n\n  _.extend(_.prototype, {\n\n    // Start chaining a wrapped Underscore object.\n    chain: function() {\n      this._chain = true;\n      return this;\n    },\n\n    // Extracts the result from a wrapped and chained object.\n    value: function() {\n      return this._wrapped;\n    }\n\n  });\n\n}).call(this);\n\n})()","(function() {\n  var active, createSearch, emptyPage, util, wiki;\n\n  wiki = require('./wiki');\n\n  util = require('./util');\n\n  active = require('./active');\n\n  emptyPage = require('./page').emptyPage;\n\n  createSearch = function(_arg) {\n    var neighborhood, performSearch;\n    neighborhood = _arg.neighborhood;\n    performSearch = function(searchQuery) {\n      var $resultPage, result, resultPage, searchResults, tally, _i, _len, _ref;\n      searchResults = neighborhood.search(searchQuery);\n      tally = searchResults.tally;\n      resultPage = emptyPage();\n      resultPage.setTitle(\"Search for '\" + searchQuery + \"'\");\n      resultPage.addParagraph(\"String '\" + searchQuery + \"' found on \" + (tally.finds || 'none') + \" of \" + (tally.pages || 'no') + \" pages from \" + (tally.sites || 'no') + \" sites.\\nText matched on \" + (tally.title || 'no') + \" titles, \" + (tally.text || 'no') + \" paragraphs, and \" + (tally.slug || 'no') + \" slugs.\\nElapsed time \" + tally.msec + \" milliseconds.\");\n      _ref = searchResults.finds;\n      for (_i = 0, _len = _ref.length; _i < _len; _i++) {\n        result = _ref[_i];\n        resultPage.addItem({\n          \"type\": \"reference\",\n          \"site\": result.site,\n          \"slug\": result.page.slug,\n          \"title\": result.page.title,\n          \"text\": result.page.synopsis || ''\n        });\n      }\n      $resultPage = wiki.createPage(resultPage.getSlug()).addClass('ghost');\n      $resultPage.appendTo($('.main'));\n      wiki.buildPage(resultPage, $resultPage);\n      return active.set($('.page').last());\n    };\n    return {\n      performSearch: performSearch\n    };\n  };\n\n  module.exports = createSearch;\n\n}).call(this);\n","(function() {\n  var active, createSearch, neighborhood, nextAvailableFetch, nextFetchInterval, populateSiteInfoFor, util, wiki, _,\n    __hasProp = {}.hasOwnProperty;\n\n  _ = require('underscore');\n\n  wiki = require('./wiki');\n\n  active = require('./active');\n\n  util = require('./util');\n\n  createSearch = require('./search');\n\n  module.exports = neighborhood = {};\n\n  if (wiki.neighborhood == null) {\n    wiki.neighborhood = {};\n  }\n\n  nextAvailableFetch = 0;\n\n  nextFetchInterval = 2000;\n\n  populateSiteInfoFor = function(site, neighborInfo) {\n    var fetchMap, now, transition;\n    if (neighborInfo.sitemapRequestInflight) {\n      return;\n    }\n    neighborInfo.sitemapRequestInflight = true;\n    transition = function(site, from, to) {\n      return $(\".neighbor[data-site=\\\"\" + site + \"\\\"]\").find('div').removeClass(from).addClass(to);\n    };\n    fetchMap = function() {\n      var request, sitemapUrl;\n      sitemapUrl = \"http://\" + site + \"/system/sitemap.json\";\n      transition(site, 'wait', 'fetch');\n      request = $.ajax({\n        type: 'GET',\n        dataType: 'json',\n        url: sitemapUrl\n      });\n      return request.always(function() {\n        return neighborInfo.sitemapRequestInflight = false;\n      }).done(function(data) {\n        neighborInfo.sitemap = data;\n        transition(site, 'fetch', 'done');\n        return $('body').trigger('new-neighbor-done', site);\n      }).fail(function(data) {\n        return transition(site, 'fetch', 'fail');\n      });\n    };\n    now = Date.now();\n    if (now > nextAvailableFetch) {\n      nextAvailableFetch = now + nextFetchInterval;\n      return setTimeout(fetchMap, 100);\n    } else {\n      setTimeout(fetchMap, nextAvailableFetch - now);\n      return nextAvailableFetch += nextFetchInterval;\n    }\n  };\n\n  wiki.registerNeighbor = neighborhood.registerNeighbor = function(site) {\n    var neighborInfo;\n    if (wiki.neighborhood[site] != null) {\n      return;\n    }\n    neighborInfo = {};\n    wiki.neighborhood[site] = neighborInfo;\n    populateSiteInfoFor(site, neighborInfo);\n    return $('body').trigger('new-neighbor', site);\n  };\n\n  neighborhood.listNeighbors = function() {\n    return _.keys(wiki.neighborhood);\n  };\n\n  neighborhood.search = function(searchQuery) {\n    var finds, match, matchingPages, neighborInfo, neighborSite, sitemap, start, tally, tick, _ref;\n    finds = [];\n    tally = {};\n    tick = function(key) {\n      if (tally[key] != null) {\n        return tally[key]++;\n      } else {\n        return tally[key] = 1;\n      }\n    };\n    match = function(key, text) {\n      var hit;\n      hit = (text != null) && text.toLowerCase().indexOf(searchQuery.toLowerCase()) >= 0;\n      if (hit) {\n        tick(key);\n      }\n      return hit;\n    };\n    start = Date.now();\n    _ref = wiki.neighborhood;\n    for (neighborSite in _ref) {\n      if (!__hasProp.call(_ref, neighborSite)) continue;\n      neighborInfo = _ref[neighborSite];\n      sitemap = neighborInfo.sitemap;\n      if (sitemap != null) {\n        tick('sites');\n      }\n      matchingPages = _.each(sitemap, function(page) {\n        tick('pages');\n        if (!(match('title', page.title) || match('text', page.synopsis) || match('slug', page.slug))) {\n          return;\n        }\n        tick('finds');\n        return finds.push({\n          page: page,\n          site: neighborSite,\n          rank: 1\n        });\n      });\n    }\n    tally['msec'] = Date.now() - start;\n    return {\n      finds: finds,\n      tally: tally\n    };\n  };\n\n  $(function() {\n    var $neighborhood, flag, search;\n    $neighborhood = $('.neighborhood');\n    flag = function(site) {\n      return \"<span class=\\\"neighbor\\\" data-site=\\\"\" + site + \"\\\">\\n  <div class=\\\"wait\\\">\\n    <img src=\\\"http://\" + site + \"/favicon.png\\\" title=\\\"\" + site + \"\\\">\\n  </div>\\n</span>\";\n    };\n    $('body').on('new-neighbor', function(e, site) {\n      return $neighborhood.append(flag(site));\n    }).delegate('.neighbor img', 'click', function(e) {\n      return wiki.doInternalLink('welcome-visitors', null, this.title);\n    });\n    search = createSearch({\n      neighborhood: neighborhood\n    });\n    return $('input.search').on('keypress', function(e) {\n      var searchQuery;\n      if (e.keyCode !== 13) {\n        return;\n      }\n      searchQuery = $(this).val();\n      search.performSearch(searchQuery);\n      return $(this).val(\"\");\n    });\n  });\n\n}).call(this);\n"]}
3607
- ;