Fix for Uporzadkuj.
[redakcja.git] / redakcja / static / edumed / js / edumed.coffee
1
2 $ = jQuery
3
4 class Binding
5   constructor: (@handler, @element) ->
6     $(@element).data(@handler, this)
7
8
9 class EduModule extends Binding
10   constructor: (element) ->
11     super 'edumodule', element
12
13     # $("[name=teacher-toggle]").change (ev) =>
14     #   if $(ev.target).is(":checked")
15     #     $(".teacher", @element).addClass "show"
16     #   else
17     #     $(".teacher", @element).removeClass "show"
18
19
20 class Exercise extends Binding
21   constructor: (element) ->
22     super 'exercise', element
23     # just save the html to reset the exercise
24     $(@element).data("exercise-html", $(@element).html())
25
26     $(".check", @element).click (ev) =>
27       @check()
28       $(".retry", @element).show()
29       $(".check", @element).hide()
30     $(".retry", @element).click (ev) =>
31       @retry()
32     $('.solutions', @element).click =>
33       @show_solutions()
34       $(".comment", @element).show()
35     $('.reset', @element).click =>
36       @reset()
37
38   retry: ->
39     $(".correct, .incorrect", @element).removeClass("correct incorrect")
40     $(".check", @element).show()
41     $(".retry", @element).hide()
42
43   reset: ->
44     $(@element).html($(@element).data('exercise-html'))
45     exercise @element
46
47   piece_correct: (qpiece) ->
48     $(qpiece).removeClass('incorrect').addClass('correct')
49
50   piece_incorrect: (qpiece) ->
51     $(qpiece).removeClass('correct').addClass('incorrect')
52
53   check: ->
54     scores = []
55     $(".question", @element).each (i, question) =>
56       scores.push(@check_question question)
57
58     score = [0, 0, 0]
59     $.each scores, (i, s) ->
60       score[0] += s[0]
61       score[1] += s[1]
62       score[2] += s[2]
63     @show_score(score)
64
65   show_solutions: ->
66     @reset()
67     $(".question", @element).each (i, question) =>
68       @solve_question question
69
70   # Parses a list of values, separated by space or comma.
71   # The list is read from data attribute of elem using data_key
72   # Returns a list with elements
73   # eg.: things_i_need: "house bike tv playstation"
74   # yields ["house", "bike", "tv", "playstation"]
75   # If optional numbers argument is true, returns list of numbers
76   # instead of strings
77   get_value_list: (elem, data_key, numbers) ->
78     vl = $(elem).attr("data-" + data_key).split(/[ ,]+/).map($.trim) #.map((x) -> parseInt(x))
79     if numbers
80       vl = vl.map((x) -> parseInt(x))
81     return vl
82
83   # Parses a list of values, separated by space or comma.
84   # The list is read from data attribute of elem using data_key
85   # Returns a 2-element list with mandatory and optional
86   # items. optional items are marked with a question mark on the end
87   # eg.: things_i_need: "house bike tv? playstation?"
88   # yields [[ "house", "bike"], ["tv", "playstation"]]
89   get_value_optional_list: (elem, data_key) ->
90     vals = @get_value_list(elem, data_key)
91     mandat = []
92     opt = []
93     for v in vals
94       if v.slice(-1) == "?"
95         opt.push v.slice(0, -1)
96       else
97         mandat.push v
98     return [mandat, opt]
99
100   show_score: (score) ->
101     $msg = $(".message", @element)
102     $msg.text("Wynik: #{score[0]} / #{score[2]}")
103     if score[0] >= score[2] and score[1] == 0
104       $msg.addClass("maxscore")
105     else
106       $msg.removeClass("maxscore")
107
108
109   draggable_equal: ($draggable1, $draggable2) ->
110     return false
111
112   draggable_accept: ($draggable, $droppable) ->
113     dropped = $droppable.closest("ul, ol").find(".draggable")
114     for d in dropped
115       if @draggable_equal $draggable, $(d)
116         return false
117     return true
118
119   draggable_move: ($draggable, $placeholder, ismultiple) ->
120     $added = $draggable.clone()
121     $added.data("original", $draggable.get(0))
122     if not ismultiple
123       $draggable.addClass('disabled').draggable('disable')
124
125     $placeholder.after($added)
126     if not $placeholder.hasClass('multiple')
127       $placeholder.hide()
128     if $added.is(".add-li")
129       $added.wrap("<li/>")
130
131     $added.append('<span class="remove">x</span><div class="clr"></div>')
132     $('.remove', $added).click (ev) =>
133       @retry()
134       if not ismultiple
135         $($added.data('original')).removeClass('disabled').draggable('enable')
136
137       if $added.is(".add-li")
138         $added = $added.closest('li')
139       $added.prev(".placeholder:not(.multiple)").show()
140       $added.remove()
141
142
143 ## XXX co z issortable?
144   dragging: (ismultiple, issortable) ->
145     $(".question", @element).each (i, question) =>
146       draggable_opts =
147         revert: 'invalid'
148         helper: 'clone'
149         start: @retry
150
151       $(".draggable", question).draggable(draggable_opts)
152       self = this
153       $(".placeholder", question).droppable
154         accept: (draggable) ->
155           $draggable = $(draggable)
156           is_accepted = true
157
158           if not $draggable.is(".draggable")
159             is_accepted = false
160
161           if is_accepted
162             is_accepted= self.draggable_accept $draggable, $(this)
163
164           if is_accepted
165             $(this).addClass 'accepting'
166           else
167             $(this).removeClass 'accepting'
168           return is_accepted
169
170         drop: (ev, ui) =>
171           $(ev.target).removeClass 'accepting dragover'
172
173           @draggable_move $(ui.draggable), $(ev.target), ismultiple
174
175           # $added = $(ui.draggable).clone()
176           # $added.data("original", ui.draggable)
177           # if not ismultiple
178           #   $(ui.draggable).addClass('disabled').draggable('disable')
179
180           # $(ev.target).after(added)
181           # if not $(ev.target).hasClass('multiple')
182           #   $(ev.target).hide()
183           # $added.append('<span class="remove">x</span>')
184           # $('.remove', added).click (ev) =>
185           #   $added.prev(".placeholder:not(.multiple)").show()
186           #   if not ismultiple
187           #     $added.data('original').removeClass('disabled').draggable('enable')
188           #   $(added).remove()
189
190         over: (ev, ui) ->
191           $(ev.target).addClass 'dragover'
192
193
194         out: (ev, ui) ->
195           $(ev.target).removeClass 'dragover'
196
197
198
199 class Wybor extends Exercise
200   constructor: (element) ->
201     super element
202     $(".question-piece input", element).change(@retry);
203
204
205   check_question: (question) ->
206     all = 0
207     good = 0
208     bad = 0
209     solution = @get_value_list(question, 'solution')
210     $(".question-piece", question).each (i, qpiece) =>
211       piece_no = $(qpiece).attr 'data-no'
212       piece_name = $(qpiece).attr 'data-name'
213       if piece_name
214         should_be_checked = solution.indexOf(piece_name) >= 0
215       else
216         should_be_checked = solution.indexOf(piece_no) >= 0
217       is_checked = $("input", qpiece).is(":checked")
218
219       if should_be_checked
220         all += 1
221
222       if is_checked
223         if should_be_checked
224           good += 1
225           @piece_correct qpiece
226         else
227           bad += 1
228           @piece_incorrect qpiece
229       else
230         $(qpiece).removeClass("correct,incorrect")
231
232     return [good, bad, all]
233
234   solve_question: (question) ->
235     solution = @get_value_list(question, 'solution')
236     $(".question-piece", question).each (i, qpiece) =>
237       piece_no = $(qpiece).attr 'data-no'
238       piece_name = $(qpiece).attr 'data-name'
239       if piece_name
240         should_be_checked = solution.indexOf(piece_name) >= 0
241       else
242         should_be_checked = solution.indexOf(piece_no) >= 0
243       console.log("check " + $("input[type=checkbox]", qpiece).attr("id") + " -> " + should_be_checked)
244       $("input[type=checkbox],input[type=radio]", qpiece).prop 'checked', should_be_checked
245
246
247
248 class Uporzadkuj extends Exercise
249   constructor: (element) ->
250     super element
251     $('ol, ul', @element).sortable({ items: "> li", start: @retry })
252
253   check_question: (question) ->
254     positions = @get_value_list(question, 'original', true)
255     sorted = positions.sort (a, b) -> a - b
256     pkts = $('.question-piece', question)
257
258     correct = 0
259     bad = 0
260     all = 0
261
262     for pkt in [0...pkts.length]
263       all += 1
264       if pkts.eq(pkt).data('pos') == sorted[pkt]
265         correct += 1
266         @piece_correct pkts.eq(pkt)
267       else
268         bad += 1
269         @piece_incorrect pkts.eq(pkt)
270     return [correct, bad, all]
271
272   solve_question: (question) ->
273     pkts = $('.question-piece', question)
274     pkts.sort (a, b) -> return $(a).data('pos') - $(b).data('pos')
275     parent = pkts.eq(0).parent()
276     for p in pkts
277       parent.append(p)
278
279
280 # XXX propozycje="1/0"
281 class Luki extends Exercise
282   constructor: (element) ->
283     super element
284     @dragging false, false
285
286   check: ->
287     all = $(".placeholder", @element).length
288     correct = 0
289     bad = 0
290     $(".placeholder + .question-piece", @element).each (i, qpiece) =>
291       $placeholder = $(qpiece).prev(".placeholder")
292       if $placeholder.data('solution') == $(qpiece).data('no')
293         @piece_correct qpiece
294         correct += 1
295       else
296         bad += 1
297         @piece_incorrect qpiece
298
299     @show_score [correct, bad, all]
300
301   solve_question: (question) ->
302     $(".placeholder", question).each (i, placeholder) =>
303       $qp = $(".question-piece[data-no=" + $(placeholder).data('solution') + "]", question)
304       @draggable_move $qp, $(placeholder), false
305
306
307 class Zastap extends Exercise
308   constructor: (element) ->
309     super element
310     $(".paragraph", @element).each (i, par) =>
311       @wrap_words $(par), $('<span class="placeholder zastap"/>')
312     @dragging false, false
313
314   check: ->
315     all = 0
316     correct = 0
317     bad = 0
318
319     $(".paragraph", @element).each (i, par) =>
320       $(".placeholder", par).each (j, qpiece) =>
321         $qp = $(qpiece)
322         $dragged = $qp.next(".draggable")
323         if $qp.data("solution")
324           if $dragged and $qp.data("solution") == $dragged.data("no")
325             @piece_correct $dragged
326             correct += 1
327 #          else -- we dont mark enything here, so not to hint user about solution. He sees he hasn't used all the draggables
328
329           all += 1
330
331     @show_score [correct, bad, all]
332
333   show_solutions: ->
334     @reset()
335     $(".paragraph", @element).each (i, par) =>
336       $(".placeholder[data-solution]", par).each (j, qpiece) =>
337         $qp = $(qpiece)
338         $dr = $(".draggable[data-no=" + $qp.data('solution') + "]", @element)
339         @draggable_move $dr, $qp, false
340
341
342   wrap_words: (element, wrapper) ->
343     # This function wraps each word of element in wrapper, but does not descend into child-tags of element.
344     # It doesn't wrap things between words (defined by ignore RE below). Warning - ignore must begin with ^
345     ignore = /^[ \t.,:;()]+/
346
347     insertWrapped = (txt, elem) ->
348       nw = wrapper.clone()
349       $(document.createTextNode(txt))
350         .wrap(nw).parent().attr("data-original", txt).insertBefore(elem)
351
352     for j in [element.get(0).childNodes.length-1..0]
353       chld = element.get(0).childNodes[j]
354       if chld.nodeType == document.TEXT_NODE
355         len = chld.textContent.length
356         wordb = 0
357         i = 0
358         while i < len
359           space = ignore.exec(chld.textContent.substr(i))
360           if space?
361             if wordb < i
362               insertWrapped(chld.textContent.substr(wordb, i-wordb), chld)
363
364             $(document.createTextNode(space[0])).insertBefore(chld)
365             i += space[0].length
366             wordb = i
367           else
368             i = i + 1
369         if wordb < len - 1
370           insertWrapped(chld.textContent.substr(wordb, len - 1 - wordb), chld)
371         $(chld).remove()
372
373
374 class Przyporzadkuj extends Exercise
375   is_multiple: ->
376     for qp in $(".question-piece", @element)
377       if $(qp).attr('data-solution').split(/[ ,]+/).length > 1
378         return true
379     return false
380
381   constructor: (element) ->
382     super element
383
384     @multiple = @is_multiple()
385
386     @dragging @multiple, true
387
388   draggable_equal: (d1, d2) ->
389     return d1.data("no") == d2.data("no")
390
391
392   check_question: (question) ->
393     # subjects placed in predicates
394     minimum = $(question).data("minimum")
395     count = 0
396     bad_count = 0
397     all = 0
398     if not minimum
399       self = this
400       $(".subject .question-piece", question).each (i, el) ->
401         v = self.get_value_optional_list el, 'solution'
402         mandatory = v[0]
403         all += mandatory.length
404
405     for pred in $(".predicate [data-predicate]", question)
406       pn = $(pred).attr('data-predicate')
407       #if minimum?
408       #  all += minimum
409
410       for qp in $(".question-piece", pred)
411         v = @get_value_optional_list qp, 'solution'
412         mandatory = v[0]
413         optional = v[1]
414
415         if mandatory.indexOf(pn) >= 0 or (minimum and optional.indexOf(pn) >= 0)
416           count += 1
417           @piece_correct qp
418         else
419           bad_count += 1
420           @piece_incorrect qp
421
422     return [count, bad_count, all]
423
424   solve_question: (question) ->
425     minimum = $(question).data("min")
426
427     for qp in $(".subject .question-piece", question)
428       v = @get_value_optional_list qp, 'solution'
429       mandatory = v[0]
430       optional = v[1]
431
432       if minimum
433         draggables = mandatory.count(optional)[0...minimum]
434       else
435         draggables = mandatory
436       for m in draggables
437         $pr = $(".predicate [data-predicate=" + m + "]", question)
438         $ph = $pr.find ".placeholder:visible"
439         @draggable_move $(qp), $ph.eq(0), @multiple
440
441
442
443 class PrawdaFalsz extends Exercise
444   constructor: (element) ->
445     super element
446
447     for qp in $(".question-piece", @element)
448       $(".true", qp).click (ev) =>
449         ev.preventDefault()
450         @retry()
451         $(ev.target).closest(".question-piece").data("value", "true")
452         $(ev.target).addClass('chosen').siblings('a').removeClass('chosen')
453       $(".false", qp).click (ev) =>
454         ev.preventDefault()
455         @retry()
456         $(ev.target).closest(".question-piece").data("value", "false")
457         $(ev.target).addClass('chosen').siblings('a').removeClass('chosen')
458
459
460   check_question: ->
461     all = 0
462     good = 0
463     bad = 0
464     for qp in $(".question-piece", @element)
465       if $(qp).data("solution").toString() == $(qp).data("value")
466         good += 1
467         @piece_correct qp
468       else
469         bad += 1
470         @piece_incorrect qp
471
472       all += 1
473
474     return [good, bad, all]
475
476   show_solutions: ->
477     @reset()
478     for qp in $(".question-piece", @element)
479       if $(qp).data('solution') == true
480         $(".true", qp).click()
481       else
482         $(".false", qp).click()
483
484
485 ##########
486
487 exercise = (ele) ->
488   es =
489     wybor: Wybor
490     uporzadkuj: Uporzadkuj
491     luki: Luki
492     zastap: Zastap
493     przyporzadkuj: Przyporzadkuj
494     prawdafalsz: PrawdaFalsz
495
496
497   cls = es[$(ele).attr('data-type')]
498   new cls(ele)
499
500
501 window.edumed =
502   'EduModule': EduModule
503
504
505
506
507 $(document).ready () ->
508   new EduModule($("#book-text"))
509
510   $(".exercise").each (i, el) ->
511     exercise(this)