@pie-lib/rubric 1.0.0-beta.4 → 1.0.0-next.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.
- package/CHANGELOG.json +1 -407
- package/CHANGELOG.md +332 -4
- package/LICENSE.md +5 -0
- package/NEXT.CHANGELOG.json +1 -0
- package/lib/authoring.js +493 -0
- package/lib/authoring.js.map +1 -0
- package/lib/index.js +20 -0
- package/lib/index.js.map +1 -0
- package/lib/point-menu.js +125 -0
- package/lib/point-menu.js.map +1 -0
- package/package.json +18 -7
- package/src/__tests__/rubric.test.jsx +179 -0
- package/src/authoring.jsx +211 -159
- package/src/index.js +1 -0
- package/src/point-menu.jsx +8 -15
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
|
|
4
|
+
Object.defineProperty(exports, "__esModule", {
|
|
5
|
+
value: true
|
|
6
|
+
});
|
|
7
|
+
exports["default"] = exports.IconMenu = void 0;
|
|
8
|
+
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
|
|
9
|
+
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
|
|
10
|
+
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
|
|
11
|
+
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
|
|
12
|
+
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
|
|
13
|
+
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
|
|
14
|
+
var _Menu = _interopRequireDefault(require("@mui/material/Menu"));
|
|
15
|
+
var _MenuItem = _interopRequireDefault(require("@mui/material/MenuItem"));
|
|
16
|
+
var _MoreVert = _interopRequireDefault(require("@mui/icons-material/MoreVert"));
|
|
17
|
+
var _MoreHoriz = _interopRequireDefault(require("@mui/icons-material/MoreHoriz"));
|
|
18
|
+
var _IconButton = _interopRequireDefault(require("@mui/material/IconButton"));
|
|
19
|
+
var _propTypes = _interopRequireDefault(require("prop-types"));
|
|
20
|
+
var _react = _interopRequireDefault(require("react"));
|
|
21
|
+
function _callSuper(t, o, e) { return o = (0, _getPrototypeOf2["default"])(o), (0, _possibleConstructorReturn2["default"])(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], (0, _getPrototypeOf2["default"])(t).constructor) : o.apply(t, e)); }
|
|
22
|
+
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
|
|
23
|
+
var IconMenu = exports.IconMenu = /*#__PURE__*/function (_React$Component) {
|
|
24
|
+
function IconMenu(props) {
|
|
25
|
+
var _this;
|
|
26
|
+
(0, _classCallCheck2["default"])(this, IconMenu);
|
|
27
|
+
_this = _callSuper(this, IconMenu, [props]);
|
|
28
|
+
(0, _defineProperty2["default"])(_this, "handleClick", function (event) {
|
|
29
|
+
return _this.setState({
|
|
30
|
+
open: true,
|
|
31
|
+
anchorEl: event.currentTarget
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
(0, _defineProperty2["default"])(_this, "handleRequestClose", function () {
|
|
35
|
+
return _this.setState({
|
|
36
|
+
open: false
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
_this.state = {
|
|
40
|
+
anchorEl: undefined,
|
|
41
|
+
open: false
|
|
42
|
+
};
|
|
43
|
+
return _this;
|
|
44
|
+
}
|
|
45
|
+
(0, _inherits2["default"])(IconMenu, _React$Component);
|
|
46
|
+
return (0, _createClass2["default"])(IconMenu, [{
|
|
47
|
+
key: "render",
|
|
48
|
+
value: function render() {
|
|
49
|
+
var _this2 = this;
|
|
50
|
+
var _this$props = this.props,
|
|
51
|
+
opts = _this$props.opts,
|
|
52
|
+
onClick = _this$props.onClick;
|
|
53
|
+
var _this$state = this.state,
|
|
54
|
+
open = _this$state.open,
|
|
55
|
+
anchorEl = _this$state.anchorEl;
|
|
56
|
+
var keys = Object.keys(opts) || [];
|
|
57
|
+
var handleMenuClick = function handleMenuClick(key) {
|
|
58
|
+
return function () {
|
|
59
|
+
onClick(key);
|
|
60
|
+
_this2.handleRequestClose();
|
|
61
|
+
};
|
|
62
|
+
};
|
|
63
|
+
var iconColor = open ? 'inherit' : 'disabled';
|
|
64
|
+
return /*#__PURE__*/_react["default"].createElement("div", null, /*#__PURE__*/_react["default"].createElement("div", {
|
|
65
|
+
onClick: this.handleClick
|
|
66
|
+
}, /*#__PURE__*/_react["default"].createElement(_IconButton["default"], {
|
|
67
|
+
size: "large"
|
|
68
|
+
}, open ? /*#__PURE__*/_react["default"].createElement(_MoreVert["default"], {
|
|
69
|
+
color: iconColor
|
|
70
|
+
}) : /*#__PURE__*/_react["default"].createElement(_MoreHoriz["default"], {
|
|
71
|
+
color: iconColor
|
|
72
|
+
}))), /*#__PURE__*/_react["default"].createElement(_Menu["default"], {
|
|
73
|
+
id: "point-menu",
|
|
74
|
+
anchorEl: anchorEl,
|
|
75
|
+
open: open,
|
|
76
|
+
onClose: this.handleRequestClose,
|
|
77
|
+
style: {
|
|
78
|
+
transform: 'translate(-15px, -15px)'
|
|
79
|
+
},
|
|
80
|
+
transformOrigin: {
|
|
81
|
+
vertical: 'center',
|
|
82
|
+
horizontal: 'right'
|
|
83
|
+
}
|
|
84
|
+
}, keys.map(function (k, index) {
|
|
85
|
+
return /*#__PURE__*/_react["default"].createElement(_MenuItem["default"], {
|
|
86
|
+
key: index,
|
|
87
|
+
onClick: handleMenuClick(k)
|
|
88
|
+
}, opts[k]);
|
|
89
|
+
})));
|
|
90
|
+
}
|
|
91
|
+
}]);
|
|
92
|
+
}(_react["default"].Component);
|
|
93
|
+
(0, _defineProperty2["default"])(IconMenu, "propTypes", {
|
|
94
|
+
opts: _propTypes["default"].object,
|
|
95
|
+
onClick: _propTypes["default"].func.isRequired
|
|
96
|
+
});
|
|
97
|
+
var PointMenu = exports["default"] = /*#__PURE__*/function (_React$Component2) {
|
|
98
|
+
function PointMenu() {
|
|
99
|
+
(0, _classCallCheck2["default"])(this, PointMenu);
|
|
100
|
+
return _callSuper(this, PointMenu, arguments);
|
|
101
|
+
}
|
|
102
|
+
(0, _inherits2["default"])(PointMenu, _React$Component2);
|
|
103
|
+
return (0, _createClass2["default"])(PointMenu, [{
|
|
104
|
+
key: "render",
|
|
105
|
+
value: function render() {
|
|
106
|
+
var _this$props2 = this.props,
|
|
107
|
+
onChange = _this$props2.onChange,
|
|
108
|
+
showSampleAnswer = _this$props2.showSampleAnswer;
|
|
109
|
+
var sampleText = showSampleAnswer ? 'Provide Sample Response' : 'Remove Sample Response';
|
|
110
|
+
return /*#__PURE__*/_react["default"].createElement(IconMenu, {
|
|
111
|
+
onClick: function onClick(key) {
|
|
112
|
+
return onChange(key);
|
|
113
|
+
},
|
|
114
|
+
opts: {
|
|
115
|
+
sample: sampleText
|
|
116
|
+
}
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
}]);
|
|
120
|
+
}(_react["default"].Component);
|
|
121
|
+
(0, _defineProperty2["default"])(PointMenu, "propTypes", {
|
|
122
|
+
onChange: _propTypes["default"].func.isRequired,
|
|
123
|
+
showSampleAnswer: _propTypes["default"].bool.isRequired
|
|
124
|
+
});
|
|
125
|
+
//# sourceMappingURL=point-menu.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"point-menu.js","names":["_Menu","_interopRequireDefault","require","_MenuItem","_MoreVert","_MoreHoriz","_IconButton","_propTypes","_react","_callSuper","t","o","e","_getPrototypeOf2","_possibleConstructorReturn2","_isNativeReflectConstruct","Reflect","construct","constructor","apply","Boolean","prototype","valueOf","call","IconMenu","exports","_React$Component","props","_this","_classCallCheck2","_defineProperty2","event","setState","open","anchorEl","currentTarget","state","undefined","_inherits2","_createClass2","key","value","render","_this2","_this$props","opts","onClick","_this$state","keys","Object","handleMenuClick","handleRequestClose","iconColor","createElement","handleClick","size","color","id","onClose","style","transform","transformOrigin","vertical","horizontal","map","k","index","React","Component","PropTypes","object","func","isRequired","PointMenu","_React$Component2","arguments","_this$props2","onChange","showSampleAnswer","sampleText","sample","bool"],"sources":["../src/point-menu.jsx"],"sourcesContent":["import Menu from '@mui/material/Menu';\nimport MenuItem from '@mui/material/MenuItem';\nimport MoreVertIcon from '@mui/icons-material/MoreVert';\nimport MoreHorizIcon from '@mui/icons-material/MoreHoriz';\nimport IconButton from '@mui/material/IconButton';\nimport PropTypes from 'prop-types';\nimport React from 'react';\n\nexport class IconMenu extends React.Component {\n static propTypes = {\n opts: PropTypes.object,\n onClick: PropTypes.func.isRequired,\n };\n\n constructor(props) {\n super(props);\n this.state = {\n anchorEl: undefined,\n open: false,\n };\n }\n\n handleClick = (event) => this.setState({ open: true, anchorEl: event.currentTarget });\n\n handleRequestClose = () => this.setState({ open: false });\n\n render() {\n const { opts, onClick } = this.props;\n const { open, anchorEl } = this.state;\n const keys = Object.keys(opts) || [];\n\n const handleMenuClick = (key) => () => {\n onClick(key);\n this.handleRequestClose();\n };\n\n const iconColor = open ? 'inherit' : 'disabled';\n\n return (\n <div>\n <div onClick={this.handleClick}>\n <IconButton size=\"large\">\n {open ? <MoreVertIcon color={iconColor} /> : <MoreHorizIcon color={iconColor} />}\n </IconButton>\n </div>\n <Menu\n id=\"point-menu\"\n anchorEl={anchorEl}\n open={open}\n onClose={this.handleRequestClose}\n style={{ transform: 'translate(-15px, -15px)' }}\n transformOrigin={{\n vertical: 'center',\n horizontal: 'right',\n }}\n >\n {keys.map((k, index) => (\n <MenuItem key={index} onClick={handleMenuClick(k)}>\n {opts[k]}\n </MenuItem>\n ))}\n </Menu>\n </div>\n );\n }\n}\n\nexport default class PointMenu extends React.Component {\n static propTypes = {\n onChange: PropTypes.func.isRequired,\n showSampleAnswer: PropTypes.bool.isRequired,\n };\n\n render() {\n const { onChange, showSampleAnswer } = this.props;\n const sampleText = showSampleAnswer ? 'Provide Sample Response' : 'Remove Sample Response';\n\n return (\n <IconMenu\n onClick={(key) => onChange(key)}\n opts={{\n sample: sampleText,\n }}\n />\n );\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
package/package.json
CHANGED
|
@@ -1,20 +1,31 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pie-lib/rubric",
|
|
3
|
-
"version": "1.0.0-
|
|
3
|
+
"version": "1.0.0-next.0",
|
|
4
4
|
"publishConfig": {
|
|
5
5
|
"access": "public"
|
|
6
6
|
},
|
|
7
7
|
"main": "lib/index.js",
|
|
8
8
|
"module": "src/index.js",
|
|
9
9
|
"dependencies": {
|
|
10
|
-
"@
|
|
11
|
-
"@
|
|
12
|
-
"@
|
|
10
|
+
"@emotion/react": "^11.14.0",
|
|
11
|
+
"@emotion/style": "^0.8.0",
|
|
12
|
+
"@hello-pangea/dnd": "^18.0.1",
|
|
13
|
+
"@mui/icons-material": "^7.3.4",
|
|
14
|
+
"@mui/material": "^7.3.4",
|
|
15
|
+
"@pie-lib/editable-html": "11.21.3-next.15",
|
|
13
16
|
"debug": "^4.1.1",
|
|
14
17
|
"lodash": "^4.17.11",
|
|
15
18
|
"prop-types": "^15.7.2",
|
|
16
|
-
"react": "^
|
|
17
|
-
"react-
|
|
18
|
-
|
|
19
|
+
"react": "^18.2.0",
|
|
20
|
+
"react-dom": "^18.2.0"
|
|
21
|
+
},
|
|
22
|
+
"gitHead": "e0f8f2250addd014938f4a631ab86cafe16b2730",
|
|
23
|
+
"exports": {
|
|
24
|
+
".": {
|
|
25
|
+
"require": "./lib/index.js",
|
|
26
|
+
"import": "./src/index.js",
|
|
27
|
+
"default": "./lib/index.js"
|
|
28
|
+
},
|
|
29
|
+
"./esm": "./esm/index.js"
|
|
19
30
|
}
|
|
20
31
|
}
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
import { render, screen } from '@testing-library/react';
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { RawAuthoring } from '../authoring';
|
|
4
|
+
import _ from 'lodash';
|
|
5
|
+
import { ThemeProvider, createTheme } from '@mui/material/styles';
|
|
6
|
+
|
|
7
|
+
// Mock dependencies
|
|
8
|
+
jest.mock('@pie-lib/editable-html', () => {
|
|
9
|
+
return function EditableHtml(props) {
|
|
10
|
+
return <div data-testid="editable-html" data-markup={props.markup} />;
|
|
11
|
+
};
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
jest.mock('@pie-lib/config-ui', () => ({
|
|
15
|
+
FeedbackConfig: ({ feedback }) => <div data-testid="feedback-config">{JSON.stringify(feedback)}</div>,
|
|
16
|
+
}));
|
|
17
|
+
|
|
18
|
+
jest.mock('@hello-pangea/dnd', () => ({
|
|
19
|
+
DragDropContext: ({ children }) => <div data-testid="drag-drop-context">{children}</div>,
|
|
20
|
+
Droppable: ({ children }) => children({ droppableProps: {}, innerRef: () => {} }, {}),
|
|
21
|
+
Draggable: ({ children, index }) =>
|
|
22
|
+
children(
|
|
23
|
+
{
|
|
24
|
+
innerRef: () => {},
|
|
25
|
+
draggableProps: { 'data-draggable-index': index },
|
|
26
|
+
dragHandleProps: {}
|
|
27
|
+
},
|
|
28
|
+
{}
|
|
29
|
+
),
|
|
30
|
+
}));
|
|
31
|
+
|
|
32
|
+
describe('Rubric', () => {
|
|
33
|
+
const points = ['nothing right', 'a teeny bit right', 'mostly right', 'bingo'];
|
|
34
|
+
const sampleAnswers = [null, 'just right', 'not left', null];
|
|
35
|
+
const theme = createTheme();
|
|
36
|
+
|
|
37
|
+
const renderComponent = (value = {}, props = {}) => {
|
|
38
|
+
const defaultProps = {
|
|
39
|
+
classes: {},
|
|
40
|
+
onChange: jest.fn(),
|
|
41
|
+
className: 'className',
|
|
42
|
+
value: {
|
|
43
|
+
excludeZero: false,
|
|
44
|
+
points,
|
|
45
|
+
sampleAnswers,
|
|
46
|
+
...value,
|
|
47
|
+
},
|
|
48
|
+
...props,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
return {
|
|
52
|
+
...render(
|
|
53
|
+
<ThemeProvider theme={theme}>
|
|
54
|
+
<RawAuthoring {...defaultProps} />
|
|
55
|
+
</ThemeProvider>
|
|
56
|
+
),
|
|
57
|
+
onChange: defaultProps.onChange,
|
|
58
|
+
props: defaultProps,
|
|
59
|
+
};
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
describe('render', () => {
|
|
63
|
+
it('renders rubric title and main structure', () => {
|
|
64
|
+
renderComponent();
|
|
65
|
+
expect(screen.getByText('Rubric')).toBeInTheDocument();
|
|
66
|
+
expect(screen.getByLabelText('Max Points')).toBeInTheDocument();
|
|
67
|
+
expect(screen.getByLabelText('Exclude zeros')).toBeInTheDocument();
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it('renders all point configurations', () => {
|
|
71
|
+
const { container } = renderComponent();
|
|
72
|
+
// Verify DragDropContext is rendered
|
|
73
|
+
expect(container.querySelector('[data-testid="drag-drop-context"]')).toBeInTheDocument();
|
|
74
|
+
|
|
75
|
+
// Check that point labels are rendered (4 points = "3 pts", "2 pts", "1 pt", "0 pt")
|
|
76
|
+
// Note: The PointConfig component uses singular "pt" for 0 and 1, plural "pts" for 2+
|
|
77
|
+
expect(screen.getByText('3 pts')).toBeInTheDocument();
|
|
78
|
+
expect(screen.getByText('2 pts')).toBeInTheDocument();
|
|
79
|
+
expect(screen.getByText('1 pt')).toBeInTheDocument();
|
|
80
|
+
expect(screen.getByText('0 pt')).toBeInTheDocument();
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
describe('draggable', () => {
|
|
84
|
+
it('renders 3 draggable items when excludeZero is true', () => {
|
|
85
|
+
const { container } = renderComponent({ excludeZero: true });
|
|
86
|
+
|
|
87
|
+
// When excludeZero is true, the last point (0 pts) should not be rendered
|
|
88
|
+
// So we should have 3 draggable items
|
|
89
|
+
const draggableItems = container.querySelectorAll('[data-draggable-index]');
|
|
90
|
+
expect(draggableItems.length).toEqual(3);
|
|
91
|
+
|
|
92
|
+
// Verify the 0 pt label is not rendered
|
|
93
|
+
expect(screen.queryByText('0 pt')).not.toBeInTheDocument();
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
it('renders 4 draggable items when excludeZero is false', () => {
|
|
97
|
+
const { container } = renderComponent({ excludeZero: false });
|
|
98
|
+
|
|
99
|
+
// When excludeZero is false, all points including 0 should be rendered
|
|
100
|
+
const draggableItems = container.querySelectorAll('[data-draggable-index]');
|
|
101
|
+
expect(draggableItems.length).toEqual(4);
|
|
102
|
+
|
|
103
|
+
// Verify all point labels are rendered
|
|
104
|
+
expect(screen.getByText('3 pts')).toBeInTheDocument();
|
|
105
|
+
expect(screen.getByText('2 pts')).toBeInTheDocument();
|
|
106
|
+
expect(screen.getByText('1 pt')).toBeInTheDocument();
|
|
107
|
+
expect(screen.getByText('0 pt')).toBeInTheDocument();
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
describe('logic', () => {
|
|
113
|
+
describe('changeMaxPoints', () => {
|
|
114
|
+
const testChangeMaxPoints = (maxPoints, excludeZero, expectedPoints, expectedSampleAnswers) => {
|
|
115
|
+
it(`maxPoints=${maxPoints}, excludeZero=${excludeZero} calls onChange correctly`, () => {
|
|
116
|
+
const { onChange, container } = renderComponent({ excludeZero });
|
|
117
|
+
|
|
118
|
+
// Get the component instance through the container
|
|
119
|
+
// We need to call the method directly since we're testing internal behavior
|
|
120
|
+
const instance = container.querySelector('[class*="MuiBox-root"]')?._owner;
|
|
121
|
+
|
|
122
|
+
// Since we can't easily access instance methods in RTL, we'll test the behavior
|
|
123
|
+
// by verifying the onChange prop receives the correct values
|
|
124
|
+
// For now, we'll directly test the logic
|
|
125
|
+
const component = new RawAuthoring({
|
|
126
|
+
value: { excludeZero, points, sampleAnswers },
|
|
127
|
+
onChange,
|
|
128
|
+
classes: {},
|
|
129
|
+
className: 'className'
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
component.changeMaxPoints(maxPoints);
|
|
133
|
+
|
|
134
|
+
expect(onChange).toHaveBeenCalledWith({
|
|
135
|
+
excludeZero,
|
|
136
|
+
points: expectedPoints,
|
|
137
|
+
sampleAnswers: expectedSampleAnswers,
|
|
138
|
+
maxPoints: expectedPoints.length - 1,
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
testChangeMaxPoints(1, false, _.takeRight(points, 2), _.takeRight(sampleAnswers, 2));
|
|
144
|
+
testChangeMaxPoints(1, true, _.takeRight(points, 2), _.takeRight(sampleAnswers, 2));
|
|
145
|
+
testChangeMaxPoints(2, true, _.takeRight(points, 3), _.takeRight(sampleAnswers, 3));
|
|
146
|
+
testChangeMaxPoints(2, false, _.takeRight(points, 3), _.takeRight(sampleAnswers, 3));
|
|
147
|
+
testChangeMaxPoints(5, false, ['', ''].concat(points), [null, null].concat(sampleAnswers));
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
describe('changeSampleResponse', () => {
|
|
151
|
+
const testChangeSampleResponse = (index, clickedItem, excludeZero, expectedPoints, expectedSampleAnswers) => {
|
|
152
|
+
it(`Point ${index} with clickedItem="${clickedItem}" calls onChange correctly`, () => {
|
|
153
|
+
const { onChange } = renderComponent({ excludeZero });
|
|
154
|
+
|
|
155
|
+
// Test the logic by creating a component instance and calling the method
|
|
156
|
+
const component = new RawAuthoring({
|
|
157
|
+
value: { excludeZero, points, sampleAnswers },
|
|
158
|
+
onChange,
|
|
159
|
+
classes: {},
|
|
160
|
+
className: 'className'
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
component.onPointMenuChange(index, clickedItem);
|
|
164
|
+
|
|
165
|
+
expect(onChange).toHaveBeenCalledWith({
|
|
166
|
+
excludeZero,
|
|
167
|
+
points: expectedPoints,
|
|
168
|
+
sampleAnswers: expectedSampleAnswers,
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
};
|
|
172
|
+
|
|
173
|
+
testChangeSampleResponse(0, 'sample', false, points, ['', 'just right', 'not left', null]);
|
|
174
|
+
testChangeSampleResponse(3, 'sample', false, points, [null, 'just right', 'not left', '']);
|
|
175
|
+
testChangeSampleResponse(1, 'sample', true, points, [null, null, 'not left', null]);
|
|
176
|
+
testChangeSampleResponse(3, 'sample', true, points, [null, 'just right', 'not left', '']);
|
|
177
|
+
});
|
|
178
|
+
});
|
|
179
|
+
});
|