japanese-conjugation-drill/conjugation/drill.js
2017-12-26 17:34:46 +00:00

767 lines
23 KiB
JavaScript
Executable File
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

// drill.js
var transformations = [
{ from: "negative", to: "dictionary" },
{ from: "past", to: "dictionary" },
{ from: "polite", to: "dictionary" },
{ from: "te-form", to: "dictionary" },
{ from: "potential", to: "dictionary" },
{ from: "imperative", to: "dictionary" },
{ from: "causative", to: "dictionary" },
{ from: "passive", to: "dictionary" },
{ from: "progressive", to: "dictionary" },
{ from: "dictionary", to: "negative" },
{ from: "past negative", to: "negative" },
{ from: "polite negative", to: "negative" },
{ from: "te-form negative", to: "negative" },
{ from: "potential negative", to: "negative" },
{ from: "imperative negative", to: "negative" },
{ from: "causative negative", to: "negative" },
{ from: "passive negative", to: "negative" },
{ from: "progressive negative", to: "negative" },
{ from: "dictionary", to: "past" },
{ from: "past negative", to: "past" },
{ from: "polite past", to: "past" },
{ from: "progressive past", to: "past" },
{ from: "negative", to: "past negative" },
{ from: "past", to: "past negative" },
{ from: "polite past negative", to: "past negative" },
{ from: "progressive past negative", to: "past negative" },
{ from: "dictionary", to: "polite" },
{ from: "polite past", to: "polite" },
{ from: "polite negative", to: "polite" },
{ from: "polite progressive", to: "polite" },
{ from: "past", to: "polite past" },
{ from: "polite", to: "polite past" },
{ from: "polite past negative", to: "polite past" },
{ from: "polite progressive past", to: "polite past" },
{ from: "negative", to: "polite negative" },
{ from: "polite", to: "polite negative" },
{ from: "polite past negative", to: "polite negative" },
{ from: "polite progressive negative", to: "polite negative" },
{ from: "past negative", to: "polite past negative" },
{ from: "polite past", to: "polite past negative" },
{ from: "polite negative", to: "polite past negative" },
{ from: "polite progressive past negative", to: "polite past negative" },
{ from: "dictionary", to: "te-form" },
{ from: "te-form negative", to: "te-form" },
{ from: "te-form", to: "te-form negative" },
{ from: "negative", to: "te-form negative" },
{ from: "dictionary", to: "potential" },
{ from: "potential negative", to: "potential" },
{ from: "potential", to: "potential negative" },
{ from: "negative", to: "potential negative" },
{ from: "dictionary", to: "imperative" },
{ from: "imperative negative", to: "imperative" },
{ from: "imperative", to: "imperative negative" },
{ from: "negative", to: "imperative negative" },
{ from: "dictionary", to: "causative" },
{ from: "causative negative", to: "causative" },
{ from: "causative passive", to: "causative" },
{ from: "causative", to: "causative negative" },
{ from: "negative", to: "causative negative" },
{ from: "causative passive negative", to: "causative negative" },
{ from: "dictionary", to: "passive" },
{ from: "passive negative", to: "passive" },
{ from: "causative passive", to: "passive" },
{ from: "passive", to: "passive negative" },
{ from: "negative", to: "passive negative" },
{ from: "causative passive negative", to: "passive negative" },
{ from: "causative", to: "causative passive" },
{ from: "passive", to: "causative passive" },
{ from: "causative passive negative", to: "causative passive" },
{ from: "causative negative", to: "causative passive negative" },
{ from: "passive negative", to: "causative passive negative" },
{ from: "causative passive", to: "causative passive negative" },
{ from: "dictionary", to: "progressive" },
{ from: "polite progressive", to: "progressive" },
{ from: "progressive negative", to: "progressive" },
{ from: "progressive past", to: "progressive" },
{ from: "progressive", to: "progressive negative" },
{ from: "negative", to: "progressive negative" },
{ from: "progressive past negative", to: "progressive negative" },
{ from: "polite progressive negative", to: "progressive negative" },
{ from: "progressive", to: "polite progressive" },
{ from: "polite", to: "polite progressive" },
{ from: "polite progressive negative", to: "polite progressive" },
{ from: "polite progressive past", to: "polite progressive" },
{ from: "polite progressive", to: "polite progressive negative" },
{ from: "polite negative", to: "polite progressive negative" },
{ from: "progressive negative", to: "polite progressive negative" },
{ from: "polite progressive past negative", to: "polite progressive negative" },
{ from: "past", to: "progressive past" },
{ from: "progressive", to: "progressive past" },
{ from: "progressive past negative", to: "progressive past" },
{ from: "polite progressive past", to: "progressive past" },
{ from: "past negative", to: "progressive past negative" },
{ from: "progressive negative", to: "progressive past negative" },
{ from: "progressive past", to: "progressive past negative" },
{ from: "polite progressive past negative", to: "progressive past negative" },
{ from: "polite past", to: "polite progressive past" },
{ from: "polite progressive", to: "polite progressive past" },
{ from: "progressive past", to: "polite progressive past" },
{ from: "polite progressive past negative", to: "polite progressive past" },
{ from: "polite past negative", to: "polite progressive past negative" },
{ from: "progressive past negative", to: "polite progressive past negative" },
{ from: "polite progressive past", to: "polite progressive past negative" },
{ from: "polite progressive negative", to: "polite progressive past negative" },
];
var log;
Array.prototype.randomElement = function () {
return this[Math.floor(Math.random() * this.length)]
}
// From: http://stackoverflow.com/a/2897510
new function ($) {
$.fn.getCursorPosition = function () {
var input = this.get(0);
if (!input) return; // No (input) element found
if ('selectionStart' in input) {
// Standard-compliant browsers
return input.selectionStart;
} else if (document.selection) {
// IE
input.fmcus();
var sel = document.selection.createRange();
var selLen = document.selection.createRange().text.length;
sel.moveStart('character', -input.value.length);
return sel.text.length - selLen;
}
}
}(jQuery);
// From: http://stackoverflow.com/questions/499126/jquery-set-cursor-position-in-text-area
new function ($) {
$.fn.setCursorPosition = function (pos) {
if (this.setSelectionRange) {
this.setSelectionRange(pos, pos);
} else if (this.createTextRange) {
var range = this.createTextRange();
range.collapse(true);
if (pos < 0) {
pos = $(this).val().length + pos;
}
range.moveEnd('character', pos);
range.moveStart('character', pos);
range.select();
}
}
}(jQuery);
function commaList(items, conjunction) {
if (conjunction == undefined) {
conjunction = "and";
}
var result = "";
for (var i = 0; i < items.length; i++) {
result = result + items[i];
if (i < (items.length - 2)) {
result += ", ";
}
if (i == (items.length - 2)) {
result += " " + conjunction + " ";
}
}
return result;
}
function resetLog() {
log = { "history": [] };
}
function kanaForm(words) {
if (words.constructor !== Array) {
words = [words];
}
return words.map(function (word) { return word.split(/.\[([^\]]*)\]/).join(""); } );
}
function kanjiForm(words) {
if (words.constructor !== Array) {
words = [words];
}
return words.map(function (word) { return word.split(/(.)\[[^\]]*\]/).join(""); } );
}
function getVerbForms(entry) {
var result = {
"kanji": {},
"hiragana": {},
"furigana": {}
};
Object.keys(words[entry].conjugations).forEach(function (key) {
result["kanji"][key] = kanjiForm(words[entry].conjugations[key]);
result["hiragana"][key] = kanaForm(words[entry].conjugations[key]);
result["furigana"][key] = words[entry].conjugations[key];
});
return result;
}
function wordWithFurigana(words) {
if (words.constructor !== Array) {
words = [words];
}
return words.map(function (word) {
var bits = word.split(/(.)\[([^\]]*)\]/);
while (bits.length > 1) {
bits[0] = bits[0] + "<span class='tooltip-w" + bits[2].length + "' tooltip='" + bits[2] + "'>" + bits[1] + "</span>" + bits[3];
bits.splice(1, 3);
}
return bits[0];
});
}
function processAnswerKey() {
var el = $('#answer');
var pos = el.getCursorPosition();
var val = el.val();
var last1 = val.slice(pos - 1, pos);
var last2 = val.slice(pos - 2, pos);
var last3 = val.slice(pos - 3, pos);
var replace1 = {
"a": "あ", "i": "い", "u": "う", "e": "え", "o": "お"
};
var replace2 = {
"ka": "か", "ki": "き", "ku": "く", "ke": "け", "ko": "こ",
"sa": "さ", "si": "し", "su": "す", "se": "せ", "so": "そ",
"ta": "た", "ti": "ち", "tu": "つ", "te": "て", "to": "と",
"na": "な", "ni": "に", "nu": "ぬ", "ne": "ね", "no": "の",
"ha": "は", "hi": "ひ", "hu": "ふ", "he": "へ", "ho": "ほ",
"ma": "ま", "mi": "み", "mu": "む", "me": "め", "mo": "も",
"ra": "ら", "ri": "り", "ru": "る", "re": "れ", "ro": "ろ",
"ga": "が", "gi": "ぎ", "gu": "ぐ", "ge": "げ", "go": "ご",
"za": "ざ", "zi": "じ", "zu": "ず", "ze": "ぜ", "zo": "ぞ",
"da": "だ", "di": "ぢ", "du": "づ", "de": "で", "do": "ど",
"ba": "ば", "bi": "び", "bu": "ぶ", "be": "べ", "bo": "ぼ",
"pa": "ぱ", "pi": "ぴ", "pu": "ぷ", "pe": "ぺ", "po": "ぽ",
"qa": "くぁ", "qi": "くぃ", "qu": "く", "qe": "くぇ", "qo": "くぉ",
"wa": "わ", "wi": "うぃ", "wu": "う", "we": "うぇ", "wo": "を",
"ya": "や", "yi": "い", "yu": "ゆ", "ye": "いぇ", "yo": "よ",
"fa": "ふぁ", "fi": "ふぃ", "fu": "ふ", "fe": "ふぇ", "fo": "ふぉ",
"ja": "じゃ", "ji": "じ", "ju": "じゅ", "je": "じぇ", "jo": "じょ",
"la": "ぁ", "li": "ぃ", "lu": "ぅ", "le": "ぇ", "lo": "ぉ",
"za": "ざ", "zi": "じ", "zu": "ず", "ze": "ぜ", "zo": "ぞ",
"xa": "ぁ", "xi": "ぃ", "xu": "ぅ", "xe": "ぇ", "xo": "ぉ",
"ca": "か", "ci": "し", "cu": "く", "ce": "せ", "co": "こ",
"va": "ヴぁ", "vi": "ヴぃ", "vu": "ヴ", "ve": "ヴぇ", "vo": "ヴぉ",
"lu": "っ",
"nn": "ん", "n'": "ん",
"nb": "んb", "nc": "んc", "nd": "んd", "nf": "んf", "ng": "んg",
"nh": "んh", "nj": "んj", "nk": "んk", "nl": "んl", "nm": "んm",
"np": "んp", "nq": "んq", "nr": "んr", "ns": "んs", "nt": "んt",
"nv": "んv", "nw": "んw", "nx": "んx", "nz": "んz",
"aa": "っa", "bb": "っb", "cc": "っc", "dd": "っd", "ee": "っe",
"ff": "っf", "gg": "っg", "hh": "っh", "ii": "っi", "jj": "っj",
"kk": "っk", "ll": "っl", "mm": "っm", "oo": "っo", "pp": "っp",
"qq": "っq", "rr": "っr", "ss": "っs", "tt": "っt", "uu": "っu",
"vv": "っv", "ww": "っw", "xx": "っx", "yy": "っy", "zz": "っz",
};
var replace3 = {
"kya": "きゃ", "kyi": "きぃ", "kyu": "きゅ", "kye": "きぇ", "kyo": "きょ",
"sha": "しゃ", "shi": "し", "shu": "しゅ", "she": "しぇ", "sho": "しょ",
"cha": "ちゃ", "chi": "ち", "chu": "ちゅ", "che": "ちぇ", "cho": "ちょ",
"nya": "にゃ", "nyi": "にぃ", "nyu": "にゅ", "nye": "にぇ", "nyo": "にょ",
"hya": "ひゃ", "hyi": "ひぃ", "hyu": "ひゅ", "hye": "ひぇ", "hyo": "ひょ",
"mya": "みゃ", "myi": "みぃ", "myu": "みゅ", "mye": "みぇ", "myo": "みょ",
"rya": "りゃ", "ryi": "りぃ", "ryu": "りゅ", "rye": "りぇ", "ryo": "りょ",
"gya": "ぎゃ", "gyi": "ぎぃ", "gyu": "ぎゅ", "gye": "ぎぇ", "gyo": "ぎょ",
"zya": "じゃ", "zyi": "じぃ", "zyu": "じゅ", "zye": "じぇ", "zyo": "じょ",
"dya": "ぢゃ", "dyi": "ぢぃ", "dyu": "ぢゅ", "dye": "ぢぇ", "dyo": "ぢょ",
"bya": "びゃ", "byi": "びぃ", "byu": "びゅ", "bye": "びぇ", "byo": "びょ",
"pya": "ぴゃ", "pyi": "ぴぃ", "pyu": "ぴゅ", "pye": "ぴぇ", "pyo": "ぴょ",
"shi": "し",
"tsu": "つ",
};
if (replace3[last3]) {
val = val.slice(0, pos - 3) + replace3[last3] + val.slice(pos, -1);
el.val(val);
el.setCursorPosition(pos - 3 + replace3[last3].length);
} else if (replace2[last2]) {
val = val.slice(0, pos - 2) + replace2[last2] + val.slice(pos, -1);
el.val(val);
el.setCursorPosition(pos - 2 + replace2[last2].length);
} else if (replace1[last1]) {
val = val.slice(0, pos - 1) + replace1[last1] + val.slice(pos, -1);
el.val(val);
el.setCursorPosition(pos - 1 + replace1[last1].length);
}
}
function validQuestion(entry, forms, transformation, options) {
var valid = true;
transformation.tags.forEach(function (type) {
if (options[type] == false) {
valid = false;
}
});
if (options[words[entry].group] == false) {
valid = false;
}
if (!forms["furigana"][transformation.from])
valid = false;
if (!forms["furigana"][transformation.to])
valid = false;
return valid;
}
function generateQuestion() {
var entry;
var to_form;
var from_form;
var forms;
var count = 0;
while (true) {
if (count++ == 10000) {
showSplash();
return;
}
entry = Object.keys(words).randomElement();
transformation = transformations.randomElement();
from_form = transformation.from;
to_form = transformation.to;
forms = getVerbForms(entry);
var valid = validQuestion(entry, forms, transformation, getOptions());
// Modify the chance of trick questions so that they appear on average 25%
// of the time. When trick questions are active then 50% of the
// transformation structure are trick questions and so a 33% filter here
// will achieve the 25% because this test is only performed when a trick
// question has been selected.
if (transformation.tags.indexOf('trick') != -1) {
if (Math.random() > 0.333) {
valid = false;
}
}
if (valid) {
break;
}
}
var kanjiForms = forms["kanji"];
var kanaForms = forms["hiragana"];
var furiganaForms = forms["furigana"];
var givenWord = wordWithFurigana(furiganaForms[from_form]).randomElement();
var question = "What is the " + transformation.phrase + " version of " +
givenWord + "?";
var answer = kanjiForms[to_form];
var answer2 = kanaForms[to_form];
$('#question').html(question);
window.questionData = {
entry: entry,
transformation: transformation,
question: question,
answer: kanjiForms[to_form],
answer2: answer2,
answerWithFurigana: wordWithFurigana(furiganaForms[to_form]),
givenWord: givenWord,
};
// Construct the explanation page.
var data = window.questionData;
var groupLabels = {
"godan" : "godan verb",
"ichidan" : "ichidan verb",
"iku" : "godan verb",
"suru" : "suru verb",
"kuru" : "special verb",
"i-adjective" : "い adjective",
"ii" : "i-adjective",
"na-adjective" : "な adjective",
};
var dictionary = words[data.entry].conjugations["dictionary"]
if (words[data.entry].group == "na-adjective") {
dictionary = dictionary.replace(/だ$/, '')
}
dictionary = wordWithFurigana(dictionary);
$('#explain-given').html(givenWord);
$('#explain-given-tags').html(data.transformation.from_tags.map(function (tag) { return "<span class='tag'>" + tag + "</span>"; }).join(" "));
$('.explain-given-dictionary').html(dictionary);
$('#explain-group').html(groupLabels[words[data.entry].group]);
$('.explain-transform').html(data.transformation.phrase);
$('.explain-answer-tags').html(data.transformation.to_tags.map(function (tag) { return "<span class='tag'>" + tag + "</span>"; }).join(" "));
$('.explain-answer-tags2').html(data.transformation.to_tags.join(" "));
$('.explain-answer').html(commaList(questionData.answerWithFurigana, "or"));
$('.explain-answer-as-list').empty();
questionData.answerWithFurigana.forEach(function (answer) {
console.log("Doing answer");
$('.explain-answer-as-list').append("<li>" + answer);
});
if (window.questionData.transformation.tags.indexOf("trick") != -1) {
$('.explain-trick').show();
$('.explain-no-trick').hide();
} else {
$('.explain-trick').hide();
$('.explain-no-trick').show();
}
if (data.transformation.to == "dictionary") {
$('.explain-hide-end').hide();
} else {
$('.explain-hide-end').show();
}
if (data.answer.length == 1) {
$('.explain-answer-single').show();
$('.explain-answer-multiple').hide();
} else {
$('.explain-answer-single').hide();
$('.explain-answer-multiple').show();
}
$('#next').prop('disabled', true);
$('#response').html("");
$('#message').html("");
$('#proceed').hide();
$('#explanation').hide();
$('#input').show();
$('#answer').focus();
$('#answer').on('input', processAnswerKey);
}
function processAnswer() {
var questionData = window.questionData;
var response = $('#answer').val().trim();
if (response == "")
return;
var correct = ((questionData.answer.indexOf(response) != -1) || (questionData.answer2.indexOf(response) != -1));
var klass = correct ? "correct" : "incorrect";
log.history.push({
"question": questionData.question,
"response": response,
"answer": questionData.answerWithFurigana,
"kana": questionData.answer2,
"correct": correct
});
$('#answer').val("");
$('#response').prop('class', klass).text(response);
$('#next').prop('disabled', false);
if (correct) {
$('#message').html("");
} else {
$('#message').html("<div>The correct answer was " + commaList(questionData.answerWithFurigana, "or") + " <input type='button' value='Explain' onclick='explain()'></div>");
}
$('#input').hide();
$('#proceed').show();
$('#explanation').hide();
$('#proceed button').focus();
updateHistoryView(log);
}
function updateHistoryView(log) {
var review = $('<table>');
var total = 0;
var correct = 0;
var header_tr = $('<tr>');
header_tr.append($('<th>Question</th>'));
header_tr.append($('<th>Answer</th>'));
header_tr.append($('<th>Response</th>'));
review.append(header_tr);
log.history.forEach(function (entry) {
total++;
if (entry.correct) {
correct++;
}
var tr = $('<tr>');
var td1 = $('<td>');
var td2 = $('<td>');
var td3 = $('<td>');
td1.html(entry.question);
td2.html(commaList(entry.answer, "or"));
td3.text(entry.response);
tr.append(td1);
tr.append(td2);
tr.append(td3);
if (entry.correct) {
td3.append("<span class='answer-correct'> ○</span>");
} else {
td3.append("<span class='answer-wrong'> ×</span>");
}
review.append(tr);
});
$('#history').empty().append(review);
$('#history').append("<p>" + correct + " of " + total + " correct.</p>");
}
function proceed() {
if (log.history.length == $('#numQuestions').val()) {
endQuiz();
} else {
generateQuestion();
}
}
function showSplash() {
$('#splash').show();
$('#quizSection').hide();
$('#scoreSection').hide();
$('#go').focus();
}
function startQuiz() {
$('#splash').hide();
$('#quizSection').show();
$('#scoreSection').hide();
resetLog();
generateQuestion();
}
function endQuiz() {
$('#splash').hide();
$('#quizSection').hide();
$('#scoreSection').show();
$('#backToStart').focus();
}
function arrayDifference(a, b) {
// From http://stackoverflow.com/a/1723220
return a.filter(function (x) { return b.indexOf(x) < 0 });
}
function arrayUnique(arr) {
return arr.filter(function (value, index, self) {
return self.indexOf(value) === index;
});
}
function calculateTransitions() {
function getTags(str) {
var tags = str.split(" ");
if ((tags.length == 1) && (tags[0] == "plain")) {
tags = [];
}
return tags;
}
function calculateTags(tags) {
tags = tags.split(" ");
if (tags.indexOf("polite") == -1) {
tags.splice(0, 0, "plain");
}
if (tags.indexOf("dictionary") != -1) {
tags.splice(tags.indexOf("dictionary"), 1);
}
return tags;
}
transformations.forEach(function (transformation) {
var from = getTags(transformation.from);
var to = getTags(transformation.to);
var from_extra = {
"negative": "affirmative",
"past": "present",
"polite": "plain",
"te-form": "non-て",
"potential": "non-potential",
"imperative": "non-imperative",
"causative": "non-causative",
"passive": "active",
"progressive": "non-progressive",
};
var to_extra = {
"negative": "negative",
"past": "past",
"polite": "polite",
"te-form": "て",
"potential": "potential",
"imperative": "imperative",
"causative": "causative",
"passive": "passive",
"progressive": "progressive",
};
var phrase;
phrase = phrase || from_extra[arrayDifference(from, to)[0]];
phrase = phrase || to_extra[arrayDifference(to, from)[0]];
transformation.phrase = phrase;
transformation.from_tags = calculateTags(transformation.from);
transformation.to_tags = calculateTags(transformation.to);
transformation.tags = arrayUnique(calculateTags(transformation.from).concat(calculateTags(transformation.to)));
});
// Add trick forms
var trick_forms = [];
transformations.forEach(function (transformation) {
trick_forms.push({
from: transformation.to,
to: transformation.to,
phrase: transformation.phrase,
from_tags: transformation.to_tags,
to_tags: transformation.to_tags,
tags: transformation.tags.concat(["trick"])
});
});
transformations = transformations.concat(trick_forms);
}
function updateOptionSummary() {
// Calculate how many questions will apply
var options = getOptions();
var applicable = 0;
Object.keys(words).forEach(function (word) {
var forms = getVerbForms(word);
transformations.forEach(function (transformation) {
if (validQuestion(word, forms, transformation, options)) {
applicable++;
}
});
});
$("#questionCount").text(applicable);
}
function explain() {
$('#explanation').show();
$('#explain-proceed-button').focus();
}
function getOptions() {
var options = ["plain", "polite", "negative", "past", "te-form",
"progressive", "potential", "imperative", "passive", "causative",
"godan", "ichidan", "iku", "kuru", "suru", "i-adjective", "na-adjective",
"ii", "trick"];
var result = {};
options.forEach(function (option) {
result[option] = $('#' + option).is(':checked') != false;
});
return result;
}
$('window').ready(function () {
calculateTransitions();
$('#go').click(startQuiz);
$('#backToStart').click(showSplash);
$('div.options input').click(updateOptionSummary);
$('input#trick').click(updateOptionSummary);
updateOptionSummary();
showSplash();
});