describe('MarkdownEditingDescriptor', function() {
describe('save stores the correct data', function() {
it('saves markdown from markdown editor', function() {
loadFixtures('problem-with-markdown.html');
this.descriptor = new MarkdownEditingDescriptor($('.problem-editor'));
const saveResult = this.descriptor.save();
expect(saveResult.metadata.markdown).toEqual('markdown');
expect(saveResult.data).toXMLEqual('\n
markdown
\n');
});
it('clears markdown when xml editor is selected', function() {
loadFixtures('problem-with-markdown.html');
this.descriptor = new MarkdownEditingDescriptor($('.problem-editor'));
this.descriptor.createXMLEditor('replace with markdown');
const saveResult = this.descriptor.save();
expect(saveResult.nullout).toEqual(['markdown']);
expect(saveResult.data).toEqual('replace with markdown');
});
it('saves xml from the xml editor', function() {
loadFixtures('problem-without-markdown.html');
this.descriptor = new MarkdownEditingDescriptor($('.problem-editor'));
const saveResult = this.descriptor.save();
expect(saveResult.nullout).toEqual(['markdown']);
expect(saveResult.data).toEqual('xml only');
});
});
describe('advanced editor opens correctly', () =>
it('click on advanced editor should work', function() {
loadFixtures('problem-with-markdown.html');
this.descriptor = new MarkdownEditingDescriptor($('.problem-editor'));
spyOn(this.descriptor, 'confirmConversionToXml').and.returnValue(true);
expect(this.descriptor.confirmConversionToXml).not.toHaveBeenCalled();
const e = jasmine.createSpyObj('e', [ 'preventDefault' ]);
this.descriptor.onShowXMLButton(e);
expect(e.preventDefault).toHaveBeenCalled();
expect(this.descriptor.confirmConversionToXml).toHaveBeenCalled();
expect($('.editor-bar').length).toEqual(0);
})
);
describe('insertMultipleChoice', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertMultipleChoice('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.multipleChoiceTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertMultipleChoice('foo\nbar');
expect(revisedSelection).toEqual('( ) foo\n( ) bar\n');
});
it('recognizes x as a selection if there is non-whitespace after x', function() {
const revisedSelection = MarkdownEditingDescriptor.insertMultipleChoice('a\nx b\nc\nx \nd\n x e');
expect(revisedSelection).toEqual('( ) a\n(x) b\n( ) c\n( ) x \n( ) d\n(x) e\n');
});
it('recognizes x as a selection if it is first non whitespace and has whitespace with other non-whitespace', function() {
const revisedSelection = MarkdownEditingDescriptor.insertMultipleChoice(' x correct\n x \nex post facto\nb x c\nx c\nxxp');
expect(revisedSelection).toEqual('(x) correct\n( ) x \n( ) ex post facto\n( ) b x c\n(x) c\n( ) xxp\n');
});
it('removes multiple newlines but not last one', function() {
const revisedSelection = MarkdownEditingDescriptor.insertMultipleChoice('a\nx b\n\n\nc\n');
expect(revisedSelection).toEqual('( ) a\n(x) b\n( ) c\n');
});
});
describe('insertCheckboxChoice', function() {
// Note, shares code with insertMultipleChoice. Therefore only doing smoke test.
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertCheckboxChoice('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.checkboxChoiceTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertCheckboxChoice('foo\nbar');
expect(revisedSelection).toEqual('[ ] foo\n[ ] bar\n');
});
});
describe('insertStringInput', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertStringInput('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.stringInputTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertStringInput('my text');
expect(revisedSelection).toEqual('= my text');
});
});
describe('insertNumberInput', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertNumberInput('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.numberInputTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertNumberInput('my text');
expect(revisedSelection).toEqual('= my text');
});
});
describe('insertSelect', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertSelect('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.selectTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertSelect('my text');
expect(revisedSelection).toEqual('[[my text]]');
});
});
describe('insertHeader', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertHeader('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.headerTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertHeader('my text');
expect(revisedSelection).toEqual('my text\n====\n');
});
});
describe('insertExplanation', function() {
it('inserts the template if selection is empty', function() {
const revisedSelection = MarkdownEditingDescriptor.insertExplanation('');
expect(revisedSelection).toEqual(MarkdownEditingDescriptor.explanationTemplate);
});
it('wraps existing text', function() {
const revisedSelection = MarkdownEditingDescriptor.insertExplanation('my text');
expect(revisedSelection).toEqual('[explanation]\nmy text\n[explanation]');
});
});
describe('markdownToXml', function() {
it('converts raw text to paragraph', function() {
const data = MarkdownEditingDescriptor.markdownToXml('foo');
expect(data).toXMLEqual('\n
foo
\n');
});
// test default templates
it('converts numerical response to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`A numerical response problem accepts a line of text input from the student, and evaluates the input for correctness based on its numerical value.
The answer is correct if it is within a specified numerical tolerance of the expected answer.
Enter the numerical value of Pi:
= 3.14159 +- .02
Enter the approximate value of 502*9:
= 502*9 +- 15%
Enter the approximate number of atoms in a mol:
= 6.022e23 +- 10%
Enter the number of fingers on a human hand:
= 5
Range tolerance case
= [6, 7]
= (1, 2)
If first and last symbols are not brackets, or they are not closed, stringresponse will appear.
= (7), 7
= (1+2
Just input 100 test. Stringresponse will appear:
= 100 test
[Explanation]
Pi, or the the ratio between a circle's circumference to its diameter, is an irrational number known to extreme precision. It is value is approximately equal to 3.14.
Although you can get an exact value by typing 502*9 into a calculator, the result will be close to 500*10, or 5,000. The grader accepts any response within 15% of the true value, 4518, so that you can use any estimation technique that you like.
If you look at your hand, you can count that you have five fingers.
[Explanation]\
`);
expect(data).toXMLEqual(`
A numerical response problem accepts a line of text input from the student, and evaluates the input for correctness based on its numerical value.
The answer is correct if it is within a specified numerical tolerance of the expected answer.
Enter the numerical value of Pi:
Enter the approximate value of 502*9:
Enter the approximate number of atoms in a mol:
Enter the number of fingers on a human hand:
Range tolerance case
If first and last symbols are not brackets, or they are not closed, stringresponse will appear.
Just input 100 test. Stringresponse will appear:
Explanation
Pi, or the the ratio between a circle's circumference to its diameter, is an irrational number known to extreme precision. It is value is approximately equal to 3.14.
Although you can get an exact value by typing 502*9 into a calculator, the result will be close to 500*10, or 5,000. The grader accepts any response within 15% of the true value, 4518, so that you can use any estimation technique that you like.
If you look at your hand, you can count that you have five fingers.
`);
});
it('will convert 0 as a numerical response (instead of string response)', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Enter 0 with a tolerance:
= 0 +- .02\
`);
expect(data).toXMLEqual(`
Enter 0 with a tolerance:
`);
});
it('markup with additional answer does not break numerical response', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Enter 1 with a tolerance:
= 1 +- .02
or= 2\
`);
expect(data).toXMLEqual(`
Enter 1 with a tolerance:
`
);
});
it('markup for numerical with multiple additional answers renders correctly', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Enter 1 with a tolerance:
= 1 +- .02
or= 2
or= 3\
`);
expect(data).toXMLEqual(`
Enter 1 with a tolerance:
`
);
});
it('Do not render ranged/tolerance/alphabetical additional answers for numerical response', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Enter 1 with a tolerance:
= 1 +- .02
or= 2
or= 3 +- 0.1
or= [4,6]
or= ABC
or= 7\
`);
expect(data).toXMLEqual(`
Enter 1 with a tolerance:
`
);
});
it('markup with feedback renders correctly in additional answer for numerical response', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Enter 1 with a tolerance:
= 100 +- .02 {{ main feedback }}
or= 10 {{ additional feedback }}\
`);
expect(data).toXMLEqual(`
Enter 1 with a tolerance:
additional feedbackmain feedback`
);
});
it('converts multiple choice to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`A multiple choice problem presents radio buttons for student input. Students can only select a single option presented. Multiple Choice questions have been the subject of many areas of research due to the early invention and adoption of bubble sheets.
One of the main elements that goes into a good multiple choice question is the existence of good distractors. That is, each of the alternate responses presented to the student should be the result of a plausible mistake that a student might make.
>>What Apple device competed with the portable CD player?<<
( ) The iPad
( ) Napster
(x) The iPod
( ) The vegetable peeler
( ) Android
( ) The Beatles
[Explanation]
The release of the iPod allowed consumers to carry their entire music library with them in a format that did not rely on fragile and energy-intensive spinning disks.
[Explanation]\
`);
expect(data).toXMLEqual(`
A multiple choice problem presents radio buttons for student input. Students can only select a single option presented. Multiple Choice questions have been the subject of many areas of research due to the early invention and adoption of bubble sheets.
One of the main elements that goes into a good multiple choice question is the existence of good distractors. That is, each of the alternate responses presented to the student should be the result of a plausible mistake that a student might make.
The iPadNapsterThe iPodThe vegetable peelerAndroidThe Beatles
Explanation
The release of the iPod allowed consumers to carry their entire music library with them in a format that did not rely on fragile and energy-intensive spinning disks.
`);
});
it('converts multiple choice shuffle to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`A multiple choice problem presents radio buttons for student input. Students can only select a single option presented. Multiple Choice questions have been the subject of many areas of research due to the early invention and adoption of bubble sheets.
One of the main elements that goes into a good multiple choice question is the existence of good distractors. That is, each of the alternate responses presented to the student should be the result of a plausible mistake that a student might make.
What Apple device competed with the portable CD player?
(!x@) The iPad
(@) Napster
() The iPod
( ) The vegetable peeler
( ) Android
(@) The Beatles
[Explanation]
The release of the iPod allowed consumers to carry their entire music library with them in a format that did not rely on fragile and energy-intensive spinning disks.
[Explanation]\
`);
expect(data).toXMLEqual(`\
A multiple choice problem presents radio buttons for student input. Students can only select a single option presented. Multiple Choice questions have been the subject of many areas of research due to the early invention and adoption of bubble sheets.
One of the main elements that goes into a good multiple choice question is the existence of good distractors. That is, each of the alternate responses presented to the student should be the result of a plausible mistake that a student might make.
What Apple device competed with the portable CD player?
The iPadNapsterThe iPodThe vegetable peelerAndroidThe Beatles
Explanation
The release of the iPod allowed consumers to carry their entire music library with them in a format that did not rely on fragile and energy-intensive spinning disks.
`);
});
it('converts a series of multiplechoice to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`bleh
(!x) a
() b
() c
yatta
( ) x
( ) y
(x) z
testa
(!) i
( ) ii
(x) iii
[Explanation]
When the student is ready, the explanation appears.
[Explanation]\
`);
expect(data).toXMLEqual(`\
bleh
abc
yatta
xyz
testa
iiiiii
Explanation
When the student is ready, the explanation appears.
`);
});
it('converts OptionResponse to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`OptionResponse gives a limited set of options for students to respond with, and presents those options in a format that encourages them to search for a specific answer rather than being immediately presented with options from which to recognize the correct answer.
The answer options and the identification of the correct answer is defined in the optioninput tag.
Translation between Option Response and __________ is extremely straightforward:
[[(Multiple Choice), String Response, Numerical Response, External Response, Image Response]]
[Explanation]
Multiple Choice also allows students to select from a variety of pre-written responses, although the format makes it easier for students to read very long response options. Optionresponse also differs slightly because students are more likely to think of an answer and then search for it rather than relying purely on recognition to answer the question.
[Explanation]\
`);
expect(data).toXMLEqual(`\
OptionResponse gives a limited set of options for students to respond with, and presents those options in a format that encourages them to search for a specific answer rather than being immediately presented with options from which to recognize the correct answer.
The answer options and the identification of the correct answer is defined in the optioninput tag.
Translation between Option Response and __________ is extremely straightforward:
Explanation
Multiple Choice also allows students to select from a variety of pre-written responses, although the format makes it easier for students to read very long response options. Optionresponse also differs slightly because students are more likely to think of an answer and then search for it rather than relying purely on recognition to answer the question.
`);
});
it('converts StringResponse to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`A string response problem accepts a line of text input from the student, and evaluates the input for correctness based on an expected answer within each input box.
The answer is correct if it matches every character of the expected answer. This can be a problem with international spelling, dates, or anything where the format of the answer is not clear.
Which US state has Lansing as its capital?
= Michigan
[Explanation]
Lansing is the capital of Michigan, although it is not Michgan's largest city, or even the seat of the county in which it resides.
[Explanation]\
`);
expect(data).toXMLEqual(`\
A string response problem accepts a line of text input from the student, and evaluates the input for correctness based on an expected answer within each input box.
The answer is correct if it matches every character of the expected answer. This can be a problem with international spelling, dates, or anything where the format of the answer is not clear.
Which US state has Lansing as its capital?
Explanation
Lansing is the capital of Michigan, although it is not Michgan's largest city, or even the seat of the county in which it resides.
`);
});
it('converts StringResponse with regular expression to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`Who lead the civil right movement in the United States of America?
= | \w*\.?\s*Luther King\s*.*
[Explanation]
Test Explanation.
[Explanation]\
`);
expect(data).toXMLEqual(`\
Who lead the civil right movement in the United States of America?
Explanation
Test Explanation.
`);
});
it('converts StringResponse with multiple answers to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`Who lead the civil right movement in the United States of America?
= Dr. Martin Luther King Jr.
or= Doctor Martin Luther King Junior
or= Martin Luther King
or= Martin Luther King Junior
[Explanation]
Test Explanation.
[Explanation]\
`);
expect(data).toXMLEqual(`\
Who lead the civil right movement in the United States of America?
Explanation
Test Explanation.
`);
});
it('converts StringResponse with multiple answers and regular expressions to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`Write a number from 1 to 4.
=| ^One$
or= two
or= ^thre+
or= ^4|Four$
[Explanation]
Test Explanation.
[Explanation]\
`);
expect(data).toXMLEqual(`\
Write a number from 1 to 4.
Explanation
Test Explanation.
`);
});
// test labels
it('converts markdown labels to label attributes', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`>>Who lead the civil right movement in the United States of America?<<
= | \w*\.?\s*Luther King\s*.*
[Explanation]
Test Explanation.
[Explanation]\
`);
expect(data).toXMLEqual(`\
Explanation
Test Explanation.
`);
});
it('handles multiple questions with labels', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
France is a country in Europe.
>>What is the capital of France?<<
= Paris
Germany is a country in Europe, too.
>>What is the capital of Germany?<<
( ) Bonn
( ) Hamburg
(x) Berlin
( ) Donut\
`);
expect(data).toXMLEqual(`\
France is a country in Europe.
Germany is a country in Europe, too.
BonnHamburgBerlinDonut`);
});
it('tests multiple questions with only one label', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
France is a country in Europe.
>>What is the capital of France?<<
= Paris
Germany is a country in Europe, too.
What is the capital of Germany?
( ) Bonn
( ) Hamburg
(x) Berlin
( ) Donut\
`);
expect(data).toXMLEqual(`\
France is a country in Europe.
Germany is a country in Europe, too.
What is the capital of Germany?
BonnHamburgBerlinDonut`);
});
it('adds labels to formulae', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
>>Enter the numerical value of Pi:<<
= 3.14159 +- .02\
`);
expect(data).toXMLEqual(``);
});
// test oddities
it('converts headers and oddities to xml', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`Not a header
A header
==============
Multiple choice w/ parentheticals
( ) option (with parens)
( ) xd option (x)
()) parentheses inside
() no space b4 close paren
Choice checks
[ ] option1 [x]
[x] correct
[x] redundant
[(] distractor
[] no space
Option with multiple correct ones
[[one option, (correct one), (should not be correct)]]
Option with embedded parens
[[My (heart), another, (correct)]]
What happens w/ empty correct options?
[[()]]
[Explanation]see[/expLanation]
[explanation]
orphaned start
No p tags in the below
But in this there should be
Great ideas require offsetting.
bad tests require drivel
[code]
Code should be nicely monospaced.
[/code]\
`);
expect(data).toXMLEqual(`\
Not a header
A header
Multiple choice w/ parentheticals
option (with parens)xd option (x)parentheses insideno space b4 close paren
Choice checks
option1 [x]correctredundantdistractorno space
Option with multiple correct ones
Option with embedded parens
What happens w/ empty correct options?
Explanation
see
[explanation]
orphaned start
No p tags in the below
But in this there should be
Great ideas require offsetting.
bad tests require drivel
Code should be nicely monospaced.
`);
});
it('can separate responsetypes based on ---', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Multiple choice problems allow learners to select only one option. Learners can see all the options along with the problem text.
>>Which of the following countries has the largest population?<<
( ) Brazil {{ timely feedback -- explain why an almost correct answer is wrong }}
( ) Germany
(x) Indonesia
( ) Russia
[explanation]
According to September 2014 estimates:
The population of Indonesia is approximately 250 million.
The population of Brazil is approximately 200 million.
The population of Russia is approximately 146 million.
The population of Germany is approximately 81 million.
[explanation]
---
Checkbox problems allow learners to select multiple options. Learners can see all the options along with the problem text.
>>The following languages are in the Indo-European family:<<
[x] Urdu
[ ] Finnish
[x] Marathi
[x] French
[ ] Hungarian
Note: Make sure you select all of the correct options—there may be more than one!
[explanation]
Urdu, Marathi, and French are all Indo-European languages, while Finnish and Hungarian are in the Uralic family.
[explanation]
\
`);
expect(data).toXMLEqual(`\
Multiple choice problems allow learners to select only one option. Learners can see all the options along with the problem text.
Brazil timely feedback -- explain why an almost correct answer is wrongGermanyIndonesiaRussia
Explanation
According to September 2014 estimates:
The population of Indonesia is approximately 250 million.
The population of Brazil is approximately 200 million.
The population of Russia is approximately 146 million.
The population of Germany is approximately 81 million.
Checkbox problems allow learners to select multiple options. Learners can see all the options along with the problem text.
UrduFinnishMarathiFrenchHungarian
Note: Make sure you select all of the correct options—there may be more than one!
Explanation
Urdu, Marathi, and French are all Indo-European languages, while Finnish and Hungarian are in the Uralic family.
\
`);
});
it('can separate other things based on ---', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
Multiple choice problems allow learners to select only one option. Learners can see all the options along with the problem text.
---
>>Which of the following countries has the largest population?<<
( ) Brazil {{ timely feedback -- explain why an almost correct answer is wrong }}
( ) Germany
(x) Indonesia
( ) Russia
[explanation]
According to September 2014 estimates:
The population of Indonesia is approximately 250 million.
The population of Brazil is approximately 200 million.
The population of Russia is approximately 146 million.
The population of Germany is approximately 81 million.
[explanation]\
`);
expect(data).toXMLEqual(`\
Multiple choice problems allow learners to select only one option. Learners can see all the options along with the problem text.
Brazil timely feedback -- explain why an almost correct answer is wrongGermanyIndonesiaRussia
Explanation
According to September 2014 estimates:
The population of Indonesia is approximately 250 million.
The population of Brazil is approximately 200 million.
The population of Russia is approximately 146 million.
The population of Germany is approximately 81 million.
\
`);
});
it('can do separation if spaces are present around ---', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
>>The following languages are in the Indo-European family:||There are three correct choices.<<
[x] Urdu
[ ] Finnish
[x] Marathi
[x] French
[ ] Hungarian
---
>>Which of the following countries has the largest population?||You have only choice.<<
( ) Brazil {{ timely feedback -- explain why an almost correct answer is wrong }}
( ) Germany
(x) Indonesia
( ) Russia\
`);
expect(data).toXMLEqual(`\
There are three correct choices.UrduFinnishMarathiFrenchHungarianYou have only choice.Brazil
timely feedback -- explain why an almost correct answer is wrongGermanyIndonesiaRussia\
`);
});
it('can extract question description', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
>>The following languages are in the Indo-European family:||Choose wisely.<<
[x] Urdu
[ ] Finnish
[x] Marathi
[x] French
[ ] Hungarian\
`);
expect(data).toXMLEqual(`\
Choose wisely.UrduFinnishMarathiFrenchHungarian\
`);
});
it('can handle question and description spanned across multiple lines', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
>>The following languages
are in the
Indo-European family:
||
first second
third
<<
[x] Urdu
[ ] Finnish
[x] Marathi\
`);
expect(data).toXMLEqual(`\
first second thirdUrduFinnishMarathi\
`);
});
it('will not add empty description', function() {
const data = MarkdownEditingDescriptor.markdownToXml(`\
>>The following languages are in the Indo-European family:||<<
[x] Urdu
[ ] Finnish\
`);
expect(data).toXMLEqual(`\
UrduFinnish\
`);
});
it('should throw error if an option does not have any text associated with it', function() {
let problemContent = `\
>>The following languages are in the Indo-European family:||<<
[ ]
[x] Urdu
[ ] Finnish\
`
expect(function(){ MarkdownEditingDescriptor.markdownToXml(problemContent); }).toThrow(
new Error(gettext("An answer option has been left blank. Please review and edit the component."))
);
problemContent = `\
>>The following languages are in the Indo-European family:||<<
( )
(x) Urdu
( ) Finnish\
`
expect(function(){ MarkdownEditingDescriptor.markdownToXml(problemContent); }).toThrow(
new Error(gettext("An answer option has been left blank. Please review and edit the component."))
);
});
});
});