-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathqoolbar.js
899 lines (857 loc) · 43.2 KB
/
qoolbar.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
// qoolbar.js - a menu of qiki verb icons to apply to things on a web page.
'use strict';
/**
* @param window
* @param qoolbar
* @param $
* @param $.ui
*/
(function (window, qoolbar, $) {
if (typeof qoolbar !== 'object') {
console.error("The qoolbar.js module doesn't appear to be included.");
}
if (typeof $ !== 'function') {
console.error("The qoolbar.js module requires jQuery.");
} else if (typeof $.ui === 'undefined') {
console.warn("The qoolbar.js module wants to use jQuery UI.");
// TODO: Why? List uses here. Then get rid of them.
}
qoolbar.is_ready = false;
qoolbar.MAX_DROP_FILE_SIZE = 1000;
qoolbar._ajax_url = null;
qoolbar.ajax_url = function qoolbar_ajax_url(url) {
qoolbar._ajax_url = url;
};
qoolbar.ICON_ENTRY_TOP_FUDGE = -3; // Correspond to bottom padding and margin
qoolbar.ICON_ENTRY_LEFT_FUDGE = 1; // Correspond to left padding and margin
var UNICODE = {
TIMES: '\u00D7', // aka ×
VERTICAL_ELLIPSIS: '\u22EE', // 3 vertical dots, aka ⋮ ⋮ ⋮
EQUIVALENT: '\u2261', // 3 horizontal bars, aka ≡ ≡ ≡
TRIGRAM_FOR_HEAVEN: '\u2630', /// 3 horizontal bars
TETRAGRAM_FOR_CENTRE: '\ud834\udf06' // 4 horiz bars -- '\u{1d306}' is not supported by IE11
};
$(function document_ready() {
/**
* Build the qoolbar DOM, given an array of verb names and icons.
*
* @param verbs[]
* @param verbs.length
* @param verbs.name -- e.g. 'like'
* @param verbs.idn
* @param verbs.icon_url -- from the iconify sentence
* @param verbs.qool_num -- 0 if deleted, 1 if not
* @returns {*|HTMLElement}
* @private
*/
// NOTE: Why does verbs.name work and not verbs[].name?
// http://usejsdoc.org/tags-param.html#parameters-with-properties
function qoolbar_build(verbs) {
qoolbar._verb_dicts = {};
var $qoolbar = $('<div>', {'class': 'qoolbar fade_until_hover'});
// THANKS: class is a reserved word,
// https://api.jquery.com/jQuery/#creating-new-elements
// 'The name "class" must be quoted in the object since it is a JavaScript
// reserved word, and "className" cannot be used since it refers to the
// DOM property, not the attribute.'
var $qoolbar_head = $('<div>', {
'class': 'qoolbar-head qool-more-expanse',
'title': "This is your qoolbar. Drag these verbs onto the page."
}).text("qoolbar");
var $qoolbar_body = $('<div>', {'class': 'qoolbar-body'});
$qoolbar.append($qoolbar_head);
$qoolbar.append($qoolbar_body);
var num_verbs = verbs.length;
for (var i_verb=0 ; i_verb < num_verbs ; i_verb++) {
// THANKS: (avoiding for-in loop on arrays) http://stackoverflow.com/a/3010848/673991
var verb = verbs[i_verb];
qoolbar._verb_dicts[verb.idn] = verb;
var tool_classes = 'qool-verb qool-verb-' + verb.name;
if (verb.qool_num === 0) {
tool_classes += ' ' + 'verb-deleted';
}
var $verb_tool = $('<div>', {
'class': tool_classes,
'data-verb': verb.name,
'data-vrb-idn': verb.idn
});
$verb_tool.append(verb_icon(verb));
var $verb_delete = $('<span>', {
'class': 'verb-deletion hide_until_more',
'title': "remove '" + verb.name + "' from your qoolbar"
}).text(UNICODE.TIMES);
$verb_tool.prepend($verb_delete);
$qoolbar_body.append($verb_tool);
}
// noinspection RequiredAttributes
$qoolbar_body.append(
$('<div>', {
'class': 'qool-more-switch',
'title': "more options"
// }).text(UNICODE.VERTICAL_ELLIPSIS)
})
.append($('<span>', {'class': 'qool-more-contract'}).text(UNICODE.VERTICAL_ELLIPSIS))
.append($('<span>', {'class': 'qool-more-expanse'}).text(UNICODE.TRIGRAM_FOR_HEAVEN))
);
$qoolbar_body.append(
$('<div>', {
'class': 'qool-more-foot qool-more-expanse',
'title': "Enter a name for a new verb."
}).append(
$('<input>', {
id: 'qool-new-verb',
type: 'text',
placeholder: "new verb"
})
)
);
qool_more_toggle(false);
// console.log('_verb_dicts', qoolbar._verb_dicts)
// EXAMPLE: _verb_dicts {
// 0q82_86: {qool_num: 1, icon_url: "http://.../thumbsup_16.png", name: "like", idn: "0q82_86"}
// 0q82_89: {qool_num: 1, icon_url: "http://.../delete_16.png", name: "delete", idn: "0q82_89"}
// 0q83_01FC: {qool_num: 0, icon_url: "...mCC", name: "laugh", idn: "0q83_01FC"}
// 0q83_0301: {qool_num: 1, icon_url: "...g==", name: "spam", idn: "0q83_0301"}
// 0q83_0335: {qool_num: 1, icon_url: "...II=", name: "laugh", idn: "0q83_0335"}
// }
return $qoolbar;
}
/**
* Decorate qiki-word-associated elements with their verbs and scores.
*
* Requirements before calling:
* - The elements are expected to already have attributes.
* - data-idn - idn qstring of the associated word
* - data-jbo - JSON of array of qoolifying verbs (jQuery .data() makes it an object)
* - qoolbar.html() has finished (its built_callback was called),
* indicating the qoolbar elements have been constructed and
* and qoolbar._verb_dicts[] is populated.
*
* data-jbo e.g.
* [
* {"sbj": "0q82_A7__8A059E058E6A6308C8B0_1D0B00", "vrb": "0q82_86", "txt": "", "num": 1, "idn": "0q83_0188"},
* {"sbj": "0q82_A7__8A059E058E6A6308C8B0_1D0B00", "vrb": "0q82_86", "txt": "", "num": 2, "idn": "0q83_01CD"},
* {"sbj": "0q82_A7__8A059E058E6A6308C8B0_1D0B00", "vrb": "0q82_86", "txt": "", "num": 3, "idn": "0q83_01D1"},
* {"sbj": "0q82_A7__8A059E058E6A6308C8B0_1D0B00", "vrb": "0q82_86", "txt": "", "num": 4, "idn": "0q83_01FA"}
* ]
*
* Append the qoolbar icons that have been applied to them, based on their data-jbo.
* @param selector
*/
qoolbar.bling = function qoolbar_bling(selector) {
$(selector).each(function () {
var $element = $(this);
var jbo = $element.data('jbo');
if (typeof jbo === 'string') {
console.error("data-jbo not converted, still a string:'" + jbo + "'");
jbo = JSON.parse(jbo);
}
console.assert(typeof jbo === 'object', typeof jbo, jbo);
var scores = scorer(jbo);
var verb_widgets = [];
for (var vrb in scores) {
if (scores.hasOwnProperty(vrb)) {
var score = scores[vrb];
if (qoolbar._verb_dicts.hasOwnProperty(vrb)) {
var verb_dict = qoolbar._verb_dicts[vrb];
// console.log("bling", vrb, verb_dict);
// EXAMPLE: 0q82_86 {idn: "0q82_86",
// icon_url: "http://tool.qiki.info/icon/thumbsup_16.png",
// name: "like"}
var $verb_icon = verb_icon(verb_dict);
safe_set_attr(
$verb_icon,
'title',
verb_dict.name + ": " + score.history.join("-")
);
var $my_score = $('<span>')
.addClass('icon-sup')
.text(str(score.my));
var $everybody_score = $('<span>', {
'class': 'icon-sub'
}).text(score.sum.toString());
var $icon_bling = $('<span>')
.addClass('icon-bling')
.append($my_score)
.append($everybody_score);
var $verb = $('<span>')
.addClass('qool-icon')
.data('num', score.my) // Never shows up as data-num attribute, unfortunately.
.data('vrb-idn', vrb)
.append($verb_icon)
.append($icon_bling);
verb_widgets.push($verb);
if (score.my !== 0) {
// TODO: Resolve whether to do this with deleted verbs.
// E.g. if a user deletes the delete verb (red x),
// Should his up-to-then deleted objects remain deleted forever?
// Or should his notion of deletion itself be deleted,
// thereby mass-un-deleting?? Probably not that...
// TODO: If "deleted" is unchecked, I need to hide both
// - what the viewer has deleted
// - what the AUTHOR has deleted
// but ignore any deletions by non-authors (who are not me)
// this calls out for a bot!
// Maybe I should NEVER show author deletions to non-authors,
// even when the box IS checked.
var data_name_me = 'data-qool-' + verb_dict.name + '-me';
// $element.attr(data_name_me, score.my);
safe_set_attr($element, data_name_me, score.my)
}
var score_they = score.sum - score.my;
if (score_they !== 0) {
var data_name_they = 'data-qool-' + verb_dict.name + '-they';
// $element.attr(data_name_they, score_they);
safe_set_attr($element, data_name_they, score_they);
}
} else {
console.warn("Verb", vrb, "not in qoolbar");
}
}
}
var $bling = $('<span>', {'class': 'qool-bling'});
$bling.append(verb_widgets);
$element.children('.qool-bling').remove(); // remove old bling
$element.append($bling)
});
};
function safe_set_attr(element, attr_name, attr_value) {
// TODO: Test and use for attr GETTING too.
var return_value = null;
try {
return_value = $(element).attr(attr_name, attr_value);
} catch (e) {
try {
return_value = $(element).attr(attr_name, "(error)");
console.warn(
".attr() can't set",
attr_name.toString(),
"to",
attr_value.toString(),
"because:\n",
e.message
);
} catch (ee) {
try {
console.warn(
".attr() cannot set",
attr_name.toString(),
"at all, because:\n",
ee.message
);
} catch (eee) {
console.error(".attr() meltdown, type", typeof attr_name, eee.message);
}
}
}
return return_value;
}
/**
* Get the list of qool verbs from the lex. Build the qoolbar and stick them in there.
*
* Should be called from application inside $(function document_ready() { ... });
*
* @param selector - empty div wherein to place it.
* @param built_callback - _verb_dicts are ready. Time to qoolbar.bling the qool targets.
*/
qoolbar.html = function qoolbar_html(selector, built_callback) {
qoolbar.post(
'qoolbar_list',
{},
/**
*
* @param response
* @param response.verbs -- (if is_valid) -- qool verbs, see qoolbar_build()
*/
function qoolbar_list_done(response) {
$(selector).append(qoolbar_build(response.verbs));
//noinspection JSUnusedGlobalSymbols,JSValidateTypes
$(selector + ' .qool-verb').draggable({
helper: 'clone',
cursor: '-moz-grabbing',
// TODO: grabby cursor? -webkit-grab? move? http://stackoverflow.com/a/26811031/673991
scroll: false,
appendTo: 'body',
// THANKS: helper-clone on top, https://stackoverflow.com/q/17977799/673991
start: function () {
association_in_progress();
},
stop: function () {
association_resolved();
}
});
$('.qoolbar')
.on('mousedown', '.qool-more-switch', function qool_more_click(event) {
qool_more_toggle();
event.stopPropagation(); // avoid document click's hiding of qool-more
event.preventDefault();
// THANKS: no text select, https://stackoverflow.com/a/43321596/673991
// mousedown preventDefault avoids double-click
})
.on('mousedown', '.verb-deletion', function verb_delete(event) {
var $qool_verb = $(this).closest('.qool-verb');
if ($qool_verb.length === 1) {
var verb_idn = $qool_verb.data('vrb-idn');
qoolbar.post(
'delete_verb',
{idn: verb_idn},
function delete_verb_done(response) {
console.debug("delete verb done", response);
// TODO: Remove from qoolbar._verb_dicts,
// and maybe qoolbar_build()
// so we don't have to reload.
}
);
} else {
console.error("Delete not inside a qool verb", $(this));
}
event.stopPropagation(); // avoid document click's hide
event.preventDefault();
})
.on('dragover dragleave', '.qool-verb', function (e) {
e.stopPropagation();
e.preventDefault();
})
.on('drop', '.qool-verb', function drop_onto_qoolbar_verb(e) {
e.stopPropagation();
e.preventDefault();
var qool_verb_name = $(this).data('verb');
var qool_verb_idn = $(this).data('vrb-idn');
var files = e.originalEvent.target.files || e.originalEvent.dataTransfer.files;
if (files.length >= 1) {
var f = files[0];
console.log("Drop", qool_verb_name, qool_verb_idn, files.length, "files", f.type, f.size);
if (f.size <= qoolbar.MAX_DROP_FILE_SIZE) {
var reader = new FileReader();
/**
* Read icon image file.
*
* @param file_event
* @param file_event.target
* @param file_event.target.result
* @param file_event.target.result[]
*
* @property reader
* @property reader.result
* @property reader.result[]
*/
reader.onload = function (file_event) {
console.log("dropped", file_event.target.result.length, "bytes", file_event.target.result);
console.log("dropped", reader.result.length, "bytes", reader.result);
// TODO: Convert reader.result to data:image
var data_image = data_image_from(reader.result, files[0].type);
post_icon(qool_verb_idn, data_image);
};
reader.readAsBinaryString(f);
} else {
console.warn(
"File is",
f.size,
"bytes, it should be less than",
qoolbar.MAX_DROP_FILE_SIZE
);
}
} else {
var dropped_html = e.originalEvent.dataTransfer.getData("text/html");
var $dropped_container = $('<div>').append(dropped_html);
// FIXME: security concerns
var image_url = $dropped_container.find("img").attr('src');
console.log("dropped url", image_url, dropped_html);
// EXAMPLE:  ... ggg==
// <img class="irc_mut" alt="Image result for laughing emoji" onload="..."
// src="... gg==" width="16" height="16"
// style="margin-top: 169px;">
// EXAMPLE: https
// https://lh4.googleusercontent.com/proxy/pBKC ... -no-nu
// <img class="irc_mi" src="https://lh4.googleusercontent.com/proxy/pBKC ... -no-nu"
// onload="..." width="16" height="16" style="margin-top: 169px;"
// alt="Image result for laughing emoji">
post_icon(qool_verb_idn, image_url);
}
})
;
if (typeof built_callback === 'function') {
built_callback();
}
}
);
};
/**
* Iconify: associate an icon URL with a verb.
*
* @param qool_verb_idn - the qstring of the idn of the word defining the verb
* @param image_url - e.g. http://... or data:image/png;base65,...
* @private
*/
function post_icon(qool_verb_idn, image_url) {
qoolbar.sentence(
{
vrb_txt: 'iconify',
obj_idn: qool_verb_idn,
txt: image_url,
use_already: true // Use an old sentence if same txt and num.
},
/**
* @param new_word
*/
function iconify_done(new_word) {
console.log("Yay iconify", new_word.idn);
// TODO: re-bling all words that use qool_verb_idn, or something
}
);
}
/**
* Convert image to data:image URL.
*
* @param bytes - raw contents of the image
* @param mime_type - e.g. 'image/png'
* @return {string}
* @private
*/
function data_image_from(bytes, mime_type) {
var base64 = btoa(bytes);
return (
'data:' +
mime_type +
';base64,' +
base64
);
}
/**
* Identify qoolbar drop-targets.
*
* If someone drops a qool verb on them in the future,
* they will become the object of a new qool sentence.
*
* qoolbar.bling() - show past sentences
* qoolbar.target() - prepare for future sentences
*
* May be called right after calling qoolbar.html(). (No need to wait for its callback.)
*
* @param selector - e.g. ".word" - all elements must have a data-idn attribute already.
*/
qoolbar.target = function qoolbar_target(selector) {
var $objects = $(selector);
var objects_without_idn = $objects.filter(':not([data-idn])');
if (objects_without_idn.length > 0) {
console.error(
"Drop objects need a data-idn attribute. " +
objects_without_idn.length + " out of " +
$objects.length + " are missing one."
);
}
// TODO: Drag without jQueryUI, https://stackoverflow.com/q/9435051/673991
// https://stackoverflow.com/q/8569095/673991
//noinspection JSUnusedGlobalSymbols,JSUnresolvedFunction
$objects.droppable({
accept: ".qool-verb",
hoverClass: 'drop-hover',
drop: function qool_verb_drop(event, ui) {
var $source = ui.draggable;
var $destination = $(event.target);
var vrb_idn = $source.data('vrb-idn');
var destination_idn = $destination.data('idn');
qoolbar.sentence(
{
// vrb_txt: verb_name, No, this may get a different verb by the same name. Use idn.
// NOTE: This solved a long-fought bug with multiple laugh verbs.
// The new one was in the qoolbar, but lex['laugh'] got the old one.
vrb_idn: vrb_idn,
obj_idn: destination_idn,
num_add: '1',
txt: ''
},
/**
* @param new_word
*/
function qool_verb_drop_done(new_word) {
valid_sentence_response(new_word, vrb_idn, $destination);
}
);
}
});
};
/**
* Arrange so that clicking on bling allows you to edit your rating.
*
* @param selector - things with bling
*/
qoolbar.click_to_edit = function qoolbar_click_to_edit(selector) {
// TODO: Can we really rely on $(selector) to contain the $(.qool-icon)s?
// Seriously not D.R.Y.
// Because that containment is expressed in
// word-diagram-call.html
// --> icon_diagram
// --> playground_extras.py
// --> icon-diagram-call.html
// Is the solution to develop a REST-full API??
// (WTF DOES THIS MEAN? Is this vestigial?)
//noinspection JSJQueryEfficiency
$(selector).on('mousedown', '.qool-icon', function bling_mousedown() {
var was_already_editing = $(this).hasClass('qool-editing');
$(this).data('was_already_editing', was_already_editing);
});
// THANKS: mousedown before blur, http://stackoverflow.com/a/10653160/673991
// then after blur (if any) comes click
$(selector).on('click', '.qool-icon', function bling_click(event) {
// TODO: Shouldn't selector events be bound in qoolbar.target()??
var $qool_icon = $(this);
var was_already_editing = $qool_icon.data('was_already_editing');
$qool_icon.removeData('was_already_editing');
if (was_already_editing === undefined) {
console.warn("Qool icon click without a preceding mousedown?");
}
if (!was_already_editing) {
$qool_icon.addClass('qool-editing');
qoolbar._is_anybody_editing = true;
var old_num = $qool_icon.data('num');
var $input = $('<input>', {
type: 'text',
// TODO: IE7 needs type in definition
// SEE: http://stackoverflow.com/questions/9898442/jquery-create-element-with-attribute-differences
'class': 'qool-icon-entry',
value: old_num
});
$qool_icon.find('.icon-bling').append($input);
$input.select();
var $icon_sup = $qool_icon.find('.icon-sup');
var icon_sup_pos = $icon_sup.offset();
$input.offset({
top:
icon_sup_pos.top
- $input.outerHeight()
+ $icon_sup.outerHeight()
+ qoolbar.ICON_ENTRY_TOP_FUDGE,
left:
icon_sup_pos.left
+ qoolbar.ICON_ENTRY_LEFT_FUDGE
});
}
event.stopPropagation();
});
//noinspection JSJQueryEfficiency
$(selector).on('click', '.qool-icon-entry', function bling_score_click(event) {
// Clicking the input field itself should not cancel editing.
// THANKS: For nested click ignoring, http://stackoverflow.com/a/2364639/673991
event.stopPropagation();
});
//noinspection JSJQueryEfficiency
$('body').on('keydown', '.qool-icon-entry', 'return', function bling_score_return(event) {
// TODO: Use evt.key == 'Enter' instead of jquery.hotkeys.js, https://stackoverflow.com/a/3369624/673991
event.preventDefault();
var new_num = $(this).val().trim();
if (new_num === '') {
end_all_editing();
return
}
var $qool_icon = $(this).closest('.qool-icon');
var vrb_idn = $qool_icon.data('vrb-idn');
if (typeof vrb_idn !== 'string') {
console.error("qool-icon element needs a data-vrb-idn attribute");
}
var $destination = $(this).closest(selector);
var obj_idn = $destination.data('idn');
// TODO: Search instead for a class that qoolbar.target() installed? Better D.R.Y.
qoolbar.sentence(
{
vrb_idn: vrb_idn,
obj_idn: obj_idn,
num: new_num, // FIXME: This could be a q-string! Dangerous??
txt: '' // TODO: Room for a comment?
// Ooh, yes, pop up a box for that too!
// Or just allow it after the number!!!
// [42 - that's my answer and I'm sticking to it]
// So it'd be handy if the input box grew as you typed
// even multiple lines!
},
function bling_score_change_done(new_word) {
valid_sentence_response(new_word, vrb_idn, $destination);
end_all_editing();
}
);
});
//noinspection JSJQueryEfficiency
$('body').on('keydown', '.qool-icon-entry', 'esc', function bling_score_esc(event) {
// TODO: Use evt.key == 'Escape' instead of jquery.hotkeys.js, https://stackoverflow.com/a/3369624/673991
event.preventDefault();
end_all_editing();
});
$(window.document).on('blur', '.qool-icon-entry', function bling_score_blur() {
// THANKS: For event on dynamic selector, http://stackoverflow.com/a/1207393/673991
if (qoolbar) {
end_all_editing();
}
});
};
$(window.document).on('click', '.qoolbar', function qoolbar_background_click(event) {
event.stopPropagation();
});
$(window.document).on('click', function document_background_click() {
qool_more_toggle(false);
});
$(window.document).on('keypress', '#qool-new-verb', function new_verb_keypress(event) {
// TODO: Use evt.key == 'Enter' instead of jquery.hotkeys.js, https://stackoverflow.com/a/3369624/673991
if (event.keyCode === 13) {
var verb_name = $(this).val();
$(this).val("");
qoolbar.post(
'new_verb',
{name: verb_name},
function new_verb_done(response) {
console.debug("new verb done", response);
// TODO: Add to qoolbar._verb_dicts, and maybe call qoolbar_build()
// so we don't have to reload to see the new verb.
}
);
}
});
function qool_more_toggle(do_expand) {
$('.qoolbar').toggleClass('qool-more-expand', do_expand);
}
function valid_sentence_response(new_word, vrb_idn, $destination) {
var jbo = $destination.data('jbo');
console.assert(typeof jbo === 'object', typeof jbo);
jbo.push(new_word);
$destination.data('jbo', jbo); // Necessary? Or was array modified in place?
qoolbar.bling($destination);
}
//
// function json_concat(ja1, ja2) {
// console.assert(typeof ja1 === 'string');
// console.assert(typeof ja2 === 'string');
// return JSON.stringify($.parseJSON(ja1) + $.parseJSON(ja2));
// }
qoolbar.page_reload = function qoolbar_page_reload() {
// noinspection JSDeprecatedSymbols
window.location.reload(true);
// TODO: Reload Chrome, https://stackoverflow.com/q/10719505/673991
};
// FIXME: Implement qoolbar.create_word() where qoolbar.sentence() is now called.
// Steal it from unslumping.js
// /**
// * Create a new word in the lex.
// *
// * The null contingency helps the caller when they have follow-up to do whether or not
// * they need to create a sentence. then_what() callback is called in either case.
// *
// * @param sentence_or_null - object, verb, etc., or null to ignore
// * (sbj will be the current user)
// * vrb_idn or vrb_txt - required
// * obj_idn - required
// * txt - required
// * num or num_add - optional (defaults to 1)
// * use_already - optional (defaults to false)
// * @param done_callback - callback async follow-up function, with the new word including
// * its idn, or null if the input sentence was null.
// * @param fail_callback {function=} - optional handler of various failure conditions
// * If specified, error_message is passed.
// * It will NOT have been displayed.
// * If unspecified, error_message WILL be displayed
// * with console.error().
// */
// // TODO: Still need the null option? Caller post_it_done_2() used to need it, doesn't now.
// qoolbar.sentence = function qoolbar_sentence(
// sentence_or_null,
// done_callback,
// fail_callback
// ) {
// done_callback = done_callback || function () {};
// fail_callback = fail_callback || function (m) { console.error(m); };
//
// if (sentence_or_null === null) {
// done_callback(null);
// } else {
// /**
// * @param response_object
// * @param response_object.new_words[]
// */
// qoolbar.post(
// 'sentence',
// sentence_or_null,
// function (response) {
// // var new_words = JSON.parse(response.new_words);
// var new_words = response.new_words;
// if (typeof new_words === 'object' && new_words.length === 1) {
// var new_word = new_words[0];
// done_callback(new_word);
// } else {
// var not_1_word =
// "qoolbar sentence " +
// sentence_or_null.toString() +
// " expects 1 word, not " +
// JSON.stringify(response);
// fail_callback(not_1_word);
// }
// },
// fail_callback
// );
// }
// };
/**
*
* action variables response (when response.is_valid is true)
* ------ ---------
* 'qoolbar_list' verbs: [{idn: ...}, {idn: ...}, ...]
* 'new_verb' name: string idn: (qstring)
* 'delete_verb' idn: q-string idn: (qstring)
* 'answer' question: string message: "Question (path) answer (text)"
* answer: string
***'sentence' vrb_idn: q-string \ pick new_words: [{idn: ...}]
*** vrb_txt: string / one
*** obj_idn: q-string
*** num: number \ pick one
*** num_add: number / (default 1)
*** txt: string (default "")
*** use_already: bool (default false)
*** - meaning use an old sentence if there is one with the same s,v,o,t,n.
*
* @param {string} action \ see table
* @param {object} variables / above
* @param {function} done_callback(response)
* response.is_valid {boolean} will be true
* response.<other_stuff> - depending on the action
* @param {function=} fail_callback(error_message) (optional)
* error_message will have already been passed to console.error()
*/
qoolbar.post = function qoolbar_post(action, variables, done_callback, fail_callback) {
done_callback = done_callback || function () {};
fail_callback = fail_callback || function (m) { console.error(m); };
variables.action = action;
// NOTE: Was this ever good for anything?
// variables.csrfmiddlewaretoken = $.cookie('csrftoken');
if (typeof qoolbar._ajax_url !== 'string') {
console.warn("Missing qoolbar.ajax_url()?");
}
/**
* @param response_object
* @param response_object.is_valid -- true or false
* @param response_object.error_message -- (if not is_valid)
*/
$.post(
qoolbar._ajax_url,
variables
).done(function (response_body) {
var response_object = JSON.parse(response_body);
response_object.original_json = response_body;
if (response_object.is_valid) {
done_callback(response_object);
} else {
var post_invalid =
"qoolbar post " +
action +
" invalid: " +
response_object.error_message;
fail_callback(post_invalid);
}
}).fail(function (jqXHR) {
var post_failed =
"qoolbar post " +
action +
" failed: " +
jqXHR.responseText;
fail_callback(post_failed);
});
};
/**
* Construct a DOM icon for the verb.
*
* @param verb -- associative array of the verb
* @param verb.name -- e.g. 'like'
* @param verb.icon_url -- from the latest iconify sentence, or null if there isn't any
* @return {jQuery}
*/
function verb_icon(verb) {
var $verb_icon;
if (verb.icon_url === null) {
var cap_first_letter = verb.name.charAt(0).toUpperCase();
$verb_icon = $('<div>', {'class': 'letter-icon', title: verb.name}).text(cap_first_letter);
} else {
// noinspection RequiredAttributes, HtmlRequiredAltAttribute
$verb_icon = $('<img>', {src: verb.icon_url, title: verb.name});
}
return $verb_icon;
}
qoolbar.i_am = function qoolbar_i_am(me_idn) {
qoolbar._me_idn = me_idn;
};
function str(x) {
if (x == null) {
return ''
} else {
return x.toString()
}
}
/**
* Tally scores for a word, from its qoolifying words.
*
* @param jbo -- array of qoolifying words (properties idn, sbj, vrb, txt, num)
* @return {{}} -- associative object whose properties (keys) are verb qstrings (e.g. '0q82_25')
* and whose values are objects with these properties associated with that verb:
* sum -- sum of each user's latest num
* my -- latest num from the current user, identified by qoolbar.i_am()
* history -- array of nums in qoolifying words (ignoring user)
* TODO: Associate nums with each user. Requires authentication.
* @property word
* @property word.sbj
* @property word.vrb
*/
function scorer(jbo) {
var scores = {};
var jbo_dict = {}; // 2D array of scores, indexed by verb idn, then subject idn
console.assert(typeof jbo === 'object', "Cannot score a " + typeof jbo);
$.each(jbo, function (_, word) {
if (!(word.vrb in scores)) {
scores[word.vrb] = {'sum': 0, 'my': null, 'history': []};
}
if (!(word.vrb in jbo_dict)) {
jbo_dict[word.vrb] = {}
}
jbo_dict[word.vrb][word.sbj] = {'num': word.num};
scores[word.vrb].history.push(word.num)
});
for (var vrb in jbo_dict) {
// Icons will appear in chronological order, if browser cooperates.
if (jbo_dict.hasOwnProperty(vrb)) {
for (var sbj in jbo_dict[vrb]) {
if (jbo_dict[vrb].hasOwnProperty(sbj)) {
var author_entry = jbo_dict[vrb][sbj];
if (qoolbar._me_idn === sbj) {
scores[vrb].my = author_entry.num;
}
scores[vrb].sum += author_entry.num;
}
}
}
}
return scores;
}
function association_in_progress() { // indicating either (1) nouns are selected,
// or (2) a verb is dragging
$(window.document.body).addClass('target-in-progress');
}
function association_resolved() { // indicating normalcy
$(window.document.body).removeClass('target-in-progress');
}
qoolbar._is_anybody_editing = false;
function end_all_editing() {
if (qoolbar._is_anybody_editing) {
qoolbar._is_anybody_editing = false;
// TODO: _is_anybody_editing -- is this now obviated?
// This _is_anybody_editing flag presumably makes end_all_editing() less of a drag.
// Otherwise there might be frequent expensive jQuery removing.
// But that's when it used to be called every document click.
// Now with a saner on-blur dynamically filtered by .qool-icon-entry,
// it would get called a lot less often.
// So it's possible end_all_editing() should be shortened
// to just the two remove calls.
// Sheesh yes, I think this flag is always true when this function is called!
$('.qool-editing').removeClass('qool-editing');
$('.qool-icon-entry').remove();
}
}
qoolbar.is_ready = true;
});
}(window, window.qoolbar = window.qoolbar || {}, window.jQuery));
// THANKS: http://www.adequatelygood.com/JavaScript-Module-Pattern-In-Depth.html
// THANKS: http://appendto.com/2010/10/how-good-c-habits-can-encourage-bad-javascript-habits-part-1/