// ==UserScript==
// @name Wanikani Self-Study Plus
// @namespace wkselfstudyplus
// @description Adds an option to add and review your own custom vocabulary
// @include *.wanikani.com/*
// @include *.wanikani.com/chat/*
// @exclude *.wanikani.com
// @include *.wanikani.com/dashboard*
// @include *.wanikani.com/community*
// @version 0.2.0
// @author shudouken and Ethan
// @run-at document-end
// @grant none
// ==/UserScript==
/*
* This script is licensed under the Creative Commons License
* "Attribution-NonCommercial 3.0 Unported"
*
* More information at:
* http://creativecommons.org/licenses/by-nc/3.0/
*/
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
var buildNode = require('./buildnode.js');
// Create DOM for 'add' window
var addElement = buildNode('div', {id: "add", className: "WKSS"});
var formElement = buildNode('form', {id: "addForm"});
addElement.appendChild(formElement);
var buttonElement = buildNode('button', {id: "AddCloseBtn", className: "wkss-close", type: "reset"});
formElement.appendChild(buttonElement);
var iconElement = buildNode('i', {className: "icon-remove"});
buttonElement.appendChild(iconElement);
var headerElement = buildNode('h1');
formElement.appendChild(headerElement);
var headerText = document.createTextNode("Add a new Item");
headerElement.appendChild(headerText);
var inputKanjiElement = buildNode('input', {id: "addKanji", type: "text", placeholder: "Enter 漢字, ひらがな or カタカナ"});
formElement.appendChild(inputKanjiElement);
var inputReadingElement = buildNode('input', {id: "addReading", type: "text", title: "Leave empty to add vocabulary like する (to do)", placeholder: "Enter reading"});
formElement.appendChild(inputReadingElement);
var inputMeaningElement = buildNode('input', {id: "addMeaning", type: "text", placeholder: "Enter meaning"});
formElement.appendChild(inputMeaningElement);
var pElement = buildNode('p', {id: "addStatus"});
formElement.appendChild(pElement);
var pText = document.createTextNode("Ready to add..");
pElement.appendChild(pText);
var execButtonElement = buildNode('button', {id: "AddItemBtn", type: "button"});
formElement.appendChild(execButtonElement);
var execText = document.createTextNode("Add new Item");
execButtonElement.appendChild(execText);
module.exports = addElement;
},{"./buildnode.js":2}],2:[function(require,module,exports){
/** Builds a node element with an id and className and other attributes if provided
* @param {string} type - The type of element to create ('div', 'p', etc...)
* @param {object} [options]
* @param {string} options.id - The id of the node
* @param {string} options.className - One or more classes for the element seperated by spaces
* @returns {HTMLElement} The node built as specified
*/
var buildNode = function(type, options){
var node = document.createElement(type);
for (var option in options) if (options.hasOwnProperty(option)) {
if (option === "className" || option === "id"){
node[option] = options[option];
}
else {
node.setAttribute(option, options[option]);
}
}
return node;
};
module.exports = buildNode;
},{}],3:[function(require,module,exports){
var buildNode = require('./buildnode.js');
// tag, id, className, other, childNodes
/* IWindow:
{
id: "WKSS-edit",
className: "WKSS",
childNodes:[{
tag: 'form',
id: "WKSS-editForm",
childNodes:[{
tag: 'button',
id: "WKSS-editCloseBtn",
className: "WKSS-close"
childNodes:[{
tag: 'i',
className: "icon-remove"
}]
},{
tag: 'h1',
childNodes:[
"Edit your Vocab" <--- string types for text node
]
},{
tag: 'select',
id: "editWindow",
other: {size: "8"} <--- 'other' avoids clashes with HTMLElement attributes just in case
},{
tag: 'input',
other:{
type: "text",
name: "",
size: "40",
placeholder: "Select vocab, click edit, change and save!"
},
id: "editItem"
},{
tag: 'p',
id: "editStatus"
childNodes:["Ready to edit..."]
},{
tag: 'button',
id: "EditEditBtn",
other: {type: "button"},
childNodes:["Edit"]
},{
tag: 'button',
id: "EditSaveBtn",
other:{type: "button"},
childNodes:["Save"]
},{
tag: 'button',
id: "EditDeleteBtn",
other: {type: "button", title: "Delete selected item"},
childNodes:["Delete"]
},{
tag: 'button',
id: "EditDeleteAllBtn",
other: {type: "button", title: "本当にやるの?"},
childNodes:["Delete All"]
},{
tag: 'button',
id: "ResetLevelsBtn",
other: {type: "button"},
childNodes:["Reset levels"]
}]
}]
}
*/
var struct = {
id: "WKSS-edit",
className: "WKSS",
childNodes:[{
tag: 'form',
id: "WKSS-editForm",
childNodes:[{
tag: 'button',
id: "WKSS-editCloseBtn",
className: "WKSS-close",
childNodes:[{
tag: 'i',
className: "icon-remove"
}]
},{
tag: 'h1',
childNodes:["Edit your Vocab"]
},{
tag: 'select',
id: "editWindow",
other: {size: "8"}
},{
tag: 'input',
other:{
type: "text",
name: "",
size: "40",
placeholder: "Select vocab, click edit, change and save!"
},
id: "editItem"
},{
tag: 'p',
id: "editStatus",
childNodes:["Ready to edit..."]
},{
tag: 'button',
id: "EditEditBtn",
other: {type: "button"},
childNodes:["Edit"]
},{
tag: 'button',
id: "EditSaveBtn",
other:{type: "button"},
childNodes:["Save"]
},{
tag: 'button',
id: "EditDeleteBtn",
other: {type: "button", title: "Delete selected item"},
childNodes:["Delete"]
},{
tag: 'button',
id: "EditDeleteAllBtn",
other: {type: "button", title: "本当にやるの?"},
childNodes:["Delete All"]
},{
tag: 'button',
id: "ResetLevelsBtn",
other: {type: "button"},
childNodes:["Reset levels"]
}]
}]
};
// 'this' context is node to attach to
var attachChildNode = function(childNode){
var el;
if ("string" === typeof childNode){ //TextNode
el = document.createTextNode(childNode);
}
else{
el = buildNode(childNode.tag, {id: childNode.id, className: childNode.className});
for (var attr in childNode.other){
el.setAttribute(attr, childNode.other[attr]);
}
if (childNode.childNodes){
childNode.childNodes.forEach(attachChildNode, el);
}
}
this.appendChild(el);
};
/** Takes a JSON object with the structure of the window to create and builds a DIVElement from that
* @param {IWindow} windowStructure
* @returns {DIVElement} The specified window.
*/
var buildWindow = function(windowStructure) {
var resultWindow = buildNode('div', {id: windowStructure.id, className: windowStructure.className});
for (var attr in windowStructure.other){
resultWindow.setAttribute(attr, windowStructure.other[attr]);
}
if (windowStructure.childNodes){
windowStructure.childNodes.forEach(attachChildNode, resultWindow);
}
return resultWindow;
};
/*
{
var editForm = buildNode('form', {id: "WKSS-editForm"});
editWindow.appendChild(editForm);
var editCloseButton = buildNode('button', {id: "WKSS-editCloseBtn", className: "WKSS-close"});
editForm.appendChild(editCloseButton);
editCloseButton.appendChild(buildNode('i', {className: "icon-remove"}));
var h1Element = buildNode('h1');
editForm.appendChild(h1Element);
h1Element.appendChild(document.createTextNode("Edit your Vocab"));
var selectElement = buildNode('select', {id: "editWindow", size: "8"});
editForm.appendChild(selectElement);
var editItemText = buildNode('input', {type: "text" id: "editItem" name: "" size: "40" placeholder: "Select vocab, click edit, change and save!"});
editForm.appendChild(editItemText);//..
var editStatus = buildNode('p', {id: "editStatus"});
editForm.appendChild(editStatus);
editStatus.appendChild(document.createTextNode("Ready to edit.."));
var editButton = buildNode('button', {id: "EditEditBtn", type: "button"});
editForm.appendChild(editButton);
editButton.appendChild(document.createTextNode("Edit"));
var editSave = buildNode('button', {id: "EditSaveBtn", type: "button"});
editForm.appendChild(editSave);
editSave.appendChild(document.createTextNode("Save"));
var editDelete = buildNode('button', {id: "EditDeleteBtn", type: "button", title: "Delete selected item"});
editForm.appendChild(editDelete);
editDelete.appendChild(document.createTextNode("Delete"));
var editDeleteAll = buildNode('button', {id: "EditDeleteAllBtn", type: "button", title: "本当にやるの?"});
editForm.appendChild(editDeleteAll);
editDeleteAll.appendChild(document.createTextNode("Delete All"));
var editResetLevels = buildNode('button', {id: "ResetLevelsBtn", type: "button"});
editForm.appendChild(editResetLevels);
editResetLevels.appendChild(document.createTextNode("Reset levels"));
return editWindow;
};*/
module.exports = buildWindow;
},{"./buildnode.js":2}],4:[function(require,module,exports){
module.exports = function(){
var kanji = $("#addKanji").val().toLowerCase();
var reading = $("#addReading").val().toLowerCase().split(/[,、]+\s*/); //split at , or 、followed by 0 or any number of spaces
var meaning = $("#addMeaning").val().toLowerCase().split(/[,、]+\s*/);
var success = false; //initalise values
var meanlen = 0;
var i = meaning.length;
while (i--){
meanlen += meaning[i].length;
}
//input is invalid: prompt user for valid input
var item = {};
if (kanji.length === 0 || meanlen === 0) {
$("#addStatus").text("One or more required fields are empty!");
if (kanji.length === 0) {
$("#addKanji").addClass("error");
} else {
$("#addKanji").removeClass("error");
}
if (meanlen === 0) {
$("#addMeaning").addClass("error");
} else {
$("#addMeaning").removeClass("error");
}
} else {
if (debugging) {console.log("building item: "+kanji);}
item.kanji = kanji;
item.reading = reading; //optional
item.meaning = meaning;
success = true;
if (debugging) {console.log("item is valid");}
}
//on successful creation of item
if (success) {
//clear error layout to required fields
$("#addKanji").removeClass("error");
$("#addMeaning").removeClass("error");
//if there are already user items, retrieve vocabList
// var vocabList = [];
var vocabList = getFullList();
if (debugging) {console.log("vocabList retrieved, length: "+vocabList.length);}
//check stored user items for duplicates ****************** to do: option for editing duplicate item with new input
if(checkForDuplicates(vocabList,item)) {
$("#addStatus").text("Duplicate Item detected!");
$("#addKanji").addClass("error");
return;
}
setVocItem(item);
if (debugging) {console.log("clear form");}
$("#addForm")[0].reset();
//--------------------------------------------------------------------------------------------------------
if (item.manualLock === "yes" || item.manualLock === "DB" && lockDB){
$("#addStatus").html("<i class=\"icon-lock\"></i> Added locked item");
} else {
$("#addStatus").html("<i class=\"icon-unlock\"></i>Added successfully");
}
//--------------------------------------------------------------------------------------------------------
}
};
},{}],5:[function(require,module,exports){
module.exports = function(){
// {"level":"17","meaning_explanation":"This word consists of kanji with hiragana attached. Because the hiragana ends with an [ja]う[/ja] sound, you know this word is a verb. The kanji itself means [kanji]flourish[/kanji] or [kanji]prosperity[/kanji], so the verb vocab versions of these would be [vocabulary]to flourish[/vocabulary] or [vocabulary]to prosper[/vocabulary].","reading_explanation":"Since this word consists of a kanji with hiragana attached, you can bet that it will use the kun'yomi reading. You didn't learn that reading with this kanji, so here's a mnemonic to help you: What do you flourish at? You're an amazing [vocabulary]soccer[/vocabulary] ([ja]さか[/ja]) player who flourishes and prospers no matter where you go to play this wonderful (but not as good as baseball) sport.","en":"To Flourish, To Prosper","kana":"さかえる","sentences":[["中国には、覚せい剤の生産で栄えていた村がありました。","There was a village in China flourishing on their production of stimulants. "]],"parts_of_speech_ids":["4","19"],"part_of_speech":"Intransitive Verb, Ichidan Verb","audio":"2e194cbf194371cd478480d6ea67769da623e99a.mp3","meaning_note":null,"reading_note":null,"related":[{"kan":"栄","en":"Prosperity, Flourish","slug":"栄"}]}
if (typeof $.mockjax === "function"){
$.mockjax({
url: /^\/json\/progress\?vWKSS(.+)\[\]=(.+)&vWKSS.+\[\]=(.+)$/,
urlParams:["WKSSid", "MeaningWrong", "ReadingWrong"],
response: function(settings) {
// do any required cleanup
var id = Number(settings.urlParams.WKSSid);
var Mw = Number(settings.urlParams.MeaningWrong);
var Rw = Number(settings.urlParams.ReadingWrong);
var UserVocab = localGet("User-Vocab")||[];
console.log("is this your card?", UserVocab[id]);
if (UserVocab[id].due < Date.now()){//double check that item was due for review
if (Mw||Rw){
//drop levels if wrong
//Adapted from WaniKani's srs to authentically mimic level downs
var o = (Mw||0)+(Rw||0);
var t = UserVocab[id].level;
var r=t>=5?2*Math.round(o/2):1*Math.round(o/2);
var n=t-r<1?1:t-r;//don't stay on 'started'
UserVocab[id].level = n;
}else{
//increase level if none wrong
UserVocab[id].level++;
}
//Put UserVocab back in storage
UserVocab[id].date = Date.now();
UserVocab[id].due = Date.now() + srsintervals[UserVocab[id].level];
localSet("User-Vocab", UserVocab);
console.log(UserVocab[id].due +" > "+ Date.now() + " (" + ms2str(UserVocab[id].due - Date.now())+")");
}else{
console.log("This item is not due for review yet, discarding results");
}
this.responseText = '{"vWKSS'+id.toString()+'":["'+Mw.toString()+'","'+Rw.toString()+'"]}';
}
});
$.mockjax({
url: /^\/json\/vocabulary\/WKSS(.+)/,
urlParams:["WKSSid"],
response: function(settings) {
// Investigate the `settings` to determine the response...
var id = settings.urlParams.WKSSid.toString();
var currentItem = $.jStorage.get("currentItem");
if (currentItem.id === "WKSS"+id){
console.log("as expected");
}
var related = '[';
for (i = 0; i < currentItem.components.length; i++){
related += '{"kan":"'+currentItem.components[i]+'","en":"","slug":"'+currentItem.components[i]+'"}';
related += (i+1<currentItem.components.length)?',':'';
}
related += ']';
var respText = JSON.stringify({
level: "U",
meaning_explanation: "This is user-defined item. Meaning explanations are not supported at this time. [id: "+id+"]",
reading_explanation: "This is user-defined item. Reading explanations are not supported at this time. [id: "+id+"]",
en: currentItem.en.join(", "),
kana: currentItem.kana.join(", "),
sentences:[],
parts_of_speech_ids:[],
part_of_speech:[],
audio:null,
meaning_note:null,
reading_note:null,
related:JSON.parse(related)
});
this.responseText = respText;
},
onAfterComplete: function() {
// do any required cleanup
$(".user-synonyms").remove();
// keeping the hooks for Community Mnemonics
$("#note-meaning, #note-reading").html("");
}
});
}
};
//--------------End Insert Into WK Review Functions--------------
},{}],6:[function(require,module,exports){
var ImportUtil = {
fileUpload: function(ev){
var csvHeader = true; //first row contains stuff like "Kanji/Vocab, Reading, Meaning" etc
var tsvfile; //tabs separate fields, commas seperate values? or false for vice versa
var CSVs = ev.target.files;
var name =CSVs[0].name;
var colsplit, vsplit;
if (name.substr(name.lastIndexOf("."),4)===".csv"){
tsvfile = false;
colsplit = ",";
vsplit = "\t";
}else{
tsvfile = true;
colsplit = "\t";
vsplit = ",";
}
if (debugging) { console.log("tsvfile: "); }
if (debugging) { console.log("file uploaded: "+CSVs[0].name); }
var reader = new FileReader();
reader.readAsText(CSVs[0]);
reader.onload = function(ev){
var csvString = ev.target.result;
var csvRow = csvString.split("\n");
//default column rows
var k = 0;
var r = 1;
var m = 2;
var i = csvRow.length;
//process header, changing k,r,m if necessary
var JSONimport = [];
while(i--){
var row = csvRow[i];
if ((csvHeader === true && i === 0)|| // Skip header
(row === "") // Skip empty rows
){
if (debugging) { console.log("Skipping row #"+i); }
}else{
console.log(row);
var elem = row.split(colsplit);
var item = {};
var c;
if (elem[k]){
item.kanji = elem[k].trim();
if (elem[r]){
if (elem[r].indexOf(vsplit)>-1){
// eg 'reading 1[tab]reading 2[tab]reading 3'
item.reading = elem[r].split(vsplit);
}else{ //no tabs in string, single value
item.reading=[elem[r]];
}
}else{
item.reading=[""];
}
if (elem[m]){
if (elem[m].indexOf(vsplit)>-1){
// eg 'meaning 1[tab]meaning 2[tab]meaning 3'
item.meaning = elem[m].split("\t");
}else{ //no tabs in string, single value
item.meaning=[elem[m]];
}
c = item.meaning.length;
while(c--){
console.log("item.meaning["+c+"]: "+item.meaning[c]);
}
}else{//todo: provide overwrite option on forced meaning
item.meaning=[""];
}
JSONimport.push(item);
}else{ // corrupt row ('kanji' is mandatory (can be kana-only word), is not present on row, skip
}
}
}
var JSONstring = JSON.stringify(JSONimport);
console.log(JSONimport);
if (JSONstring.length !== 0) {
try {
var add = JSON.parse(JSONstring.toLowerCase());
/*//---------/-------------
if (!checkAdd(add)) {
$("#importStatus").text("No valid input (duplicates?)!");
return;
}
//----------------------*/
var a = add.length;
while(a--){
StorageUtil.setVocItem(add[a]);
}
$("#importStatus").text("Import successful!");
$("#importForm")[0].reset();
$("#importArea").text("");
}
catch (e) {
$("#importStatus").text("Parsing Error!");
console.log(e);
}
}
else {
$("#importStatus").text("Nothing to import :( Please paste your stuff first");
}
};
}
};
module.exports = ImportUtil;
},{}],7:[function(require,module,exports){
var ReviewUtil = {
/** Takes an array of strings and returns the portions before left brackets '(' but only for strings that have them. It is used to add synonym values to the answer list.
* @param {Array.<string>} solution - An array of acceptable answers for the Task
* @returns {Array.<string>} Parts of the solution left of left bracket in strings where it exists
* @example unbracketSolution(["newspaper", "reading Stick (this text won't get through)"]) // ["reading stick"]
*/
unbracketSolution: function(solution){
var unbracketed = solution.filter(function(ans){
var openBracket = ans.indexOf("(");
if (openBracket !== -1){ //string contains a bracket
return ans.toLowerCase().substr(0, openBracket).trim();
}
}, this);
return unbracketed;
},
inputCorrect: function() {
var input = $("#rev-input").val().toLowerCase().trim();
var solution = document.getElementById('rev-solution').innerHTML.split(/[,、]+\s*/);
var correctCharCount = 0;
var returnvalue = false;
console.log("Input: " + input);
var append = this.unbracketSolution(solution);
solution = solution.concat(append);
var i = solution.length;
while(i--){
var threshold = 0;//how many characters can be wrong
if(document.getElementById('rev-type').innerHTML == "Meaning") {
threshold = Math.floor(solution[i].length / errorAllowance);
}
console.log("Checking " + solution[i] + " with threshold: " + threshold);
var j;
var lengthDiff = Math.abs(input.length - solution[i].length);
if (lengthDiff > threshold){
returnvalue = returnvalue || false;
console.log("false at if branch " + input.length + " < " + JSON.stringify(solution[i]));//.length );//- threshold));
} else { //difference in response length is within threshold
j = input.length;
while (j--) {
if (input[j] == solution[i][j]) {
console.log (input[j] +" == "+ solution[i][j]);
correctCharCount++;
}
}
if (correctCharCount >= solution[i].length - threshold){
returnvalue = true;
}
}
}
console.log("Returning " + returnvalue);
return returnvalue;
},
submitResponse: function (e) {
//functions:
// inputCorrect()
//check if key press was 'enter' (keyCode 13) on the way up
//and keystate true (answer being submitted)
//and cursor is focused in reviewfield
if (e.keyCode == 13 && submit === true) {
var input = $("#rev-input").val();
var reviewList = sessionGet('User-Review')||[];
var rnd = sessionStorage.getItem('WKSS-rnd')||0;
var item = sessionGet('WKSS-item');
//-- starting implementation of forgiveness protocol
item.forgive = [];//"ゆるす"]; //placeholder (許す to forgive)
if (item === null){
alert("Item Null??");
reviewList.splice(rnd, 1);
}
else{
//handle grading and storing solution
//check for input, do nothing if none
if(input.length === 0){
return;
}
//disable input after submission
//document.getElementById('rev-input').disabled = true;
//was the input correct?
var correct = this.inputCorrect();
//was the input forgiven?
var forgiven = (item.forgive.indexOf(input) !== -1);
if (correct) {
//highlight in (default) green
$("#rev-input").addClass("correct");
//show answer
$("#rev-solution").addClass("info");
} else if (forgiven){
$("#rev-input").addClass("caution");
} else {
//highight in red
$("#rev-input").addClass("error");
//show answer
$("#rev-solution").addClass("info");
}
//remove from sessionList if correct
if (correct) {
console.log("correct answer");
if (reviewList !== null){
var oldlen = reviewList.length;
reviewList.splice(rnd, 1);
console.log("sessionList.length: "+ oldlen +" -> "+reviewList.length);
//replace shorter (by one) sessionList to session
if (reviewList.length !== 0) {
console.log("sessionList.length: "+ reviewList.length);
sessionSet('User-Review', JSON.stringify(reviewList));
} else {
//reveiw over, delete sessionlist from session
sessionStorage.removeItem('User-Review');
}
}else{
console.error("Error: no review session found");
}
}
else{
// if(forgiven){
// console.log(input +" has been forgiven. "+item.type);
// return;
// }
console.log("wrong answer");
}
item = markAnswer(item);
sessionSet(item.index, item);
var list = JSON.parse(sessionStorage.getItem("User-Stats"))||[];
var found = false;
if (list){
var i = list.length;
while(i--){
if (list[i].index == item.index) {
list[i] = item; //replace item if it exists
found = true;
break;
}
}
if(!found){
list = saveToSortedList(list,item);
}
} else {
list = [item];
}
sessionSet("User-Stats", JSON.stringify(list));
//playAudio();
//answer submitted, next 'enter' proceeds with script
submit = false;
}//null garbage collection
}
else if (e.keyCode == 13 && submit === false) {
console.log("keystat = " + submit);
//there are still more reviews in session?
if (sessionStorage.getItem('User-Review')) {
// console.log("found a 'User-Review': " + sessionStorage.getItem('User-Review'));
setTimeout(function () {
console.log("refreshing reviewList from storage");
var reviewList = JSON.parse(sessionStorage.getItem('User-Review'));
//cue up first remaining review
nextReview(reviewList);
console.log("checking for empty reviewList");
if (reviewList.length === 0){
console.log("session over. reviewList: "+JSON.stringify(reviewList));
sessionStorage.removeItem("User-Review");
}
// document.getElementById('rev-input').disabled = true;
$("#rev-solution").removeClass("info");
$("#selfstudy").hide().fadeIn('fast');
}, 1);
}
else {
// no review stored in session, review is over
setTimeout(function () {
$("#selfstudy").hide();
//document.getElementById('rev-input').disabled = false;
$("#rev-solution").removeClass("info");
console.log("showResults");
showResults();
$("#resultwindow").show();
console.log("showResults completed");
//*/ //clear session
sessionStorage.clear();
reviewActive = false;
}, 1);
}
submit = true;
}
}
};
module.exports = ReviewUtil;
},{}],8:[function(require,module,exports){
var StorageUtil = {
/** Initialise User-Vocab
*/
initStorage: function(){
if (!this.localGet("User-Vocab")){
this.localSet("User-Vocab", []);
}
},
parseString: function(strObj){
//avoids duplication of code for sesssionGet and localGet
var obj;
try {
obj = JSON.parse(strObj);
console.log("Variable is of type " + typeof obj);
}
catch(e){
if (e.name === "SyntaxError"){
console.log(strObj + " is an ordinary string that cannot be parsed.");
obj = strObj;
}
else{
console.error("Could not parse " + strObj + ". Error: ", e);
}
}
return obj;
},
/**
*/
localGet: function(strName){
var strObj = localStorage.getItem(strName);
return this.parseString(strObj);
},
/** Sets strings and objects into browser storage
* @requires localStorage
* @requires JSON
*/
localSet: function(strName, obj){
localStorage.setItem(strName, typeof obj === "string"? obj : JSON.stringify(obj));
},
/**
*/
sessionGet: function(strName){
var strObj = sessionStorage.getItem(strName);
return this.parseString(strObj);
},
/** Sets strings and objects into browser session storage
* @requires localStorage
* @requires JSON
*/
sessionSet: function(strName, obj){
sessionStorage.setItem(strName, typeof obj === "string"? obj : JSON.stringify(obj));
},
/**
*/
getVocList: function(){
var vocList = JSON.parse(localStorage.getItem('User-Vocab'))||[];
if (vocList){
var v=vocList.length;
while(v--){
vocList[v].i = v; //set index for item (->out)
}
}
return vocList;
},
setVocList: function(vocList){
this.localSet('User-Vocab', vocList);
},
/**
*/
setVocItem: function(item){
//Assumption: item comes only with kanji, reading and meaning
item.level = 0;
item.date = Date.now();
item.manualLock = "";
item = setLocks(item);
//0.1.9 adding in 'due' property to make review building simpler
item.due = item.date + srsObject[item.level].duration;
var vocList = localGet('User-Vocab')||[];
var found = vocList.find(function(task){
return task.kanji === item.kanji;
}, this);
//add meaning and reading to existing item
// vocList[v].meaning = item.meaning;
// vocList[v].reading = item.reading;
if (!found) {
//provide index for faster searches
console.log(item.kanji +" not found in vocablist, adding now");
item.i = vocList.length;
vocList.push(item);
localSet('User-Vocab',vocList);
}
}
};
module.exports = StorageUtil;
},{}],9:[function(require,module,exports){
/* This is the original code that I am breaking into bite size bits */
//NEED TO MAKE SURE BROWSERIFY PUTS THIS ON THE TOP
/** Describes any object that can be reviewed or learned, includes IRadical, IKanji, and IVocabulary
* @typedef {Object} Task
* @property {boolean|string} locked - locked
* @property {boolean|string} manualLock - manualLock
*/
var StorageUtil = require('./storageutil.js');
var ImportUtil = require('./importutil.js');
var WanikaniUtil = require('./wanikaniutil.js');
var ReviewUtil = require('./reviewutil.js');
function main(){
"use strict";
$("head").prepend("<script src='https://cdn.jsdelivr.net/jquery.mockjax/1.6.1/jquery.mockjax.js'></script>");
var APIkey = "YOUR_API_HERE";
var locksOn = true; //Disable vocab locks (unlocked items persist until deleted)
var lockDB = true; //Set to false to unlock Kanji is not available on WaniKani (ie. not returned by API)
var reverse = true; //Include English to ひらがな reading reviews
var debugging = true;
var asWK = true; //Push user reviews into the main WK review queue
// shut up JSHint
/* jshint multistr: true , jquery: true, expr: true, indent:2 */
/* global window, wanakana, XDomainRequest */
/** Debugging
*/
console.log = debugging ? function (msg) {
if (typeof msg === 'string') {
window.console.log("WKSS: " + msg);
}
else {
window.console.log("WKSS: ", msg);
}
} : function () {
};
$("head").prepend('<script src="https://rawgit.com/WaniKani/WanaKana/master/lib/wanakana.js" type="text/javascript"></script>');
var localSet = function(strName, obj){
debugging&&console.log(strName + " is of type " + typeof obj);
if (typeof obj === "object")
obj=JSON.stringify(obj);
localStorage.setItem(strName, obj);
};
//track versions & datatypes
localSet("WKSSdata", {
v: "0.1.13",
propertyType: {
meaning: "array", reading: "array", kanji: "string", i:"number", components: "array", date: "number", due: "number", locked: "string", manualLock: "string"
},
propertyDesc: {
meaning: "list of meanings", reading: "list of readings", kanji: "item prompt", i:"item index", components: "kanji found in word", date: "timestamp of new level", due: "timestamp of item's next review", locked: "indicator of whether components are eligible", manualLock: "latch for 'locked' so failing components don't re-lock the item"
}
});
/** Settings and constants
*/
var errorAllowance = 4; //every x letters, you can make one mistake when entering the meaning
//srs 4h, 8h, 24h, 3d (guru), 1w, 2w (master), 1m (enlightened), 4m (burned)
var hrs = 60*60*1000;
var days = 24*hrs;
var weeks = 7*days;
var srsObject = [
{level: 0, rank: "Started", duration: 0},
{level: 1, rank: "Apprentice", duration: 4*hrs},
{level: 2, rank: "Apprentice", duration: 8*hrs},
{level: 3, rank: "Apprentice", duration: 1*days},
{level: 4, rank: "Apprentice", duration: 3*days},
{level: 5, rank: "Guru", duration: 1*weeks},
{level: 6, rank: "Guru", duration: 2*weeks},
{level: 7, rank: "Master", duration: 730*hrs},
{level: 8, rank: "Enlightened", duration: 2922*hrs},
{level: 9, rank: "Burned"}
];
var localGet = function(strName){
var strObj = localStorage.getItem(strName);
return parseString(strObj);
};
// Initialise User-Vocab
StorageUtil.initStorage();
//GM_addStyle shim for compatibility with greasemonkey
var gM_addStyle = function(CssString){
//get DOM head
var head = document.getElementsByTagName('head')[0];
if (head) {
//build style tag
var style = document.createElement('style');
style.setAttribute('type', 'text/css');
style.textContent = CssString;
//insert DOM style into head
head.appendChild(style);
}
};
/** JQuery fixes
*/
$("[placeholder]").focus(function () {
var input = $(this);
if (input.val() == input.attr("placeholder")) {
input.val("''");
input.removeClass("'placeholder'");
}
}).blur(function () {
var input = $(this);
if (input.val() == "''" || input.val() == input.attr("placeholder")) {
input.addClass("placeholder");
input.val(input.attr("placeholder"));
}
}).blur();
$("[placeholder]").parents("form").submit(function () {
$(this).find("[placeholder]").each(function () {
var input = $(this);
if (input.val() == input.attr("placeholder")) {
input.val("");
}
});
});
/** Handle the users API key.
* @param {string} APIkey - the users API key to set. If given "YOUR_API_HERE", it will return the key in browser storage.
* @returns {string} the users API key as supplied and stored, or in the case of "YOUR_API_HERE" being passed, the stored key.
*/
var getSetApi = function(APIkey){
var storedAPI = localStorage.getItem('WaniKani-API');
if (APIkey === "YOUR_API_HERE"){
if (storedAPI !== null){
APIkey = storedAPI;
}
}
else{
//API has been set in code.
if (storedAPI !== APIkey){
localSet('WaniKani-API', APIkey);//overwrite with new API
}
}
return APIkey;
};
APIkey = getSetApi(APIkey);
//--------------Start Insert Into WK Review Functions--------------
/** Messing around with vanilla WaniKani review variables
*/
var joinReviews = function(WKItems){
console.log("joining reviews");
$.jStorage.stopListening("reviewQueue", joinReviews);
var WKreview = $.jStorage.get("reviewQueue")||[];
var WKcombined = WKreview.concat(WKItems);
$.jStorage.set("reviewQueue", WKcombined);
};
var WKItems = [];
console.groupCollapsed("Loading Items");
var wKSS_to_WK = function(WKSSItem){
var WKItem = {};
// WKItem.aud = "";
WKItem.en = WKSSItem.meaning.map(function(s) {
//trim whitespace and capitalize words
return s.trim().replace(/\b\w/g , function(m){
return m.toUpperCase();
});
});
WKItem.id = "WKSS" + WKSSItem.i;
WKItem.kana = WKSSItem.reading;
WKItem.srs = WKSSItem.level+1;//WK starts levels from 1, WKSS starts them from 0
WKItem.voc = WKSSItem.kanji;
WKItem.components = WKSSItem.components;
WKItem.syn = [];
//Add synonyms of strings without bracketed info to get around checking the full string including brackets
WKSSItem.meaning.forEach(function(meaning){
var openBracket = meaning.indexOf("(");
if (openBracket !== -1 && meaning.indexOf(")") !== -1){
WKItem.syn.push(meaning.substr(0, openBracket).trim().replace(/\b\w/g , function(m){ return m.toUpperCase();}));
}
}, this);
return WKItem;
};
var loadTasks = function(userVocab, i, userVocabs){
var dueNow = (userVocab.locked === "no" && userVocab.level < 9 && Date.now() > userVocab.due);
if (dueNow){
if (userVocab.kanji.length * userVocab.meaning[0].length * userVocab.reading[0].length){
//Sorry, we need all three to add to WK review, no kana only without readings etc.
debugging&&console.log("item:" + userVocab.kanji + ", " + userVocab.locked +" === \"no\" && " + userVocab.level + " < 9 && " + Date.now() + " > " + userVocab.due);
debugging&&console.log(dueNow);
WKItems.push(wKSS_to_WK(userVocab));
}else{
debugging&&console.log("Item " + userVocab.kanji + " could not be added, it is missing one or more of the essential fields for a WK vocabulary review");
}
}
};
var userVocabs = StorageUtil.getVocList();
userVocabs.forEach(loadTasks);//, this);
console.groupEnd();
//where the magic happens
if (asWK){
$.jStorage.listenKeyChange("reviewQueue", function(){joinReviews(WKItems);});
}
var sessionSet = function(strName, obj){
debugging&&console.log(strName + " is of type " + typeof obj);
if (typeof obj === "object")
obj=JSON.stringify(obj);
sessionStorage.setItem(strName, obj);
};
var sessionGet = function(strName){
var strObj = sessionStorage.getItem(strName);
return parseString(strObj);
};
var generateReviewList = function(reviewActive) {
//don't interfere with an active session
if (reviewActive){
document.getElementById('user-review').innerHTML = "Review in Progress";
return;
}
debugging&&console.log("generateReviewList()");
// function generateReviewList() builds a review session and updates the html menu to show number waiting.
var numReviews = 0;
var soonest = Infinity;
var next;
var reviewList = [];
//check to see if there is vocab already in offline storage
if (localStorage.getItem('User-Vocab')) {
var vocabList = StorageUtil.getVocList();
debugging&&console.log(vocabList);
var now = Date.now();
//for each vocab in storage, get the amount of time vocab has lived
//var i = vocabList.length;
//while(i--){
vocabList.forEach(function(task, i){
var due = task.date + srsObject[task.level].duration;
// if tem is unlocked and unburned
if (task.level < 9 &&
(task.manualLock === "no" || task.manualLock === "n" ||
task.manualLock ==="DB" && !lockDB )){
// if it is past review time
if(now >= due) {
// count vocab up for review
numReviews++;
// add item-meaning object to reviewList
// have made this optional for surname lists etc.
if (task.meaning[0] !== "") {
//Rev_Item object args: prompt, kanji, type, solution, index
var revItem = new Rev_Item(task.kanji, task.kanji, "Meaning", task.meaning, i);
reviewList.push(revItem);
}
// reading is optional, if there is a reading for the vocab, add its object.
if (task.reading[0] !== "") {
//Rev_Item object args: prompt, kanji, type, solution, index
var revItem2 = new Rev_Item(task.kanji, task.kanji, "Reading", task.reading, i);
reviewList.push(revItem2);
}
//if there is a meaning and reading, and reverse flag is true, test reading from english
if (task.reading[0] !== "" && task.meaning[0] !== "" && reverse){
//Rev_Item object args: prompt, kanji, type, solution, index
var revItem3 = new Rev_Item(task.meaning.join(", "), task.kanji, "Reverse", task.reading, i);
reviewList.push(revItem3);
}
}
else{//unlocked/unburned but not time to review yet
debugging&&console.log("setting soonest");
next = due - now;
soonest = Math.min(soonest, next);
}
}//end if item is up for review
}, this);// end iterate through vocablist
}// end if localStorage
if (reviewList.length !== 0){
//store reviewList in current session
sessionSet('User-Review', JSON.stringify(reviewList));
debugging&&console.log(reviewList);
}
else{
debugging&&console.log("reviewList is empty: "+JSON.stringify(reviewList));
document.getElementById('user-review').innerHTML = soonest<Infinity? "Next Review in "+ms2str(soonest) : "No Reviews Available";
}
var strReviews = numReviews.toString();
/* If you want to do the 42+ thing.
if (numReviews > 42) {
strReviews = "42+"; //hail the crabigator!
}
//*/
// return the number of reviews
debugging&&console.log(numReviews.toString() +" reviews created");
if (numReviews > 0){
var reviewString = (soonest !== void 0)? "<br/>\
More to come in "+ms2str(soonest):"";
document.getElementById('user-review').innerHTML = "Review (" + strReviews + ")" + reviewString;
}
};
/*
* populate reviews when menu button pressed
*/
window.generateReviewList = function() {
//if menu is invisible, it is about to be visible
if ( $("#WKSS_dropdown").is(":hidden") ){
//This is really the only time it needs to run
//unless we want to start updating in realtime by keeping track of the soonest item
generateReviewList();
}
};
/*
* Add Item
*/
// event function to open "add window" and close any other window that might be open at the time.
window.WKSS_add = function () {
//show the add window
$("#add").show();
//hide other windows
$("#export").hide();
$("#import").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
//'add window' html text
var addHtml = '\n\
<div id="add" class="WKSS">\n\
<form id="addForm">\n\
<button id="AddCloseBtn" class="wkss-close" type="reset"><i class="icon-remove"></i></button>\n\
<h1>Add a new Item</h1>\n\
<input type="text" id="addKanji" placeholder="Enter 漢字, ひらがな or カタカナ">\n\
<input type="text" id="addReading" title="Leave empty to add vocabulary like する (to do)" placeholder="Enter reading">\n\
<input type="text" id="addMeaning" placeholder="Enter meaning">\n\
\n\
<p id="addStatus">Ready to add..</p>\n\
<button id="AddItemBtn" type="button">Add new Item</button>\n\
</form>\n\
</div>\n';
//add html to page source
$("body").append(addHtml);
//hide add window ("div add" code that was just appended)
$("#add").hide();
//function to fire on click event for "Add new Item"
$("#AddItemBtn").click(function () {
handleAddClick();
});
$("#AddCloseBtn").click(function () {
$("#add").hide();
$("#addForm")[0].reset();
$("#addStatus").text('Ready to add..');
$("#addKanji").removeClass("error");
$("#addMeaning").removeClass("error");
});
//---Function wrappers to facilitate use of one localstorage array
//---Maintains data integrity between previously two (vocab and srs)
function setSrsItem(srsitem,srsList){
var index = srsitem.i;
debugging&&console.log("setSrsItem: ");
if(srsList){
if(srsList[index].kanji===srsitem.kanji){// try search by index
debugging&&console.log("success: "+srsitem.kanji+" found at index "+ index);
//replace only the srs parts of the item
srsList[index].date = srsitem.date;
srsList[index].level = srsitem.level;
srsList[index].locked = srsitem.locked;
srsList[index].manualLock = srsitem.manualLock;
}else{ //backup plan (cycle through list?)
debugging&&console.log("SRS Kanji not found in vocablist, needs work");
}
debugging&&console.log("item: ");
return srsList;
}
}
function getSrsList(){
var srsList = StorageUtil.getVocList();
return srsList;
}
function getFullList(){
var fullList = JSON.parse(localStorage.getItem('User-Vocab'))||[];
if(!fullList){
fullList=[];
}
return fullList;
}
//--------
/*
* Edit Items
*/
window.WKSS_edit = function () {
generateEditOptions();
$("#edit").show();
//hide other windows
$("#export").hide();
$("#import").hide();
$("#add").hide();
$("#selfstudy").hide();
};
$("body").append(" \
<div id=\"edit\" class=\"WKSS\"> \
<form id=\"editForm\"> \
<button id=\"EditCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\
<h1>Edit your Vocab</h1> \
<select id=\"editWindow\" size=\"8\"></select>\
<input type=\"text\" id=\"editItem\" name=\"\" size=\"40\" placeholder=\"Select vocab, click edit, change and save!\">\
\
<p id=\"editStatus\">Ready to edit..</p>\
<button id=\"EditEditBtn\" type=\"button\">Edit</button>\
<button id=\"EditSaveBtn\" type=\"button\">Save</button> \
<button id=\"EditDeleteBtn\" type=\"button\" title=\"Delete selected item\">Delete</button> \
<button id=\"EditDeleteAllBtn\" type=\"button\" title=\"本当にやるの?\">Delete All</button> \
<button id=\"ResetLevelsBtn\" type=\"button\">Reset levels</button> \
</form> \
</div>");
$("#edit").hide();
$("#EditEditBtn").click(function () {
//get handle for 'select' area
var select = document.getElementById("editWindow");
//get the index for the currently selected item
var index = select.selectedIndex; //select.options[select.selectedIndex].value is not required, option values are set to index
var vocabList = StorageUtil.getVocList();
vocabList = vocabList.reverse();
document.getElementById("editItem").value = JSON.stringify(vocabList[index]);
document.getElementById("editItem").name = index; //using name to save the index
$("#editStatus").text('Loaded item to edit');
});
$("#EditSaveBtn").click(function () {
if ($("#editItem").val().length !== 0) {
//-- be aware
//deleting one item may cause mismatch if i is property of item in list
try {
var index = document.getElementById("editItem").name;
var item = JSON.parse(document.getElementById("editItem").value.toLowerCase());
var m = item.meaning.length;
while(m--){
if (item.meaning[m] === ""){
delete item.meaning[m];
}
}
var fullList = getFullList().reverse();
if (isItemValid(item) &&//item is valid
!(checkForDuplicates(fullList,item) && //kanji (if changed) is not already in the list
fullList[index].kanji !== item.kanji)) {//unless it is the item being edited
var srslist = getSrsList().reverse();
//get srs components of item(list)
fullList[index] = item;//does not have srs stuff, re-add it now
debugging&&console.log(fullList[index]);
debugging&&console.log(srslist[index]);
fullList[index].date = srslist[index].date;
fullList[index].level = srslist[index].level;
fullList[index].locked = srslist[index].locked;
fullList[index].manualLock = srslist[index].manualLock;
fullList = fullList.reverse(); //reset order of array
localSet('User-Vocab', fullList);
generateEditOptions();
$("#editStatus").html('Saved changes!');
document.getElementById("editItem").value = "";
document.getElementById("editItem").name = "";
}
else{
$("#editStatus").text('Invalid item or duplicate!');
alert(isItemValid(item).toString() +" && !("+ checkForDuplicates(fullList,item).toString()+" && !("+fullList[index].kanji+" !== "+item.kanji+")");
}
}
catch (e) {
$("#editStatus").text(e);
}
}
});
$("#EditDeleteBtn").click(function () {
//select options element window
var select = document.getElementById("editWindow");
//index of selected item
var item = select.options[select.selectedIndex].value;
//fetch JSON strings from storage and convert them into Javascript literals
var vocabList = getFullList();
//starting at selected index, remove 1 entry (the selected index).
if (item > -1) {
if (vocabList !== null){
vocabList.splice(item, 1);
}
}
//yuck
if (vocabList.length !== 0) {
localSet('User-Vocab', vocabList);
}
else {
localStorage.removeItem('User-Vocab');
}
updateEditGUI();
$("#editStatus").text('Item deleted!');
});
$("#EditDeleteAllBtn").click(function () {
var deleteAll = confirm("Are you sure you want to delete all entries?");
if (deleteAll) {
//drop local storage
localStorage.removeItem('User-Vocab');
updateEditGUI();
$("#editStatus").text('All items deleted!');
}
});
$("#EditCloseBtn").click(function () {
$("#edit").hide();
$("#editForm")[0].reset();
$("#editStatus").text('Ready to edit..');
});
/*
* Export
*/
window.WKSS_export = function () {
$("#export").show();
//hide other windows
$("#add").hide();
$("#import").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
$("body").append(' \
<div id="export" class="WKSS"> \
<form id="exportForm"> \
<button id="ExportCloseBtn" class="wkss-close" type="button"><i class="icon-remove"></i></button>\
<h1>Export Items</h1> \
<textarea cols="50" rows="18" id="exportArea" placeholder="Export your stuff! Sharing is caring ;)"></textarea> \
\
<p id="exportStatus">Ready to export..</p> \
<button id="ExportItemsBtn" type="button">Export Items</button>\
<button id="ExportSelectAllBtn" type="button">Select All</button>\
<button id="ExportCsvBtn" type="button">Export CSV</button>\
</form> \
</div>');
$("#export").hide();
$("#ExportItemsBtn").click(function () {
if (localStorage.getItem('User-Vocab')) {
$("#exportForm")[0].reset();
var vocabList = StorageUtil.getVocList();
$("#exportArea").text(JSON.stringify(vocabList));
$("#exportStatus").text("Copy this text and share it with others!");
}
else {
$("#exportStatus").text("Nothing to export yet :(");
}
});
$("#ExportSelectAllBtn").click(function () {
if ($("#exportArea").val().length !== 0) {
select_all("exportArea");
$("#exportStatus").text("Don't forget to CTRL + C!");
}
});
$("#ExportCsvBtn").click(function () {
var vocabList = getFullList();
var CsvFile = createCSV(vocabList);
window.open(CsvFile);
});
$("#ExportCloseBtn").click(
function () {
$("#export").hide();
$("#exportForm")[0].reset();
$("#exportArea").text("");
$("#exportStatus").text('Ready to export..');
}
);
/*
* Import
*/
window.WKSS_import = function () {
$("#import").show();
//hide other windows
$("#add").hide();
$("#export").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
$("body").append(' \
<div id="import" class="WKSS"> \
<form id="importForm"> \
<button id="ImportCloseBtn" class="wkss-close" type="reset"><i class="icon-remove"></i></button>\
<h1>Import Items</h1>\
<textarea cols="50" rows="18" id="importArea" placeholder="Paste your stuff and hit the import button! Use with caution!"></textarea> \
\
<p id="importStatus">Ready to import..</p> \
<label class="button" id="ImportItemsBtn" style="display:inline;">Import Items</label>\
\
<label id="ImportCsvBtn" class="button" style="display:inline;cursor: pointer;">Import CSV \
\
<input type="file" id="upload" accept=".csv,.tsv" style="height:0px;width:0px;background:red;opacity:0;filter:opacity(1);" />\
\
</label>\
\
<label class="button" id="ImportWKBtn" style="display:inline;"><i class="icon-download-alt"></i> WK</label>\
</form> \
</div>');
$("#import").hide();
document.getElementById("upload") && document.getElementById("upload").addEventListener('change', ImportUtil.fileUpload, false);
$("#ImportCsvBtn").click(function () {
});
$("#ImportWKBtn").click(function(){
WanikaniUtil.getServerResp(APIkey,"vocabulary");
debugging&&console.log("maybe?");
});
$("#ImportItemsBtn").click(function () {
if ($("#importArea").val().length !== 0) {
try {
var add = JSON.parse($("#importArea").val().toLowerCase());
alert(JSON.stringify(add));
if (checkAdd(add)) {
$("#importStatus").text("No valid input (duplicates?)!");
return;
}
var newlist;
var srslist = [];
if (localStorage.getItem('User-Vocab')) {
var vocabList = StorageUtil.getVocList();
srslist = getSrsList();
newlist = vocabList.concat(add);
}
else {
newlist = add;
}
var i = add.length;
while(i--){
StorageUtil.setVocItem(add[i]);
}
$("#importStatus").text("Import successful!");
$("#importForm")[0].reset();
$("#importArea").text("");
}
catch (e) {
$("#importStatus").text("Parsing Error!");
debugging&&console.log(e);
}
}
else {
$("#importStatus").text("Nothing to import :( Please paste your stuff first");
}
});
$("#ImportCloseBtn").click(function () {
$("#import").hide();
$("#importForm")[0].reset();
$("#importArea").text("");
$("#importStatus").text('Ready to import..');
});
/*
* Review Items
*/
window.WKSS_review = function () {
//is there a session waiting in storage?
if(sessionStorage.getItem('User-Review')) {
//show the selfstudy window
$("#selfstudy").show();
//hide other windows
$("#add").hide();
$("#export").hide();
$("#edit").hide();
$("#import").hide();
startReview();
}
};
$("body").append(' \
<div id="selfstudy" class="WKSS">\
<button id="SelfstudyCloseBtn" class="wkss-close" type="button"><i class="icon-remove"></i></button>\
<h1>Review<span id="RevNum"></span></h1>\
<div id="wkss-kanji">\
<span id="rev-kanji"></span>\
</div><div id="wkss-type">\
<span id="rev-type"></span><br />\
</div><div id="wkss-solution">\
<span id="rev-solution"></span>\
</div><div id="wkss-input">\
<input type="text" id="rev-input" size="40" placeholder="">\
</div><span id="rev-index" style="display: block;"></span>\
\
<form id="audio-form">\
<label id="AudioButton" class="button">Play audio</label>\
<label id="WrapUpBtn" class="button">Wrap Up</label>\
</form>\
<div id="rev-audio" style="display:none;"></div>\
</div>');
$("#selfstudy").hide();
$("#SelfstudyCloseBtn").click(function () {
$("#selfstudy").hide();
$("#rev-input").val("");
reviewActive = false;
});
$("#WrapUpBtn").click(function() {
var sessionList = sessionGet('User-Review')||[];
var statsList = sessionGet('User-Stats')||[];
//if an index in sessionList matches one in statsList, don't delete
var sessionI = sessionList.length;
var item = sessionGet('WKSS-item')||[];
var arr2 = [];
//for every item in sessionList, look for index in statsList,
//if not there (-1) delete item from sessionList
while (sessionI--){
var index = findIndex(statsList,sessionList[sessionI]);
if ((Math.sign(1/index) !== -1)||(sessionList[sessionI].index == item.index)){
arr2.push(sessionList[sessionI]);
}
}
debugging&&console.log(arr2);
sessionSet('User-Review', JSON.stringify(arr2));
});
$("#AudioButton").click(function () {
openInNewTab(document.getElementById('rev-audio').innerHTML);
});
$("body").append(' \
<div id="resultwindow" class="WKSS"> \
<button id="ReviewresultsCloseBtn" class="wkss-close" type="button"><i class="icon-remove"></i></button>\
<h1>Review Results</h1>\
<h2>All</h2>\
<div id="stats-a"></div>\
</div>');
$("#resultwindow").hide();
$("#ReviewresultsCloseBtn").click(function () {
$("#resultwindow").hide();
document.getElementById("stats-a").innerHTML = "";
});
//declare global values for keyup event
//is an answer being submitted?
var submit = true;
//jquery keyup event
$("#rev-input").keyup(function (e) {
//functions:
// inputCorrect()
//check if key press was 'enter' (keyCode 13) on the way up
//and keystate true (answer being submitted)
//and cursor is focused in reviewfield
if (e.keyCode == 13 && submit === true) {
var input = $("#rev-input").val();
var reviewList = sessionGet('User-Review')||[];
var rnd = sessionStorage.getItem('WKSS-rnd')||0;
var item = sessionGet('WKSS-item');
//-- starting implementation of forgiveness protocol
item.forgive = [];//"ゆるす"]; //placeholder (許す to forgive)
if (item === null){
alert("Item Null??");
reviewList.splice(rnd, 1);
}else{
//handle grading and storing solution
//check for input, do nothing if none
if(input.length === 0){
return;
}
//disable input after submission
//document.getElementById('rev-input').disabled = true;
//was the input correct?
var correct = inputCorrect();
//was the input forgiven?
var forgiven = (item.forgive.indexOf(input) !== -1);
if (correct) {
//highlight in (default) green
$("#rev-input").addClass("correct");
//show answer
$("#rev-solution").addClass("info");
} else if (forgiven){
$("#rev-input").addClass("caution");
} else {
//highight in red
$("#rev-input").addClass("error");
//show answer
$("#rev-solution").addClass("info");
}
//remove from sessionList if correct
if (correct) {
debugging&&console.log("correct answer");
if (reviewList !== null){
var oldlen = reviewList.length;
reviewList.splice(rnd, 1);
debugging&&console.log("sessionList.length: "+ oldlen +" -> "+reviewList.length);
//replace shorter (by one) sessionList to session
if (reviewList.length !== 0) {
debugging&&console.log("sessionList.length: "+ reviewList.length);
sessionSet('User-Review', JSON.stringify(reviewList));
} else {
//reveiw over, delete sessionlist from session
sessionStorage.removeItem('User-Review');
}
}else{
console.error("Error: no review session found");
}
}else{
// if(forgiven){
// debugging&&console.log(input +" has been forgiven. "+item.type);
// return;
//}
debugging&&console.log("wrong answer");
}
item = markAnswer(item);
sessionSet(item.index, item);
var list = JSON.parse(sessionStorage.getItem("User-Stats"))||[];
var found = false;
if (list){
var i = list.length;
while(i--){
if (list[i].index == item.index) {
list[i] = item; //replace item if it exists
found = true;
break;
}
}
if(!found){
list = saveToSortedList(list,item);
}
} else {
list = [item];
}
sessionSet("User-Stats", JSON.stringify(list));
//playAudio();
//answer submitted, next 'enter' proceeds with script
submit = false;
}//null garbage collection
}
else if (e.keyCode == 13 && submit === false) {
debugging&&console.log("keystat = " + submit);
//there are still more reviews in session?
if (sessionStorage.getItem('User-Review')) {
// debugging&&console.log("found a 'User-Review': " + sessionStorage.getItem('User-Review'));
setTimeout(function () {
debugging&&console.log("refreshing reviewList from storage");
var reviewList = JSON.parse(sessionStorage.getItem('User-Review'));
//cue up first remaining review
nextReview(reviewList);
debugging&&console.log("checking for empty reviewList");
if (reviewList.length === 0){
debugging&&console.log("session over. reviewList: "+JSON.stringify(reviewList));
sessionStorage.removeItem("User-Review");
}
// document.getElementById('rev-input').disabled = true;
$("#rev-solution").removeClass("info");
$("#selfstudy").hide().fadeIn('fast');
}, 1);
}
else {
// no review stored in session, review is over
setTimeout(function () {
$("#selfstudy").hide();
//document.getElementById('rev-input').disabled = false;
$("#rev-solution").removeClass("info");
debugging&&console.log("showResults");
showResults();
$("#resultwindow").show();
debugging&&console.log("showResults completed");
//*/ //clear session
sessionStorage.clear();
reviewActive = false;
}, 1);
}
submit = true;
}
});
/** populate reviews when menu button pressed
*/
window.generateReviewList = function() {
//if menu is invisible, it is about to be visible
if ( $("#WKSS_dropdown").is(":hidden") ){
//This is really the only time it needs to run
//unless we want to start updating in realtime by keeping track of the soonest item
generateReviewList(reviewActive);
}
};
/** Add Item: event function to open "add window" and close any other window that might be open at the time.
*/
window.WKSS_add = function () {
//show the add window
$("#add").show();
//hide other windows
$("#export").hide();
$("#import").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
var addElement = require('./addelement.js');
//add html to page source
$("body").append(addElement);
//hide add window ("div add" code that was just appended)
$("#add").hide();
var handleAddClick = require('./handleAddClick.js');
//function to fire on click event for "Add new Item"
$("#AddItemBtn").click(function () {
handleAddClick();
});
$("#AddCloseBtn").click(function () {
$("#add").hide();
$("#addForm")[0].reset();
$("#addStatus").text('Ready to add..');
$("#addKanji").removeClass("error");
$("#addMeaning").removeClass("error");
});
/** Keeps legacy srsList updated.
* @depreciate
* @param {SrsItem} srsitem
* @param {Array.<SrsItem>} srsList
* @returns {Array.<SrsItem>} The srs data for a task. Or null if no srsList was provided.
*/
var updateSrsInList = function(srsitem, srsList){
var index = srsitem.i;
if(srsList){
if(srsList[index].kanji===srsitem.kanji){// try search by index
debugging&&console.log("success: "+srsitem.kanji+" found at index "+ index);
//replace only the srs parts of the item
srsList[index].date = srsitem.date;
srsList[index].level = srsitem.level;
srsList[index].locked = srsitem.locked;
srsList[index].manualLock = srsitem.manualLock;
}
return srsList;
}
else{
return null;
}
};
/** Checks if an item's kanji is represented in a list
* @returns {boolean}
*/
var checkForDuplicates = function(list, item){
return list.some(function(a){return a.kanji === item.kanji;});
};
/** Creates a lookup array for each kanji with its srs level. Used for displaying component levels.
* @param item
* @param kanjilist
* @returns An array of the kanji with SRS values for each kanji component.
* @example
eg. 折り紙:
compSRS = [{"kanji": "折", "srs": "guru"}, {"kanji": "紙", "srs": "apprentice"}]
*/
var getCompKanji = function(item, kanjiList){
if (!kanjiList){
kanjiList = [];
}
debugging&&console.log("getCompKanji(item, kanjiList)");
var compSRS = [];
var kanjiReady = false; //indicates if the kanjiList has been populated
var userGuppy = false; //indicates if kanjiList has less than 100 items
var kanjiObj = {};
//has the server responded yet
if (kanjiList.length > 0){
debugging&&console.log("kanjiList is > 0");
kanjiReady = true;
//create lookup object
for (var k=0;k<kanjiList.length;k++){
kanjiObj[kanjiList[k].character] = kanjiList[k];
}
//is there less than 100 kanji in the response
if (kanjiList.length < 100){
debugging&&console.log("kanjiList is < 100");
userGuppy = true;
}
}
var components = item.components;
//for each kanji character component
// this is the outer loop since there will be far less of them than kanjiList
for(var i = 0; i < components.length; i++){
var matched = false;
//for each kanji returned by the server
// for(var j=0; j<kanjiList.length; j++){
//if the kanji returned by the server matches the character in the item
if (typeof kanjiObj[components[i]] !== 'undefined'){
// if (kanjiList[j].character == components[i]){
compSRS[i] = {"kanji": components[i], "srs": kanjiObj[components[i]].srs};
matched = true;
// break; //kanji found: 'i' is its position in item components; 'j' is its postion in the 'kanjiList' server response
}
// }
if (matched === false){ // character got all the way through kanjiList without a match.
if (kanjiReady){ //was there a server response?
if (userGuppy){ //is the user a guppy (kanji probably matches a turtles response)
debugging&&console.log("matched=false, kanjiList.length: "+kanjiList.length);
compSRS[i] = {"kanji": components[i], "srs": "noMatchGuppy"};
}
else{ //user is a turtle, kanji must not have been added to WK (yet)
debugging&&console.log("matched=false, kanjiList.length: "+kanjiList.length);
compSRS[i] = {"kanji": components[i], "srs": "noMatchWK"};
}
}
else{
debugging&&console.log("matched=false, kanjiReady=false, noServerResp");
compSRS[i] = {"kanji": components[i], "srs": "noServerResp"};
}
}
}
return compSRS;
};
var isKanjiLocked = function(srsitem, kanjiList, locksOn){
//item unlocked by default
//may have no kanji, only unlocked kanji will get through the code unflagged
// Enumeration "yes", "no", "DB"
var locked = "no";
if (locksOn){
//get the kanji characters in the word.
var componentList = getCompKanji(srsitem, kanjiList);
// eg: componentList = getCompKanji("折り紙", kanjiList);
// componentList = [{"kanji": "折", "srs": "guru"}, {"kanji": "紙", "srs": "apprentice"}]
var c = componentList.length;
while(c--){
//look for locked kanji in list
if (componentList[c].srs == "apprentice" ||
componentList[c].srs == "noServerResp"||
componentList[c].srs == "unreached"
){
//----could be apprentice etc.
//Simple: lock is 'yes'
locked = "yes";
// "yes": item will be locked while there is no database connection.
// if the server response indicates that it has been unlocked, only then will it be available for review
debugging&&console.log("test srs for apprentice etc. 'locked': "+ locked);
debugging&&console.log(componentList[c].kanji +": "+componentList[c].srs +" -> "+ locked);
break; // as soon as one kanji is locked, the whole item is locked
}
//DB locks get special state
if (componentList[c].srs == "noMatchWK" || componentList[c].srs == "noMatchGuppy"){
locked = "DB";
//"DB" : database limitations, one of two things
//a. the kanji isn't in the database and the user is a guppy --could change if user subscribes or first two levels change/expand
//b. the kanji isn't in the database and the user is a turtle --could change if more kanji added.
debugging&&console.log("test srs for unmatched kanji. 'locked': "+ locked);
debugging&&console.log(componentList[c].kanji +": "+componentList[c].srs +" -> "+ locked);
}
} //for char in componentList
debugging&&console.log("out of character loop");
}
//locked will be either "yes","no", or "DB"
return [locked];
};
/** Gets the Kanji characters in a given string.
* @param {string} vocabString -
* @return {Array.<string>} An array of the kanji components in the given string
*/
var getComponents = function(vocabString){
return Array.prototype.filter.call(vocabString, function(ch){
return /^[\u4e00-\u9faf]+$/.test(ch);
}, this);
};
/** Manages the locked and manualLock properties of srsitem. This is to stop items being locked again after they have been unlocked if any of the kanji used falls below the unlock threshold (eg. if the 勉 in 勉強 falls back to apprentice, we do not want to lock up 勉強 again.)
* @param {Object} item
* @param {string} item.locked - (String enumeration) A real time evaluation of the item (is any of the kanji in the word locked?)
* @param {string} item.manualLock - (String enumeration) Will return 'no' if .locked has ever returned 'no'.
* @returns {ITask} item
*/
var setLocks = function(item){
//once manualLock is "no" it stays "no"
if (item.manualLock !== false && item.manualLock !== "no" && item.manualLock !== "n"){
var kanjiList = localGet('User-KanjiList')||[];
item.components = getComponents(item.kanji);
var kanjiLockedResult = isKanjiLocked(item, kanjiList, locksOn);
item.locked = kanjiLockedResult[0];
item.manualLock = item.locked;
}else{
item.manualLock = false;
}
debugging&&console.log("setting locks for "+ item.kanji +": locked: "+item.locked+", manualLock: "+ item.manualLock);
return item;
};
/** Converts number of milliseconds into a readable string
* @param {number} milliseconds - The number of milliseconds to approximate
* @returns {string} Readable time frame ('2 months', '3 hours', '1 week' etc).
*/
var ms2str = function(milliseconds){
var num; //number of months weeks hours etc
//more time has elapsed than required for the level
if(milliseconds <= 0) {
return "Now" ;
}
if(milliseconds > 2628000000) {//About a month
num = Math.floor(milliseconds/2628000000).toString()+" month";
if (num !== "1 month"){
return num+"s";
}else{
return num;
}
}
if(milliseconds > 604800000) {//A week
num = Math.floor(milliseconds/604800000).toString()+" week";
if (num !== "1 week"){
return num+"s";
}else{
return num;
}
}
if(milliseconds > 86400000) {//A day
num = Math.floor(milliseconds/86400000).toString()+" day";
if (num !== "1 day"){
return num+"s";
}else{
return num;
}
}
if(milliseconds > 3600000) {//An hour
num = Math.floor(milliseconds/3600000).toString()+" hour";
if (num !== "1 hour"){
return num+"s";
}else{
return num;
}
}
if(milliseconds > 60000) {//A minute
num = Math.floor(milliseconds/60000).toString()+" minute";
if (num !== "1 minute"){
return num+"s";
}else{
return num;
}
}
if(milliseconds > 0) {//A second is 1000, but need to return something for less than one too
num = Math.floor(milliseconds/1000).toString()+" second";
if (num !== "1 second"){
return num+"s";
}else{
return num;
}
}
};
/** Retrieves values from storage to populate 'editItems' menu
*/
var generateEditOptions = function() {
var select = document.getElementById('editWindow');
//clear the editWindow
while (select.firstChild) {
select.removeChild(select.firstChild);
}
//check for items to add
if (localStorage.getItem('User-Vocab')) {
//retrieve from local storage
var vocabList = StorageUtil.getVocList();
var srslist = StorageUtil.getVocList();
var options = [];
//build option string
//var i = vocabList.length;
//while (i--){
vocabList.forEach(function(task){
//form element to save string
var opt = document.createElement('option');
//dynamic components of string
//when is this item up for review
var due = task.due||task.date + srsObject[task.level].duration;
var review = "";
//no future reviews if burned
if(task.level >= 9) {
review = "Never";
}
//calculate next relative review time
//current timestamp is past due date.
else if(Date.now() >= due) {
review = "Now" ;
}
else {
review = ms2str(due - Date.now());
}//end if review is not 'never' or 'now'
var text = task.kanji + " & " +
task.reading + " & " +
task.meaning + " (" +
srsObject[task.level].rank +
" - Review: " +
review + ") Locked: " +
task.manualLock;
opt.value = i;
opt.innerHTML = text;
options.push(opt);//for future use (sorting data etc)
select.appendChild(opt);//export item to option menu
}, this);
}
};
/** Edit Items
*/
window.WKSS_edit = function () {
generateEditOptions();
$("#edit").show();
//hide other windows
$("#export").hide();
$("#import").hide();
$("#add").hide();
$("#selfstudy").hide();
};
var buildNode = require('./buildnode.js');
var buildWindow = require('./buildwindow.js');
/*var addEditWindow = function() {
var editWindow = buildNode('div', {id: "WKSS-edit", className: "WKSS"});
var editForm = buildNode('form', {id: "WKSS-editForm"});
editWindow.appendChild(editForm);
var editCloseButton = buildNode('button', {id: "WKSS-editCloseBtn", className: "WKSS-close"});
editForm.appendChild(editCloseButton);
editCloseButton.appendChild(buildNode('i', {className: "icon-remove"}));
var h1Element = buildNode('h1');
editForm.appendChild(h1Element);
h1Element.appendChild(document.createTextNode("Edit your Vocab"));
var selectElement = buildNode('select', {id: "editWindow", size: "8"});
editForm.appendChild(selectElement);
var editItemText = buildNode('input', {type: "text" id: "editItem" name: "" size: "40" placeholder: "Select vocab, click edit, change and save!"});
editForm.appendChild(editItemText);
var editStatus = buildNode('p', {id: "editStatus"});
editForm.appendChild(editStatus);
editStatus.appendChild(document.createTextNode("Ready to edit.."));
var editButton = buildNode('button', {id: "EditEditBtn", type: "button"});
editForm.appendChild(editButton);
editButton.appendChild(document.createTextNode("Edit"));
var editSave = buildNode('button', {id: "EditSaveBtn", type: "button"});
editForm.appendChild(editSave);
editSave.appendChild(document.createTextNode("Save"));
var editDelete = buildNode('button', {id: "EditDeleteBtn", type: "button", title: "Delete selected item"});
editForm.appendChild(editDelete);
editDelete.appendChild(document.createTextNode("Delete"));
var editDeleteAll = buildNode('button', {id: "EditDeleteAllBtn", type: "button", title: "本当にやるの?"});
editForm.appendChild(editDeleteAll);
editDeleteAll.appendChild(document.createTextNode("Delete All"));
var editResetLevels = buildNode('button', {id: "ResetLevelsBtn", type: "button"});
editForm.appendChild(editResetLevels);
editResetLevels.appendChild(document.createTextNode("Reset levels"));
return editWindow;
};
*/
var editWindowStructure = {
id: "WKSS-edit",
className: "WKSS",
childNodes:[{
tag: 'form',
id: "WKSS-editForm",
childNodes:[{
tag: 'button',
id: "WKSS-editCloseBtn",
className: "WKSS-close",
childNodes:[{
tag: 'i',
className: "icon-remove"
}]
},{
tag: 'h1',
childNodes:["Edit your Vocab"]
},{
tag: 'select',
id: "editWindow",
other: {size: "8"}
},{
tag: 'input',
other:{
type: "text",
name: "",
size: "40",
placeholder: "Select vocab, click edit, change and save!"
},
id: "editItem"
},{
tag: 'p',
id: "editStatus",
childNodes:["Ready to edit..."]
},{
tag: 'button',
id: "EditEditBtn",
other: {type: "button"},
childNodes:["Edit"]
},{
tag: 'button',
id: "EditSaveBtn",
other:{type: "button"},
childNodes:["Save"]
},{
tag: 'button',
id: "EditDeleteBtn",
other: {type: "button", title: "Delete selected item"},
childNodes:["Delete"]
},{
tag: 'button',
id: "EditDeleteAllBtn",
other: {type: "button", title: "本当にやるの?"},
childNodes:["Delete All"]
},{
tag: 'button',
id: "ResetLevelsBtn",
other: {type: "button"},
childNodes:["Reset levels"]
}]
}]
};
var addEditWindow = buildWindow(editWindowStructure);
$("body").append(addEditWindow);
$("#WKSS-edit").hide();
/** Resets the levels of all tasks and re-indexes them in storage.
* @param {Event} evt - Click event (not used)
*/
var resetLevels = function (evt) {
var vocList = StorageUtil.getVocList().map(function(vocItem, i){
vocItem.level = 0;
debugging&&console.log("vocList[i].i before: "+vocItem.i);
vocItem.i=i;
debugging&&console.log("vocList[i].i after: "+vocItem.i);
return vocItem;
}, this);
StorageUtil.setVocList(vocList);
};
$("#ResetLevelsBtn").click(resetLevels);
$("#EditEditBtn").click(function () {
//get handle for 'select' area
var select = document.getElementById("editWindow");
//get the index for the currently selected item
var index = select.selectedIndex; //select.options[select.selectedIndex].value is not required, option values are set to index
var vocabList = StorageUtil.getVocList();
vocabList = vocabList.reverse();
document.getElementById("editItem").value = JSON.stringify(vocabList[index]);
document.getElementById("editItem").name = index; //using name to save the index
$("#editStatus").text('Loaded item to edit');
});
var isEmpty = function(value) {
return (value === void 0 || value === null);
};
var isArray = function(arg){
return Array.isArray ? Array.isArray(arg) : Object.prototype.toString.call(arg) === '[object Array]';
};
/** Validates a task object
* @param {Task} add - The Task being verified
* @returns {Boolean} If the provided task has all the necessary properties to be added to the review list.
*/
var isItemValid = function(add) {
return (!isEmpty(add.kanji) && //kanji property exists
!isEmpty(add.meaning) && //meaning property exists
!isEmpty(add.reading) && //reading property exists
isArray(add.meaning) &&//meaning is an array
isArray(add.reading));//reading is an array
};
$("#EditSaveBtn").click(function () {
//-- be aware
//deleting one item may cause mismatch if i is property of item in list
try {
if ($("#editItem").val().length !== 0) {
var editItem = document.getElementById("editItem");
var index = editItem.name;
var item = JSON.parse(editItem.value.toLowerCase());
// Make sure that the word 'meaning' is immutable, so it exists to trim
if (item.meaning){
item.meaning.forEach(function(meaning, m, meanings){
if (meaning === ""){
delete meanings[m];
}
}, this);
}
var fullList = StorageUtil.getVocList().reverse();
if (isItemValid(item) &&//item is valid
!(checkForDuplicates(fullList,item) && //kanji (if changed) is not already in the list
fullList[index].kanji !== item.kanji)) {//unless it is the item being edited
var srslist = StorageUtil.getVocList().reverse();
//get srs components of item(list)
fullList[index] = item;//does not have srs stuff, re-add it now
fullList[index].date = srslist[index].date;
fullList[index].level = srslist[index].level;
fullList[index].locked = srslist[index].locked;
fullList[index].manualLock = srslist[index].manualLock;
fullList = fullList.reverse(); //reset order of array
localSet('User-Vocab', fullList);
generateEditOptions();
$("#editStatus").html('Saved changes!');
document.getElementById("editItem").value = "";
document.getElementById("editItem").name = "";
}
else{
$("#editStatus").text('Invalid item or duplicate!');
alert(isItemValid(item).toString() +" && !("+ checkForDuplicates(fullList,item).toString()+" && !("+fullList[index].kanji+" !== "+item.kanji+")");
}
}
}
catch (e) {
$("#editStatus").text(e);
}
});
var updateEditGUI = function(){
generateEditOptions();
document.getElementById("editItem").value = "";
document.getElementById("editItem").name = "";
};
var editDelete = function () {
//select options element window
var select = document.getElementById("editWindow");
//index of selected item
var item = select.options[select.selectedIndex].value;
//fetch JSON strings from storage and convert them into Javascript literals
var vocabList = StorageUtil.getVocList();
//starting at selected index, remove 1 entry (the selected index).
if (item > -1) {
if (vocabList !== null){
vocabList.splice(item, 1);
}
}
//yuck
if (vocabList.length !== 0) {
localSet('User-Vocab', vocabList);
}
else {
localStorage.removeItem('User-Vocab');
}
updateEditGUI();
$("#editStatus").text('Item deleted!');
};
$("#EditDeleteBtn").click(editDelete);
var editDeleteAll = function () {
var deleteAll = confirm("Are you sure you want to delete all entries?");
if (deleteAll) {
//drop local storage
localStorage.removeItem('User-Vocab');
updateEditGUI();
$("#editStatus").text('All items deleted!');
}
};
$("#EditDeleteAllBtn").click(editDeleteAll);
$("#EditCloseBtn").click(function () {
$("#edit").hide();
$("#editForm")[0].reset();
$("#editStatus").text('Ready to edit..');
});
/** Export
*/
window.WKSS_export = function () {
$("#export").show();
//hide other windows
$("#add").hide();
$("#import").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
var exportWindowStructure = {
id: "WKSS-export",
className: "WKSS",
childNodes:[{
tag: 'form',
id: "WKSS-exportForm",
childNodes:[
{
tag: 'button',
id: "WKSS-exportCloseBtn",
className: "WKSS-close",
childNodes:[{
tag: 'i',
className: "icon-remove"
}]
},
{
tag: 'h1',
childNodes:["Export Items"]
},
{
tag: 'textarea',
id: "exportArea",
other: {cols: "50", rows: "18", placeholder: "Export your stuff! Sharing is caring ;)"}
},
{
tag: 'p',
id: "exportStatus",
childNodes:["Ready to export..."]
},
{
tag: 'button',
id: "ExportItemsBtn",
other: {type: "button"},
childNodes:["Export Items"]
},
{
tag: 'button',
id: "ExportSelectAllBtn",
other:{type: "button"},
childNodes:["Select All"]
},
{
tag: 'button',
id: "ExportCsvBtn",
other: {type: "button"},
childNodes:["Export CSV"]
}
]
}]
};
var exportWindow = buildWindow(exportWindowStructure);
$("body").append(exportWindow);
$("#WKSS-export").hide();
$("#ExportItemsBtn").click(function () {
if (localStorage.getItem('User-Vocab')) {
$("#exportForm")[0].reset();
var vocabList = StorageUtil.getVocList();
$("#exportArea").text(JSON.stringify(vocabList));
$("#exportStatus").text("Copy this text and share it with others!");
}
else {
$("#exportStatus").text("Nothing to export yet :(");
}
});
var select_all = function(str) {
var text_val = document.getElementById(str);
debugging&&console.log(text_val);
text_val.focus();
text_val.select();
};
$("#ExportSelectAllBtn").click(function () {
if ($("#exportArea").val().length !== 0) {
select_all("exportArea");
$("#exportStatus").text("Don't forget to CTRL + C!");
}
});
var createCSV = function(JSONstring){
var JSONobject = (typeof JSONstring === 'string') ? JSON.parse(JSONstring) : JSONstring;
var key;
var CSVarray = [];
var header = [];
var id = JSONobject.length;
if (id){//object not empty
for (key in JSONobject[0]){
if (JSONobject[0].hasOwnProperty(key)){
header.push(key);
}
}
}
CSVarray.push(header.join(','));
while(id--){
var line = [];
var h = header.length;
while(h--){// only do keys in header, in the header's order. //JSONobject[id]){
key = header[h];
if(JSONobject[id][key] !== undefined){
if (Array.isArray(JSONobject[id][key])){
//parse array here
line.push(JSONobject[id][key].join("\t"));
}else{
line.push(JSONobject[id][key]);
}
}
}line = line.reverse();
CSVarray.push(line.join(','));
}
var CSVstring = CSVarray.join("\r\n");
return encodeURI("data:text/csv;charset=utf-8," + CSVstring);
};
$("#ExportCsvBtn").click(function () {
var vocabList = StorageUtil.getVocList();
var CsvFile = createCSV(vocabList);
window.open(CsvFile);
});
$("#ExportCloseBtn").click(function () {
$("#export").hide();
$("#exportForm")[0].reset();
$("#exportArea").text("");
$("#exportStatus").text('Ready to export..');
});
/** Import
*/
window.WKSS_import = function () {
$("#import").show();
//hide other windows
$("#add").hide();
$("#export").hide();
$("#edit").hide();
$("#selfstudy").hide();
};
var importWindowStructure = {
id: "WKSS-import",
className: "WKSS",
childNodes:[{
tag: 'form',
id: "WKSS-importForm",
childNodes:[
{
tag: 'button',
id: "WKSS-importCloseBtn",
className: "WKSS-close",
childNodes:[{
tag: 'i',
className: "icon-remove"
}]
},
{
tag: 'h1',
childNodes:["Import Items"]
},
{
tag: 'textarea',
id: "importArea",
other: {cols: "50", rows: "18", placeholder: "Paste your stuff and hit the import button! Use with caution!"}
},
{
tag: 'p',
id: "importStatus",
childNodes:["Ready to import..."]
},
{
tag: 'label',
id: "ImportItemsBtn",
className: "button",
other: {type: "button", style: "display:inline;"},
childNodes:["Import Items"]
},
{
tag: 'label',
id: "ImportCsvBtn",
className: "button",
other: {style:"display:inline; cursor: pointer;"},
childNodes:["Import CSV",
{
tag: 'input',
id: "upload",
other: {
type: "file", accept: ".csv, .tsv",
style: "height:0px;width:0px;background:red;opacity:0;filter:opacity(1);"
}
}
]
},
{
tag: 'label',
id: "ImportWKBtn",
className: "button",
other: {style: "display:inline;"},
childNodes:[
{
tag:'i',
className: "icon-download-alt"
},
"WK"
]
}
]
}]
};
var importWindow = buildWindow(importWindowStructure);
$("body").append(importWindow);
$("#import").hide();
var checkAdd = function(add) {
//take a JSON object (parsed from import window) and check with stored items for any duplicates
// Returns true if each item in 'add' array is valid and
//at least one of them already exists in storage
var i = add.length;
if(localStorage.getItem('User-Vocab')) {
var vocabList = StorageUtil.getVocList();
while(i--){
if (isItemValid(add[i]) &&
checkForDuplicates(vocabList,add[i]))
return true;
}
}
return false;
};
document.getElementById("upload") && document.getElementById("upload").addEventListener('change', ImportUtil.fileUpload, false);
var refreshLocks = function(){
var vocList = StorageUtil.getVocList().map(function(vocItem){
debugging&&console.log("vocList[i] = setLocks(vocList[i]);");
vocItem = setLocks(vocItem);
return vocItem;
}, this);
console.groupEnd();
StorageUtil.setVocList(vocList);
};
$("#ImportWKBtn").click(function(){
WanikaniUtil.getServerResp(APIkey,"vocabulary");
debugging&&console.log("maybe?");
});
$("#ImportItemsBtn").click(function () {
if ($("#importArea").val().length !== 0) {
try {
var add = JSON.parse($("#importArea").val().toLowerCase());
alert(JSON.stringify(add));
if (checkAdd(add)) {
$("#importStatus").text("No valid input (duplicates?)!");
return;
}
var newlist;
var srslist = [];
if (localStorage.getItem('User-Vocab')) {
var vocabList = StorageUtil.getVocList();
srslist = StorageUtil.getVocList();
newlist = vocabList.concat(add);
}
else {
newlist = add;
}
var i = add.length;
while(i--){
StorageUtil.setVocItem(add[i]);
}
$("#importStatus").text("Import successful!");
$("#importForm")[0].reset();
$("#importArea").text("");
}
catch (e) {
$("#importStatus").text("Parsing Error!");
debugging&&console.log(e);
}
}
else {
$("#importStatus").text("Nothing to import :( Please paste your stuff first");
}
});
$("#ImportCloseBtn").click(function () {
$("#import").hide();
$("#importForm")[0].reset();
$("#importArea").text("");
$("#importStatus").text('Ready to import..');
});
var playAudio = function() {
var kanji = document.getElementById('rev-kanji').innerHTML;
var kana = (document.getElementById('rev-solution').innerHTML.split(/[,、]+\s*/))[0];
document.getElementById('rev-audio').innerHTML = "";
document.getElementById('audio-form').action = "";
//document.getElementById('AudioButton').disabled = true;
if( !kanji.match(/[a-zA-Z]+/i) && !kana.match(/[a-zA-Z]+/i)) {
kanji = encodeURIComponent(kanji);
kana = encodeURIComponent(kana);
var i;
var newkanji = "";
for(i = 1; i < kanji.length; i = i+3) {
newkanji = newkanji.concat(kanji[i-1]);
newkanji = newkanji.concat('2');
newkanji = newkanji.concat('5');
newkanji = newkanji.concat(kanji[i]);
newkanji = newkanji.concat(kanji[i+1]);
}
var newkana = "";
for(i = 1; i < kana.length; i = i+3) {
newkana = newkana.concat(kana[i-1]);
newkana = newkana.concat('2');
newkana = newkana.concat('5');
newkana = newkana.concat(kana[i]);
newkana = newkana.concat(kana[i+1]);
}
var url = "http://www.csse.monash.edu.au/~jwb/audiock.swf?u=kana=" + newkana + "%26kanji=" + newkanji;
debugging&&console.log("Audio URL: " + url);
document.getElementById('AudioButton').disabled = false;
document.getElementById('rev-audio').innerHTML = url;
}
};
var nextReview = function(reviewList) {
//sets up the next item for review
//uses functions:
// wanakana.bind/unbind
var rnd = Math.floor(Math.random()*reviewList.length);
var item = reviewList[rnd];
sessionSet('WKSS-item', JSON.stringify(item));
sessionSet('WKSS-rnd', rnd);
if (sessionStorage.getItem('User-Stats')){
$("#RevNum").innerHtml = sessionGet('User-Stats').length;
}
document.getElementById('rev-kanji').innerHTML = item.prompt;
document.getElementById('rev-type').innerHTML = item.type;
var typeBgColor = 'grey';
if (item.type.toLowerCase() == 'meaning'){
typeBgColor = 'blue';
} else if (item.type.toLowerCase() == 'reading'){
typeBgColor = 'orange';
} else if (item.type.toLowerCase() == 'reverse'){
typeBgColor = 'orange';
}
document.getElementById('wkss-type').style.backgroundColor = typeBgColor;
$("#rev-solution").removeClass("info");
document.getElementById('rev-solution').innerHTML = item.solution;
document.getElementById('rev-index').innerHTML = item.index;
//initialise the input field
$("#rev-input").focus();
$("#rev-input").removeClass("caution");
$("#rev-input").removeClass("error");
$("#rev-input").removeClass("correct");
$("#rev-input").val("");
//check for alphabet letters and decide to bind or unbind wanakana
if (item.solution[0].match(/[a-zA-Z]+/i)) {
wanakana.unbind(document.getElementById('rev-input'));
$('#rev-input').attr('placeholder','Your response');
$('#rev-input').attr('lang','en');
}
else {
wanakana.bind(document.getElementById('rev-input'));
$('#rev-input').attr('placeholder','答え');
$('#rev-input').attr('lang','ja');
}
playAudio();
};
//global to keep track of when a review is in session.
var reviewActive = false;
var startReview = function() {
debugging&&console.log("startReview()");
submit = true;
reviewActive = true;
//get the review 'list' from session storage, line up the first item in queue
var reviewList = sessionGet('User-Review')||[];
nextReview(reviewList);
};
/** Review Items
*/
window.WKSS_review = function () {
//is there a session waiting in storage?
if(sessionStorage.getItem('User-Review')) {
//show the selfstudy window
$("#selfstudy").show();
//hide other windows
$("#add").hide();
$("#export").hide();
$("#edit").hide();
$("#import").hide();
startReview();
}
};
$("body").append(' \
<div id="selfstudy" class="WKSS">\
<button id="SelfstudyCloseBtn" class="wkss-close" type="button"><i class="icon-remove"></i></button>\
<h1>Review<span id="RevNum"></span></h1>\
<div id="wkss-kanji">\
<span id="rev-kanji"></span>\
</div><div id="wkss-type">\
<span id="rev-type"></span><br />\
</div><div id="wkss-solution">\
<span id="rev-solution"></span>\
</div><div id="wkss-input">\
<input type="text" id="rev-input" size="40" placeholder="">\
</div><span id="rev-index" style="display: block;"></span>\
\
<form id="audio-form">\
<label id="AudioButton" class="button">Play audio</label>\
<label id="WrapUpBtn" class="button">Wrap Up</label>\
</form>\
<div id="rev-audio" style="display:none;"></div>\
</div>');
$("#selfstudy").hide();
$("#SelfstudyCloseBtn").click(function () {
$("#selfstudy").hide();
$("#rev-input").val("");
reviewActive = false;
});
var binarySearch = function(values, target, start, end) {
//debugging&&console.log("binarySearch(values: ,target: , start: "+start+", end: "+end+")");
if (start > end) {
//start has higher value than target, end has lower value
//item belongs between
// need to return 'start' with a flag that it hasn't been found
//invert sign :)
return -(start);
//for testing truths
// return String(end)+" < "+item.index+" < "+String(start);
} //does not exist
var middle = Math.floor((start + end) / 2);
var value = values[middle];
/*debugging&&console.log("start.index: "+values[start].index);
debugging&&console.log("middle.index: "+values[middle].index);
debugging&&console.log("end.index: "+values[end].index);
*/
if (Number(value.index) > Number(target.index)) {
return binarySearch(values, target, start, middle-1);
}
if (Number(value.index) < Number(target.index)) {
return binarySearch(values, target, middle+1, end);
}
return middle; //found!
};
var findIndex = function(values, target) {
return binarySearch(values, target, 0, values.length - 1);
};
$("#WrapUpBtn").click(function() {
var sessionList = sessionGet('User-Review')||[];
var statsList = sessionGet('User-Stats')||[];
//if an index in sessionList matches one in statsList, don't delete
var sessionI = sessionList.length;
var item = sessionGet('WKSS-item')||[];
var arr2 = [];
//for every item in sessionList, look for index in statsList,
//if not there (-1) delete item from sessionList
while (sessionI--){
var index = findIndex(statsList,sessionList[sessionI]);
if ((Math.sign(1/index) !== -1)||(sessionList[sessionI].index == item.index)){
arr2.push(sessionList[sessionI]);
}
}
debugging&&console.log(arr2);
sessionSet('User-Review', JSON.stringify(arr2));
});
/** Save to list based on .index property
* @param {Array.<task>} eList
* @param {task} eItem
*/
var saveToSortedList = function(eList,eItem){
var get = findIndex(eList,eItem);
if (Math.sign(1/get) === -1){
eList.splice(-get,0,eItem);
}
return eList;
};
//-------
var openInNewTab = function(url) {
var win=window.open(url, '_blank');
win.focus();
};
$("#AudioButton").click(function () {
openInNewTab(document.getElementById('rev-audio').innerHTML);
});
var Rev_Item = function(prompt, kanji, type, solution, index){
this.prompt = prompt;
this.kanji = kanji;
this.type = type;
this.solution = solution;
this.index = index;
};
var updateSRS = function(stats, voclist) {
var now = Date.now();
if (voclist[stats.index].due < now){ //double check that the item was really up for review.
if(!stats.numWrong && voclist[stats.index].level < 9) {//all correct (none wrong)
voclist[stats.index].level++;
}
else {
stats.numWrong = {};
//Adapted from WaniKani's srs to authentically mimic level downs
var o = (stats.numWrong.Meaning||0)+(stats.numWrong.Reading||0)+(stats.numWrong.Reverse||0);
var t = voclist[stats.index].level;
var r=t>=5?2*Math.round(o/2):1*Math.round(o/2);
var n=t-r<1?1:t-r;
voclist[stats.index].level = n;//don't stay on 'started'
}
voclist[stats.index].date = now;
voclist[stats.index].due = now + srsObject[voclist[stats.index].level].duration;
console.log("Next review in "+ms2str(srsObject[voclist[stats.index].level].duration));
return voclist;
}
};
var showResults = function() {
var statsList = sessionGet('User-Stats')||[];
sessionStorage.clear();
console.log("statslist", statsList);
var voclist = StorageUtil.getVocList();
statsList.forEach(function(stats, i, statsList){
debugging&&console.log("stats",stats);
var altText = voclist[stats.index].level;//+stats.type;
if (stats.numWrong) {
if (stats.numWrong.Meaning)
altText = altText + " Meaning Wrong x"+stats.numWrong.Meaning +"\n";
if (stats.numWrong.Reading)
altText = altText + " Reading Wrong x"+stats.numWrong.Reading +"\n";
if (stats.numWrong.Reverse)
altText = altText + " Reverse Wrong x"+stats.numWrong.Reverse +"\n";
}
if (stats.numCorrect){
if (stats.numCorrect.Meaning)
altText = altText + " Meaning Correct x"+stats.numCorrect.Meaning +"\n";
if (stats.numCorrect.Reading)
altText = altText + " Reading Correct x"+stats.numCorrect.Reading +"\n";
if (stats.numCorrect.Reverse)
altText = altText + " Reverse Correct x"+stats.numCorrect.Reverse +"\n";
}
console.log(stats);
//TODO sort into apprentice, guru, etc
document.getElementById("stats-a").innerHTML +=
"<span class=" +
(stats.numWrong? "\"rev-error\"":"\"rev-correct\"") +
" title='"+altText+"'>" + stats.kanji + "</span>";
//map with side effects?
statsList[i] = updateSRS(stats, voclist);
}, this);
sessionSet("User-Stats",statsList);
localSet("User-Vocab", voclist);
};
$("body").append(' \
<div id="resultwindow" class="WKSS"> \
<button id="ReviewresultsCloseBtn" class="wkss-close" type="button"><i class="icon-remove"></i></button>\
<h1>Review Results</h1>\
<h2>All</h2>\
<div id="stats-a"></div>\
</div>');
$("#resultwindow").hide();
$("#ReviewresultsCloseBtn").click(function () {
$("#resultwindow").hide();
document.getElementById("stats-a").innerHTML = "";
});
var markAnswer = function(item) {
//evaluate 'item' against the question.
// match by index
// get type of question
// determine if right or wrong and return result appropriately
//get the question
//var prompt = document.getElementById('rev-kanji').innerHTML.trim();
//get the answer
var answer = $("#rev-input").val().toLowerCase();
//get the index
var index = document.getElementById('rev-index').innerHTML.trim();
//get the question type
var type = document.getElementById('rev-type').innerHTML.trim();
//var vocab = localGet("User-Vocab");
//get the item if it is in the current session
var storedItem = sessionGet(item.index);
if (storedItem){
item.numCorrect = storedItem.numCorrect;
item.numWrong = storedItem.numWrong;
}
if (index == item.index){//-------------
if (inputCorrect()){
debugging&&console.log(answer+"/"+item.solution[0]);
if (!item.numCorrect){
debugging&&console.log("initialising numCorrect");
item.numCorrect={};
}
debugging&&console.log("Correct: "+ type);
if (type == "Meaning"){
if (!item.numCorrect.Meaning)
item.numCorrect.Meaning = 0;
item.numCorrect.Meaning++;
}
if (type == "Reading"){
if (!item.numCorrect.Reading)
item.numCorrect.Reading = 0;
item.numCorrect.Reading++;
}
if (type == "Reverse"){
if (!item.numCorrect.Reverse)
item.numCorrect.Reverse = 0;
item.numCorrect.Reverse++;
}
}else{
debugging&&console.log(answer+"!="+item.solution);
if (!item.numWrong){
debugging&&console.log("initialising numCorrect");
item.numWrong={};
}
debugging&&console.log("Wrong: "+ type);
if (type == "Meaning"){
if (!item.numWrong.Meaning)
item.numWrong.Meaning = 0;
item.numWrong.Meaning++;
}
if (type == "Reading"){
if (!item.numWrong.Reading)
item.numWrong.Reading = 0;
item.numWrong.Reading++;
}
if (type == "Reverse"){
if (!item.numWrong.Reverse)
item.numWrong.Reverse = 0;
item.numWrong.Reverse++;
}
}
}
else {
console.error("Error: indexes don't match");
}
return item;
};
//jquery keyup event
$("#rev-input").keyup(ReviewUtil.submitResponse);
/** Adds the Button
*/
var addUserVocabButton = function() {
debugging&&console.log("addUserVocabButton()");
//Functions (indirect)
// WKSS_add()
// WKSS_edit()
// WKSS_export()
// WKSS_import()
// WKSS_lock()
// WKSS_review()
var nav = document.getElementsByClassName('nav');
debugging&&console.log("generating review list because: initialising script and populating reviews");
if (nav&&nav.length>2) {
nav[2].innerHTML = nav[2].innerHTML + "\n\
<li class=\"dropdown custom\">\n\
<a class=\"dropdown-toggle custom\" data-toggle=\"dropdown\" href=\"#\" onclick=\"generateReviewList();\">\n\
<span lang=\"ja\">自習</span>\n\
Self-Study <i class=\"icon-chevron-down\"></i>\n\
</a>\n\
<ul class=\"dropdown-menu\" id=\"WKSS_dropdown\">\n\
<li class=\"nav-header\">Customize</li>\n\
<li><a id=\"click\" href=\"#\" onclick=\"WKSS_add();\">Add</a></li>\n\
<li><a href=\"#\" onclick=\"WKSS_edit();\">Edit</a></li>\n\
<li><a href=\"#\" onclick=\"WKSS_export();\">Export</a></li>\n\
<li><a href=\"#\" onclick=\"WKSS_import();\">Import</a></li>\n\
<!--// <li><a href=\"#\" onclick=\"WKSS_lock();\">Server Settings</a></li>//-->\n\
<li class=\"nav-header\">Learn</li>\n\
<li><a id=\"user-review\" href=\"#\" onclick=\"WKSS_review();\">Please wait...</a></li>\n\
</ul>\n\
</li>";
}else{
console.error("could not find nav", nav);
}
console.log("addUserVocab");
};
/** Error handling
* Can use 'error.stack', not cross-browser (though it should work on Firefox and Chrome)
*/
var logError = function(error) {
debugging&&console.log("logError(error)");
var stackMessage = "";
if ("stack" in error)
stackMessage = "\n\tStack: " + error.stack;
debugging&&console.log("WKSS: Error: " + error.name + "\n\tMessage: " + error.message + stackMessage);
console.error("WKSS: Error: " + error.name + "\n\tMessage: " + error.message + stackMessage);
};
/** Prepares the script
*/
var scriptInit = function() {
debugging&&console.log("scriptInit()");
//functions:
// addUserVocabButton()
// logError(err)
debugging&&console.log("Initializing Wanikani UserVocab Script!");
gM_addStyle(".custom .dropdown-menu {background-color: #DBA901 !important;}");
gM_addStyle(".custom .dropdown-menu:after {border-bottom-color: #DBA901 !important;");
gM_addStyle(".custom .dropdown-menu:before {border-bottom-color: #DBA901 !important;");
gM_addStyle(".open .dropdown-toggle.custom {background-color: #FFC400 !important;}");
gM_addStyle(".custom .dropdown-menu a:hover {background-color: #A67F00 !important;}");
gM_addStyle(".custom:hover {color: #FFC400 !important;}");
gM_addStyle(".custom:hover span {border-color: #FFC400 !important;}");
gM_addStyle(".custom:focus {color: #FFC400 !important;}");
gM_addStyle(".custom:focus span {border-color: #FFC400 !important;}");
gM_addStyle(".open .custom span {border-color: #FFFFFF !important;}");
gM_addStyle(".open .custom {color: #FFFFFF !important}");
var wkStyleCSS = require('./wkstyle.js');
gM_addStyle(wkStyleCSS);
// Set up buttons
try {
if (typeof localStorage !== "undefined") {
addUserVocabButton();
//provide warning to users trying to use the (incomplete) script.
debugging&&console.log("this script is still incomplete: \n\
It is provided as is without warranty express or implied\n\
in the hope that you may find it useful.");
}
else {
debugging&&console.log("Wanikani Self-Study: Your browser does not support localStorage.. Sorry :(");
}
}
catch (err) {
logError(err);
}
};
console.info(document.readyState);
console.log("adding DOM listener", document.readyState);
// Check for file API support.
if (window.File && window.FileReader && window.FileList && window.Blob) {
}
else {
alert('The File APIs are not fully supported in this browser.');
}
/** Start the script
*/
//unless the user navigated from the review directory, they are unlikely to have unlocked any kanji
var noNewStuff = /^https:\/\/.*\.wanikani\.com\/.*/.test(document.referrer)&&!(/https:\/\/.*\.wanikani\.com\/review.*/.test(document.referrer));
var usingHTTPS = /^https:/.test(window.location.href);
console.info(usingHTTPS, window.location.href);
if (usingHTTPS){
if (!noNewStuff){ //Don't waste time if user is browsing site
WanikaniUtil.getServerResp(APIkey);
}else{
debugging&&console.log("User is unlikely to have new kanji unlocked");
}
debugging&&console.info("WaniKani Self-Study Plus is about to start");
scriptInit();
}else{
debugging&&console.warn("It appears that you are not using https protocol. Attempting to redirect to https now.");
window.location.href = window.location.href.replace(/^http/, "https");
}
}
if (document.readyState === 'complete'){
console.info("About to initialise WKSS+");
main();
} else {
window.addEventListener("load", main, false);
}
},{"./addelement.js":1,"./buildnode.js":2,"./buildwindow.js":3,"./handleAddClick.js":4,"./importutil.js":6,"./reviewutil.js":7,"./storageutil.js":8,"./wanikaniutil.js":10,"./wkstyle.js":12}],10:[function(require,module,exports){
/** Utilities for interaction with the Wanikani API and general website.
*/
var WanikaniUtil = {
hijackRequests: require('./hijackrequests.js'),
createCORSRequest: function(method, url){
var xhr = new XMLHttpRequest();
if ("withCredentials" in xhr){
xhr.open(method, url, true);
}
else if (typeof XDomainRequest !== "undefined"){
xhr = new XDomainRequest();
xhr.open(method, url);
}
else {
xhr = null;
}
return xhr;
},
/** Gets the user information using the Wanikani API and stores them directly into browser storage.
* @param
* @param
*/
getServerResp: function(APIkey, requestedItem){
requestedItem = requestedItem === void 0 ? 'kanji' :requestedItem;
if (APIkey !== "test"){
var levels = (requestedItem ==="kanji")? "/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":
"/1,2,3,4,5,6,7,8,9,10";
var xhrk = this.createCORSRequest("get", "https://www.wanikani.com/api/user/" + APIkey + "/" + requestedItem + levels);
if (!isEmpty(xhrk)){
xhrk.onreadystatechange = function() {
if (xhrk.readyState == 4){
var kanjiList = this.handleReadyStateFour(xhrk,requestedItem);
if (requestedItem === 'kanji'){
localSet('User-KanjiList', kanjiList);
console.log("kanjiList from server", kanjiList);
//update locks in localStorage
//pass kanjilist into this function
//(don't shift things through storage unecessarily)
refreshLocks();
}
else{
var v = kanjiList.length;
console.log(v + " items found, attempting to import");
while (v--){
StorageUtil.setVocItem(kanjiList[v]);
}
}
}
};
xhrk.send();
console.log("below");
}
}
else {
//dummy server response for testing.
setTimeout(function () {
var kanjiList = [];
console.log("creating dummy response");
kanjiList.push({"character": "猫", "srs": "noServerResp"});
var SRS = "apprentice"; //prompt("enter SRS for 子", "guru");
kanjiList.push({"character": "子", "srs": SRS});
kanjiList.push({"character": "品", "srs": "guru"});
kanjiList.push({"character": "供", "srs": "guru"});
kanjiList.push({"character": "本", "srs": "guru"});
kanjiList.push({"character": "聞", "srs": "apprentice"});
kanjiList.push({"character": "人", "srs": "enlightened"});
kanjiList.push({"character": "楽", "srs": "burned"});
kanjiList.push({"character": "相", "srs": "guru"});
kanjiList.push({"character": "卒", "srs": "noMatchWK"});
kanjiList.push({"character": "無", "srs": "noMatchGuppy"});
console.log("Server responded with dummy kanjiList: \n"+JSON.stringify(kanjiList));
localSet('User-KanjiList', kanjiList);
//update locks in localStorage
refreshLocks();
}, 10000);
}
},
handleReadyStateFour: function(xhrk, requestedItem){
var localkanjiList = [];
console.log("readystate: "+ xhrk.readyState);
var resp = JSON.parse(xhrk.responseText);
console.log("about to loop through requested information");
if (resp.requested_information && resp.requested_information.length){
localkanjiList = resp.requested_information.map(function(requestedTask){
if (requestedItem === "kanji"){
if (requestedTask.user_specific !== null){
return {
character: requestedTask.character,
srs: requestedTask.user_specific.srs,
reading: requestedTask[requestedTask.important_reading].split(",")[0],
meaning: requestedTask.meaning.split(",")[0]
};
}
else{
return {
character: requestedTask.character,
srs: "unreached"
};
}
}
else if(requestedItem === "vocabulary"){
if (requestedTask.user_specific !== null||true){ //--
//build vocablist
return {
kanji: requestedTask.character,
reading: requestedTask.kana.split(","),
meaning: requestedTask.meaning.split(",")
};
}
}
}, this);
}
//return kanjiList
// console.log("Server responded with new kanjiList: \n"+JSON.stringify(kanjiList));
return localkanjiList;
}
};
module.exports = WanikaniUtil;
},{"./hijackrequests.js":5}],11:[function(require,module,exports){
// Window Configs
module.exports = {
add:{height: "300px", width: "300px"},
exportImport:{height: "275px", width: "390px"},
edit:{height: "380px", width: "800px"},
study:{height: "auto", width: "600px"}, //height : auto
result:{height: "500px", width: "700px"}
};
},{}],12:[function(require,module,exports){
/* jshint multistr: true */
// Config for window sizes in pixels
var windowConfig = require('./windowconfig.js');
//.WKSS
var classWKSS = [
{position: "fixed"},
{zIndex: "2"},
{top: "125px"},
{left: "50%"},
{margin: "0px"},
{background: "#FFF"},
{padding: "5px"},
{font: "12px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"},
{color: "#888"},
{textShadow: "1px 1px 1px #FFF"},
{border: "1px solid #DDD"},
{borderRadius: "5px"},
{WebkitBorderRadius: "5px"},
{MozBorderRadius: "5px"},
{boxShadow: "10px 10px 5px #888888"}
];
//.WKSS h1
var classWKSS_h1 = [
{font: "25px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"},
{paddingLeft: "5px"},
{display: "block"},
{borderBottom: "1px solid #DADADA"},
{margin: "0px"},
{color: "#888"}
];
//.WKSS h1 > span
var classWKSS_h1_direct_Span = [
{display: "block"},
{fontSize: "11px"}
];
//.WKSS label
var w = [
{display: "block"},
{margin: "0px 0px 5px"},
];
//.WKSS label>span
w = [
{float: "left"},
{width: "80px"},
{textAlign: "right"},
{paddingRight: "10px"},
{marginTop: "10px"},
{color: "#333"},
{fontFamily: "\"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"},
{fontWeight: "bold"}
];
//.WKSS input[type=\"text\"], .WKSS input[type=\"email\"], .WKSS textarea
w = [
{border: "1px solid #CCC"},
{color: "#888"},
{height: "20px"},
{marginBottom: "16px"},
{marginRight: "6px"},
{marginTop: "2px"},
{outline: "0 none"},
{padding: "6px 12px"},
{width: "80%"},
{borderRadius: "4px"},
{lineHeight: "normal !important"},
{WebkitBorderRadius: "4px"},
{MozBorderRadius: "4px"},
{font: "normal 14px/14px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"},
{WebkitBoxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"},
{boxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"},
{MozBoxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"}
];
//.WKSS select
w = [
{border: "1px solid \"#CCC\""},
{color: "#888"},
{outline: "0 none"},
{padding: "6px 12px"},
{height: "160px !important"},
{width: "95%"},
{borderRadius: "4px"},
{WebkitBorderRadius: "4px"},
{MozBorderRadius: "4px"},
{font: "normal 14px/14px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"},
{WebkitBoxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"},
{boxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"},
{MozBoxShadow: "inset 0 1px 1px rgba(0, 0, 0, 0.075)"},
{background: "#FFF url('down-arrow.png') no-repeat right"},
{appearance: "none"},
{WebkitAppearance: "none"},
{MozAppearance: "none"},
{textIndent: "0.01px"},
{textOverflow: "''"}
];
//.WKSS textarea
w = [
{height: "100px"}
];
//.WKSS button, .button
w = [
{position: "relative"},
{background: "#FFF"},
{border: "1px solid #CCC"},
{padding: "10px 25px 10px 25px"},
{color: "#333"},
{borderRadius: "4px"},
{display: "inline !important"}
];
//.WKSS button:disabled
w = [
{background: "#EBEBEB"},
{border: "1px solid #CCC"},
{padding: "10px 25px 10px 25px"},
{color: "#333"},
{borderRadius: "4px"}
];
//.WKSS .button:hover, button:hover:enabled
w = [
{color: "#333"},
{backgroundColor: "#EBEBEB"},
{borderColor: "#ADADAD"}
];
//.WKSS button:hover:disabled
w = [
{cursor: "default"}
];
//.error
w = [
{borderColor:"#F00 !important"},
{color: "#F00 !important"}
];
//.caution
w = [
{borderColor: "#F90 !important"},
{color: "#F90 !important"}
];
//.correct
w = [
{borderColor: "#0F0 !important"},
{color: "#0F0 !important"}
];
//.info
w = [
{borderColor: "#696969 !important"},
{color: "#696969 !important"}
];
//.rev-error
w = [
{textShadow: "none"},
{border: "1px solid #F00 !important"},
{borderRadius: "10px"},
{backgroundColor: "#F00"},
{padding: "4px"},
{margin: "4px"},
{color: "#FFFFFF"},
{font: "normal 18px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"}
];
//.rev-correct
w = [
{textShadow:"none"},
{border: "1px"},
{solid: "#088A08 !important"},
{borderRadius: "10px"},
{backgroundColor: "#088A08"},
{padding: "4px"},
{margin:"4px"},
{color: "#FFFFFF"},
{font: "normal 18px \"ヒラギノ角ゴ Pro W3\", \"Hiragino Kaku Gothic Pro\",Osaka, \"メイリオ\", Meiryo, \"MS Pゴシック\", \"MS PGothic\", sans-serif"}
];
//#add
w = [
{width: windowConfig.add.width},
{height: windowConfig.add.height},
{marginLeft: -windowConfig.add.width/2}
];
//#export, #import
w = [
{background: "#fff"},
{width: windowConfig.exportImport.width},
{height: windowConfig.exportImport.height},
{marginLeft: -windowConfig.exportImport.width/2}
];
//#edit
w = [
{width: windowConfig.edit.width},
{height: windowConfig.edit.height},
{marginLeft: -windowConfig.edit.width/2}
];
//#selfstudy
w = [
{left: "50%"},
{width: windowConfig.study.width},
{height: windowConfig.study.height},
{marginLeft: -windowConfig.study.width/2}
];
//#resultwindow
w = [
{left:"50%"},
{width: windowConfig.result.width + "px"},
{height: windowConfig.result.height + "px"},
{marginLeft: -windowConfig.result.width/2 + "px"}
];
//#AudioButton
w = [
{marginTop: "35px"},
{position: "relative"},
{display: "inline !important"},
{WebkitMarginBefore: "50px"}
];
//button.wkss-close
w = [
{float: "right"},
{backgroundColor: "#ff4040"},
{color: "#fff"},
{padding: "0px"},
{height: "27px"},
{width: "27px"}
];
//#wkss-close
w = [
{float: "right"},
{backgroundColor: "#ff4040"},
{color: "#fff"},
{padding: "0px"},
{height: "27px"},
{width: "27px"}
];
//#wkss-kanji, #rev-kanji
w = [
{textAlign: "center !important"},
{fontSize: "50px !important"},
{backgroundColor: "#9400D3 !important"},
{color: "#FFFFFF !important"},
{borderRadius: "10px 10px 0px 0px"}
];
//#wkss-solution, #rev-solution
w = [
{textAlign: "center !important"},
{fontSize: "30px !important"},
{color: "#FFFFFF"},
{padding: "2px"}
];
//#wkss-type
w = [
{textAlign: "center !important"},
{fontSize: "24px !important"},
{backgroundColor: "#696969"},
{color: "#FFFFFF !important"},
{borderRadius: "0px 0px 10px 10px"}
];
//#rev-type
w = [
{textAlign: "center !important"},
{fontSize: "24px !important"},
{color: "#FFFFFF !important"},
{borderRadius: "0px 0px 10px 10px"}
];
//#wkss-input
w = [
{textAlign: "center !important"},
{fontSize: "40px !important"},
{height: "80px !important"},
{lineHeight: "normal !important"}
];
//#rev-input
w = [
{textAlign: "center !important"},
{fontSize: "40px !important"},
{height: "60px !important"},
{lineHeight: "normal !important"}
];
//----
module.exports = classWKSS;
//module.exports = wkstyleCSS;
},{"./windowconfig.js":11}]},{},[9])
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/addelement.js","src/buildnode.js","src/buildwindow.js","src/handleAddClick.js","src/hijackrequests.js","src/importutil.js","src/reviewutil.js","src/storageutil.js","src/trunk.js","src/wanikaniutil.js","src/windowconfig.js","src/wkstyle.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACt8EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var buildNode = require('./buildnode.js');\r\n\r\n// Create DOM for 'add' window\r\nvar addElement = buildNode('div', {id: \"add\", className: \"WKSS\"});\r\n\r\nvar formElement = buildNode('form', {id: \"addForm\"});\r\naddElement.appendChild(formElement);\r\n\r\nvar buttonElement = buildNode('button', {id: \"AddCloseBtn\", className: \"wkss-close\", type: \"reset\"});\r\nformElement.appendChild(buttonElement);\r\n\r\nvar iconElement = buildNode('i', {className: \"icon-remove\"});\r\nbuttonElement.appendChild(iconElement);\r\n\r\nvar headerElement = buildNode('h1');\r\nformElement.appendChild(headerElement);\r\n\r\nvar headerText = document.createTextNode(\"Add a new Item\");\r\nheaderElement.appendChild(headerText);\r\n\r\nvar inputKanjiElement = buildNode('input', {id: \"addKanji\", type: \"text\", placeholder: \"Enter 漢字, ひらがな or カタカナ\"});\r\nformElement.appendChild(inputKanjiElement);\r\n\r\nvar inputReadingElement = buildNode('input', {id: \"addReading\", type: \"text\", title: \"Leave empty to add vocabulary like する (to do)\", placeholder: \"Enter reading\"});\r\nformElement.appendChild(inputReadingElement);\r\n\r\nvar inputMeaningElement = buildNode('input', {id: \"addMeaning\", type: \"text\", placeholder: \"Enter meaning\"});\r\nformElement.appendChild(inputMeaningElement);\r\n\r\nvar pElement = buildNode('p', {id: \"addStatus\"});\r\nformElement.appendChild(pElement);\r\n\r\nvar pText = document.createTextNode(\"Ready to add..\");\r\npElement.appendChild(pText);\r\n\r\nvar execButtonElement = buildNode('button', {id: \"AddItemBtn\", type: \"button\"});\r\nformElement.appendChild(execButtonElement);\r\n\r\nvar execText = document.createTextNode(\"Add new Item\");\r\nexecButtonElement.appendChild(execText);\r\n\r\nmodule.exports = addElement;","/** Builds a node element with an id and className and other attributes if provided\r\n* @param {string} type - The type of element to create ('div', 'p', etc...)\r\n* @param {object} [options]\r\n* @param {string} options.id - The id of the node\r\n* @param {string} options.className - One or more classes for the element seperated by spaces\r\n* @returns {HTMLElement} The node built as specified\r\n*/\r\nvar buildNode = function(type, options){\r\n\tvar node = document.createElement(type);\r\n\tfor (var option in options) if (options.hasOwnProperty(option)) {\r\n\t\tif (option === \"className\" || option === \"id\"){\r\n\t\t\tnode[option] = options[option];\r\n\t\t}\r\n\t\telse {\r\n\t\t\tnode.setAttribute(option, options[option]);\r\n\t\t}\r\n\t}\r\n\treturn node;\r\n};\r\n\r\nmodule.exports = buildNode;","var buildNode = require('./buildnode.js');\r\n\r\n// tag, id, className, other, childNodes\r\n\r\n/* IWindow:\r\n\r\n{\r\n\tid: \"WKSS-edit\",\r\n\tclassName: \"WKSS\",\r\n\tchildNodes:[{\r\n\t\ttag: 'form',\r\n\t\tid: \"WKSS-editForm\",\r\n\t\tchildNodes:[{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"WKSS-editCloseBtn\",\r\n\t\t\tclassName: \"WKSS-close\"\r\n\t\t\tchildNodes:[{\r\n\t\t\t\ttag: 'i',\r\n\t\t\t\tclassName: \"icon-remove\"\r\n\t\t\t}]\r\n\t\t},{\r\n\t\t\ttag: 'h1',\r\n\t\t\tchildNodes:[\r\n\t\t\t\t\"Edit your Vocab\"                 <--- string types for text node\r\n\t\t\t]\r\n\t\t},{\r\n\t\t\ttag: 'select',\r\n\t\t\tid: \"editWindow\",\r\n\t\t\tother: {size: \"8\"}\t\t\t\t\t<--- 'other' avoids clashes with HTMLElement attributes just in case\r\n\t\t},{\r\n\t\t\ttag: 'input', \r\n\t\t\tother:{\r\n\t\t\t\ttype: \"text\",\r\n\t\t\t\tname: \"\",\r\n\t\t\t\tsize: \"40\",\r\n\t\t\t\tplaceholder: \"Select vocab, click edit, change and save!\"\r\n\t\t\t},\r\n\t\t\tid: \"editItem\"\r\n\t\t},{\r\n\t\t\ttag: 'p', \r\n\t\t\tid: \"editStatus\"\r\n\t\t\tchildNodes:[\"Ready to edit...\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditEditBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Edit\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditSaveBtn\",\r\n\t\t\tother:{type: \"button\"},\r\n\t\t\tchildNodes:[\"Save\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteBtn\",\r\n\t\t\tother: {type: \"button\", title: \"Delete selected item\"},\r\n\t\t\tchildNodes:[\"Delete\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteAllBtn\",\r\n\t\t\tother: {type: \"button\", title: \"本当にやるの？\"},\r\n\t\t\tchildNodes:[\"Delete All\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"ResetLevelsBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Reset levels\"]\r\n\t\t}]\r\n\t}]\r\n}\r\n\r\n*/\r\n\r\nvar struct = {\r\n\tid: \"WKSS-edit\",\r\n\tclassName: \"WKSS\",\r\n\tchildNodes:[{\r\n\t\ttag: 'form',\r\n\t\tid: \"WKSS-editForm\",\r\n\t\tchildNodes:[{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"WKSS-editCloseBtn\",\r\n\t\t\tclassName: \"WKSS-close\",\r\n\t\t\tchildNodes:[{\r\n\t\t\t\ttag: 'i',\r\n\t\t\t\tclassName: \"icon-remove\"\r\n\t\t\t}]\r\n\t\t},{\r\n\t\t\ttag: 'h1',\r\n\t\t\tchildNodes:[\"Edit your Vocab\"]\r\n\t\t},{\r\n\t\t\ttag: 'select',\r\n\t\t\tid: \"editWindow\",\r\n\t\t\tother: {size: \"8\"}\r\n\t\t},{\r\n\t\t\ttag: 'input', \r\n\t\t\tother:{\r\n\t\t\t\ttype: \"text\",\r\n\t\t\t\tname: \"\",\r\n\t\t\t\tsize: \"40\",\r\n\t\t\t\tplaceholder: \"Select vocab, click edit, change and save!\"\r\n\t\t\t},\r\n\t\t\tid: \"editItem\"\r\n\t\t},{\r\n\t\t\ttag: 'p', \r\n\t\t\tid: \"editStatus\",\r\n\t\t\tchildNodes:[\"Ready to edit...\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditEditBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Edit\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditSaveBtn\",\r\n\t\t\tother:{type: \"button\"},\r\n\t\t\tchildNodes:[\"Save\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteBtn\",\r\n\t\t\tother: {type: \"button\", title: \"Delete selected item\"},\r\n\t\t\tchildNodes:[\"Delete\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteAllBtn\",\r\n\t\t\tother: {type: \"button\", title: \"本当にやるの？\"},\r\n\t\t\tchildNodes:[\"Delete All\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"ResetLevelsBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Reset levels\"]\r\n\t\t}]\r\n\t}]\r\n};\r\n\r\n// 'this' context is node to attach to\r\nvar attachChildNode = function(childNode){\r\n\tvar el;\r\n\tif (\"string\" === typeof childNode){ //TextNode\r\n\t\t el = document.createTextNode(childNode);\r\n\t}\r\n\telse{\r\n\t\tel = buildNode(childNode.tag, {id: childNode.id, className: childNode.className});\r\n\t\tfor (var attr in childNode.other){\r\n\t\t\tel.setAttribute(attr, childNode.other[attr]);\r\n\t\t}\r\n\t\tif (childNode.childNodes){\r\n\t\t\tchildNode.childNodes.forEach(attachChildNode, el);\r\n\t\t}\r\n\t}\r\n\tthis.appendChild(el);\r\n};\r\n\r\n/** Takes a JSON object with the structure of the window to create and builds a DIVElement from that\r\n* @param {IWindow} windowStructure\r\n* @returns {DIVElement} The specified window.\r\n*/\r\nvar buildWindow = function(windowStructure) {\r\n\t\r\n\tvar resultWindow = buildNode('div', {id: windowStructure.id, className: windowStructure.className});\r\n\tfor (var attr in windowStructure.other){\r\n\t\tresultWindow.setAttribute(attr, windowStructure.other[attr]);\r\n\t}\r\n\tif (windowStructure.childNodes){\r\n\t\twindowStructure.childNodes.forEach(attachChildNode, resultWindow);\r\n\t}\r\n\treturn resultWindow;\r\n};\r\n\r\n/*\t\r\n{\t\r\n\t\tvar editForm = buildNode('form', {id: \"WKSS-editForm\"});\r\n\teditWindow.appendChild(editForm);\r\n\t\t\tvar editCloseButton = buildNode('button', {id: \"WKSS-editCloseBtn\", className: \"WKSS-close\"});\r\n\t\teditForm.appendChild(editCloseButton);\r\n\t\r\n\t\t\teditCloseButton.appendChild(buildNode('i', {className: \"icon-remove\"}));\r\n\t\t\tvar h1Element = buildNode('h1');\r\n\t\teditForm.appendChild(h1Element);\r\n\t\t\th1Element.appendChild(document.createTextNode(\"Edit your Vocab\"));\r\n\t\t\tvar selectElement = buildNode('select', {id: \"editWindow\", size: \"8\"});\r\n\t\teditForm.appendChild(selectElement);\r\n\t\t\tvar editItemText = buildNode('input', {type: \"text\" id: \"editItem\" name: \"\" size: \"40\" placeholder: \"Select vocab, click edit, change and save!\"});\r\n\t\teditForm.appendChild(editItemText);//..\r\n\t\t\tvar editStatus = buildNode('p', {id: \"editStatus\"});\r\n\t\teditForm.appendChild(editStatus);\r\n\t\t\teditStatus.appendChild(document.createTextNode(\"Ready to edit..\"));\r\n\t\r\n\t\t\tvar editButton = buildNode('button', {id: \"EditEditBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editButton);\r\n\t\t\teditButton.appendChild(document.createTextNode(\"Edit\"));\r\n\t\t\tvar editSave = buildNode('button', {id: \"EditSaveBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editSave);\r\n\t\t\teditSave.appendChild(document.createTextNode(\"Save\"));\r\n\t\t\tvar editDelete = buildNode('button', {id: \"EditDeleteBtn\", type: \"button\", title: \"Delete selected item\"});\r\n\t\teditForm.appendChild(editDelete);\r\n\t\t\teditDelete.appendChild(document.createTextNode(\"Delete\"));\r\n\t\t\tvar editDeleteAll = buildNode('button', {id: \"EditDeleteAllBtn\", type: \"button\", title: \"本当にやるの？\"});\r\n\t\teditForm.appendChild(editDeleteAll);\r\n\t\t\teditDeleteAll.appendChild(document.createTextNode(\"Delete All\"));\r\n\t\t\tvar editResetLevels = buildNode('button', {id: \"ResetLevelsBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editResetLevels);\r\n\t\t\teditResetLevels.appendChild(document.createTextNode(\"Reset levels\"));\r\n\t\t\r\n\treturn editWindow;\r\n};*/\r\nmodule.exports = buildWindow;","module.exports = function(){\r\n\r\n\tvar kanji = $(\"#addKanji\").val().toLowerCase();\r\n\tvar reading = $(\"#addReading\").val().toLowerCase().split(/[,、]+\\s*/); //split at , or 、followed by 0 or any number of spaces\r\n\tvar meaning = $(\"#addMeaning\").val().toLowerCase().split(/[,、]+\\s*/);\r\n\tvar success = false; //initalise values\r\n\tvar meanlen = 0;\r\n\r\n\tvar i = meaning.length;\r\n\twhile (i--){\r\n\t\tmeanlen += meaning[i].length;\r\n\t}\r\n\r\n\t//input is invalid: prompt user for valid input\r\n\tvar item = {};\r\n\tif (kanji.length === 0 || meanlen === 0) {\r\n\t\t$(\"#addStatus\").text(\"One or more required fields are empty!\");\r\n\t\tif (kanji.length === 0) {\r\n\t\t\t$(\"#addKanji\").addClass(\"error\");\r\n\t\t} else {\r\n\t\t\t$(\"#addKanji\").removeClass(\"error\");\r\n\t\t}\r\n\t\tif (meanlen === 0) {\r\n\t\t\t$(\"#addMeaning\").addClass(\"error\");\r\n\t\t} else {\r\n\t\t\t$(\"#addMeaning\").removeClass(\"error\");\r\n\t\t}\r\n\t} else {\r\n\t\tif (debugging) {console.log(\"building item: \"+kanji);}\r\n\t\titem.kanji = kanji;\r\n\t\titem.reading = reading; //optional\r\n\t\titem.meaning = meaning;\r\n\r\n\t\tsuccess = true;\r\n\t\tif (debugging) {console.log(\"item is valid\");}\r\n\t}\r\n\r\n\t//on successful creation of item\r\n\tif (success) {\r\n\t\t//clear error layout to required fields\r\n\t\t$(\"#addKanji\").removeClass(\"error\");\r\n\t\t$(\"#addMeaning\").removeClass(\"error\");\r\n\r\n\r\n\r\n\t\t//if there are already user items, retrieve vocabList\r\n\t\t// var vocabList = [];\r\n\t\tvar vocabList = getFullList();\r\n\r\n\t\tif (debugging) {console.log(\"vocabList retrieved, length: \"+vocabList.length);}\r\n\t\t//check stored user items for duplicates ****************** to do: option for editing duplicate item with new input\r\n\t\tif(checkForDuplicates(vocabList,item)) {\r\n\t\t\t$(\"#addStatus\").text(\"Duplicate Item detected!\");\r\n\t\t\t$(\"#addKanji\").addClass(\"error\");\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tsetVocItem(item);\r\n\r\n\t\tif (debugging) {console.log(\"clear form\");}\r\n\t\t$(\"#addForm\")[0].reset();\r\n\r\n\t\t//--------------------------------------------------------------------------------------------------------\r\n\t\tif (item.manualLock === \"yes\" || item.manualLock === \"DB\" && lockDB){\r\n\t\t\t$(\"#addStatus\").html(\"<i class=\\\"icon-lock\\\"></i> Added locked item\");\r\n\t\t} else {\r\n\t\t\t$(\"#addStatus\").html(\"<i class=\\\"icon-unlock\\\"></i>Added successfully\");\r\n\t\t}\r\n\t\t//--------------------------------------------------------------------------------------------------------\r\n\t}\r\n};","module.exports = function(){\r\n\t// {\"level\":\"17\",\"meaning_explanation\":\"This word consists of kanji with hiragana attached. Because the hiragana ends with an [ja]う[/ja] sound, you know this word is a verb. The kanji itself means [kanji]flourish[/kanji] or [kanji]prosperity[/kanji], so the verb vocab versions of these would be [vocabulary]to flourish[/vocabulary] or [vocabulary]to prosper[/vocabulary].\",\"reading_explanation\":\"Since this word consists of a kanji with hiragana attached, you can bet that it will use the kun'yomi reading. You didn't learn that reading with this kanji, so here's a mnemonic to help you: What do you flourish at? You're an amazing [vocabulary]soccer[/vocabulary] ([ja]さか[/ja]) player who flourishes and prospers no matter where you go to play this wonderful (but not as good as baseball) sport.\",\"en\":\"To Flourish, To Prosper\",\"kana\":\"さかえる\",\"sentences\":[[\"中国には、覚せい剤の生産で栄えていた村がありました。\",\"There was a village in China flourishing on their production of stimulants. \"]],\"parts_of_speech_ids\":[\"4\",\"19\"],\"part_of_speech\":\"Intransitive Verb, Ichidan Verb\",\"audio\":\"2e194cbf194371cd478480d6ea67769da623e99a.mp3\",\"meaning_note\":null,\"reading_note\":null,\"related\":[{\"kan\":\"栄\",\"en\":\"Prosperity, Flourish\",\"slug\":\"栄\"}]}\r\n\r\n\r\n\tif (typeof $.mockjax === \"function\"){\r\n\t\t$.mockjax({\r\n\t\t\turl: /^\\/json\\/progress\\?vWKSS(.+)\\[\\]=(.+)&vWKSS.+\\[\\]=(.+)$/,\r\n\t\t\turlParams:[\"WKSSid\", \"MeaningWrong\", \"ReadingWrong\"],\r\n\t\t\tresponse: function(settings) {\r\n\t\t\t\t// do any required cleanup\r\n\t\t\t\tvar id = Number(settings.urlParams.WKSSid);\r\n\t\t\t\tvar Mw = Number(settings.urlParams.MeaningWrong);\r\n\t\t\t\tvar Rw = Number(settings.urlParams.ReadingWrong);\r\n\t\t\t\tvar UserVocab = localGet(\"User-Vocab\")||[];\r\n\r\n\t\t\t\tconsole.log(\"is this your card?\", UserVocab[id]);\r\n\t\t\t\tif (UserVocab[id].due < Date.now()){//double check that item was due for review\r\n\t\t\t\t\tif (Mw||Rw){\r\n\t\t\t\t\t\t//drop levels if wrong\r\n\r\n\t\t\t\t\t\t//Adapted from WaniKani's srs to authentically mimic level downs\r\n\t\t\t\t\t\tvar o = (Mw||0)+(Rw||0);\r\n\t\t\t\t\t\tvar t = UserVocab[id].level;\r\n\t\t\t\t\t\tvar r=t>=5?2*Math.round(o/2):1*Math.round(o/2);\r\n\t\t\t\t\t\tvar n=t-r<1?1:t-r;//don't stay on 'started'\r\n\r\n\t\t\t\t\t\tUserVocab[id].level = n;\r\n\t\t\t\t\t}else{\r\n\t\t\t\t\t\t//increase level if none wrong\r\n\t\t\t\t\t\tUserVocab[id].level++;\r\n\t\t\t\t\t}\r\n\t\t\t\t\t//Put UserVocab back in storage\r\n\t\t\t\t\tUserVocab[id].date = Date.now();\r\n\t\t\t\t\tUserVocab[id].due = Date.now() + srsintervals[UserVocab[id].level];\r\n\t\t\t\t\tlocalSet(\"User-Vocab\", UserVocab);\r\n\t\t\t\t\tconsole.log(UserVocab[id].due +\" > \"+ Date.now() + \" (\" + ms2str(UserVocab[id].due - Date.now())+\")\");\r\n\r\n\t\t\t\t}else{\r\n\t\t\t\t\tconsole.log(\"This item is not due for review yet, discarding results\");\r\n\t\t\t\t}\r\n\t\t\t\tthis.responseText = '{\"vWKSS'+id.toString()+'\":[\"'+Mw.toString()+'\",\"'+Rw.toString()+'\"]}';\r\n\r\n\t\t\t}\r\n\t\t});\r\n\r\n\t\t$.mockjax({\r\n\t\t\turl: /^\\/json\\/vocabulary\\/WKSS(.+)/,\r\n\t\t\turlParams:[\"WKSSid\"],\r\n\t\t\tresponse: function(settings) {\r\n\r\n\t\t\t\t// Investigate the `settings` to determine the response...\r\n\t\t\t\tvar id = settings.urlParams.WKSSid.toString();\r\n\t\t\t\tvar currentItem = $.jStorage.get(\"currentItem\");\r\n\t\t\t\tif (currentItem.id === \"WKSS\"+id){\r\n\t\t\t\t\tconsole.log(\"as expected\");\r\n\t\t\t\t}\r\n\t\t\t\tvar related = '[';\r\n\t\t\t\tfor (i = 0; i < currentItem.components.length; i++){\r\n\t\t\t\t\trelated += '{\"kan\":\"'+currentItem.components[i]+'\",\"en\":\"\",\"slug\":\"'+currentItem.components[i]+'\"}';\r\n\t\t\t\t\trelated += (i+1<currentItem.components.length)?',':'';\r\n\t\t\t\t}\r\n\t\t\t\trelated += ']';\r\n\r\n\t\t\t\tvar respText = JSON.stringify({\r\n\t\t\t\t\tlevel: \"U\",\r\n\t\t\t\t\tmeaning_explanation: \"This is user-defined item. Meaning explanations are not supported at this time. [id: \"+id+\"]\",\r\n\t\t\t\t\treading_explanation: \"This is user-defined item. Reading explanations are not supported at this time. [id: \"+id+\"]\",\r\n\t\t\t\t\ten: currentItem.en.join(\", \"),\r\n\t\t\t\t\tkana: currentItem.kana.join(\", \"),\r\n\t\t\t\t\tsentences:[],\r\n\t\t\t\t\tparts_of_speech_ids:[],\r\n\t\t\t\t\tpart_of_speech:[],\r\n\t\t\t\t\taudio:null,\r\n\t\t\t\t\tmeaning_note:null,\r\n\t\t\t\t\treading_note:null,\r\n\t\t\t\t\trelated:JSON.parse(related)\r\n\t\t\t\t});\r\n\t\t\t\tthis.responseText = respText;\r\n\t\t\t},\r\n\t\t\tonAfterComplete: function() {\r\n\t\t\t\t// do any required cleanup\r\n\t\t\t\t$(\".user-synonyms\").remove();\r\n\t\t\t\t// keeping the hooks for Community Mnemonics\r\n\t\t\t\t$(\"#note-meaning, #note-reading\").html(\"\");\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n};\r\n//--------------End Insert Into WK Review Functions--------------\r\n\r\n","\r\nvar ImportUtil = {\r\n\tfileUpload: function(ev){\r\n\t\tvar csvHeader = true;        //first row contains stuff like \"Kanji/Vocab, Reading, Meaning\" etc\r\n\t\tvar tsvfile;          //tabs separate fields, commas seperate values? or false for vice versa\r\n\t\tvar CSVs = ev.target.files;\r\n\t\tvar name =CSVs[0].name;\r\n\t\tvar colsplit, vsplit;\r\n\t\tif (name.substr(name.lastIndexOf(\".\"),4)===\".csv\"){\r\n\t\t\ttsvfile = false;\r\n\t\t\tcolsplit = \",\";\r\n\t\t\tvsplit = \"\\t\";\r\n\t\t}else{\r\n\t\t\ttsvfile = true;\r\n\t\t\tcolsplit = \"\\t\";\r\n\t\t\tvsplit = \",\";\r\n\t\t}\r\n\r\n\t\tif (debugging) { console.log(\"tsvfile: \"); }\r\n\t\tif (debugging) { console.log(\"file uploaded: \"+CSVs[0].name); }\r\n\t\tvar reader = new FileReader();\r\n\t\treader.readAsText(CSVs[0]);\r\n\t\treader.onload = function(ev){\r\n\t\t\tvar csvString = ev.target.result;\r\n\t\t\tvar csvRow = csvString.split(\"\\n\");\r\n\t\t\t//default column rows\r\n\t\t\tvar k = 0;\r\n\t\t\tvar r = 1;\r\n\t\t\tvar m = 2;\r\n\r\n\t\t\tvar i = csvRow.length;\r\n\t\t\t//process header, changing k,r,m if necessary\r\n\t\t\tvar JSONimport = [];\r\n\t\t\twhile(i--){\r\n\t\t\t\tvar row = csvRow[i];\r\n\t\t\t\tif ((csvHeader === true && i === 0)||  //  Skip header\r\n\t\t\t\t\t(row === \"\") // Skip empty rows\r\n\t\t\t\t   ){\r\n\t\t\t\t\tif (debugging) { console.log(\"Skipping row #\"+i); }\r\n\r\n\t\t\t\t}else{\r\n\t\t\t\t\tconsole.log(row);\r\n\r\n\r\n\t\t\t\t\tvar elem = row.split(colsplit);\r\n\t\t\t\t\tvar item = {};\r\n\t\t\t\t\tvar c;\r\n\r\n\t\t\t\t\tif (elem[k]){\r\n\t\t\t\t\t\titem.kanji = elem[k].trim();\r\n\r\n\t\t\t\t\t\tif (elem[r]){\r\n\r\n\t\t\t\t\t\t\tif (elem[r].indexOf(vsplit)>-1){\r\n\t\t\t\t\t\t\t\t// eg 'reading 1[tab]reading 2[tab]reading 3'\r\n\r\n\t\t\t\t\t\t\t\titem.reading = elem[r].split(vsplit);\r\n\t\t\t\t\t\t\t}else{ //no tabs in string, single value\r\n\t\t\t\t\t\t\t\titem.reading=[elem[r]];\r\n\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t}else{\r\n\t\t\t\t\t\t\titem.reading=[\"\"];\r\n\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\tif (elem[m]){\r\n\r\n\t\t\t\t\t\t\tif (elem[m].indexOf(vsplit)>-1){\r\n\t\t\t\t\t\t\t\t// eg 'meaning 1[tab]meaning 2[tab]meaning 3'\r\n\r\n\t\t\t\t\t\t\t\titem.meaning = elem[m].split(\"\\t\");\r\n\t\t\t\t\t\t\t}else{ //no tabs in string, single value\r\n\t\t\t\t\t\t\t\titem.meaning=[elem[m]];\r\n\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t\tc = item.meaning.length;\r\n\r\n\t\t\t\t\t\t\twhile(c--){\r\n\t\t\t\t\t\t\t\tconsole.log(\"item.meaning[\"+c+\"]: \"+item.meaning[c]);\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t}else{//todo: provide overwrite option on forced meaning\r\n\t\t\t\t\t\t\titem.meaning=[\"\"];\r\n\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\tJSONimport.push(item);\r\n\t\t\t\t\t}else{ // corrupt row ('kanji' is mandatory (can be kana-only word), is not present on row, skip\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t\tvar JSONstring = JSON.stringify(JSONimport);\r\n\t\t\tconsole.log(JSONimport);\r\n\r\n\t\t\tif (JSONstring.length !== 0) {\r\n\t\t\t\ttry {\r\n\t\t\t\t\tvar add = JSON.parse(JSONstring.toLowerCase());\r\n\t\t\t\t\t/*//---------/-------------\r\n\t\t\t\t if (!checkAdd(add)) {\r\n\t\t\t\t $(\"#importStatus\").text(\"No valid input (duplicates?)!\");\r\n\t\t\t\t return;\r\n\t\t\t\t }\r\n\t\t\t\t //----------------------*/\r\n\r\n\t\t\t\t\tvar a = add.length;\r\n\t\t\t\t\twhile(a--){\r\n\t\t\t\t\t\tStorageUtil.setVocItem(add[a]);\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\t$(\"#importStatus\").text(\"Import successful!\");\r\n\r\n\t\t\t\t\t$(\"#importForm\")[0].reset();\r\n\t\t\t\t\t$(\"#importArea\").text(\"\");\r\n\r\n\t\t\t\t}\r\n\t\t\t\tcatch (e) {\r\n\t\t\t\t\t$(\"#importStatus\").text(\"Parsing Error!\");\r\n\t\t\t\t\tconsole.log(e);\r\n\t\t\t\t}\r\n\r\n\t\t\t}\r\n\t\t\telse {\r\n\t\t\t\t$(\"#importStatus\").text(\"Nothing to import :( Please paste your stuff first\");\r\n\t\t\t}\r\n\r\n\t\t};\r\n\t}\r\n};\r\n\r\nmodule.exports = ImportUtil;","var ReviewUtil = {\r\n\t/** Takes an array of strings and returns the portions before left brackets '(' but only for strings that have them. It is used to add synonym values to the answer list.\r\n\t* @param {Array.<string>} solution - An array of acceptable answers for the Task\r\n\t* @returns {Array.<string>} Parts of the solution left of left bracket in strings where it exists\r\n\t* @example unbracketSolution([\"newspaper\", \"reading Stick (this text won't get through)\"]) // [\"reading stick\"]\r\n\t*/\r\n\tunbracketSolution: function(solution){\r\n        var unbracketed = solution.filter(function(ans){\r\n            var openBracket = ans.indexOf(\"(\");\r\n            if (openBracket !== -1){ //string contains a bracket\r\n                return ans.toLowerCase().substr(0, openBracket).trim();\r\n            } \r\n        }, this);\r\n        return unbracketed;\r\n    },\r\n\r\n\tinputCorrect: function() {\r\n        var input = $(\"#rev-input\").val().toLowerCase().trim();\r\n        var solution = document.getElementById('rev-solution').innerHTML.split(/[,、]+\\s*/);\r\n        var correctCharCount = 0;\r\n        var returnvalue = false;\r\n\r\n        console.log(\"Input: \" + input);\r\n\r\n        var append = this.unbracketSolution(solution);\r\n        solution = solution.concat(append);\r\n        var i = solution.length;\r\n        while(i--){\r\n\r\n            var threshold = 0;//how many characters can be wrong\r\n            if(document.getElementById('rev-type').innerHTML == \"Meaning\") {\r\n                threshold = Math.floor(solution[i].length / errorAllowance);\r\n            }\r\n\r\n            console.log(\"Checking \" + solution[i] + \" with threshold: \" + threshold);\r\n\r\n            var j;\r\n            var lengthDiff = Math.abs(input.length - solution[i].length);\r\n            if (lengthDiff > threshold){\r\n                returnvalue = returnvalue || false;\r\n                console.log(\"false at if branch \" + input.length + \" < \" + JSON.stringify(solution[i]));//.length );//- threshold));\r\n            } else { //difference in response length is within threshold\r\n                j = input.length;\r\n                while (j--) {\r\n                    if (input[j] == solution[i][j]) {\r\n                        console.log (input[j] +\" == \"+ solution[i][j]);\r\n                        correctCharCount++;\r\n                    }\r\n                }\r\n                if (correctCharCount >= solution[i].length - threshold){\r\n                    returnvalue = true;\r\n                }\r\n            }\r\n\r\n        }\r\n\r\n        console.log(\"Returning \" + returnvalue);\r\n        return returnvalue;\r\n    },\r\n\r\n\tsubmitResponse: function (e) {\r\n\t\t//functions:\r\n\t\t//  inputCorrect()\r\n\r\n\t\t//check if key press was 'enter' (keyCode 13) on the way up\r\n\t\t//and keystate true (answer being submitted)\r\n\t\t//and cursor is focused in reviewfield\r\n\t\tif (e.keyCode == 13 && submit === true) {\r\n\t\t\tvar input = $(\"#rev-input\").val();\r\n\t\t\tvar reviewList = sessionGet('User-Review')||[];\r\n\t\t\tvar rnd = sessionStorage.getItem('WKSS-rnd')||0;\r\n\r\n\t\t\tvar item = sessionGet('WKSS-item');\r\n\r\n\t\t\t//-- starting implementation of forgiveness protocol\r\n\r\n\t\t\titem.forgive = [];//\"ゆるす\"]; //placeholder (許す to forgive)\r\n\r\n\t\t\tif (item === null){\r\n\t\t\t\talert(\"Item Null??\");\r\n\t\t\t\treviewList.splice(rnd, 1);\r\n\t\t\t}\r\n\t\t\telse{\r\n\t\t\t\t//handle grading and storing solution\r\n\r\n\t\t\t\t//check for input, do nothing if none\r\n\t\t\t\tif(input.length === 0){\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\r\n\t\t\t\t//disable input after submission\r\n\t\t\t\t//document.getElementById('rev-input').disabled = true;\r\n\r\n\r\n\t\t\t\t//was the input correct?\r\n\t\t\t\tvar correct = this.inputCorrect();\r\n\r\n\t\t\t\t//was the input forgiven?\r\n\t\t\t\tvar forgiven = (item.forgive.indexOf(input) !== -1);\r\n\r\n\t\t\t\tif (correct) {\r\n\t\t\t\t\t//highlight in (default) green\r\n\t\t\t\t\t$(\"#rev-input\").addClass(\"correct\");\r\n\t\t\t\t\t//show answer\r\n\t\t\t\t\t$(\"#rev-solution\").addClass(\"info\");\r\n\t\t\t\t} else if (forgiven){\r\n\t\t\t\t\t$(\"#rev-input\").addClass(\"caution\");\r\n\t\t\t\t} else {\r\n\t\t\t\t\t//highight in red\r\n\t\t\t\t\t$(\"#rev-input\").addClass(\"error\");\r\n\t\t\t\t\t//show answer\r\n\t\t\t\t\t$(\"#rev-solution\").addClass(\"info\");\r\n\t\t\t\t}\r\n\r\n\t\t\t\t//remove from sessionList if correct\r\n\t\t\t\tif (correct) {\r\n\t\t\t\t\tconsole.log(\"correct answer\");\r\n\t\t\t\t\tif (reviewList !== null){\r\n\t\t\t\t\t\tvar oldlen = reviewList.length;\r\n\r\n\t\t\t\t\t\treviewList.splice(rnd, 1);\r\n\t\t\t\t\t\tconsole.log(\"sessionList.length: \"+ oldlen +\" -> \"+reviewList.length);\r\n\r\n\t\t\t\t\t\t//replace shorter (by one) sessionList to session\r\n\t\t\t\t\t\tif (reviewList.length !== 0) {\r\n\t\t\t\t\t\t\tconsole.log(\"sessionList.length: \"+ reviewList.length);\r\n\t\t\t\t\t\t\tsessionSet('User-Review', JSON.stringify(reviewList));\r\n\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t//reveiw over, delete sessionlist from session\r\n\t\t\t\t\t\t\tsessionStorage.removeItem('User-Review');\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}else{\r\n\t\t\t\t\t\tconsole.error(\"Error: no review session found\");\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t\telse{\r\n\t\t\t\t\t//   if(forgiven){\r\n\t\t\t\t\t//     console.log(input +\" has been forgiven. \"+item.type);\r\n\t\t\t\t\t//   return;\r\n\t\t\t\t\t// }\r\n\t\t\t\t\tconsole.log(\"wrong answer\");\r\n\t\t\t\t}\r\n\r\n\t\t\t\titem = markAnswer(item);\r\n\r\n\t\t\t\tsessionSet(item.index, item);\r\n\r\n\r\n\t\t\t\tvar list = JSON.parse(sessionStorage.getItem(\"User-Stats\"))||[];\r\n\t\t\t\tvar found = false;\r\n\r\n\t\t\t\tif (list){\r\n\t\t\t\t\tvar i = list.length;\r\n\t\t\t\t\twhile(i--){\r\n\t\t\t\t\t\tif (list[i].index == item.index) {\r\n\t\t\t\t\t\t\tlist[i] = item;\t\t\t\t\t\t\t\t//replace item if it exists\r\n\t\t\t\t\t\t\tfound = true;\r\n\t\t\t\t\t\t\tbreak;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif(!found){\r\n\t\t\t\t\t\tlist = saveToSortedList(list,item);\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t} else {\r\n\t\t\t\t\tlist = [item];\r\n\t\t\t\t}\r\n\r\n\t\t\t\tsessionSet(\"User-Stats\", JSON.stringify(list));\r\n\t\t\t\t//playAudio();\r\n\r\n\t\t\t\t//answer submitted, next 'enter' proceeds with script\r\n\t\t\t\tsubmit = false;\r\n\t\t\t}//null garbage collection\r\n\t\t}\r\n\t\telse if (e.keyCode == 13 && submit === false) {\r\n\t\t\tconsole.log(\"keystat = \" + submit);\r\n\r\n\t\t\t//there are still more reviews in session?\r\n\t\t\tif (sessionStorage.getItem('User-Review')) {\r\n\t\t\t\t// console.log(\"found a 'User-Review': \" + sessionStorage.getItem('User-Review'));\r\n\r\n\t\t\t\tsetTimeout(function () {\r\n\t\t\t\t\tconsole.log(\"refreshing reviewList from storage\");\r\n\t\t\t\t\tvar reviewList = JSON.parse(sessionStorage.getItem('User-Review'));\r\n\r\n\t\t\t\t\t//cue up first remaining review\r\n\t\t\t\t\tnextReview(reviewList);\r\n\t\t\t\t\tconsole.log(\"checking for empty reviewList\");\r\n\t\t\t\t\tif (reviewList.length === 0){\r\n\r\n\t\t\t\t\t\tconsole.log(\"session over. reviewList: \"+JSON.stringify(reviewList));\r\n\t\t\t\t\t\tsessionStorage.removeItem(\"User-Review\");\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\t//         document.getElementById('rev-input').disabled = true;\r\n\t\t\t\t\t$(\"#rev-solution\").removeClass(\"info\");\r\n\t\t\t\t\t$(\"#selfstudy\").hide().fadeIn('fast');\r\n\r\n\t\t\t\t}, 1);\r\n\t\t\t}\r\n\t\t\telse {\r\n\t\t\t\t// no review stored in session, review is over\r\n\t\t\t\tsetTimeout(function () {\r\n\r\n\t\t\t\t\t$(\"#selfstudy\").hide();\r\n\t\t\t\t\t//document.getElementById('rev-input').disabled = false;\r\n\t\t\t\t\t$(\"#rev-solution\").removeClass(\"info\");\r\n\t\t\t\t\tconsole.log(\"showResults\");\r\n\t\t\t\t\tshowResults();\r\n\t\t\t\t\t$(\"#resultwindow\").show();\r\n\t\t\t\t\tconsole.log(\"showResults completed\");\r\n\r\n\t\t\t\t\t//*/  //clear session\r\n\t\t\t\t\tsessionStorage.clear();\r\n\t\t\t\t\treviewActive = false;\r\n\r\n\r\n\t\t\t\t}, 1);\r\n\t\t\t}\r\n\t\t\tsubmit = true;\r\n\r\n\t\t}\r\n\t}\r\n};\r\n\r\nmodule.exports = ReviewUtil;","\r\nvar StorageUtil = {\r\n\t/** Initialise User-Vocab\r\n\t*/\r\n\tinitStorage: function(){\r\n\t\tif (!this.localGet(\"User-Vocab\")){\r\n\t\t\tthis.localSet(\"User-Vocab\", []);\r\n\t\t}\r\n\t},\r\n\tparseString: function(strObj){\r\n        //avoids duplication of code for sesssionGet and localGet\r\n        var obj;\r\n        try {\r\n            obj = JSON.parse(strObj);\r\n            console.log(\"Variable is of type \" + typeof obj);\r\n        }\r\n\t\tcatch(e){\r\n            if (e.name === \"SyntaxError\"){\r\n                console.log(strObj + \" is an ordinary string that cannot be parsed.\");\r\n                obj = strObj;\r\n            }\r\n\t\t\telse{\r\n                console.error(\"Could not parse \" + strObj + \". Error: \", e);\r\n            }\r\n        }\r\n        return obj;\r\n    },\r\n\t/**\r\n\t*/\r\n\tlocalGet: function(strName){\r\n        var strObj = localStorage.getItem(strName);\r\n        return this.parseString(strObj);\r\n    },\r\n\t/** Sets strings and objects into browser storage\r\n\t* @requires localStorage\r\n\t* @requires JSON\r\n\t*/\r\n\tlocalSet: function(strName, obj){\r\n        localStorage.setItem(strName, typeof obj === \"string\"? obj : JSON.stringify(obj));\r\n    },\r\n\t/**\r\n\t*/\r\n\tsessionGet: function(strName){\r\n        var strObj = sessionStorage.getItem(strName);\r\n        return this.parseString(strObj);\r\n    },\r\n\t/** Sets strings and objects into browser session storage\r\n\t* @requires localStorage\r\n\t* @requires JSON\r\n\t*/\r\n\tsessionSet: function(strName, obj){\r\n        sessionStorage.setItem(strName, typeof obj === \"string\"? obj : JSON.stringify(obj));\r\n    },\r\n\t/**\r\n\t*/\r\n\tgetVocList: function(){\r\n        var vocList = JSON.parse(localStorage.getItem('User-Vocab'))||[];\r\n        if (vocList){\r\n            var v=vocList.length;\r\n            while(v--){\r\n                vocList[v].i = v; //set index for item (->out)\r\n            }\r\n        }\r\n        return vocList;\r\n    },\r\n\tsetVocList: function(vocList){\r\n\t\tthis.localSet('User-Vocab', vocList);\r\n\t},\r\n\t/**\r\n\t*/\r\n\tsetVocItem: function(item){\r\n        //Assumption: item comes only with kanji, reading and meaning\r\n        item.level = 0;\r\n        item.date = Date.now();\r\n        item.manualLock = \"\";\r\n        item = setLocks(item);\r\n\t\t //0.1.9 adding in 'due' property to make review building simpler\r\n        item.due = item.date + srsObject[item.level].duration;\r\n\r\n        var vocList = localGet('User-Vocab')||[];\r\n\r\n\t\tvar found = vocList.find(function(task){\r\n            return task.kanji === item.kanji;\r\n        }, this);\r\n\t\t//add meaning and reading to existing item\r\n\t\t//        vocList[v].meaning = item.meaning;\r\n\t\t//      vocList[v].reading = item.reading;\r\n        if (!found) {\r\n            //provide index for faster searches\r\n            console.log(item.kanji +\" not found in vocablist, adding now\");\r\n            item.i = vocList.length;\r\n            vocList.push(item);\r\n\r\n            localSet('User-Vocab',vocList);\r\n        }\r\n    }\r\n};\r\n\r\nmodule.exports = StorageUtil;","/*  This is the original code that I am breaking into bite size bits */\r\n//NEED TO MAKE SURE BROWSERIFY PUTS THIS ON THE TOP\r\n\r\n \r\n /** Describes any object that can be reviewed or learned, includes IRadical, IKanji, and IVocabulary\r\n * @typedef {Object} Task\r\n * @property {boolean|string} locked - locked\r\n * @property {boolean|string} manualLock - manualLock\r\n */\r\n \r\nvar StorageUtil = require('./storageutil.js');\r\nvar ImportUtil = require('./importutil.js');\r\nvar WanikaniUtil = require('./wanikaniutil.js');\r\nvar ReviewUtil = require('./reviewutil.js');\r\n\r\nfunction main(){\r\n    \"use strict\";\r\n\r\n    $(\"head\").prepend(\"<script src='https://cdn.jsdelivr.net/jquery.mockjax/1.6.1/jquery.mockjax.js'></script>\");\r\n\r\n    var APIkey = \"YOUR_API_HERE\";\r\n    var locksOn = true; //Disable vocab locks (unlocked items persist until deleted)\r\n    var lockDB = true; //Set to false to unlock Kanji is not available on WaniKani (ie. not returned by API)\r\n    var reverse = true; //Include English to ひらがな reading reviews\r\n    var debugging = true;\r\n    var asWK = true; //Push user reviews into the main WK review queue\r\n\r\n    // shut up JSHint\r\n    /* jshint multistr: true , jquery: true, expr: true, indent:2 */\r\n    /* global window, wanakana, XDomainRequest */\r\n\r\n    /** Debugging\r\n\t */\r\n\tconsole.log = debugging ? function (msg) {\r\n\t\tif (typeof msg === 'string') {\r\n\t\t\twindow.console.log(\"WKSS: \" + msg);\r\n\t\t}\r\n\t\telse {\r\n\t\t\twindow.console.log(\"WKSS: \", msg);\r\n\t\t}\r\n\t} : function () {\r\n\t};\r\n\t\r\n    $(\"head\").prepend('<script src=\"https://rawgit.com/WaniKani/WanaKana/master/lib/wanakana.js\" type=\"text/javascript\"></script>');\r\n    \r\n    var localSet = function(strName, obj){\r\n        debugging&&console.log(strName + \" is of type \" + typeof obj);\r\n        if (typeof obj === \"object\")\r\n            obj=JSON.stringify(obj);\r\n        localStorage.setItem(strName, obj);\r\n    };\r\n\r\n\t//track versions & datatypes\r\n\tlocalSet(\"WKSSdata\", {\r\n        v: \"0.1.13\",\r\n        propertyType: {\r\n\t\t\tmeaning: \"array\", reading: \"array\", kanji: \"string\", i:\"number\", components: \"array\", date: \"number\", due: \"number\", locked: \"string\", manualLock: \"string\"\r\n\t\t},\r\n        propertyDesc: {\r\n\t\t\tmeaning: \"list of meanings\", reading: \"list of readings\", kanji: \"item prompt\", i:\"item index\", components: \"kanji found in word\", date: \"timestamp of new level\", due: \"timestamp of item's next review\", locked: \"indicator of whether components are eligible\", manualLock: \"latch for 'locked' so failing components don't re-lock the item\"\r\n\t\t}\r\n    });\r\n\r\n\r\n    /** Settings and constants\r\n\t */\r\n    var errorAllowance = 4; //every x letters, you can make one mistake when entering the meaning\r\n\r\n    //srs 4h, 8h, 24h, 3d (guru), 1w, 2w (master), 1m (enlightened), 4m (burned)\r\n    \r\n    var hrs = 60*60*1000;\r\n    var days = 24*hrs;\r\n    var weeks = 7*days;\r\n\tvar srsObject = [\r\n\t\t{level: 0, rank: \"Started\",\t\tduration: 0}, \r\n\t\t{level: 1, rank: \"Apprentice\",\tduration: 4*hrs},\r\n\t\t{level: 2, rank: \"Apprentice\",\tduration: 8*hrs},\r\n\t\t{level: 3, rank: \"Apprentice\",\tduration: 1*days},\r\n\t\t{level: 4, rank: \"Apprentice\",\tduration: 3*days},\r\n\t\t{level: 5, rank: \"Guru\",\t\tduration: 1*weeks},\r\n\t\t{level: 6, rank: \"Guru\",\t\tduration: 2*weeks},\r\n\t\t{level: 7, rank: \"Master\",\t\tduration: 730*hrs},\r\n\t\t{level: 8, rank: \"Enlightened\",\tduration: 2922*hrs},\r\n\t\t{level: 9, rank: \"Burned\"}\r\n\t];\r\n\r\n\r\n\r\n\tvar localGet = function(strName){\r\n        var strObj = localStorage.getItem(strName);\r\n        return parseString(strObj);\r\n    };\r\n    \r\n\t// Initialise User-Vocab\r\n\tStorageUtil.initStorage();\r\n\t\r\n\t//GM_addStyle shim for compatibility with greasemonkey\r\n    var gM_addStyle = function(CssString){\r\n        //get DOM head\r\n        var head = document.getElementsByTagName('head')[0];\r\n        if (head) {\r\n            //build style tag\r\n            var style = document.createElement('style');\r\n            style.setAttribute('type', 'text/css');\r\n            style.textContent = CssString;\r\n            //insert DOM style into head\r\n            head.appendChild(style);\r\n        }\r\n    };\r\n\r\n    /**  JQuery fixes\r\n\t */\r\n    $(\"[placeholder]\").focus(function () {\r\n        var input = $(this);\r\n        if (input.val() == input.attr(\"placeholder\")) {\r\n            input.val(\"''\");\r\n            input.removeClass(\"'placeholder'\");\r\n        }\r\n    }).blur(function () {\r\n        var input = $(this);\r\n        if (input.val() == \"''\" || input.val() == input.attr(\"placeholder\")) {\r\n            input.addClass(\"placeholder\");\r\n            input.val(input.attr(\"placeholder\"));\r\n        }\r\n    }).blur();\r\n\r\n    $(\"[placeholder]\").parents(\"form\").submit(function () {\r\n        $(this).find(\"[placeholder]\").each(function () {\r\n            var input = $(this);\r\n            if (input.val() == input.attr(\"placeholder\")) {\r\n                input.val(\"\");\r\n            }\r\n        });\r\n    });\r\n\r\n\t/** Handle the users API key.\r\n\t* @param {string} APIkey - the users API key to set. If given \"YOUR_API_HERE\", it will return the key in browser storage.\r\n\t* @returns {string} the users API key as supplied and stored, or in the case of \"YOUR_API_HERE\" being passed, the stored key.\r\n\t*/\r\n    var getSetApi = function(APIkey){\r\n        var storedAPI = localStorage.getItem('WaniKani-API');\r\n        if (APIkey === \"YOUR_API_HERE\"){\r\n            if (storedAPI !== null){\r\n                APIkey = storedAPI;\r\n            }\r\n        }\r\n\t\telse{\r\n            //API has been set in code.\r\n            if (storedAPI !== APIkey){\r\n                localSet('WaniKani-API', APIkey);//overwrite with new API\r\n            }\r\n        }\r\n        return APIkey;\r\n    };\r\n    APIkey = getSetApi(APIkey);\r\n\r\n    //--------------Start Insert Into WK Review Functions--------------\r\n\r\n\t/** Messing around with vanilla WaniKani review variables\r\n\t*/\r\n    var joinReviews = function(WKItems){\r\n        console.log(\"joining reviews\");\r\n        $.jStorage.stopListening(\"reviewQueue\", joinReviews);\r\n        var WKreview = $.jStorage.get(\"reviewQueue\")||[];\r\n        var WKcombined = WKreview.concat(WKItems);\r\n        $.jStorage.set(\"reviewQueue\", WKcombined);\r\n    };\r\n\r\n    var WKItems = [];\r\n    console.groupCollapsed(\"Loading Items\");\r\n\t\r\n\tvar wKSS_to_WK = function(WKSSItem){\r\n        var WKItem = {};\r\n        //    WKItem.aud = \"\";\r\n        WKItem.en = WKSSItem.meaning.map(function(s) {\r\n\t\t\t //trim whitespace and capitalize words\r\n\t\t\t return s.trim().replace(/\\b\\w/g , function(m){\r\n\t\t\t\treturn m.toUpperCase();\r\n\t\t\t});\r\n\t\t});\r\n        WKItem.id = \"WKSS\" + WKSSItem.i;\r\n        WKItem.kana = WKSSItem.reading;\r\n        WKItem.srs = WKSSItem.level+1;//WK starts levels from 1, WKSS starts them from 0\r\n        WKItem.voc = WKSSItem.kanji;\r\n        WKItem.components = WKSSItem.components;\r\n\r\n        WKItem.syn = [];\r\n        //Add synonyms of strings without bracketed info to get around checking the full string including brackets\r\n        WKSSItem.meaning.forEach(function(meaning){\r\n            var openBracket = meaning.indexOf(\"(\");\r\n            if (openBracket !== -1 && meaning.indexOf(\")\") !== -1){\r\n                WKItem.syn.push(meaning.substr(0, openBracket).trim().replace(/\\b\\w/g , function(m){ return m.toUpperCase();}));\r\n            }\r\n        }, this);\r\n\r\n        return WKItem;\r\n    };\r\n\r\n\tvar loadTasks = function(userVocab, i, userVocabs){\r\n        var dueNow = (userVocab.locked === \"no\" && userVocab.level < 9 && Date.now() > userVocab.due);\r\n\r\n        if (dueNow){\r\n            if (userVocab.kanji.length * userVocab.meaning[0].length * userVocab.reading[0].length){\r\n                //Sorry, we need all three to add to WK review, no kana only without readings etc.\r\n                debugging&&console.log(\"item:\" + userVocab.kanji + \", \" + userVocab.locked +\" === \\\"no\\\" && \" + userVocab.level + \" < 9 && \" + Date.now() + \" > \" + userVocab.due);\r\n                debugging&&console.log(dueNow);\r\n                WKItems.push(wKSS_to_WK(userVocab));\r\n            }else{\r\n                debugging&&console.log(\"Item \" + userVocab.kanji + \" could not be added, it is missing one or more of the essential fields for a WK vocabulary review\");\r\n            }\r\n        }\r\n    };\r\n\t\r\n    var userVocabs = StorageUtil.getVocList();\r\n    userVocabs.forEach(loadTasks);//, this);\r\n    console.groupEnd();\r\n\t\r\n    //where the magic happens\r\n    if (asWK){\r\n        $.jStorage.listenKeyChange(\"reviewQueue\", function(){joinReviews(WKItems);});\r\n    }\r\n\r\n    var sessionSet = function(strName, obj){\r\n        debugging&&console.log(strName + \" is of type \" + typeof obj);\r\n        if (typeof obj === \"object\")\r\n            obj=JSON.stringify(obj);\r\n        sessionStorage.setItem(strName, obj);\r\n    };\r\n\t\r\n    var sessionGet = function(strName){\r\n        var strObj = sessionStorage.getItem(strName);\r\n        return parseString(strObj);\r\n    };\r\n\r\n\tvar generateReviewList = function(reviewActive) {\r\n        //don't interfere with an active session\r\n        if (reviewActive){\r\n            document.getElementById('user-review').innerHTML = \"Review in Progress\";\r\n            return;\r\n        }\r\n\r\n        debugging&&console.log(\"generateReviewList()\");\r\n        // function generateReviewList() builds a review session and updates the html menu to show number waiting.\r\n        var numReviews = 0;\r\n        var soonest = Infinity;\r\n        var next;\r\n\r\n        var reviewList = [];\r\n\r\n        //check to see if there is vocab already in offline storage\r\n        if (localStorage.getItem('User-Vocab')) {\r\n            var vocabList = StorageUtil.getVocList();\r\n            debugging&&console.log(vocabList);\r\n            var now = Date.now();\r\n\r\n            //for each vocab in storage, get the amount of time vocab has lived\r\n            //var i = vocabList.length;\r\n            //while(i--){\r\n\t\t\tvocabList.forEach(function(task, i){\r\n                var due = task.date + srsObject[task.level].duration;\r\n\r\n                // if tem is unlocked and unburned\r\n                if (task.level < 9 &&\r\n                    (task.manualLock === \"no\" || task.manualLock === \"n\" ||\r\n                     task.manualLock ===\"DB\" && !lockDB )){\r\n                    // if it is past review time\r\n                    if(now >= due) {\r\n                        // count vocab up for review\r\n                        numReviews++;\r\n\r\n                        // add item-meaning object to reviewList\r\n                        // have made this optional for surname lists etc.\r\n                        if (task.meaning[0] !== \"\") {\r\n                            //Rev_Item object args: prompt, kanji, type, solution, index\r\n                            var revItem = new Rev_Item(task.kanji, task.kanji, \"Meaning\", task.meaning, i);\r\n                            reviewList.push(revItem);\r\n\t\t\t\t\t\t}\r\n\r\n                        // reading is optional, if there is a reading for the vocab, add its object.\r\n                        if (task.reading[0] !== \"\") {\r\n                            //Rev_Item object args: prompt, kanji, type, solution, index\r\n                            var revItem2 = new Rev_Item(task.kanji, task.kanji, \"Reading\", task.reading, i);\r\n                            reviewList.push(revItem2);\r\n                        }\r\n\r\n                        //if there is a meaning and reading, and reverse flag is true, test reading from english\r\n                        if (task.reading[0] !== \"\" && task.meaning[0] !== \"\" && reverse){\r\n                            //Rev_Item object args: prompt, kanji, type, solution, index\r\n                            var revItem3 = new Rev_Item(task.meaning.join(\", \"), task.kanji, \"Reverse\", task.reading, i);\r\n                            reviewList.push(revItem3);\r\n                        }\r\n\r\n                    }\r\n\t\t\t\t\telse{//unlocked/unburned but not time to review yet\r\n                        debugging&&console.log(\"setting soonest\");\r\n                        next = due - now;\r\n\t\t\t\t\t\tsoonest = Math.min(soonest, next);\r\n                    }\r\n\t\t\t\t}//end if item is up for review\r\n\t\t\t}, this);// end iterate through vocablist\r\n\t\t}// end if localStorage\r\n        if (reviewList.length !== 0){\r\n            //store reviewList in current session\r\n            sessionSet('User-Review', JSON.stringify(reviewList));\r\n            debugging&&console.log(reviewList);\r\n        }\r\n\t\telse{\r\n            debugging&&console.log(\"reviewList is empty: \"+JSON.stringify(reviewList));\r\n\t\t\tdocument.getElementById('user-review').innerHTML = soonest<Infinity? \"Next Review in \"+ms2str(soonest) : \"No Reviews Available\";\r\n\t\t}\r\n        var strReviews = numReviews.toString();\r\n\r\n        /* If you want to do the 42+ thing.\r\n\t\t if (numReviews > 42) {\r\n\t\t strReviews = \"42+\"; //hail the crabigator!\r\n\t\t }\r\n\t\t//*/\r\n\r\n        // return the number of reviews\r\n        debugging&&console.log(numReviews.toString() +\" reviews created\");\r\n        if (numReviews > 0){\r\n            var reviewString = (soonest !== void 0)? \"<br/>\\\r\nMore to come in \"+ms2str(soonest):\"\";\r\n            document.getElementById('user-review').innerHTML = \"Review (\" + strReviews + \")\" + reviewString;\r\n        }\r\n    };\r\n\r\n\t\r\n/*\r\n* populate reviews when menu button pressed\r\n*/\r\n\r\nwindow.generateReviewList = function() {\r\n\t//if menu is invisible, it is about to be visible\r\n\tif ( $(\"#WKSS_dropdown\").is(\":hidden\") ){\r\n\t\t//This is really the only time it needs to run\r\n\t\t//unless we want to start updating in realtime by keeping track of the soonest item\r\n\t\tgenerateReviewList();\r\n\t}\r\n};\r\n\r\n/*\r\n*  Add Item\r\n*/\r\n// event function to open \"add window\" and close any other window that might be open at the time.\r\nwindow.WKSS_add = function () {\r\n\t//show the add window\r\n\t$(\"#add\").show();\r\n\t//hide other windows\r\n\t$(\"#export\").hide();\r\n\t$(\"#import\").hide();\r\n\t$(\"#edit\").hide();\r\n\t$(\"#selfstudy\").hide();\r\n};\r\n\r\n//'add window' html text\r\nvar addHtml = '\\n\\\r\n<div id=\"add\" class=\"WKSS\">\\n\\\r\n<form id=\"addForm\">\\n\\\r\n<button id=\"AddCloseBtn\" class=\"wkss-close\" type=\"reset\"><i class=\"icon-remove\"></i></button>\\n\\\r\n<h1>Add a new Item</h1>\\n\\\r\n<input type=\"text\" id=\"addKanji\" placeholder=\"Enter 漢字, ひらがな or カタカナ\">\\n\\\r\n<input type=\"text\" id=\"addReading\" title=\"Leave empty to add vocabulary like する (to do)\" placeholder=\"Enter reading\">\\n\\\r\n<input type=\"text\" id=\"addMeaning\" placeholder=\"Enter meaning\">\\n\\\r\n\\n\\\r\n<p id=\"addStatus\">Ready to add..</p>\\n\\\r\n<button id=\"AddItemBtn\" type=\"button\">Add new Item</button>\\n\\\r\n</form>\\n\\\r\n</div>\\n';\r\n\r\n//add html to page source\r\n$(\"body\").append(addHtml);\r\n\r\n//hide add window (\"div add\" code that was just appended)\r\n$(\"#add\").hide();\r\n\r\n//function to fire on click event for \"Add new Item\"\r\n$(\"#AddItemBtn\").click(function () {\r\n\thandleAddClick();\r\n});\r\n\r\n$(\"#AddCloseBtn\").click(function () {\r\n\t$(\"#add\").hide();\r\n\t$(\"#addForm\")[0].reset();\r\n\t$(\"#addStatus\").text('Ready to add..');\r\n\t$(\"#addKanji\").removeClass(\"error\");\r\n\t$(\"#addMeaning\").removeClass(\"error\");\r\n});\r\n\r\n\r\n\r\n//---Function wrappers to facilitate use of one localstorage array\r\n//---Maintains data integrity between previously two (vocab and srs)\r\n\r\n\r\nfunction setSrsItem(srsitem,srsList){\r\n\tvar index = srsitem.i;\r\n\tdebugging&&console.log(\"setSrsItem: \");\r\n\r\n\tif(srsList){\r\n\t\tif(srsList[index].kanji===srsitem.kanji){// try search by index\r\n\r\n\t\t\tdebugging&&console.log(\"success: \"+srsitem.kanji+\" found at index \"+ index);\r\n\t\t\t//replace only the srs parts of the item\r\n\t\t\tsrsList[index].date = srsitem.date;\r\n\t\t\tsrsList[index].level = srsitem.level;\r\n\t\t\tsrsList[index].locked = srsitem.locked;\r\n\t\t\tsrsList[index].manualLock = srsitem.manualLock;\r\n\t\t}else{ //backup plan (cycle through list?)\r\n\t\t\tdebugging&&console.log(\"SRS Kanji not found in vocablist, needs work\");\r\n\r\n\t\t}\r\n\t\tdebugging&&console.log(\"item: \");\r\n\t\treturn srsList;\r\n\t}\r\n}\r\n\r\nfunction getSrsList(){\r\n\tvar srsList = StorageUtil.getVocList();\r\n\treturn srsList;\r\n}\r\n\r\n\r\nfunction getFullList(){\r\n\tvar fullList = JSON.parse(localStorage.getItem('User-Vocab'))||[];\r\n\tif(!fullList){\r\n\t\tfullList=[];\r\n\t}\r\n\treturn fullList;\r\n}\r\n\r\n//--------\r\n\r\n/*\r\n*  Edit Items\r\n*/\r\nwindow.WKSS_edit = function () {\r\n\tgenerateEditOptions();\r\n\t$(\"#edit\").show();\r\n\t//hide other windows\r\n\t$(\"#export\").hide();\r\n\t$(\"#import\").hide();\r\n\t$(\"#add\").hide();\r\n\t$(\"#selfstudy\").hide();\r\n};\r\n\r\n$(\"body\").append(\"                                                          \\\r\n<div id=\\\"edit\\\" class=\\\"WKSS\\\">                                               \\\r\n<form id=\\\"editForm\\\">                                                                    \\\r\n<button id=\\\"EditCloseBtn\\\" class=\\\"wkss-close\\\" type=\\\"button\\\"><i class=\\\"icon-remove\\\"></i></button>\\\r\n<h1>Edit your Vocab</h1>                                                \\\r\n<select id=\\\"editWindow\\\" size=\\\"8\\\"></select>\\\r\n<input type=\\\"text\\\" id=\\\"editItem\\\" name=\\\"\\\" size=\\\"40\\\" placeholder=\\\"Select vocab, click edit, change and save!\\\">\\\r\n\\\r\n<p id=\\\"editStatus\\\">Ready to edit..</p>\\\r\n<button id=\\\"EditEditBtn\\\" type=\\\"button\\\">Edit</button>\\\r\n<button id=\\\"EditSaveBtn\\\" type=\\\"button\\\">Save</button>         \\\r\n<button id=\\\"EditDeleteBtn\\\" type=\\\"button\\\" title=\\\"Delete selected item\\\">Delete</button>         \\\r\n<button id=\\\"EditDeleteAllBtn\\\" type=\\\"button\\\" title=\\\"本当にやるの？\\\">Delete All</button>   \\\r\n<button id=\\\"ResetLevelsBtn\\\" type=\\\"button\\\">Reset levels</button>         \\\r\n</form>                                                                   \\\r\n</div>\");\r\n$(\"#edit\").hide();\r\n\r\n$(\"#EditEditBtn\").click(function () {\r\n\t//get handle for 'select' area\r\n\tvar select = document.getElementById(\"editWindow\");\r\n\r\n\t//get the index for the currently selected item\r\n\tvar index = select.selectedIndex; //select.options[select.selectedIndex].value is not required, option values are set to index\r\n\tvar vocabList = StorageUtil.getVocList();\r\n\tvocabList = vocabList.reverse();\r\n\tdocument.getElementById(\"editItem\").value = JSON.stringify(vocabList[index]);\r\n\tdocument.getElementById(\"editItem\").name = index; //using name to save the index\r\n\t$(\"#editStatus\").text('Loaded item to edit');\r\n});\r\n\r\n$(\"#EditSaveBtn\").click(function () {\r\n\tif ($(\"#editItem\").val().length !== 0) {\r\n\t\t//-- be aware\r\n\t\t//deleting one item may cause mismatch if i is property of item in list\r\n\t\ttry {\r\n\t\t\tvar index = document.getElementById(\"editItem\").name;\r\n\t\t\tvar item = JSON.parse(document.getElementById(\"editItem\").value.toLowerCase());\r\n\t\t\tvar m = item.meaning.length;\r\n\t\t\twhile(m--){\r\n\t\t\t\tif (item.meaning[m] === \"\"){\r\n\t\t\t\t\tdelete item.meaning[m];\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t\tvar fullList = getFullList().reverse();\r\n\r\n\r\n\t\t\tif (isItemValid(item) &&//item is valid\r\n\t\t\t\t!(checkForDuplicates(fullList,item) && //kanji (if changed) is not already in the list\r\n\t\t\t\t  fullList[index].kanji !== item.kanji)) {//unless it is the item being edited\r\n\r\n\r\n\t\t\t\tvar srslist = getSrsList().reverse();\r\n\t\t\t\t//get srs components of item(list)\r\n\r\n\t\t\t\tfullList[index] = item;//does not have srs stuff, re-add it now\r\n\r\n\t\t\t\tdebugging&&console.log(fullList[index]);\r\n\t\t\t\tdebugging&&console.log(srslist[index]);\r\n\t\t\t\tfullList[index].date = srslist[index].date;\r\n\t\t\t\tfullList[index].level = srslist[index].level;\r\n\t\t\t\tfullList[index].locked = srslist[index].locked;\r\n\t\t\t\tfullList[index].manualLock = srslist[index].manualLock;\r\n\r\n\t\t\t\tfullList = fullList.reverse(); //reset order of array\r\n\r\n\t\t\t\tlocalSet('User-Vocab', fullList);\r\n\r\n\t\t\t\tgenerateEditOptions();\r\n\t\t\t\t$(\"#editStatus\").html('Saved changes!');\r\n\t\t\t\tdocument.getElementById(\"editItem\").value = \"\";\r\n\t\t\t\tdocument.getElementById(\"editItem\").name = \"\";\r\n\r\n\t\t\t}\r\n\t\t\telse{\r\n\t\t\t\t$(\"#editStatus\").text('Invalid item or duplicate!');\r\n\t\t\t\talert(isItemValid(item).toString() +\" && ！(\"+ checkForDuplicates(fullList,item).toString()+\" && !(\"+fullList[index].kanji+\" !== \"+item.kanji+\")\");\r\n\r\n\t\t\t}\r\n\t\t}\r\n\t\tcatch (e) {\r\n\t\t\t$(\"#editStatus\").text(e);\r\n\t\t}\r\n\t}\r\n});\r\n\r\n$(\"#EditDeleteBtn\").click(function () {\r\n\t//select options element window\r\n\tvar select = document.getElementById(\"editWindow\");\r\n\r\n\t//index of selected item\r\n\tvar item = select.options[select.selectedIndex].value;\r\n\r\n\t//fetch JSON strings from storage and convert them into Javascript literals\r\n\tvar vocabList = getFullList();\r\n\r\n\t//starting at selected index, remove 1 entry (the selected index).\r\n\tif (item > -1) {\r\n\t\tif (vocabList !== null){\r\n\t\t\tvocabList.splice(item, 1);\r\n\t\t}\r\n\t}\r\n\r\n\t//yuck\r\n\tif (vocabList.length !== 0) {\r\n\t\tlocalSet('User-Vocab', vocabList);\r\n\t}\r\n\telse {\r\n\t\tlocalStorage.removeItem('User-Vocab');\r\n\t}\r\n\r\n\tupdateEditGUI();\r\n\r\n\t$(\"#editStatus\").text('Item deleted!');\r\n});\r\n\r\n$(\"#EditDeleteAllBtn\").click(function () {\r\n\tvar deleteAll = confirm(\"Are you sure you want to delete all entries?\");\r\n\tif (deleteAll) {\r\n\r\n\t\t//drop local storage\r\n\t\tlocalStorage.removeItem('User-Vocab');\r\n\r\n\r\n\t\tupdateEditGUI();\r\n\r\n\t\t$(\"#editStatus\").text('All items deleted!');\r\n\t}\r\n});\r\n\r\n\r\n$(\"#EditCloseBtn\").click(function () {\r\n\t$(\"#edit\").hide();\r\n\t$(\"#editForm\")[0].reset();\r\n\t$(\"#editStatus\").text('Ready to edit..');\r\n});\r\n\r\n/*\r\n*  Export\r\n*/\r\nwindow.WKSS_export = function () {\r\n\t$(\"#export\").show();\r\n\t//hide other windows\r\n\t$(\"#add\").hide();\r\n\t$(\"#import\").hide();\r\n\t$(\"#edit\").hide();\r\n\t$(\"#selfstudy\").hide();\r\n};\r\n\r\n$(\"body\").append('                                                          \\\r\n<div id=\"export\" class=\"WKSS\">                                               \\\r\n<form id=\"exportForm\">                                                                    \\\r\n<button id=\"ExportCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Export Items</h1>                                                \\\r\n<textarea cols=\"50\" rows=\"18\" id=\"exportArea\" placeholder=\"Export your stuff! Sharing is caring ;)\"></textarea>                           \\\r\n\\\r\n<p id=\"exportStatus\">Ready to export..</p>                                        \\\r\n<button id=\"ExportItemsBtn\" type=\"button\">Export Items</button>\\\r\n<button id=\"ExportSelectAllBtn\" type=\"button\">Select All</button>\\\r\n<button id=\"ExportCsvBtn\" type=\"button\">Export CSV</button>\\\r\n</form>                                                                   \\\r\n</div>');\r\n$(\"#export\").hide();\r\n\r\n\r\n$(\"#ExportItemsBtn\").click(function () {\r\n\r\n\tif (localStorage.getItem('User-Vocab')) {\r\n\t\t$(\"#exportForm\")[0].reset();\r\n\t\tvar vocabList = StorageUtil.getVocList();\r\n\t\t$(\"#exportArea\").text(JSON.stringify(vocabList));\r\n\t\t$(\"#exportStatus\").text(\"Copy this text and share it with others!\");\r\n\t}\r\n\telse {\r\n\t\t$(\"#exportStatus\").text(\"Nothing to export yet :(\");\r\n\t}\r\n});\r\n\r\n$(\"#ExportSelectAllBtn\").click(function () {\r\n\tif ($(\"#exportArea\").val().length !== 0) {\r\n\t\tselect_all(\"exportArea\");\r\n\t\t$(\"#exportStatus\").text(\"Don't forget to CTRL + C!\");\r\n\t}\r\n});\r\n\r\n$(\"#ExportCsvBtn\").click(function () {\r\n\tvar vocabList = getFullList();\r\n\tvar CsvFile = createCSV(vocabList);\r\n\twindow.open(CsvFile);\r\n});\r\n\r\n$(\"#ExportCloseBtn\").click(\r\n\tfunction () {\r\n\t\t$(\"#export\").hide();\r\n\t\t$(\"#exportForm\")[0].reset();\r\n\t\t$(\"#exportArea\").text(\"\");\r\n\t\t$(\"#exportStatus\").text('Ready to export..');\r\n\t}\r\n);\r\n\r\n/*\r\n*  Import\r\n*/\r\nwindow.WKSS_import = function () {\r\n\t$(\"#import\").show();\r\n\t//hide other windows\r\n\t$(\"#add\").hide();\r\n\t$(\"#export\").hide();\r\n\t$(\"#edit\").hide();\r\n\t$(\"#selfstudy\").hide();\r\n};\r\n\r\n$(\"body\").append('                                                          \\\r\n<div id=\"import\" class=\"WKSS\">                                               \\\r\n<form id=\"importForm\">                                                                    \\\r\n<button id=\"ImportCloseBtn\" class=\"wkss-close\" type=\"reset\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Import Items</h1>\\\r\n<textarea cols=\"50\" rows=\"18\" id=\"importArea\" placeholder=\"Paste your stuff and hit the import button! Use with caution!\"></textarea>                     \\\r\n\\\r\n<p id=\"importStatus\">Ready to import..</p>                                        \\\r\n<label class=\"button\" id=\"ImportItemsBtn\" style=\"display:inline;\">Import Items</label>\\\r\n\\\r\n<label id=\"ImportCsvBtn\" class=\"button\" style=\"display:inline;cursor: pointer;\">Import CSV         \\\r\n\\\r\n<input type=\"file\" id=\"upload\" accept=\".csv,.tsv\" style=\"height:0px;width:0px;background:red;opacity:0;filter:opacity(1);\" />\\\r\n\\\r\n</label>\\\r\n\\\r\n<label class=\"button\" id=\"ImportWKBtn\" style=\"display:inline;\"><i class=\"icon-download-alt\"></i> WK</label>\\\r\n</form>                                                                   \\\r\n</div>');\r\n$(\"#import\").hide();\r\n\r\ndocument.getElementById(\"upload\") && document.getElementById(\"upload\").addEventListener('change', ImportUtil.fileUpload, false);\r\n\r\n\r\n$(\"#ImportCsvBtn\").click(function () {\r\n});\r\n\r\n$(\"#ImportWKBtn\").click(function(){\r\n\tWanikaniUtil.getServerResp(APIkey,\"vocabulary\");\r\n\tdebugging&&console.log(\"maybe?\");\r\n});\r\n\r\n$(\"#ImportItemsBtn\").click(function () {\r\n\r\n\tif ($(\"#importArea\").val().length !== 0) {\r\n\t\ttry {\r\n\t\t\tvar add = JSON.parse($(\"#importArea\").val().toLowerCase());\r\n\t\t\talert(JSON.stringify(add));\r\n\t\t\tif (checkAdd(add)) {\r\n\t\t\t\t$(\"#importStatus\").text(\"No valid input (duplicates?)!\");\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tvar newlist;\r\n\t\t\tvar srslist = [];\r\n\t\t\tif (localStorage.getItem('User-Vocab')) {\r\n\t\t\t\tvar vocabList = StorageUtil.getVocList();\r\n\t\t\t\tsrslist = getSrsList();\r\n\t\t\t\tnewlist = vocabList.concat(add);\r\n\t\t\t}\r\n\t\t\telse {\r\n\t\t\t\tnewlist = add;\r\n\r\n\r\n\t\t\t}\r\n\t\t\tvar i = add.length;\r\n\t\t\twhile(i--){\r\n\t\t\t\tStorageUtil.setVocItem(add[i]);\r\n\t\t\t}\r\n\r\n\t\t\t$(\"#importStatus\").text(\"Import successful!\");\r\n\r\n\t\t\t$(\"#importForm\")[0].reset();\r\n\t\t\t$(\"#importArea\").text(\"\");\r\n\r\n\t\t}\r\n\t\tcatch (e) {\r\n\t\t\t$(\"#importStatus\").text(\"Parsing Error!\");\r\n\t\t\tdebugging&&console.log(e);\r\n\t\t}\r\n\r\n\t}\r\n\telse {\r\n\t\t$(\"#importStatus\").text(\"Nothing to import :( Please paste your stuff first\");\r\n\t}\r\n});\r\n\r\n$(\"#ImportCloseBtn\").click(function () {\r\n\t$(\"#import\").hide();\r\n\t$(\"#importForm\")[0].reset();\r\n\t$(\"#importArea\").text(\"\");\r\n\t$(\"#importStatus\").text('Ready to import..');\r\n});\r\n\r\n/*\r\n*  Review Items\r\n*/\r\nwindow.WKSS_review = function () {\r\n\r\n\t//is there a session waiting in storage?\r\n\tif(sessionStorage.getItem('User-Review')) {\r\n\r\n\t\t//show the selfstudy window\r\n\t\t$(\"#selfstudy\").show();\r\n\r\n\t\t//hide other windows\r\n\t\t$(\"#add\").hide();\r\n\t\t$(\"#export\").hide();\r\n\t\t$(\"#edit\").hide();\r\n\t\t$(\"#import\").hide();\r\n\r\n\t\tstartReview();\r\n\t}\r\n};\r\n\r\n$(\"body\").append('                                                          \\\r\n<div id=\"selfstudy\" class=\"WKSS\">\\\r\n<button id=\"SelfstudyCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Review<span id=\"RevNum\"></span></h1>\\\r\n<div id=\"wkss-kanji\">\\\r\n<span id=\"rev-kanji\"></span>\\\r\n</div><div id=\"wkss-type\">\\\r\n<span id=\"rev-type\"></span><br />\\\r\n</div><div id=\"wkss-solution\">\\\r\n<span id=\"rev-solution\"></span>\\\r\n</div><div id=\"wkss-input\">\\\r\n<input type=\"text\" id=\"rev-input\" size=\"40\" placeholder=\"\">\\\r\n</div><span id=\"rev-index\" style=\"display: block;\"></span>\\\r\n\\\r\n<form id=\"audio-form\">\\\r\n<label id=\"AudioButton\" class=\"button\">Play audio</label>\\\r\n<label id=\"WrapUpBtn\"   class=\"button\">Wrap Up</label>\\\r\n</form>\\\r\n<div id=\"rev-audio\" style=\"display:none;\"></div>\\\r\n</div>');\r\n$(\"#selfstudy\").hide();\r\n\r\n$(\"#SelfstudyCloseBtn\").click(function () {\r\n\t$(\"#selfstudy\").hide();\r\n\t$(\"#rev-input\").val(\"\");\r\n\treviewActive = false;\r\n});\r\n\r\n$(\"#WrapUpBtn\").click(function() {\r\n\tvar sessionList = sessionGet('User-Review')||[];\r\n\tvar statsList = sessionGet('User-Stats')||[];\r\n\t//if an index in sessionList matches one in statsList, don't delete\r\n\tvar sessionI = sessionList.length;\r\n\tvar item = sessionGet('WKSS-item')||[];\r\n\tvar arr2 = [];\r\n\t//for every item in sessionList, look for index in statsList,\r\n\t//if not there (-1) delete item from sessionList\r\n\twhile (sessionI--){\r\n\t\tvar index = findIndex(statsList,sessionList[sessionI]);\r\n\t\tif ((Math.sign(1/index) !== -1)||(sessionList[sessionI].index == item.index)){\r\n\r\n\t\t\tarr2.push(sessionList[sessionI]);\r\n\t\t}\r\n\t}\r\n\r\n\r\n\tdebugging&&console.log(arr2);\r\n\tsessionSet('User-Review', JSON.stringify(arr2));\r\n});\r\n\r\n$(\"#AudioButton\").click(function () {\r\n\topenInNewTab(document.getElementById('rev-audio').innerHTML);\r\n});\r\n\r\n$(\"body\").append('                                                          \\\r\n<div id=\"resultwindow\" class=\"WKSS\">                                    \\\r\n<button id=\"ReviewresultsCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Review Results</h1>\\\r\n<h2>All</h2>\\\r\n<div id=\"stats-a\"></div>\\\r\n</div>');\r\n\r\n$(\"#resultwindow\").hide();\r\n\r\n$(\"#ReviewresultsCloseBtn\").click(function () {\r\n\t$(\"#resultwindow\").hide();\r\n\tdocument.getElementById(\"stats-a\").innerHTML = \"\";\r\n});\r\n\r\n//declare global values for keyup event\r\n//is an answer being submitted?\r\nvar submit = true;\r\n\r\n//jquery keyup event\r\n$(\"#rev-input\").keyup(function (e) {\r\n\t//functions:\r\n\t//  inputCorrect()\r\n\r\n\t//check if key press was 'enter' (keyCode 13) on the way up\r\n\t//and keystate true (answer being submitted)\r\n\t//and cursor is focused in reviewfield\r\n\tif (e.keyCode == 13 && submit === true) {\r\n\t\tvar input = $(\"#rev-input\").val();\r\n\t\tvar reviewList = sessionGet('User-Review')||[];\r\n\t\tvar rnd = sessionStorage.getItem('WKSS-rnd')||0;\r\n\r\n\t\tvar item = sessionGet('WKSS-item');\r\n\r\n\t\t//-- starting implementation of forgiveness protocol\r\n\r\n\t\titem.forgive = [];//\"ゆるす\"]; //placeholder (許す to forgive)\r\n\r\n\r\n\t\tif (item === null){\r\n\t\t\talert(\"Item Null??\");\r\n\t\t\treviewList.splice(rnd, 1);\r\n\t\t}else{\r\n\t\t\t//handle grading and storing solution\r\n\r\n\t\t\t//check for input, do nothing if none\r\n\t\t\tif(input.length === 0){\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t//disable input after submission\r\n\t\t\t//document.getElementById('rev-input').disabled = true;\r\n\r\n\r\n\t\t\t//was the input correct?\r\n\t\t\tvar correct = inputCorrect();\r\n\r\n\t\t\t//was the input forgiven?\r\n\t\t\tvar forgiven = (item.forgive.indexOf(input) !== -1);\r\n\r\n\t\t\tif (correct) {\r\n\t\t\t\t//highlight in (default) green\r\n\t\t\t\t$(\"#rev-input\").addClass(\"correct\");\r\n\t\t\t\t//show answer\r\n\t\t\t\t$(\"#rev-solution\").addClass(\"info\");\r\n\t\t\t} else if (forgiven){\r\n\t\t\t\t$(\"#rev-input\").addClass(\"caution\");\r\n\t\t\t} else {\r\n\t\t\t\t//highight in red\r\n\t\t\t\t$(\"#rev-input\").addClass(\"error\");\r\n\t\t\t\t//show answer\r\n\t\t\t\t$(\"#rev-solution\").addClass(\"info\");\r\n\t\t\t}\r\n\r\n\t\t\t//remove from sessionList if correct\r\n\t\t\tif (correct) {\r\n\t\t\t\tdebugging&&console.log(\"correct answer\");\r\n\t\t\t\tif (reviewList !== null){\r\n\t\t\t\t\tvar oldlen = reviewList.length;\r\n\r\n\t\t\t\t\treviewList.splice(rnd, 1);\r\n\t\t\t\t\tdebugging&&console.log(\"sessionList.length: \"+ oldlen +\" -> \"+reviewList.length);\r\n\r\n\t\t\t\t\t//replace shorter (by one) sessionList to session\r\n\t\t\t\t\tif (reviewList.length !== 0) {\r\n\t\t\t\t\t\tdebugging&&console.log(\"sessionList.length: \"+ reviewList.length);\r\n\t\t\t\t\t\tsessionSet('User-Review', JSON.stringify(reviewList));\r\n\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\t//reveiw over, delete sessionlist from session\r\n\t\t\t\t\t\tsessionStorage.removeItem('User-Review');\r\n\t\t\t\t\t}\r\n\t\t\t\t}else{\r\n\t\t\t\t\tconsole.error(\"Error: no review session found\");\r\n\t\t\t\t}\r\n\t\t\t}else{\r\n\t\t\t\t//   if(forgiven){\r\n\t\t\t\t//     debugging&&console.log(input +\" has been forgiven. \"+item.type);\r\n\t\t\t\t//   return;\r\n\t\t\t\t//}\r\n\t\t\t\tdebugging&&console.log(\"wrong answer\");\r\n\t\t\t}\r\n\r\n\t\t\titem = markAnswer(item);\r\n\r\n\t\t\tsessionSet(item.index, item);\r\n\r\n\r\n\t\t\tvar list = JSON.parse(sessionStorage.getItem(\"User-Stats\"))||[];\r\n\t\t\tvar found = false;\r\n\r\n\t\t\tif (list){\r\n\t\t\t\tvar i = list.length;\r\n\t\t\t\twhile(i--){\r\n\t\t\t\t\tif (list[i].index == item.index) {\r\n\t\t\t\t\t\tlist[i] = item;\t\t\t\t\t\t\t\t//replace item if it exists\r\n\t\t\t\t\t\tfound = true;\r\n\t\t\t\t\t\tbreak;\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t\tif(!found){\r\n\t\t\t\t\tlist = saveToSortedList(list,item);\r\n\t\t\t\t}\r\n\r\n\t\t\t} else {\r\n\t\t\t\tlist = [item];\r\n\t\t\t}\r\n\r\n\t\t\tsessionSet(\"User-Stats\", JSON.stringify(list));\r\n\t\t\t//playAudio();\r\n\r\n\t\t\t//answer submitted, next 'enter' proceeds with script\r\n\t\t\tsubmit = false;\r\n\t\t}//null garbage collection\r\n\t}\r\n\telse if (e.keyCode == 13 && submit === false) {\r\n\t\tdebugging&&console.log(\"keystat = \" + submit);\r\n\r\n\t\t//there are still more reviews in session?\r\n\t\tif (sessionStorage.getItem('User-Review')) {\r\n\t\t\t// debugging&&console.log(\"found a 'User-Review': \" + sessionStorage.getItem('User-Review'));\r\n\r\n\t\t\tsetTimeout(function () {\r\n\t\t\t\tdebugging&&console.log(\"refreshing reviewList from storage\");\r\n\t\t\t\tvar reviewList = JSON.parse(sessionStorage.getItem('User-Review'));\r\n\r\n\t\t\t\t//cue up first remaining review\r\n\t\t\t\tnextReview(reviewList);\r\n\t\t\t\tdebugging&&console.log(\"checking for empty reviewList\");\r\n\t\t\t\tif (reviewList.length === 0){\r\n\r\n\t\t\t\t\tdebugging&&console.log(\"session over. reviewList: \"+JSON.stringify(reviewList));\r\n\t\t\t\t\tsessionStorage.removeItem(\"User-Review\");\r\n\t\t\t\t}\r\n\r\n\t\t\t\t//         document.getElementById('rev-input').disabled = true;\r\n\t\t\t\t$(\"#rev-solution\").removeClass(\"info\");\r\n\t\t\t\t$(\"#selfstudy\").hide().fadeIn('fast');\r\n\r\n\t\t\t}, 1);\r\n\t\t}\r\n\t\telse {\r\n\t\t\t// no review stored in session, review is over\r\n\t\t\tsetTimeout(function () {\r\n\r\n\t\t\t\t$(\"#selfstudy\").hide();\r\n\t\t\t\t//document.getElementById('rev-input').disabled = false;\r\n\t\t\t\t$(\"#rev-solution\").removeClass(\"info\");\r\n\t\t\t\tdebugging&&console.log(\"showResults\");\r\n\t\t\t\tshowResults();\r\n\t\t\t\t$(\"#resultwindow\").show();\r\n\t\t\t\tdebugging&&console.log(\"showResults completed\");\r\n\r\n\t\t\t\t//*/  //clear session\r\n\t\t\t\tsessionStorage.clear();\r\n\t\t\t\treviewActive = false;\r\n\r\n\r\n\t\t\t}, 1);\r\n\t\t}\r\n\t\tsubmit = true;\r\n\r\n\t}\r\n});\r\n\t/** populate reviews when menu button pressed\r\n\t*/\r\n    window.generateReviewList = function() {\r\n        //if menu is invisible, it is about to be visible\r\n        if ( $(\"#WKSS_dropdown\").is(\":hidden\") ){\r\n            //This is really the only time it needs to run\r\n            //unless we want to start updating in realtime by keeping track of the soonest item\r\n            generateReviewList(reviewActive);\r\n        }\r\n    };\r\n\t/**  Add Item: event function to open \"add window\" and close any other window that might be open at the time.\r\n\t*/\r\n    window.WKSS_add = function () {\r\n        //show the add window\r\n        $(\"#add\").show();\r\n        //hide other windows\r\n        $(\"#export\").hide();\r\n        $(\"#import\").hide();\r\n        $(\"#edit\").hide();\r\n        $(\"#selfstudy\").hide();\r\n    };\r\n\t\r\n\tvar addElement = require('./addelement.js');\r\n\t//add html to page source\r\n    $(\"body\").append(addElement);\r\n    //hide add window (\"div add\" code that was just appended)\r\n    $(\"#add\").hide();\r\n\r\n    var handleAddClick = require('./handleAddClick.js');\r\n\t\r\n    //function to fire on click event for \"Add new Item\"\r\n    $(\"#AddItemBtn\").click(function () {\r\n        handleAddClick();\r\n    });\r\n\r\n    $(\"#AddCloseBtn\").click(function () {\r\n        $(\"#add\").hide();\r\n        $(\"#addForm\")[0].reset();\r\n        $(\"#addStatus\").text('Ready to add..');\r\n        $(\"#addKanji\").removeClass(\"error\");\r\n        $(\"#addMeaning\").removeClass(\"error\");\r\n    });\r\n\r\n    /** Keeps legacy srsList updated.\r\n\t* @depreciate\r\n\t* @param {SrsItem} srsitem\r\n\t* @param {Array.<SrsItem>} srsList\r\n\t* @returns {Array.<SrsItem>} The srs data for a task. Or null if no srsList was provided.\r\n\t*/\r\n    var updateSrsInList = function(srsitem, srsList){\r\n        var index = srsitem.i;\r\n        if(srsList){\r\n            if(srsList[index].kanji===srsitem.kanji){// try search by index\r\n                debugging&&console.log(\"success: \"+srsitem.kanji+\" found at index \"+ index);\r\n                //replace only the srs parts of the item\r\n                srsList[index].date = srsitem.date;\r\n                srsList[index].level = srsitem.level;\r\n                srsList[index].locked = srsitem.locked;\r\n                srsList[index].manualLock = srsitem.manualLock;\r\n            }\r\n            return srsList;\r\n        }\r\n\t\telse{\r\n\t\t\treturn null;\r\n\t\t}\r\n    };\r\n    /** Checks if an item's kanji is represented in a list\r\n\t* @returns {boolean}\r\n\t*/\r\n    var checkForDuplicates = function(list, item){\r\n\t\treturn list.some(function(a){return a.kanji === item.kanji;});\r\n\t};\r\n\r\n\t/** Creates a lookup array for each kanji with its srs level. Used for displaying component levels.\r\n\t* @param item\r\n\t* @param kanjilist\r\n\t* @returns An array of the kanji with SRS values for each kanji component.\r\n\t* @example\r\n        eg. 折り紙:\r\n        compSRS = [{\"kanji\": \"折\", \"srs\": \"guru\"}, {\"kanji\": \"紙\", \"srs\": \"apprentice\"}]\r\n\t*/\r\n    var getCompKanji = function(item, kanjiList){\r\n        if (!kanjiList){\r\n            kanjiList = [];\r\n        }\r\n        debugging&&console.log(\"getCompKanji(item, kanjiList)\");\r\n\r\n        var compSRS = [];\r\n        var kanjiReady = false; //indicates if the kanjiList has been populated\r\n        var userGuppy = false; //indicates if kanjiList has less than 100 items\r\n        var kanjiObj = {};\r\n\r\n        //has the server responded yet\r\n        if (kanjiList.length > 0){\r\n            debugging&&console.log(\"kanjiList is > 0\");\r\n            kanjiReady = true;\r\n\r\n            //create lookup object\r\n            for (var k=0;k<kanjiList.length;k++){\r\n                kanjiObj[kanjiList[k].character] = kanjiList[k];\r\n            }\r\n\r\n            //is there less than 100 kanji in the response\r\n            if (kanjiList.length < 100){\r\n                debugging&&console.log(\"kanjiList is < 100\");\r\n                userGuppy = true;\r\n            }\r\n        }    \r\n\r\n        var components = item.components;\r\n        //for each kanji character component\r\n        //    this is the outer loop since there will be far less of them than kanjiList\r\n        for(var i = 0; i < components.length; i++){\r\n\r\n            var matched = false;\r\n            //for each kanji returned by the server\r\n            // for(var j=0; j<kanjiList.length; j++){\r\n\r\n            //if the kanji returned by the server matches the character in the item\r\n            if (typeof kanjiObj[components[i]] !== 'undefined'){\r\n                //      if (kanjiList[j].character == components[i]){\r\n                compSRS[i] = {\"kanji\": components[i], \"srs\": kanjiObj[components[i]].srs};\r\n                matched = true;\r\n\r\n                // break; //kanji found: 'i' is its position in item components; 'j' is its postion in the 'kanjiList' server response\r\n            }\r\n            // }\r\n\r\n            if (matched === false){ // character got all the way through kanjiList without a match.\r\n                if (kanjiReady){ //was there a server response?\r\n                    if (userGuppy){ //is the user a guppy (kanji probably matches a turtles response)\r\n                        debugging&&console.log(\"matched=false, kanjiList.length: \"+kanjiList.length);\r\n                        compSRS[i] = {\"kanji\": components[i], \"srs\": \"noMatchGuppy\"};\r\n                    }\r\n\t\t\t\t\telse{ //user is a turtle, kanji must not have been added to WK (yet)\r\n                        debugging&&console.log(\"matched=false, kanjiList.length: \"+kanjiList.length);\r\n                        compSRS[i] = {\"kanji\": components[i], \"srs\": \"noMatchWK\"};\r\n                    }\r\n                }\r\n\t\t\t\telse{\r\n                    debugging&&console.log(\"matched=false, kanjiReady=false, noServerResp\");\r\n                    compSRS[i] = {\"kanji\": components[i], \"srs\": \"noServerResp\"};\r\n                }\r\n            }\r\n        }\r\n        return compSRS;\r\n    };\r\n\r\n\r\n    var isKanjiLocked = function(srsitem, kanjiList, locksOn){\r\n        //item unlocked by default\r\n        //may have no kanji, only unlocked kanji will get through the code unflagged\r\n\r\n\t\t// Enumeration \"yes\", \"no\", \"DB\"\r\n        var locked = \"no\";\r\n        if (locksOn){\r\n            //get the kanji characters in the word.\r\n            var componentList = getCompKanji(srsitem, kanjiList);\r\n            // eg: componentList = getCompKanji(\"折り紙\", kanjiList);\r\n            // componentList = [{\"kanji\": \"折\", \"srs\": \"guru\"}, {\"kanji\": \"紙\", \"srs\": \"apprentice\"}]\r\n\r\n\r\n            var c = componentList.length;\r\n            while(c--){\r\n                //look for locked kanji in list\r\n                if (componentList[c].srs == \"apprentice\" ||\r\n                    componentList[c].srs == \"noServerResp\"||\r\n                    componentList[c].srs == \"unreached\"\r\n                   ){\r\n\r\n                    //----could be apprentice etc.\r\n                    //Simple: lock is 'yes'\r\n                    locked = \"yes\";\r\n                    // \"yes\":\titem will be locked while there is no database connection.\r\n                    //\t\t\tif the server response indicates that it has been unlocked, only then will it be available for review\r\n\r\n                    debugging&&console.log(\"test srs for apprentice etc. 'locked': \"+ locked);\r\n\r\n                    debugging&&console.log(componentList[c].kanji +\": \"+componentList[c].srs +\" -> \"+ locked);\r\n\r\n                    break; // as soon as one kanji is locked, the whole item is locked\r\n                }\r\n\r\n                //DB locks get special state\r\n                if (componentList[c].srs == \"noMatchWK\" || componentList[c].srs == \"noMatchGuppy\"){\r\n\r\n                    locked = \"DB\";\r\n                    //\"DB\"\t: database limitations, one of two things\r\n                    //a. the kanji isn't in the database and the user is a guppy --could change if user subscribes or first two levels change/expand\r\n                    //b. the kanji isn't in the database and the user is a turtle --could change if more kanji added.\r\n\r\n                    debugging&&console.log(\"test srs for unmatched kanji. 'locked': \"+ locked);\r\n\r\n                    debugging&&console.log(componentList[c].kanji +\": \"+componentList[c].srs +\" -> \"+ locked);\r\n\r\n\r\n                }\r\n\r\n            } //for char in componentList\r\n            debugging&&console.log(\"out of character loop\");\r\n        }\r\n        //locked will be either \"yes\",\"no\", or \"DB\"\r\n        return [locked];\r\n    };\r\n\r\n    /** Gets the Kanji characters in a given string.\r\n\t* @param {string} vocabString -\r\n\t* @return {Array.<string>} An array of the kanji components in the given string\r\n\t*/\r\n    var getComponents = function(vocabString){\r\n        return Array.prototype.filter.call(vocabString, function(ch){\r\n            return /^[\\u4e00-\\u9faf]+$/.test(ch);\r\n        }, this);\r\n    };\r\n\r\n    /** Manages the locked and manualLock properties of srsitem. This is to stop items being locked again after they have been unlocked if any of the kanji used falls below the unlock threshold (eg. if the 勉 in 勉強 falls back to apprentice, we do not want to lock up 勉強 again.)\r\n\t* @param {Object} item\r\n\t* @param {string} item.locked - (String enumeration) A real time evaluation of the item (is any of the kanji in the word locked?)\r\n\t* @param {string} item.manualLock - (String enumeration) Will return 'no' if .locked has ever returned 'no'.\r\n\t* @returns {ITask} item\r\n\t*/\r\n    var setLocks = function(item){\r\n        //once manualLock is \"no\" it stays \"no\"\r\n        if (item.manualLock !== false && item.manualLock !== \"no\" && item.manualLock !== \"n\"){\r\n\r\n            var kanjiList = localGet('User-KanjiList')||[];\r\n\r\n            item.components = getComponents(item.kanji);\r\n\r\n            var kanjiLockedResult = isKanjiLocked(item, kanjiList, locksOn);\r\n            item.locked = kanjiLockedResult[0];\r\n\r\n            item.manualLock = item.locked;\r\n        }else{\r\n            item.manualLock = false;\r\n        }\r\n\r\n        debugging&&console.log(\"setting locks for \"+ item.kanji +\": locked: \"+item.locked+\", manualLock: \"+ item.manualLock);\r\n\r\n        return item;\r\n    };\r\n    /** Converts number of milliseconds into a readable string\r\n\t* @param {number} milliseconds - The number of milliseconds to approximate\r\n\t* @returns {string} Readable time frame ('2 months', '3 hours', '1 week' etc).\r\n\t*/\r\n\tvar ms2str = function(milliseconds){\r\n        var num; //number of months weeks hours etc\r\n        //more time has elapsed than required for the level\r\n        if(milliseconds <= 0) {\r\n            return \"Now\" ;\r\n        }\r\n        if(milliseconds > 2628000000) {//About a month\r\n            num = Math.floor(milliseconds/2628000000).toString()+\" month\";\r\n            if (num !== \"1 month\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n        if(milliseconds > 604800000) {//A week\r\n            num = Math.floor(milliseconds/604800000).toString()+\" week\";\r\n            if (num !== \"1 week\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n        if(milliseconds > 86400000) {//A day\r\n            num = Math.floor(milliseconds/86400000).toString()+\" day\";\r\n            if (num !== \"1 day\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n        if(milliseconds > 3600000) {//An hour\r\n            num = Math.floor(milliseconds/3600000).toString()+\" hour\";\r\n            if (num !== \"1 hour\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n        if(milliseconds > 60000) {//A minute\r\n            num = Math.floor(milliseconds/60000).toString()+\" minute\";\r\n            if (num !== \"1 minute\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n        if(milliseconds > 0) {//A second is 1000, but need to return something for less than one too\r\n            num = Math.floor(milliseconds/1000).toString()+\" second\";\r\n            if (num !== \"1 second\"){\r\n                return num+\"s\";\r\n            }else{\r\n                return num;\r\n            }\r\n        }\r\n    };\r\n    /** Retrieves values from storage to populate 'editItems' menu\r\n\t*/\r\n    var generateEditOptions = function() {\r\n        var select = document.getElementById('editWindow');\r\n        //clear the editWindow\r\n        while (select.firstChild) {\r\n            select.removeChild(select.firstChild);\r\n        }\r\n        //check for items to add\r\n        if (localStorage.getItem('User-Vocab')) {\r\n\r\n            //retrieve from local storage\r\n            var vocabList = StorageUtil.getVocList();\r\n            var srslist =  StorageUtil.getVocList();\r\n            var options = [];\r\n            //build option string\r\n            //var i = vocabList.length;\r\n            //while (i--){\r\n\t\t\tvocabList.forEach(function(task){\r\n                //form element to save string\r\n                var opt = document.createElement('option');\r\n\r\n                //dynamic components of string\r\n\r\n                //when is this item up for review\r\n                var due = task.due||task.date + srsObject[task.level].duration;\r\n                var review = \"\";\r\n\r\n                //no future reviews if burned\r\n                if(task.level >= 9) {\r\n                    review = \"Never\";\r\n                }\r\n\r\n                //calculate next relative review time\r\n                //current timestamp is past due date.\r\n                else if(Date.now() >= due) {\r\n                    review = \"Now\" ;\r\n                }\r\n                else {\r\n                    review = ms2str(due - Date.now());\r\n                }//end if review is not 'never' or 'now'\r\n\r\n                var text = task.kanji + \" & \" +\r\n                    task.reading + \" & \" +\r\n                    task.meaning + \" (\" +\r\n\t\t\t\t\tsrsObject[task.level].rank +\r\n\t\t\t\t\t\" - Review: \" +\r\n                    review + \") Locked: \" +\r\n                    task.manualLock;\r\n\r\n                opt.value = i;\r\n                opt.innerHTML = text;\r\n                options.push(opt);//for future use (sorting data etc)\r\n                select.appendChild(opt);//export item to option menu\r\n            }, this);\r\n        }\r\n    };\r\n    /** Edit Items\r\n\t*/\r\n    window.WKSS_edit = function () {\r\n        generateEditOptions();\r\n        $(\"#edit\").show();\r\n        //hide other windows\r\n        $(\"#export\").hide();\r\n        $(\"#import\").hide();\r\n        $(\"#add\").hide();\r\n        $(\"#selfstudy\").hide();\r\n    };\r\n\tvar buildNode = require('./buildnode.js');\r\n\r\n\tvar buildWindow = require('./buildwindow.js');\r\n\t\r\n\t/*var addEditWindow = function() {\r\n\t\tvar editWindow = buildNode('div', {id: \"WKSS-edit\", className: \"WKSS\"});\r\n\t\tvar editForm = buildNode('form', {id: \"WKSS-editForm\"});\r\n\t\teditWindow.appendChild(editForm);\r\n\t\tvar editCloseButton = buildNode('button', {id: \"WKSS-editCloseBtn\", className: \"WKSS-close\"});\r\n\t\teditForm.appendChild(editCloseButton);\r\n\t\t\r\n\t\teditCloseButton.appendChild(buildNode('i', {className: \"icon-remove\"}));\r\n\t\tvar h1Element = buildNode('h1');\r\n\t\teditForm.appendChild(h1Element);\r\n\t\th1Element.appendChild(document.createTextNode(\"Edit your Vocab\"));\r\n\t\tvar selectElement = buildNode('select', {id: \"editWindow\", size: \"8\"});\r\n\t\teditForm.appendChild(selectElement);\r\n\t\tvar editItemText = buildNode('input', {type: \"text\" id: \"editItem\" name: \"\" size: \"40\" placeholder: \"Select vocab, click edit, change and save!\"});\r\n\t\teditForm.appendChild(editItemText);\r\n\t\tvar editStatus = buildNode('p', {id: \"editStatus\"});\r\n\t\teditForm.appendChild(editStatus);\r\n\t\teditStatus.appendChild(document.createTextNode(\"Ready to edit..\"));\r\n\t\t\r\n\t\tvar editButton = buildNode('button', {id: \"EditEditBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editButton);\r\n\t\teditButton.appendChild(document.createTextNode(\"Edit\"));\r\n\t\tvar editSave = buildNode('button', {id: \"EditSaveBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editSave);\r\n\t\teditSave.appendChild(document.createTextNode(\"Save\"));\r\n\t\tvar editDelete = buildNode('button', {id: \"EditDeleteBtn\", type: \"button\", title: \"Delete selected item\"});\r\n\t\teditForm.appendChild(editDelete);\r\n\t\teditDelete.appendChild(document.createTextNode(\"Delete\"));\r\n\t\tvar editDeleteAll = buildNode('button', {id: \"EditDeleteAllBtn\", type: \"button\", title: \"本当にやるの？\"});\r\n\t\teditForm.appendChild(editDeleteAll);\r\n\t\teditDeleteAll.appendChild(document.createTextNode(\"Delete All\"));\r\n\t\tvar editResetLevels = buildNode('button', {id: \"ResetLevelsBtn\", type: \"button\"});\r\n\t\teditForm.appendChild(editResetLevels);\r\n\t\teditResetLevels.appendChild(document.createTextNode(\"Reset levels\"));\r\n\t\t\r\n\t\treturn editWindow;\r\n\t};\r\n\t*/\r\n\tvar editWindowStructure = {\r\n\tid: \"WKSS-edit\",\r\n\tclassName: \"WKSS\",\r\n\tchildNodes:[{\r\n\t\ttag: 'form',\r\n\t\tid: \"WKSS-editForm\",\r\n\t\tchildNodes:[{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"WKSS-editCloseBtn\",\r\n\t\t\tclassName: \"WKSS-close\",\r\n\t\t\tchildNodes:[{\r\n\t\t\t\ttag: 'i',\r\n\t\t\t\tclassName: \"icon-remove\"\r\n\t\t\t}]\r\n\t\t},{\r\n\t\t\ttag: 'h1',\r\n\t\t\tchildNodes:[\"Edit your Vocab\"]\r\n\t\t},{\r\n\t\t\ttag: 'select',\r\n\t\t\tid: \"editWindow\",\r\n\t\t\tother: {size: \"8\"}\r\n\t\t},{\r\n\t\t\ttag: 'input', \r\n\t\t\tother:{\r\n\t\t\t\ttype: \"text\",\r\n\t\t\t\tname: \"\",\r\n\t\t\t\tsize: \"40\",\r\n\t\t\t\tplaceholder: \"Select vocab, click edit, change and save!\"\r\n\t\t\t},\r\n\t\t\tid: \"editItem\"\r\n\t\t},{\r\n\t\t\ttag: 'p', \r\n\t\t\tid: \"editStatus\",\r\n\t\t\tchildNodes:[\"Ready to edit...\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditEditBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Edit\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditSaveBtn\",\r\n\t\t\tother:{type: \"button\"},\r\n\t\t\tchildNodes:[\"Save\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteBtn\",\r\n\t\t\tother: {type: \"button\", title: \"Delete selected item\"},\r\n\t\t\tchildNodes:[\"Delete\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"EditDeleteAllBtn\",\r\n\t\t\tother: {type: \"button\", title: \"本当にやるの？\"},\r\n\t\t\tchildNodes:[\"Delete All\"]\r\n\t\t},{\r\n\t\t\ttag: 'button',\r\n\t\t\tid: \"ResetLevelsBtn\",\r\n\t\t\tother: {type: \"button\"},\r\n\t\t\tchildNodes:[\"Reset levels\"]\r\n\t\t}]\r\n\t}]\r\n};\r\n\r\n    var addEditWindow = buildWindow(editWindowStructure);\r\n\t$(\"body\").append(addEditWindow);\r\n    $(\"#WKSS-edit\").hide();\r\n\r\n\t/** Resets the levels of all tasks and re-indexes them in storage.\r\n\t* @param {Event} evt - Click event (not used)\r\n\t*/\r\n\tvar resetLevels = function (evt) {\r\n\t\tvar vocList = StorageUtil.getVocList().map(function(vocItem, i){\r\n\t\t\tvocItem.level = 0;\r\n\t\t\tdebugging&&console.log(\"vocList[i].i before: \"+vocItem.i);\r\n\t\t\tvocItem.i=i;\r\n\t\t\tdebugging&&console.log(\"vocList[i].i after: \"+vocItem.i);\r\n\t\t\treturn vocItem;\r\n\t\t}, this);\r\n\t\tStorageUtil.setVocList(vocList);\r\n    };\r\n    $(\"#ResetLevelsBtn\").click(resetLevels);\r\n\r\n    $(\"#EditEditBtn\").click(function () {\r\n        //get handle for 'select' area\r\n        var select = document.getElementById(\"editWindow\");\r\n\r\n        //get the index for the currently selected item\r\n        var index = select.selectedIndex; //select.options[select.selectedIndex].value is not required, option values are set to index\r\n        var vocabList = StorageUtil.getVocList();\r\n        vocabList = vocabList.reverse();\r\n        document.getElementById(\"editItem\").value = JSON.stringify(vocabList[index]);\r\n        document.getElementById(\"editItem\").name = index; //using name to save the index\r\n        $(\"#editStatus\").text('Loaded item to edit');\r\n    });\r\n\r\n    var isEmpty = function(value) {\r\n        return (value === void 0 || value === null);\r\n    };\r\n\t\r\n\tvar isArray = function(arg){\r\n\t\treturn Array.isArray ? Array.isArray(arg) : Object.prototype.toString.call(arg) === '[object Array]';\r\n\t};\r\n\r\n    /** Validates a task object\r\n\t* @param {Task} add - The Task being verified\r\n\t* @returns {Boolean} If the provided task has all the necessary properties to be added to the review list.\r\n\t*/\r\n\tvar isItemValid = function(add) {\r\n        return (!isEmpty(add.kanji) && //kanji property exists\r\n\t\t\t!isEmpty(add.meaning) && //meaning property exists\r\n\t\t\t!isEmpty(add.reading) && //reading property exists\r\n\t\t\tisArray(add.meaning) &&//meaning is an array\r\n\t\t\tisArray(add.reading));//reading is an array\r\n    };\r\n\r\n    $(\"#EditSaveBtn\").click(function () {\r\n\t\t//-- be aware\r\n\t\t//deleting one item may cause mismatch if i is property of item in list\r\n\t\ttry {\r\n\t\t\tif ($(\"#editItem\").val().length !== 0) {\r\n\t\t\t\tvar editItem = document.getElementById(\"editItem\");\r\n                var index = editItem.name;\r\n\t\t\t\tvar item = JSON.parse(editItem.value.toLowerCase());\r\n                // Make sure that the word 'meaning' is immutable, so it exists to trim\r\n\t\t\t\t\r\n\t\t\t\tif (item.meaning){\r\n\t\t\t\t\titem.meaning.forEach(function(meaning, m, meanings){\r\n\t\t\t\t\t\tif (meaning === \"\"){\r\n\t\t\t\t\t\t\tdelete meanings[m];\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}, this);\r\n\t\t\t\t}\r\n                var fullList = StorageUtil.getVocList().reverse();\r\n\r\n                if (isItemValid(item) &&//item is valid\r\n                    !(checkForDuplicates(fullList,item) && //kanji (if changed) is not already in the list\r\n                      fullList[index].kanji !== item.kanji)) {//unless it is the item being edited\r\n\r\n                    var srslist = StorageUtil.getVocList().reverse();\r\n                    //get srs components of item(list)\r\n                    fullList[index] = item;//does not have srs stuff, re-add it now\r\n\r\n                    fullList[index].date = srslist[index].date;\r\n                    fullList[index].level = srslist[index].level;\r\n                    fullList[index].locked = srslist[index].locked;\r\n                    fullList[index].manualLock = srslist[index].manualLock;\r\n\r\n                    fullList = fullList.reverse(); //reset order of array\r\n\r\n                    localSet('User-Vocab', fullList);\r\n\r\n                    generateEditOptions();\r\n                    $(\"#editStatus\").html('Saved changes!');\r\n                    document.getElementById(\"editItem\").value = \"\";\r\n                    document.getElementById(\"editItem\").name = \"\";\r\n\t\t\t\t}\r\n\t\t\t\telse{\r\n                    $(\"#editStatus\").text('Invalid item or duplicate!');\r\n                    alert(isItemValid(item).toString() +\" && ！(\"+ checkForDuplicates(fullList,item).toString()+\" && !(\"+fullList[index].kanji+\" !== \"+item.kanji+\")\");\r\n                }\r\n\t\t\t}\r\n\t\t}\r\n\t\tcatch (e) {\r\n\t\t\t$(\"#editStatus\").text(e);\r\n\t\t}\r\n    });\r\n\r\n    var updateEditGUI = function(){\r\n        generateEditOptions();\r\n        document.getElementById(\"editItem\").value = \"\";\r\n        document.getElementById(\"editItem\").name = \"\";\r\n    };\r\n\tvar editDelete = function () {\r\n        //select options element window\r\n        var select = document.getElementById(\"editWindow\");\r\n\r\n        //index of selected item\r\n        var item = select.options[select.selectedIndex].value;\r\n\r\n        //fetch JSON strings from storage and convert them into Javascript literals\r\n        var vocabList = StorageUtil.getVocList();\r\n\r\n        //starting at selected index, remove 1 entry (the selected index).\r\n        if (item > -1) {\r\n            if (vocabList !== null){\r\n                vocabList.splice(item, 1);\r\n            }\r\n        }\r\n\r\n        //yuck\r\n        if (vocabList.length !== 0) {\r\n            localSet('User-Vocab', vocabList);\r\n        }\r\n        else {\r\n            localStorage.removeItem('User-Vocab');\r\n        }\r\n\r\n        updateEditGUI();\r\n\r\n        $(\"#editStatus\").text('Item deleted!');\r\n    };\r\n    $(\"#EditDeleteBtn\").click(editDelete);\r\n\r\n\tvar editDeleteAll = function () {\r\n        var deleteAll = confirm(\"Are you sure you want to delete all entries?\");\r\n        if (deleteAll) {\r\n            //drop local storage\r\n            localStorage.removeItem('User-Vocab');\r\n            updateEditGUI();\r\n            $(\"#editStatus\").text('All items deleted!');\r\n        }\r\n    };\r\n    $(\"#EditDeleteAllBtn\").click(editDeleteAll);\r\n\r\n    $(\"#EditCloseBtn\").click(function () {\r\n        $(\"#edit\").hide();\r\n        $(\"#editForm\")[0].reset();\r\n        $(\"#editStatus\").text('Ready to edit..');\r\n    });\r\n\r\n    /** Export\r\n\t*/\r\n    window.WKSS_export = function () {\r\n        $(\"#export\").show();\r\n        //hide other windows\r\n        $(\"#add\").hide();\r\n        $(\"#import\").hide();\r\n        $(\"#edit\").hide();\r\n        $(\"#selfstudy\").hide();\r\n    };\r\n\r\n\tvar exportWindowStructure = {\r\n\t\tid: \"WKSS-export\",\r\n\t\tclassName: \"WKSS\",\r\n\t\tchildNodes:[{\r\n\t\t\ttag: 'form',\r\n\t\t\tid: \"WKSS-exportForm\",\r\n\t\t\tchildNodes:[\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'button',\r\n\t\t\t\t\tid: \"WKSS-exportCloseBtn\",\r\n\t\t\t\t\tclassName: \"WKSS-close\",\r\n\t\t\t\t\tchildNodes:[{\r\n\t\t\t\t\t\ttag: 'i',\r\n\t\t\t\t\t\tclassName: \"icon-remove\"\r\n\t\t\t\t\t}]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'h1',\r\n\t\t\t\t\tchildNodes:[\"Export Items\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'textarea',\r\n\t\t\t\t\tid: \"exportArea\",\r\n\t\t\t\t\tother: {cols: \"50\", rows: \"18\", placeholder: \"Export your stuff! Sharing is caring ;)\"}\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'p', \r\n\t\t\t\t\tid: \"exportStatus\",\r\n\t\t\t\t\tchildNodes:[\"Ready to export...\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'button',\r\n\t\t\t\t\tid: \"ExportItemsBtn\",\r\n\t\t\t\t\tother: {type: \"button\"},\r\n\t\t\t\t\tchildNodes:[\"Export Items\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'button',\r\n\t\t\t\t\tid: \"ExportSelectAllBtn\",\r\n\t\t\t\t\tother:{type: \"button\"},\r\n\t\t\t\t\tchildNodes:[\"Select All\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'button',\r\n\t\t\t\t\tid: \"ExportCsvBtn\",\r\n\t\t\t\t\tother: {type: \"button\"},\r\n\t\t\t\t\tchildNodes:[\"Export CSV\"]\r\n\t\t\t\t}\r\n\t\t\t]\r\n\t\t}]\r\n\t};\r\n\tvar exportWindow = buildWindow(exportWindowStructure);\r\n\r\n    $(\"body\").append(exportWindow);\r\n    $(\"#WKSS-export\").hide();\r\n\r\n\r\n    $(\"#ExportItemsBtn\").click(function () {\r\n\r\n        if (localStorage.getItem('User-Vocab')) {\r\n            $(\"#exportForm\")[0].reset();\r\n            var vocabList = StorageUtil.getVocList();\r\n            $(\"#exportArea\").text(JSON.stringify(vocabList));\r\n            $(\"#exportStatus\").text(\"Copy this text and share it with others!\");\r\n        }\r\n        else {\r\n            $(\"#exportStatus\").text(\"Nothing to export yet :(\");\r\n        }\r\n    });\r\n\r\n\tvar select_all = function(str) {\r\n        var text_val = document.getElementById(str);\r\n        debugging&&console.log(text_val);\r\n        text_val.focus();\r\n        text_val.select();\r\n    };\r\n\r\n    $(\"#ExportSelectAllBtn\").click(function () {\r\n        if ($(\"#exportArea\").val().length !== 0) {\r\n            select_all(\"exportArea\");\r\n            $(\"#exportStatus\").text(\"Don't forget to CTRL + C!\");\r\n        }\r\n    });\r\n\r\n    var createCSV = function(JSONstring){\r\n        var JSONobject = (typeof JSONstring === 'string') ? JSON.parse(JSONstring) : JSONstring;\r\n        var key;\r\n        var CSVarray = [];\r\n        var header = [];  \r\n        var id = JSONobject.length;\r\n        if (id){//object not empty\r\n            for (key in JSONobject[0]){\r\n                if (JSONobject[0].hasOwnProperty(key)){\r\n                    header.push(key);\r\n                }\r\n            }\r\n        }\r\n        CSVarray.push(header.join(','));\r\n\r\n        while(id--){\r\n            var line = [];\r\n            var h = header.length;\r\n            while(h--){// only do keys in header, in the header's order. //JSONobject[id]){\r\n                key = header[h];\r\n                if(JSONobject[id][key] !== undefined){\r\n                    if (Array.isArray(JSONobject[id][key])){\r\n                        //parse array here\r\n                        line.push(JSONobject[id][key].join(\"\\t\"));\r\n                    }else{\r\n                        line.push(JSONobject[id][key]);\r\n                    }\r\n                }\r\n            }line = line.reverse();\r\n            CSVarray.push(line.join(','));\r\n        }\r\n        var CSVstring = CSVarray.join(\"\\r\\n\");\r\n\r\n        return encodeURI(\"data:text/csv;charset=utf-8,\" + CSVstring);\r\n    };\r\n\r\n    $(\"#ExportCsvBtn\").click(function () {\r\n        var vocabList = StorageUtil.getVocList();\r\n        var CsvFile = createCSV(vocabList);\r\n        window.open(CsvFile);\r\n    });\r\n\r\n    $(\"#ExportCloseBtn\").click(function () {\r\n\t\t$(\"#export\").hide();\r\n\t\t$(\"#exportForm\")[0].reset();\r\n\t\t$(\"#exportArea\").text(\"\");\r\n\t\t$(\"#exportStatus\").text('Ready to export..');\r\n\t});\r\n\r\n    /** Import\r\n\t*/\r\n    window.WKSS_import = function () {\r\n        $(\"#import\").show();\r\n        //hide other windows\r\n        $(\"#add\").hide();\r\n        $(\"#export\").hide();\r\n        $(\"#edit\").hide();\r\n        $(\"#selfstudy\").hide();\r\n    };\r\n\r\n \tvar importWindowStructure = {\r\n\t\tid: \"WKSS-import\",\r\n\t\tclassName: \"WKSS\",\r\n\t\tchildNodes:[{\r\n\t\t\ttag: 'form',\r\n\t\t\tid: \"WKSS-importForm\",\r\n\t\t\tchildNodes:[\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'button',\r\n\t\t\t\t\tid: \"WKSS-importCloseBtn\",\r\n\t\t\t\t\tclassName: \"WKSS-close\",\r\n\t\t\t\t\tchildNodes:[{\r\n\t\t\t\t\t\ttag: 'i',\r\n\t\t\t\t\t\tclassName: \"icon-remove\"\r\n\t\t\t\t\t}]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'h1',\r\n\t\t\t\t\tchildNodes:[\"Import Items\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'textarea',\r\n\t\t\t\t\tid: \"importArea\",\r\n\t\t\t\t\tother: {cols: \"50\", rows: \"18\", placeholder: \"Paste your stuff and hit the import button! Use with caution!\"}\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'p', \r\n\t\t\t\t\tid: \"importStatus\",\r\n\t\t\t\t\tchildNodes:[\"Ready to import...\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'label',\r\n\t\t\t\t\tid: \"ImportItemsBtn\",\r\n\t\t\t\t\tclassName: \"button\",\r\n\t\t\t\t\tother: {type: \"button\", style: \"display:inline;\"},\r\n\t\t\t\t\tchildNodes:[\"Import Items\"]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'label',\r\n\t\t\t\t\tid: \"ImportCsvBtn\",\r\n\t\t\t\t\tclassName: \"button\",\r\n\t\t\t\t\tother: {style:\"display:inline; cursor: pointer;\"},\r\n\t\t\t\t\tchildNodes:[\"Import CSV\",\r\n\t\t\t\t\t\t{\r\n\t\t\t\t\t\t\ttag: 'input',\r\n\t\t\t\t\t\t\tid: \"upload\",\r\n\t\t\t\t\t\t\tother: {\r\n\t\t\t\t\t\t\t\ttype: \"file\", accept: \".csv, .tsv\",\r\n\t\t\t\t\t\t\t\tstyle: \"height:0px;width:0px;background:red;opacity:0;filter:opacity(1);\"\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t]\r\n\t\t\t\t},\r\n\t\t\t\t{\r\n\t\t\t\t\ttag: 'label',\r\n\t\t\t\t\tid: \"ImportWKBtn\",\r\n\t\t\t\t\tclassName: \"button\",\r\n\t\t\t\t\tother: {style: \"display:inline;\"},\r\n\t\t\t\t\tchildNodes:[\r\n\t\t\t\t\t\t{\r\n\t\t\t\t\t\t\ttag:'i',\r\n\t\t\t\t\t\t\tclassName: \"icon-download-alt\"\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\"WK\"\r\n\t\t\t\t\t]\r\n\t\t\t\t}\r\n\t\t\t]\r\n\t\t}]\r\n\t};\r\n\tvar importWindow = buildWindow(importWindowStructure);\r\n\r\n    $(\"body\").append(importWindow);\r\n   $(\"#import\").hide();\r\n\r\n\tvar checkAdd = function(add) {\r\n        //take a JSON object (parsed from import window) and check with stored items for any duplicates\r\n        // Returns true if each item in 'add' array is valid and\r\n        //at least one of them already exists in storage\r\n        var i = add.length;\r\n        if(localStorage.getItem('User-Vocab')) {    \r\n            var vocabList = StorageUtil.getVocList();\r\n            while(i--){\r\n                if (isItemValid(add[i]) &&\r\n                    checkForDuplicates(vocabList,add[i]))\r\n                    return true;\r\n            }\r\n        }\r\n        return false;\r\n    };\r\n\r\n    document.getElementById(\"upload\") && document.getElementById(\"upload\").addEventListener('change', ImportUtil.fileUpload, false);\r\n\r\n\tvar refreshLocks = function(){\r\n\t\tvar vocList = StorageUtil.getVocList().map(function(vocItem){\r\n\t\t\tdebugging&&console.log(\"vocList[i] = setLocks(vocList[i]);\");\r\n\t\t\tvocItem = setLocks(vocItem);  \r\n\t\t\treturn vocItem;\r\n\t\t}, this);\r\n\t\tconsole.groupEnd();\r\n\t\tStorageUtil.setVocList(vocList);\r\n    };\r\n\r\n    \r\n    $(\"#ImportWKBtn\").click(function(){\r\n        WanikaniUtil.getServerResp(APIkey,\"vocabulary\");\r\n        debugging&&console.log(\"maybe?\");\r\n    });\r\n\r\n    $(\"#ImportItemsBtn\").click(function () {\r\n        if ($(\"#importArea\").val().length !== 0) {\r\n            try {\r\n                var add = JSON.parse($(\"#importArea\").val().toLowerCase());\r\n                alert(JSON.stringify(add));\r\n                if (checkAdd(add)) {\r\n                    $(\"#importStatus\").text(\"No valid input (duplicates?)!\");\r\n                    return;\r\n                }\r\n\r\n                var newlist;\r\n                var srslist = [];\r\n                if (localStorage.getItem('User-Vocab')) {\r\n                    var vocabList = StorageUtil.getVocList();\r\n                    srslist = StorageUtil.getVocList();\r\n                    newlist = vocabList.concat(add);\r\n                }\r\n                else {\r\n                    newlist = add;\r\n\r\n\r\n                }\r\n                var i = add.length;\r\n                while(i--){\r\n                    StorageUtil.setVocItem(add[i]);\r\n                }\r\n\r\n                $(\"#importStatus\").text(\"Import successful!\");\r\n\r\n                $(\"#importForm\")[0].reset();\r\n                $(\"#importArea\").text(\"\");\r\n\r\n            }\r\n            catch (e) {\r\n                $(\"#importStatus\").text(\"Parsing Error!\");\r\n                debugging&&console.log(e);\r\n            }\r\n\r\n        }\r\n        else {\r\n            $(\"#importStatus\").text(\"Nothing to import :( Please paste your stuff first\");\r\n        }\r\n    });\r\n\r\n    $(\"#ImportCloseBtn\").click(function () {\r\n        $(\"#import\").hide();\r\n        $(\"#importForm\")[0].reset();\r\n        $(\"#importArea\").text(\"\");\r\n        $(\"#importStatus\").text('Ready to import..');\r\n    });\r\n\r\n    var playAudio = function() {\r\n        var kanji = document.getElementById('rev-kanji').innerHTML;\r\n        var kana = (document.getElementById('rev-solution').innerHTML.split(/[,、]+\\s*/))[0];\r\n\r\n        document.getElementById('rev-audio').innerHTML = \"\";\r\n        document.getElementById('audio-form').action = \"\";\r\n        //document.getElementById('AudioButton').disabled = true;\r\n\r\n        if( !kanji.match(/[a-zA-Z]+/i) && !kana.match(/[a-zA-Z]+/i)) {\r\n\r\n            kanji = encodeURIComponent(kanji);\r\n            kana = encodeURIComponent(kana);\r\n            var i;\r\n\r\n            var newkanji = \"\";\r\n            for(i = 1; i < kanji.length; i = i+3) {\r\n                newkanji = newkanji.concat(kanji[i-1]);\r\n                newkanji = newkanji.concat('2');\r\n                newkanji = newkanji.concat('5');\r\n                newkanji = newkanji.concat(kanji[i]);\r\n                newkanji = newkanji.concat(kanji[i+1]);\r\n            }\r\n\r\n            var newkana = \"\";\r\n            for(i = 1; i < kana.length; i = i+3) {\r\n                newkana = newkana.concat(kana[i-1]);\r\n                newkana = newkana.concat('2');\r\n                newkana = newkana.concat('5');\r\n                newkana = newkana.concat(kana[i]);\r\n                newkana = newkana.concat(kana[i+1]);\r\n            }\r\n\r\n            var url = \"http://www.csse.monash.edu.au/~jwb/audiock.swf?u=kana=\" + newkana + \"%26kanji=\" + newkanji;\r\n\r\n            debugging&&console.log(\"Audio URL: \" + url);\r\n\r\n            document.getElementById('AudioButton').disabled = false;\r\n\r\n            document.getElementById('rev-audio').innerHTML = url;\r\n\r\n        }\r\n\r\n    };\r\n\r\n    var nextReview = function(reviewList) {\r\n        //sets up the next item for review\r\n        //uses functions:\r\n        //    wanakana.bind/unbind\r\n\r\n        var rnd = Math.floor(Math.random()*reviewList.length);\r\n        var item = reviewList[rnd];\r\n        sessionSet('WKSS-item', JSON.stringify(item));\r\n        sessionSet('WKSS-rnd', rnd);\r\n        if (sessionStorage.getItem('User-Stats')){\r\n            $(\"#RevNum\").innerHtml = sessionGet('User-Stats').length;\r\n        }\r\n        document.getElementById('rev-kanji').innerHTML = item.prompt;\r\n        document.getElementById('rev-type').innerHTML = item.type;\r\n        var typeBgColor = 'grey';\r\n        if (item.type.toLowerCase() == 'meaning'){\r\n            typeBgColor = 'blue';\r\n        } else if (item.type.toLowerCase() == 'reading'){\r\n            typeBgColor = 'orange';\r\n        } else if (item.type.toLowerCase() == 'reverse'){\r\n            typeBgColor = 'orange';\r\n        }\r\n        document.getElementById('wkss-type').style.backgroundColor = typeBgColor;\r\n        $(\"#rev-solution\").removeClass(\"info\");\r\n        document.getElementById('rev-solution').innerHTML = item.solution;\r\n        document.getElementById('rev-index').innerHTML = item.index;\r\n\r\n        //initialise the input field\r\n        $(\"#rev-input\").focus();\r\n        $(\"#rev-input\").removeClass(\"caution\");\r\n        $(\"#rev-input\").removeClass(\"error\");\r\n        $(\"#rev-input\").removeClass(\"correct\");\r\n        $(\"#rev-input\").val(\"\");\r\n\r\n        //check for alphabet letters and decide to bind or unbind wanakana\r\n        if (item.solution[0].match(/[a-zA-Z]+/i)) {\r\n            wanakana.unbind(document.getElementById('rev-input'));\r\n            $('#rev-input').attr('placeholder','Your response');\r\n            $('#rev-input').attr('lang','en');\r\n\r\n        }\r\n        else {\r\n            wanakana.bind(document.getElementById('rev-input'));\r\n            $('#rev-input').attr('placeholder','答え');\r\n            $('#rev-input').attr('lang','ja');\r\n\r\n        }\r\n\r\n        playAudio();\r\n    };\r\n\r\n\t//global to keep track of when a review is in session.\r\n    var reviewActive = false;\r\n\r\n    var startReview = function() {\r\n        debugging&&console.log(\"startReview()\");\r\n        submit = true;\r\n        reviewActive = true;\r\n        //get the review 'list' from session storage, line up the first item in queue\r\n        var reviewList = sessionGet('User-Review')||[];\r\n        nextReview(reviewList);\r\n    };\r\n\r\n    /** Review Items\r\n\t*/\r\n    window.WKSS_review = function () {\r\n\r\n        //is there a session waiting in storage?\r\n        if(sessionStorage.getItem('User-Review')) {\r\n\r\n            //show the selfstudy window\r\n            $(\"#selfstudy\").show();\r\n\r\n            //hide other windows\r\n            $(\"#add\").hide();\r\n            $(\"#export\").hide();\r\n            $(\"#edit\").hide();\r\n            $(\"#import\").hide();\r\n\r\n            startReview();\r\n        }\r\n    };\r\n\r\n    $(\"body\").append('                                                          \\\r\n<div id=\"selfstudy\" class=\"WKSS\">\\\r\n<button id=\"SelfstudyCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Review<span id=\"RevNum\"></span></h1>\\\r\n<div id=\"wkss-kanji\">\\\r\n<span id=\"rev-kanji\"></span>\\\r\n</div><div id=\"wkss-type\">\\\r\n<span id=\"rev-type\"></span><br />\\\r\n</div><div id=\"wkss-solution\">\\\r\n<span id=\"rev-solution\"></span>\\\r\n</div><div id=\"wkss-input\">\\\r\n<input type=\"text\" id=\"rev-input\" size=\"40\" placeholder=\"\">\\\r\n</div><span id=\"rev-index\" style=\"display: block;\"></span>\\\r\n\\\r\n<form id=\"audio-form\">\\\r\n<label id=\"AudioButton\" class=\"button\">Play audio</label>\\\r\n<label id=\"WrapUpBtn\"   class=\"button\">Wrap Up</label>\\\r\n</form>\\\r\n<div id=\"rev-audio\" style=\"display:none;\"></div>\\\r\n</div>');\r\n    $(\"#selfstudy\").hide();\r\n\r\n    $(\"#SelfstudyCloseBtn\").click(function () {\r\n        $(\"#selfstudy\").hide();\r\n        $(\"#rev-input\").val(\"\");\r\n        reviewActive = false;\r\n    });\r\n\r\n    var binarySearch = function(values, target, start, end) {\r\n        //debugging&&console.log(\"binarySearch(values: ,target: , start: \"+start+\", end: \"+end+\")\");\r\n\r\n        if (start > end) {\r\n            //start has higher value than target, end has lower value\r\n            //item belongs between\r\n            // need to return 'start' with a flag that it hasn't been found\r\n            //invert sign :)\r\n            return -(start);\r\n\r\n\r\n            //for testing truths\r\n            //    return String(end)+\" < \"+item.index+\" < \"+String(start);\r\n\r\n        } //does not exist\r\n\r\n\r\n        var middle = Math.floor((start + end) / 2);\r\n        var value = values[middle];\r\n        /*debugging&&console.log(\"start.index: \"+values[start].index);\r\n     debugging&&console.log(\"middle.index: \"+values[middle].index);\r\n     debugging&&console.log(\"end.index: \"+values[end].index);\r\n     */\r\n        if (Number(value.index) > Number(target.index)) {\r\n\t\t\treturn binarySearch(values, target, start, middle-1);\r\n\t\t}\r\n        if (Number(value.index) < Number(target.index)) {\r\n\t\t\treturn binarySearch(values, target, middle+1, end);\r\n\t\t}\r\n        return middle; //found!\r\n    };\r\n\r\n\tvar findIndex = function(values, target) {\r\n        return binarySearch(values, target, 0, values.length - 1);\r\n\t};\r\n\r\n    $(\"#WrapUpBtn\").click(function() {\r\n        var sessionList = sessionGet('User-Review')||[];\r\n        var statsList = sessionGet('User-Stats')||[];\r\n        //if an index in sessionList matches one in statsList, don't delete\r\n        var sessionI = sessionList.length;\r\n        var item = sessionGet('WKSS-item')||[];\r\n        var arr2 = [];\r\n        //for every item in sessionList, look for index in statsList,\r\n        //if not there (-1) delete item from sessionList\r\n        while (sessionI--){\r\n            var index = findIndex(statsList,sessionList[sessionI]);\r\n            if ((Math.sign(1/index) !== -1)||(sessionList[sessionI].index == item.index)){\r\n\r\n                arr2.push(sessionList[sessionI]);\r\n            }\r\n        }\r\n        debugging&&console.log(arr2);\r\n        sessionSet('User-Review', JSON.stringify(arr2));\r\n    });\r\n\r\n    /** Save to list based on .index property\r\n\t* @param {Array.<task>} eList\r\n\t* @param {task} eItem\r\n\t*/\r\n    var saveToSortedList = function(eList,eItem){\r\n        var get = findIndex(eList,eItem);\r\n        if (Math.sign(1/get) === -1){\r\n            eList.splice(-get,0,eItem);\r\n        }\r\n\t\treturn eList;\r\n    };\r\n\r\n    //-------\r\n    var openInNewTab = function(url) {\r\n        var win=window.open(url, '_blank');\r\n        win.focus();\r\n    };\r\n\r\n    $(\"#AudioButton\").click(function () {\r\n        openInNewTab(document.getElementById('rev-audio').innerHTML);\r\n    });\r\n\r\n    var Rev_Item = function(prompt, kanji, type, solution, index){\r\n        this.prompt = prompt;\r\n        this.kanji = kanji;\r\n        this.type = type;\r\n        this.solution = solution;\r\n        this.index = index;\r\n    };\r\n\r\n    var updateSRS = function(stats, voclist) {\r\n        var now = Date.now();\r\n        if (voclist[stats.index].due < now){ //double check that the item was really up for review.\r\n            if(!stats.numWrong && voclist[stats.index].level < 9) {//all correct (none wrong)\r\n                voclist[stats.index].level++;\r\n            }\r\n            else {\r\n                stats.numWrong = {};\r\n                //Adapted from WaniKani's srs to authentically mimic level downs\r\n                var o = (stats.numWrong.Meaning||0)+(stats.numWrong.Reading||0)+(stats.numWrong.Reverse||0);\r\n                var t = voclist[stats.index].level;\r\n                var r=t>=5?2*Math.round(o/2):1*Math.round(o/2);\r\n                var n=t-r<1?1:t-r;\r\n\r\n                voclist[stats.index].level = n;//don't stay on 'started'\r\n\r\n            }\r\n            voclist[stats.index].date = now;\r\n            voclist[stats.index].due = now + srsObject[voclist[stats.index].level].duration;\r\n            console.log(\"Next review in \"+ms2str(srsObject[voclist[stats.index].level].duration));\r\n\r\n            return voclist;\r\n        }\r\n    };\r\n\r\n    var showResults = function() {\r\n\r\n        var statsList = sessionGet('User-Stats')||[];\r\n        sessionStorage.clear();\r\n\r\n        console.log(\"statslist\", statsList);\r\n        var voclist = StorageUtil.getVocList();\r\n        \r\n\t\tstatsList.forEach(function(stats, i, statsList){\r\n            debugging&&console.log(\"stats\",stats);\r\n            var altText = voclist[stats.index].level;//+stats.type;\r\n\r\n            if (stats.numWrong) {\r\n                if (stats.numWrong.Meaning)\r\n                    altText = altText + \" Meaning Wrong x\"+stats.numWrong.Meaning +\"\\n\";\r\n                if (stats.numWrong.Reading)\r\n                    altText = altText + \" Reading Wrong x\"+stats.numWrong.Reading +\"\\n\";\r\n                if (stats.numWrong.Reverse)\r\n                    altText = altText + \" Reverse Wrong x\"+stats.numWrong.Reverse +\"\\n\";\r\n\t\t\t}\r\n\t\t\tif (stats.numCorrect){\r\n\t\t\t\tif (stats.numCorrect.Meaning)\r\n\t\t\t\t\taltText = altText + \" Meaning Correct x\"+stats.numCorrect.Meaning +\"\\n\";\r\n\t\t\t\tif (stats.numCorrect.Reading)\r\n\t\t\t\t\taltText = altText + \" Reading Correct x\"+stats.numCorrect.Reading +\"\\n\";\r\n\t\t\t\tif (stats.numCorrect.Reverse)\r\n\t\t\t\t\taltText = altText + \" Reverse Correct x\"+stats.numCorrect.Reverse +\"\\n\";\r\n\t\t\t}\r\n            console.log(stats);\r\n\r\n\t\t\t//TODO sort into apprentice, guru, etc\r\n\t\t\tdocument.getElementById(\"stats-a\").innerHTML +=\r\n\t\t\t\t\"<span class=\" +\r\n\t\t\t\t(stats.numWrong? \"\\\"rev-error\\\"\":\"\\\"rev-correct\\\"\") +\r\n\t\t\t\t\" title='\"+altText+\"'>\" + stats.kanji + \"</span>\";\r\n\t\t\t\r\n\t\t\t//map with side effects?\r\n            statsList[i] = updateSRS(stats, voclist);\r\n\r\n        }, this);\r\n        sessionSet(\"User-Stats\",statsList);\r\n        localSet(\"User-Vocab\", voclist);\r\n    };\r\n\r\n    $(\"body\").append('                                                          \\\r\n<div id=\"resultwindow\" class=\"WKSS\">                                    \\\r\n<button id=\"ReviewresultsCloseBtn\" class=\"wkss-close\" type=\"button\"><i class=\"icon-remove\"></i></button>\\\r\n<h1>Review Results</h1>\\\r\n<h2>All</h2>\\\r\n<div id=\"stats-a\"></div>\\\r\n</div>');\r\n\r\n    $(\"#resultwindow\").hide();\r\n\r\n    $(\"#ReviewresultsCloseBtn\").click(function () {\r\n        $(\"#resultwindow\").hide();\r\n        document.getElementById(\"stats-a\").innerHTML = \"\";\r\n    });\r\n\r\n\tvar markAnswer = function(item) {\r\n        //evaluate 'item' against the question.\r\n        // match by index\r\n        // get type of question\r\n        // determine if right or wrong and return result appropriately\r\n\r\n        //get the question\r\n        //var prompt = document.getElementById('rev-kanji').innerHTML.trim();\r\n        //get the answer\r\n        var answer = $(\"#rev-input\").val().toLowerCase();\r\n        //get the index\r\n        var index = document.getElementById('rev-index').innerHTML.trim();\r\n        //get the question type\r\n        var type  = document.getElementById('rev-type').innerHTML.trim();\r\n\r\n        //var vocab = localGet(\"User-Vocab\");\r\n\r\n        //get the item if it is in the current session\r\n        var storedItem = sessionGet(item.index);\r\n        if (storedItem){\r\n\r\n            item.numCorrect = storedItem.numCorrect;\r\n            item.numWrong = storedItem.numWrong;\r\n        }\r\n\r\n        if (index == item.index){//-------------\r\n            if (inputCorrect()){\r\n                debugging&&console.log(answer+\"/\"+item.solution[0]);\r\n                if (!item.numCorrect){\r\n                    debugging&&console.log(\"initialising numCorrect\");\r\n                    item.numCorrect={};\r\n                }\r\n\r\n                debugging&&console.log(\"Correct: \"+ type);\r\n                if (type == \"Meaning\"){\r\n                    if (!item.numCorrect.Meaning)\r\n                        item.numCorrect.Meaning = 0;\r\n\r\n                    item.numCorrect.Meaning++;\r\n\r\n                }\r\n                if (type == \"Reading\"){\r\n                    if (!item.numCorrect.Reading)\r\n                        item.numCorrect.Reading = 0;\r\n\r\n                    item.numCorrect.Reading++;\r\n                }\r\n\r\n                if (type == \"Reverse\"){\r\n                    if (!item.numCorrect.Reverse)\r\n                        item.numCorrect.Reverse = 0;\r\n\r\n                    item.numCorrect.Reverse++;\r\n                }\r\n\r\n            }else{\r\n                debugging&&console.log(answer+\"!=\"+item.solution);\r\n                if (!item.numWrong){\r\n                    debugging&&console.log(\"initialising numCorrect\");\r\n                    item.numWrong={};\r\n                }\r\n\r\n                debugging&&console.log(\"Wrong: \"+ type);\r\n                if (type == \"Meaning\"){\r\n                    if (!item.numWrong.Meaning)\r\n                        item.numWrong.Meaning = 0;\r\n\r\n                    item.numWrong.Meaning++;\r\n\r\n                }\r\n                if (type == \"Reading\"){\r\n                    if (!item.numWrong.Reading)\r\n                        item.numWrong.Reading = 0;\r\n\r\n                    item.numWrong.Reading++;\r\n\r\n                }\r\n                if (type == \"Reverse\"){\r\n                    if (!item.numWrong.Reverse)\r\n                        item.numWrong.Reverse = 0;\r\n\r\n                    item.numWrong.Reverse++;\r\n                }\r\n            }\r\n        }\r\n\t\telse {\r\n            console.error(\"Error: indexes don't match\");\r\n        }\r\n        return item;\r\n    };\r\n\r\n\t\r\n    //jquery keyup event\r\n    $(\"#rev-input\").keyup(ReviewUtil.submitResponse);\r\n\r\n\r\n    /** Adds the Button\r\n\t*/\r\n    var addUserVocabButton = function() {\r\n        debugging&&console.log(\"addUserVocabButton()\");\r\n        //Functions (indirect)\r\n        //    WKSS_add()\r\n        //    WKSS_edit()\r\n        //    WKSS_export()\r\n        //    WKSS_import()\r\n        //    WKSS_lock()\r\n        //    WKSS_review()\r\n\r\n        var nav = document.getElementsByClassName('nav');\r\n        debugging&&console.log(\"generating review list because: initialising script and populating reviews\");\r\n\r\n\r\n        if (nav&&nav.length>2) {\r\n            nav[2].innerHTML = nav[2].innerHTML + \"\\n\\\r\n<li class=\\\"dropdown custom\\\">\\n\\\r\n<a class=\\\"dropdown-toggle custom\\\" data-toggle=\\\"dropdown\\\" href=\\\"#\\\" onclick=\\\"generateReviewList();\\\">\\n\\\r\n<span lang=\\\"ja\\\">自習</span>\\n\\\r\nSelf-Study <i class=\\\"icon-chevron-down\\\"></i>\\n\\\r\n</a>\\n\\\r\n<ul class=\\\"dropdown-menu\\\" id=\\\"WKSS_dropdown\\\">\\n\\\r\n<li class=\\\"nav-header\\\">Customize</li>\\n\\\r\n<li><a id=\\\"click\\\" href=\\\"#\\\" onclick=\\\"WKSS_add();\\\">Add</a></li>\\n\\\r\n<li><a href=\\\"#\\\" onclick=\\\"WKSS_edit();\\\">Edit</a></li>\\n\\\r\n<li><a href=\\\"#\\\" onclick=\\\"WKSS_export();\\\">Export</a></li>\\n\\\r\n<li><a href=\\\"#\\\" onclick=\\\"WKSS_import();\\\">Import</a></li>\\n\\\r\n<!--//   <li><a href=\\\"#\\\" onclick=\\\"WKSS_lock();\\\">Server Settings</a></li>//-->\\n\\\r\n<li class=\\\"nav-header\\\">Learn</li>\\n\\\r\n<li><a id=\\\"user-review\\\" href=\\\"#\\\" onclick=\\\"WKSS_review();\\\">Please wait...</a></li>\\n\\\r\n</ul>\\n\\\r\n</li>\";\r\n\r\n\r\n        }else{\r\n            console.error(\"could not find nav\", nav);\r\n        }\r\n        console.log(\"addUserVocab\");\r\n    };\r\n\r\n\t    /** Error handling\r\n\t* Can use 'error.stack', not cross-browser (though it should work on Firefox and Chrome)\r\n\t*/\r\n    var logError = function(error) {\r\n        debugging&&console.log(\"logError(error)\");\r\n        var stackMessage = \"\";\r\n        if (\"stack\" in error)\r\n            stackMessage = \"\\n\\tStack: \" + error.stack;\r\n\r\n        debugging&&console.log(\"WKSS: Error: \" + error.name + \"\\n\\tMessage: \" + error.message + stackMessage);\r\n        console.error(\"WKSS: Error: \" + error.name + \"\\n\\tMessage: \" + error.message + stackMessage);\r\n    };\r\n\r\n\r\n    /**  Prepares the script\r\n\t*/\r\n    var scriptInit = function() {\r\n        debugging&&console.log(\"scriptInit()\");\r\n        //functions:\r\n        //    addUserVocabButton()\r\n        //    logError(err)\r\n\r\n        debugging&&console.log(\"Initializing Wanikani UserVocab Script!\");\r\n\r\n        gM_addStyle(\".custom .dropdown-menu {background-color: #DBA901 !important;}\");\r\n        gM_addStyle(\".custom .dropdown-menu:after {border-bottom-color: #DBA901 !important;\");\r\n        gM_addStyle(\".custom .dropdown-menu:before {border-bottom-color: #DBA901 !important;\");\r\n        gM_addStyle(\".open .dropdown-toggle.custom {background-color: #FFC400 !important;}\");\r\n        gM_addStyle(\".custom .dropdown-menu a:hover {background-color: #A67F00 !important;}\");\r\n        gM_addStyle(\".custom:hover {color: #FFC400 !important;}\");\r\n        gM_addStyle(\".custom:hover span {border-color: #FFC400 !important;}\");\r\n        gM_addStyle(\".custom:focus {color: #FFC400 !important;}\");\r\n        gM_addStyle(\".custom:focus span {border-color: #FFC400 !important;}\");\r\n        gM_addStyle(\".open .custom span {border-color: #FFFFFF !important;}\");\r\n        gM_addStyle(\".open .custom {color: #FFFFFF !important}\");\r\n\r\n\t\tvar wkStyleCSS = require('./wkstyle.js');\r\n        gM_addStyle(wkStyleCSS);\r\n        // Set up buttons\r\n        try {\r\n            if (typeof localStorage !== \"undefined\") {\r\n                addUserVocabButton();\r\n\r\n                //provide warning to users trying to use the (incomplete) script.\r\n                debugging&&console.log(\"this script is still incomplete: \\n\\\r\nIt is provided as is without warranty express or implied\\n\\\r\nin the hope that you may find it useful.\");\r\n            }\r\n            else {\r\n                debugging&&console.log(\"Wanikani Self-Study: Your browser does not support localStorage.. Sorry :(\");\r\n            }\r\n        }\r\n        catch (err) {\r\n            logError(err);\r\n        }\r\n    };\r\n\r\n\tconsole.info(document.readyState);\r\n    console.log(\"adding DOM listener\", document.readyState);\r\n    // Check for file API support.\r\n    if (window.File && window.FileReader && window.FileList && window.Blob) {\r\n\r\n    }\r\n\telse {\r\n        alert('The File APIs are not fully supported in this browser.');\r\n    }\r\n\r\n    /** Start the script\r\n    */\r\n    //unless the user navigated from the review directory, they are unlikely to have unlocked any kanji\r\n    var noNewStuff = /^https:\\/\\/.*\\.wanikani\\.com\\/.*/.test(document.referrer)&&!(/https:\\/\\/.*\\.wanikani\\.com\\/review.*/.test(document.referrer));\r\n    var usingHTTPS = /^https:/.test(window.location.href);\r\n    console.info(usingHTTPS, window.location.href);\r\n    if (usingHTTPS){\r\n        if (!noNewStuff){  //Don't waste time if user is browsing site\r\n            WanikaniUtil.getServerResp(APIkey);\r\n        }else{\r\n            debugging&&console.log(\"User is unlikely to have new kanji unlocked\");\r\n        }\r\n        debugging&&console.info(\"WaniKani Self-Study Plus is about to start\");\r\n\r\n        scriptInit();\r\n\r\n    }else{\r\n        debugging&&console.warn(\"It appears that you are not using https protocol. Attempting to redirect to https now.\");\r\n        window.location.href = window.location.href.replace(/^http/, \"https\");\r\n    }\r\n}\r\nif (document.readyState === 'complete'){\r\n    console.info(\"About to initialise WKSS+\");\r\n    main();\r\n} else {\r\n    window.addEventListener(\"load\", main, false);\r\n}\r\n","/** Utilities for interaction with the Wanikani API and general website.\r\n*/\r\nvar WanikaniUtil = {\r\n\thijackRequests: require('./hijackrequests.js'),\r\n\tcreateCORSRequest: function(method, url){\r\n        var xhr = new XMLHttpRequest();\r\n        if (\"withCredentials\" in xhr){\r\n            xhr.open(method, url, true);\r\n        }\r\n\t\telse if (typeof XDomainRequest !== \"undefined\"){\r\n            xhr = new XDomainRequest();\r\n            xhr.open(method, url);\r\n        }\r\n\t\telse {\r\n            xhr = null;\r\n        }\r\n        return xhr;\r\n    },\r\n\t/** Gets the user information using the Wanikani API and stores them directly into browser storage.\r\n\t* @param\r\n\t* @param\r\n\t*/\r\n\tgetServerResp: function(APIkey, requestedItem){\r\n\r\n        requestedItem = requestedItem === void 0 ? 'kanji' :requestedItem;\r\n\r\n        if (APIkey !== \"test\"){\r\n            var levels = (requestedItem ===\"kanji\")? \"/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\":\r\n            \"/1,2,3,4,5,6,7,8,9,10\";\r\n            var xhrk = this.createCORSRequest(\"get\", \"https://www.wanikani.com/api/user/\" + APIkey + \"/\" + requestedItem + levels);\r\n            if (!isEmpty(xhrk)){\r\n                xhrk.onreadystatechange = function() {\r\n                    if (xhrk.readyState == 4){\r\n                        var kanjiList = this.handleReadyStateFour(xhrk,requestedItem);\r\n\r\n                        if (requestedItem === 'kanji'){\r\n                            localSet('User-KanjiList', kanjiList);\r\n                            console.log(\"kanjiList from server\", kanjiList);\r\n                            //update locks in localStorage \r\n                            //pass kanjilist into this function\r\n                            //(don't shift things through storage unecessarily)\r\n                            refreshLocks();\r\n                        }\r\n\t\t\t\t\t\telse{\r\n                            var v = kanjiList.length;\r\n                            console.log(v + \" items found, attempting to import\");\r\n                            while (v--){\r\n                                StorageUtil.setVocItem(kanjiList[v]);\r\n                            }\r\n                        }\r\n                    }\r\n                };\r\n\r\n                xhrk.send();\r\n                console.log(\"below\");  \r\n            }\r\n        }\r\n\t\telse {\r\n            //dummy server response for testing.\r\n            setTimeout(function () {\r\n                var kanjiList = [];\r\n                console.log(\"creating dummy response\");\r\n                kanjiList.push({\"character\": \"猫\", \"srs\": \"noServerResp\"});\r\n                var SRS = \"apprentice\"; //prompt(\"enter SRS for 子\", \"guru\");\r\n                kanjiList.push({\"character\": \"子\", \"srs\": SRS});\r\n                kanjiList.push({\"character\": \"品\", \"srs\": \"guru\"});\r\n                kanjiList.push({\"character\": \"供\", \"srs\": \"guru\"});\r\n                kanjiList.push({\"character\": \"本\", \"srs\": \"guru\"});\r\n                kanjiList.push({\"character\": \"聞\", \"srs\": \"apprentice\"});\r\n                kanjiList.push({\"character\": \"人\", \"srs\": \"enlightened\"});\r\n                kanjiList.push({\"character\": \"楽\", \"srs\": \"burned\"});\r\n                kanjiList.push({\"character\": \"相\", \"srs\": \"guru\"});\r\n                kanjiList.push({\"character\": \"卒\", \"srs\": \"noMatchWK\"});\r\n                kanjiList.push({\"character\": \"無\", \"srs\": \"noMatchGuppy\"});\r\n\r\n                console.log(\"Server responded with dummy kanjiList: \\n\"+JSON.stringify(kanjiList));\r\n\r\n                localSet('User-KanjiList', kanjiList);\r\n\r\n                //update locks in localStorage\r\n                refreshLocks();\r\n            }, 10000);\r\n        }   \r\n    },\r\n\t\r\n\thandleReadyStateFour: function(xhrk, requestedItem){\r\n\r\n        var localkanjiList = [];\r\n        console.log(\"readystate: \"+ xhrk.readyState);\r\n        var resp = JSON.parse(xhrk.responseText);\r\n        console.log(\"about to loop through requested information\"); \r\n\t\tif (resp.requested_information && resp.requested_information.length){\r\n\t\t\t\r\n\t\t\tlocalkanjiList = resp.requested_information.map(function(requestedTask){\r\n\t\t\t\tif (requestedItem === \"kanji\"){\r\n\t\t\t\t\tif (requestedTask.user_specific !== null){\r\n\t\t\t\t\t\treturn {\r\n\t\t\t\t\t\t\tcharacter: requestedTask.character,\r\n\t\t\t\t\t\t\tsrs: requestedTask.user_specific.srs,\r\n\t\t\t\t\t\t\treading: requestedTask[requestedTask.important_reading].split(\",\")[0],\r\n\t\t\t\t\t\t\tmeaning: requestedTask.meaning.split(\",\")[0]\r\n\t\t\t\t\t\t};\r\n\t\t\t\t\t}\r\n\t\t\t\t\telse{\r\n\t\t\t\t\t\treturn {\r\n\t\t\t\t\t\t\tcharacter: requestedTask.character,\r\n\t\t\t\t\t\t\tsrs: \"unreached\"\r\n\t\t\t\t\t\t};\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t\telse if(requestedItem === \"vocabulary\"){\r\n\t\t\t\t\tif (requestedTask.user_specific !== null||true){ //--\r\n\t\t\t\t\t\t//build vocablist\r\n\t\t\t\t\t\treturn {\r\n\t\t\t\t\t\t\tkanji: requestedTask.character,\r\n\t\t\t\t\t\t\treading: requestedTask.kana.split(\",\"),\r\n\t\t\t\t\t\t\tmeaning: requestedTask.meaning.split(\",\")\r\n\t\t\t\t\t\t};\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}, this);\r\n\t\t}\r\n        //return kanjiList\r\n        //  console.log(\"Server responded with new kanjiList: \\n\"+JSON.stringify(kanjiList));\r\n        return localkanjiList;\r\n    }\r\n\r\n};\r\n\r\nmodule.exports = WanikaniUtil;","// Window Configs\r\nmodule.exports = {\r\n\tadd:{height: \"300px\", width: \"300px\"},\r\n\texportImport:{height: \"275px\", width: \"390px\"},\r\n\tedit:{height: \"380px\", width: \"800px\"},\r\n\tstudy:{height: \"auto\", width: \"600px\"}, //height : auto\r\n\tresult:{height: \"500px\", width: \"700px\"}\r\n};","/* jshint multistr: true */\r\n// Config for window sizes in pixels\r\nvar windowConfig = require('./windowconfig.js');\r\n\r\n//.WKSS\r\nvar classWKSS = [\r\n\t{position: \"fixed\"},\r\n\t{zIndex: \"2\"},\r\n\t{top: \"125px\"},\r\n\t{left: \"50%\"},\r\n\t{margin: \"0px\"},\r\n\t{background: \"#FFF\"},\r\n\t{padding: \"5px\"},\r\n\t{font: \"12px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"},\r\n\t{color: \"#888\"},\r\n\t{textShadow: \"1px 1px 1px #FFF\"},\r\n\t{border: \"1px solid #DDD\"},\r\n\t{borderRadius: \"5px\"},\r\n\t{WebkitBorderRadius: \"5px\"},\r\n\t{MozBorderRadius: \"5px\"},\r\n\t{boxShadow: \"10px 10px 5px #888888\"}\r\n];\r\n\r\n//.WKSS h1\r\nvar classWKSS_h1 = [\r\n\t{font: \"25px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"},\r\n\t{paddingLeft: \"5px\"},\r\n\t{display: \"block\"},\r\n\t{borderBottom: \"1px solid #DADADA\"},\r\n\t{margin: \"0px\"},\r\n\t{color: \"#888\"}\r\n];\r\n\r\n//.WKSS h1 > span\r\nvar classWKSS_h1_direct_Span = [\r\n\t{display: \"block\"},\r\n\t{fontSize: \"11px\"}\r\n];\r\n\r\n//.WKSS label\r\nvar w = [\r\n\t{display: \"block\"},\r\n\t{margin: \"0px 0px 5px\"},\r\n];\r\n\r\n//.WKSS label>span\r\nw = [\r\n\t{float: \"left\"},\r\n\t{width: \"80px\"},\r\n\t{textAlign: \"right\"},\r\n\t{paddingRight: \"10px\"},\r\n\t{marginTop: \"10px\"},\r\n\t{color: \"#333\"},\r\n\t{fontFamily: \"\\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"},\r\n\t{fontWeight: \"bold\"}\r\n];\r\n\r\n//.WKSS input[type=\\\"text\\\"], .WKSS input[type=\\\"email\\\"], .WKSS textarea \r\nw = [\r\n\t{border: \"1px solid #CCC\"},\r\n\t{color: \"#888\"},\r\n\t{height: \"20px\"},\r\n\t{marginBottom: \"16px\"},\r\n\t{marginRight: \"6px\"},\r\n\t{marginTop: \"2px\"},\r\n\t{outline: \"0 none\"},\r\n\t{padding: \"6px 12px\"},\r\n\t{width: \"80%\"},\r\n\t{borderRadius: \"4px\"},\r\n\t{lineHeight: \"normal !important\"},\r\n\t{WebkitBorderRadius: \"4px\"},\r\n\t{MozBorderRadius: \"4px\"},\r\n\t{font: \"normal 14px/14px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"},\r\n\t{WebkitBoxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"},\r\n\t{boxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"},\r\n\t{MozBoxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"}\r\n];\r\n\r\n//.WKSS select\r\nw = [\r\n\t{border: \"1px solid \\\"#CCC\\\"\"},\r\n\t{color: \"#888\"},\r\n\t{outline: \"0 none\"},\r\n\t{padding: \"6px 12px\"},\r\n\t{height: \"160px !important\"},\r\n\t{width: \"95%\"},\r\n\t{borderRadius: \"4px\"},\r\n\t{WebkitBorderRadius: \"4px\"},\r\n\t{MozBorderRadius: \"4px\"},\r\n\t{font: \"normal 14px/14px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"},\r\n\t{WebkitBoxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"},\r\n\t{boxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"},\r\n\t{MozBoxShadow: \"inset 0 1px 1px rgba(0, 0, 0, 0.075)\"},\r\n\t{background: \"#FFF url('down-arrow.png') no-repeat right\"},\r\n\t{appearance: \"none\"},\r\n\t{WebkitAppearance: \"none\"},\r\n\t{MozAppearance: \"none\"},\r\n\t{textIndent: \"0.01px\"},\r\n\t{textOverflow: \"''\"}\r\n];\r\n\r\n//.WKSS textarea\r\nw = [\r\n\t{height: \"100px\"}\r\n];\r\n\r\n//.WKSS button, .button\r\nw = [\r\n\t{position: \"relative\"},\r\n\t{background: \"#FFF\"},\r\n\t{border: \"1px solid #CCC\"},\r\n\t{padding: \"10px 25px 10px 25px\"},\r\n\t{color: \"#333\"},\r\n\t{borderRadius: \"4px\"},\r\n\t{display: \"inline !important\"}\r\n];\r\n\r\n//.WKSS button:disabled\r\nw = [\r\n\t{background: \"#EBEBEB\"},\r\n\t{border: \"1px solid #CCC\"},\r\n\t{padding: \"10px 25px 10px 25px\"},\r\n\t{color: \"#333\"},\r\n\t{borderRadius: \"4px\"}\r\n];\r\n\r\n//.WKSS .button:hover, button:hover:enabled\r\nw = [\r\n\t{color: \"#333\"},\r\n\t{backgroundColor: \"#EBEBEB\"},\r\n\t{borderColor: \"#ADADAD\"}\r\n];\r\n\r\n//.WKSS button:hover:disabled\r\nw = [\r\n\t{cursor: \"default\"}                             \r\n];\r\n\r\n//.error\r\nw = [\r\n\t{borderColor:\"#F00 !important\"},\r\n\t{color: \"#F00 !important\"}\r\n];\r\n\r\n//.caution\r\nw = [\r\n\t{borderColor: \"#F90 !important\"},\r\n\t{color: \"#F90 !important\"}\r\n];\r\n\r\n//.correct\r\nw = [\r\n\t{borderColor: \"#0F0 !important\"},\r\n\t{color: \"#0F0 !important\"}\r\n];\r\n\r\n//.info\r\nw = [\r\n\t{borderColor: \"#696969 !important\"},\r\n\t{color: \"#696969 !important\"}\r\n];\r\n\r\n//.rev-error\r\nw = [\r\n\t{textShadow: \"none\"},\r\n\t{border: \"1px solid #F00 !important\"},\r\n\t{borderRadius: \"10px\"},\r\n\t{backgroundColor: \"#F00\"},\r\n\t{padding: \"4px\"},\r\n\t{margin: \"4px\"},\r\n\t{color: \"#FFFFFF\"},\r\n\t{font: \"normal 18px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"}\r\n];\r\n\r\n//.rev-correct\r\nw = [\r\n\t{textShadow:\"none\"},\r\n\t{border: \"1px\"},\r\n\t{solid: \"#088A08 !important\"},\r\n\t{borderRadius: \"10px\"},\r\n\t{backgroundColor: \"#088A08\"},\r\n\t{padding: \"4px\"},\r\n\t{margin:\"4px\"},\r\n\t{color: \"#FFFFFF\"},\r\n\t{font: \"normal 18px \\\"ヒラギノ角ゴ Pro W3\\\", \\\"Hiragino Kaku Gothic Pro\\\",Osaka, \\\"メイリオ\\\", Meiryo, \\\"ＭＳ Ｐゴシック\\\", \\\"MS PGothic\\\", sans-serif\"}\r\n];\r\n\r\n//#add\r\nw = [\r\n\t{width: windowConfig.add.width},\r\n\t{height: windowConfig.add.height},\r\n\t{marginLeft: -windowConfig.add.width/2}\r\n];\r\n\r\n//#export, #import\r\nw = [\r\n{background: \"#fff\"},\r\n\t{width: windowConfig.exportImport.width},\r\n\t{height: windowConfig.exportImport.height},\r\n\t{marginLeft: -windowConfig.exportImport.width/2}\r\n];\r\n\r\n//#edit\r\nw = [\r\n\t{width: windowConfig.edit.width},\r\n\t{height: windowConfig.edit.height},\r\n\t{marginLeft: -windowConfig.edit.width/2}\r\n];\r\n\r\n//#selfstudy\r\nw = [\r\n\t{left: \"50%\"},\r\n\t{width: windowConfig.study.width},\r\n\t{height: windowConfig.study.height},\r\n\t{marginLeft: -windowConfig.study.width/2}\r\n];\r\n\r\n//#resultwindow\r\nw = [\r\n\t{left:\"50%\"},\r\n\t{width: windowConfig.result.width + \"px\"},\r\n\t{height: windowConfig.result.height + \"px\"},\r\n\t{marginLeft: -windowConfig.result.width/2 + \"px\"}\r\n];\r\n\r\n//#AudioButton\r\nw = [\r\n\t{marginTop: \"35px\"},\r\n\t{position: \"relative\"},\r\n\t{display: \"inline !important\"},\r\n\t{WebkitMarginBefore: \"50px\"}\r\n];\r\n\r\n//button.wkss-close\r\nw = [\r\n\t{float: \"right\"},\r\n\t{backgroundColor: \"#ff4040\"},\r\n\t{color: \"#fff\"},\r\n\t{padding: \"0px\"},\r\n\t{height: \"27px\"},\r\n\t{width: \"27px\"}\r\n];\r\n\r\n//#wkss-close\r\nw = [\r\n\t{float: \"right\"},\r\n\t{backgroundColor: \"#ff4040\"},\r\n\t{color: \"#fff\"},\r\n\t{padding: \"0px\"},\r\n\t{height: \"27px\"},\r\n\t{width: \"27px\"}\r\n];\r\n\r\n//#wkss-kanji, #rev-kanji\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"50px !important\"},\r\n\t{backgroundColor: \"#9400D3 !important\"},\r\n\t{color: \"#FFFFFF !important\"},\r\n\t{borderRadius: \"10px 10px 0px 0px\"}\r\n];\r\n\r\n//#wkss-solution, #rev-solution\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"30px !important\"},\r\n\t{color: \"#FFFFFF\"},\r\n\t{padding: \"2px\"}\r\n];\r\n\r\n//#wkss-type\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"24px !important\"},\r\n\t{backgroundColor: \"#696969\"},\r\n\t{color: \"#FFFFFF !important\"},\r\n\t{borderRadius: \"0px 0px 10px 10px\"}\r\n];\r\n\r\n//#rev-type\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"24px !important\"},\r\n\t{color: \"#FFFFFF !important\"},\r\n\t{borderRadius: \"0px 0px 10px 10px\"}\r\n];\r\n//#wkss-input\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"40px !important\"},\r\n\t{height: \"80px !important\"},\r\n\t{lineHeight: \"normal !important\"}\r\n];\r\n\r\n//#rev-input\r\nw = [\r\n\t{textAlign: \"center !important\"},\r\n\t{fontSize: \"40px !important\"},\r\n\t{height: \"60px !important\"},\r\n\t{lineHeight: \"normal !important\"}\r\n];\r\n\r\n//----\r\nmodule.exports = classWKSS;\r\n//module.exports = wkstyleCSS;"]}