Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Move scoring logic: Expression #2118

Merged
merged 2 commits into from
Jan 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import {getDecimalSeparator} from "@khanacademy/perseus-core";

import {MathFieldActionType} from "../../types";
import {getDecimalSeparator} from "../../utils";
import {mathQuillInstance} from "../input/mathquill-instance";

import handleArrow from "./handle-arrow";
Expand Down
7 changes: 2 additions & 5 deletions packages/math-input/src/components/keypad/button-assets.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,9 @@ asset.
In the future it would be great if these were included from files so that
no copying and pasting is necessary.
*/
import {getDecimalSeparator} from "@khanacademy/perseus-core";
import * as React from "react";

import {DecimalSeparator, getDecimalSeparator} from "../../utils";
import {useMathInputI18n} from "../i18n-context";

import type Key from "../../data/keys";
Expand Down Expand Up @@ -176,10 +176,7 @@ export default function ButtonAsset({id}: Props): React.ReactNode {
case "PERIOD":
// Different locales use different symbols for the decimal separator
// (, vs .)
if (
id === "DECIMAL" &&
getDecimalSeparator(locale) === DecimalSeparator.COMMA
) {
if (id === "DECIMAL" && getDecimalSeparator(locale) !== ".") {
// comma decimal separator
return (
<svg
Expand Down
37 changes: 0 additions & 37 deletions packages/math-input/src/utils.ts
Original file line number Diff line number Diff line change
@@ -1,40 +1,3 @@
export const DecimalSeparator = {
COMMA: ",",
PERIOD: ".",
} as const;

/**
* Get the character used for separating decimals.
*/
export const getDecimalSeparator = (locale: string): string => {
let separator: string = DecimalSeparator.PERIOD;

switch (locale) {
// TODO(somewhatabstract): Remove this when Chrome supports the `ka`
// locale properly.
// https://github.com/formatjs/formatjs/issues/1526#issuecomment-559891201
//
// Supported locales in Chrome:
// https://source.chromium.org/chromium/chromium/src/+/master:third_party/icu/scripts/chrome_ui_languages.list
case "ka":
separator = ",";
break;

default:
const numberWithDecimalSeparator = 1.1;
// TODO(FEI-3647): Update to use .formatToParts() once we no longer have to
// support Safari 12.
const match = new Intl.NumberFormat(locale)
.format(numberWithDecimalSeparator)
// 0x661 is ARABIC-INDIC DIGIT ONE
// 0x6F1 is EXTENDED ARABIC-INDIC DIGIT ONE
.match(/[^\d\u0661\u06F1]/);
separator = match?.[0] ?? ".";
}

return separator === "," ? DecimalSeparator.COMMA : DecimalSeparator.PERIOD;
};

const CDOT_ONLY = [
"az",
"cs",
Expand Down
1 change: 1 addition & 0 deletions packages/perseus-core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ export type {ErrorKind} from "./error/errors";
// Careful, `version.ts` uses this function so it _must_ be imported above it
export {addLibraryVersionToPerseusDebug} from "./utils/add-library-version-to-perseus-debug";
export {default as getMatrixSize} from "./utils/get-matrix-size";
export {default as getDecimalSeparator} from "./utils/get-decimal-separator";

export {libVersion} from "./version";

Expand Down
1 change: 1 addition & 0 deletions packages/perseus-score/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ export type * from "./validation.types";
export {default as scoreCategorizer} from "./widgets/categorizer/score-categorizer";
export {default as scoreCSProgram} from "./widgets/cs-program/score-cs-program";
export {default as scoreDropdown} from "./widgets/dropdown/score-dropdown";
export {default as scoreExpression} from "./widgets/expression/score-expression";
export {default as scoreIframe} from "./widgets/iframe/score-iframe";
export {
default as scoreLabelImage,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
import {mockStrings} from "../../strings";

import {expressionItem3Options} from "./expression.testdata";
import scoreExpression from "./score-expression";
import {expressionItem3Options} from "./score-expression.testdata";
import * as ExpressionValidator from "./validate-expression";

import type {PerseusExpressionRubric} from "@khanacademy/perseus-score";

// TODO: remove strings as a param for scorers
const mockStrings = {};

describe("scoreExpression", () => {
it("should be correctly answerable if validation passes", function () {
// Arrange
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import type {PerseusExpressionWidgetOptions} from "@khanacademy/perseus-core";

export const expressionItem3Options: PerseusExpressionWidgetOptions = {
answerForms: [
{
considered: "ungraded",
form: false,
simplify: false,
value: "x+1",
},
{
considered: "wrong",
form: false,
simplify: false,
value: "y+1",
},
{
considered: "correct",
form: false,
simplify: false,
value: "z+1",
},
{
considered: "correct",
form: false,
simplify: false,
value: "a+1",
},
],
times: false,
buttonSets: ["basic"],
functions: ["f", "g", "h"],
buttonsVisible: "focused",
visibleLabel: "number of cm",
ariaLabel: "number of centimeters",
};
Original file line number Diff line number Diff line change
@@ -1,21 +1,22 @@
import * as KAS from "@khanacademy/kas";
import {Errors} from "@khanacademy/perseus-core";
import {KhanAnswerTypes} from "@khanacademy/perseus-score";
import {
Errors,
getDecimalSeparator,
PerseusError,
} from "@khanacademy/perseus-core";
import _ from "underscore";

import {Log} from "../../logging/log";
import KhanAnswerTypes from "../../util/answer-types";

import getDecimalSeparator from "./get-decimal-separator";
import validateExpression from "./validate-expression";

import type {PerseusStrings} from "../../strings";
import type {PerseusExpressionAnswerForm} from "@khanacademy/perseus-core";
import type {Score} from "../../util/answer-types";
import type {
PerseusScore,
Score,
PerseusExpressionRubric,
PerseusExpressionUserInput,
} from "@khanacademy/perseus-score";
PerseusScore,
} from "../../validation.types";
import type {PerseusExpressionAnswerForm} from "@khanacademy/perseus-core";

/* Content creators input a list of answers which are matched from top to
* bottom. The intent is that they can include spcific solutions which should
Expand All @@ -38,7 +39,8 @@ import type {
function scoreExpression(
userInput: PerseusExpressionUserInput,
rubric: PerseusExpressionRubric,
strings: PerseusStrings,
// TODO: remove strings as a param for scorers
strings: unknown,
locale: string,
): PerseusScore {
const validationError = validateExpression(userInput);
Expand All @@ -62,12 +64,10 @@ function scoreExpression(
// in the function variables list for the expression.
if (!expression.parsed) {
/* c8 ignore next */
Log.error(
throw new PerseusError(
"Unable to parse solution answer for expression",
Errors.InvalidInput,
{loggedMetadata: {rubric: JSON.stringify(rubric)}},
);
return null;
}

return KhanAnswerTypes.expression.createValidatorFunctional(
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import type {
ValidationResult,
PerseusExpressionUserInput,
} from "@khanacademy/perseus-score";
ValidationResult,
} from "../../validation.types";

/**
* Checks user input from the expression widget to see if it is scorable.
Expand Down
17 changes: 9 additions & 8 deletions packages/perseus/src/widgets/expression/expression.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,15 @@
import * as KAS from "@khanacademy/kas";
import {KeyArray, KeypadInput, KeypadType} from "@khanacademy/math-input";
import {
getDecimalSeparator,
type PerseusExpressionWidgetOptions,
} from "@khanacademy/perseus-core";
import {linterContextDefault} from "@khanacademy/perseus-linter";
import {
scoreExpression,
type PerseusExpressionRubric,
type PerseusExpressionUserInput,
} from "@khanacademy/perseus-score";
import {View} from "@khanacademy/wonder-blocks-core";
import Tooltip from "@khanacademy/wonder-blocks-tooltip";
import {LabelSmall} from "@khanacademy/wonder-blocks-typography";
Expand All @@ -18,18 +27,10 @@ import {ApiOptions, ClassNames as ApiClassNames} from "../../perseus-api";
import a11y from "../../util/a11y";
import {getPromptJSON as _getPromptJSON} from "../../widget-ai-utils/expression/expression-ai-utils";

import getDecimalSeparator from "./get-decimal-separator";
import scoreExpression from "./score-expression";

import type {DependenciesContext} from "../../dependencies";
import type {FocusPath, Widget, WidgetExports, WidgetProps} from "../../types";
import type {ExpressionPromptJSON} from "../../widget-ai-utils/expression/expression-ai-utils";
import type {Keys as Key, KeypadConfiguration} from "@khanacademy/math-input";
import type {PerseusExpressionWidgetOptions} from "@khanacademy/perseus-core";
import type {
PerseusExpressionRubric,
PerseusExpressionUserInput,
} from "@khanacademy/perseus-score";
import type {PropsFor} from "@khanacademy/wonder-blocks-core";

type InputPath = ReadonlyArray<string>;
Expand Down
Loading