qti 2.25.0 → 2.26.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/qti/sanitizer.rb +50 -1
- data/lib/qti/v1/models/interactions/hot_spot_interaction.rb +43 -8
- data/lib/qti/version.rb +1 -1
- data/lib/qti.rb +1 -0
- data/spec/fixtures/items_1.2/hot_spot.xml +42 -3
- data/spec/lib/qti/sanitizer_spec.rb +6 -0
- data/spec/lib/qti/v1/models/interactions/hot_spot_interaction_spec.rb +32 -1
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 9f3871afe8e84ed37180922c577051db9fc09ccd3936a74353e19dfb8a688dc8
|
4
|
+
data.tar.gz: d8be8fe548bc1e89d1e8a595275272291553bb89df682942c22bf53070aa32f6
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 570712ae37a5c79f34ac12d5f1ff02f7a5591035fb0ab16d0c01066b61b4a6546317e8049c4427e8ed3c63e013c25a638fc93ec59490fd088d6d17606748d105
|
7
|
+
data.tar.gz: 48b6d948fdbd2f867e98737f7456459ab9206b1446e8c4d77c65b6928f272dce2696ba1fb1b9842b2e5d770f8d5d35c5276019da19e9bdf2d52a6d18987a3880
|
data/lib/qti/sanitizer.rb
CHANGED
@@ -20,9 +20,58 @@ module Qti
|
|
20
20
|
Sanitize::Config::RELAXED[:attributes][element] + overrides
|
21
21
|
end
|
22
22
|
|
23
|
+
# Copied from Canvas (Classic Quizzes)
|
24
|
+
# canvas_sanitize/lib/canvas_sanitize/canvas_sanitize.rb:142
|
25
|
+
MATHML_TAGS =
|
26
|
+
%w[annotation
|
27
|
+
annotationml
|
28
|
+
maction
|
29
|
+
maligngroup
|
30
|
+
malignmark
|
31
|
+
mark
|
32
|
+
math
|
33
|
+
menclose
|
34
|
+
merror
|
35
|
+
mfenced
|
36
|
+
mfrac
|
37
|
+
mglyph
|
38
|
+
mi
|
39
|
+
mlabeledtr
|
40
|
+
mlongdiv
|
41
|
+
mmultiscripts
|
42
|
+
mn
|
43
|
+
mo
|
44
|
+
mover
|
45
|
+
mpadded
|
46
|
+
mphantom
|
47
|
+
mprescripts
|
48
|
+
mroot
|
49
|
+
mrow
|
50
|
+
ms
|
51
|
+
mscarries
|
52
|
+
mscarry
|
53
|
+
msgroup
|
54
|
+
msline
|
55
|
+
mspace
|
56
|
+
msqrt
|
57
|
+
msrow
|
58
|
+
mstack
|
59
|
+
mstyle
|
60
|
+
msub
|
61
|
+
msubsup
|
62
|
+
msup
|
63
|
+
mtable
|
64
|
+
mtd
|
65
|
+
mtext
|
66
|
+
mtr
|
67
|
+
munder
|
68
|
+
munderover
|
69
|
+
none
|
70
|
+
semantics].freeze
|
71
|
+
|
23
72
|
CONFIG =
|
24
73
|
{
|
25
|
-
elements: Sanitize::Config::RELAXED[:elements] + FILTER_TAGS,
|
74
|
+
elements: Sanitize::Config::RELAXED[:elements] + MATHML_TAGS + FILTER_TAGS,
|
26
75
|
protocols:
|
27
76
|
{
|
28
77
|
'iframe' => { 'src' => PROTOCOLS },
|
@@ -31,22 +31,32 @@ module Qti
|
|
31
31
|
end
|
32
32
|
|
33
33
|
def shape_type
|
34
|
-
@shape_type ||= response_label
|
34
|
+
@shape_type ||= extract_shape_type(response_label)
|
35
35
|
end
|
36
36
|
|
37
37
|
def coordinates
|
38
|
-
@coordinates ||=
|
39
|
-
|
40
|
-
return [] unless (raw_coordinates_array.length % 2).zero?
|
38
|
+
@coordinates ||= parse_coordinates(raw_coordinates)
|
39
|
+
end
|
41
40
|
|
42
|
-
|
43
|
-
|
44
|
-
end
|
45
|
-
end
|
41
|
+
def scoring_data_structs
|
42
|
+
@scoring_data_structs ||= parse_scoring_data
|
46
43
|
end
|
47
44
|
|
48
45
|
private
|
49
46
|
|
47
|
+
def parse_scoring_data
|
48
|
+
raw_scoring_data = answer_nodes&.map do |answer|
|
49
|
+
{
|
50
|
+
id: extract_identifier(answer),
|
51
|
+
type: extract_shape_type(answer),
|
52
|
+
coordinates: parse_coordinates(answer&.text || '')
|
53
|
+
}
|
54
|
+
end
|
55
|
+
raw_scoring_data.map do |values|
|
56
|
+
Models::ScoringData.new(values, rcardinality)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
50
60
|
def response_label
|
51
61
|
@response_label ||= begin
|
52
62
|
node = @node.dup
|
@@ -55,10 +65,35 @@ module Qti
|
|
55
65
|
end
|
56
66
|
end
|
57
67
|
|
68
|
+
def answer_nodes
|
69
|
+
@answer_nodes ||= begin
|
70
|
+
node = @node.dup
|
71
|
+
presentation = node.at_xpath('.//xmlns:presentation')
|
72
|
+
presentation.xpath('.//xmlns:response_label')
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
58
76
|
def raw_coordinates
|
59
77
|
@raw_coordinates ||= response_label&.text || ''
|
60
78
|
end
|
61
79
|
|
80
|
+
def extract_shape_type(answer)
|
81
|
+
answer&.attributes&.[]('rarea')&.value
|
82
|
+
end
|
83
|
+
|
84
|
+
def extract_identifier(answer)
|
85
|
+
answer&.attributes&.[]('ident')&.value
|
86
|
+
end
|
87
|
+
|
88
|
+
def parse_coordinates(raw_coordinates)
|
89
|
+
raw_coordinates_array = raw_coordinates.split(',')
|
90
|
+
return [] unless (raw_coordinates_array.length % 2).zero?
|
91
|
+
|
92
|
+
raw_coordinates_array.each_slice(2).map do |point|
|
93
|
+
{ x: point[0].to_f, y: point[1].to_f }
|
94
|
+
end
|
95
|
+
end
|
96
|
+
|
62
97
|
def coordinate_hash(coordinate_node)
|
63
98
|
%w[x y].map do |axis|
|
64
99
|
[axis.to_sym, coordinate_node&.attributes&.[](axis)&.value&.to_f]
|
data/lib/qti/version.rb
CHANGED
data/lib/qti.rb
CHANGED
@@ -39,7 +39,7 @@
|
|
39
39
|
<material>
|
40
40
|
<matimage uri="$IMS-CC-FILEBASE$/Uploaded Media/someuuid"/>
|
41
41
|
</material>
|
42
|
-
<response_label ident="response1" rarea="
|
42
|
+
<response_label ident="response1" rarea="rectangle">0.372,0.1376,0.616,0.4256</response_label>
|
43
43
|
</render_hotspot>
|
44
44
|
</response_xy>
|
45
45
|
</flow>
|
@@ -76,7 +76,7 @@
|
|
76
76
|
<material>
|
77
77
|
<matimage uri="$IMS-CC-FILEBASE$/Uploaded Media/someuuid"/>
|
78
78
|
</material>
|
79
|
-
<response_label ident="response1" rarea="
|
79
|
+
<response_label ident="response1" rarea="rectangle">0.372</response_label>
|
80
80
|
</render_hotspot>
|
81
81
|
</response_xy>
|
82
82
|
</flow>
|
@@ -113,7 +113,46 @@
|
|
113
113
|
<material>
|
114
114
|
<matimage uri="$IMS-CC-FILEBASE$/Uploaded Media/someuuid"/>
|
115
115
|
</material>
|
116
|
-
<response_label ident="response1" rarea="
|
116
|
+
<response_label ident="response1" rarea="rectangle"></response_label>
|
117
|
+
</render_hotspot>
|
118
|
+
</response_xy>
|
119
|
+
</flow>
|
120
|
+
</presentation>
|
121
|
+
</item>
|
122
|
+
<item ident="g515255148f5a3cae1832584071aa8529" title="hot-spot with multiple selections">
|
123
|
+
<itemmetadata>
|
124
|
+
<qtimetadata>
|
125
|
+
<qtimetadatafield>
|
126
|
+
<fieldlabel>question_type</fieldlabel>
|
127
|
+
<fieldentry>hot_spot_question</fieldentry>
|
128
|
+
</qtimetadatafield>
|
129
|
+
<qtimetadatafield>
|
130
|
+
<fieldlabel>points_possible</fieldlabel>
|
131
|
+
<fieldentry>1.0</fieldentry>
|
132
|
+
</qtimetadatafield>
|
133
|
+
<qtimetadatafield>
|
134
|
+
<fieldlabel>original_answer_ids</fieldlabel>
|
135
|
+
<fieldentry/>
|
136
|
+
</qtimetadatafield>
|
137
|
+
<qtimetadatafield>
|
138
|
+
<fieldlabel>assessment_question_identifierref</fieldlabel>
|
139
|
+
<fieldentry>gd6ff15355f3dd3f532816fa268230130</fieldentry>
|
140
|
+
</qtimetadatafield>
|
141
|
+
</qtimetadata>
|
142
|
+
</itemmetadata>
|
143
|
+
<presentation>
|
144
|
+
<flow>
|
145
|
+
<response_xy ident="response1" rcardinality="Multiple" rtiming="No">
|
146
|
+
<material>
|
147
|
+
<mattext texttype="text/html"><p>point something on the image</p></mattext>
|
148
|
+
</material>
|
149
|
+
<render_hotspot>
|
150
|
+
<material>
|
151
|
+
<matimage uri="$IMS-CC-FILEBASE$/Uploaded Media/someuuid"/>
|
152
|
+
</material>
|
153
|
+
<response_label ident="response1" rarea="rectangle">0.1,0.2,0.3,0.4</response_label>
|
154
|
+
<response_label ident="response2" rarea="ellipse">0.5,0.6,0.7,0.8</response_label>
|
155
|
+
<response_label ident="response3" rarea="bounded">0.9,0.1,0.2,0.3</response_label>
|
117
156
|
</render_hotspot>
|
118
157
|
</response_xy>
|
119
158
|
</flow>
|
@@ -63,5 +63,11 @@ describe Qti::Sanitizer do
|
|
63
63
|
expect(sanitizer.clean(html)).to include 'height: 294px;'
|
64
64
|
expect(sanitizer.clean(html)).to include 'display: inline-block;'
|
65
65
|
end
|
66
|
+
|
67
|
+
Qti::Sanitizer::MATHML_TAGS.each do |tag|
|
68
|
+
it "allows MathML tag: #{tag}" do
|
69
|
+
expect(sanitizer.clean("<#{tag}>")).to eq("<#{tag}></#{tag}>")
|
70
|
+
end
|
71
|
+
end
|
66
72
|
end
|
67
73
|
end
|
@@ -28,7 +28,7 @@ describe Qti::V1::Models::Interactions::HotSpotInteraction do
|
|
28
28
|
describe '#shape_type' do
|
29
29
|
let(:loaded_class) { described_class.new(assessment_item_refs.first, test_object) }
|
30
30
|
it 'returns correct shape type' do
|
31
|
-
expect(loaded_class.shape_type).to eq '
|
31
|
+
expect(loaded_class.shape_type).to eq 'rectangle'
|
32
32
|
end
|
33
33
|
end
|
34
34
|
|
@@ -54,4 +54,35 @@ describe Qti::V1::Models::Interactions::HotSpotInteraction do
|
|
54
54
|
end
|
55
55
|
end
|
56
56
|
end
|
57
|
+
|
58
|
+
describe '#scoring_data_structs' do
|
59
|
+
context 'with a single hotspot' do
|
60
|
+
let(:loaded_class) { described_class.new(assessment_item_refs.first, test_object) }
|
61
|
+
|
62
|
+
it 'returns scoring data with a single hotspot' do
|
63
|
+
scoring_data = loaded_class.scoring_data_structs
|
64
|
+
expect(scoring_data.size).to eq(1)
|
65
|
+
expect(scoring_data.first.values[:type]).to eq('rectangle')
|
66
|
+
expect(scoring_data.first.values[:coordinates]).to eq([{ x: 0.372, y: 0.1376 }, { x: 0.616, y: 0.4256 }])
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
context 'with multiple hotspots' do
|
71
|
+
let(:loaded_class) { described_class.new(assessment_item_refs[3], test_object) }
|
72
|
+
|
73
|
+
it 'returns scoring data with multiple hotspots' do
|
74
|
+
scoring_data = loaded_class.scoring_data_structs
|
75
|
+
expect(scoring_data.size).to eq(3)
|
76
|
+
|
77
|
+
expect(scoring_data[0].values[:type]).to eq('rectangle')
|
78
|
+
expect(scoring_data[0].values[:coordinates]).to eq([{ x: 0.1, y: 0.2 }, { x: 0.3, y: 0.4 }])
|
79
|
+
|
80
|
+
expect(scoring_data[1].values[:type]).to eq('ellipse')
|
81
|
+
expect(scoring_data[1].values[:coordinates]).to eq([{ x: 0.5, y: 0.6 }, { x: 0.7, y: 0.8 }])
|
82
|
+
|
83
|
+
expect(scoring_data[2].values[:type]).to eq('bounded')
|
84
|
+
expect(scoring_data[2].values[:coordinates]).to eq([{ x: 0.9, y: 0.1 }, { x: 0.2, y: 0.3 }])
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
57
88
|
end
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: qti
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.26.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Adrian Diaz
|
@@ -12,7 +12,7 @@ authors:
|
|
12
12
|
autorequire:
|
13
13
|
bindir: bin
|
14
14
|
cert_chain: []
|
15
|
-
date:
|
15
|
+
date: 2025-03-21 00:00:00.000000000 Z
|
16
16
|
dependencies:
|
17
17
|
- !ruby/object:Gem::Dependency
|
18
18
|
name: actionview
|