type2docfx
Version:
A tool to convert json format output from TypeDoc to universal reference model for DocFx to consume.
93 lines (87 loc) • 4.06 kB
text/typescript
/**
* @module botbuilder-choices
*/
/**
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License.
*/
import { Choice, findChoices, FoundChoice, FindChoicesOptions } from './findChoices';
import { ModelResult } from './modelResult';
import * as Recognizers from '@microsoft/recognizers-text-number';
/**
* :package: **botbuilder-choices**
*
* High level function for recognizing a choice in a users utterance. This is layered above the
* `findChoices()` function and adds logic to let the user specify their choice by index (they can
* say "one" to pick `choice[0]`) or ordinal position (they can say "the second one" to pick
* `choice[1]`.) The users utterance is recognized in the following order:
*
* - By name using `findChoices()`.
* - By 1's based ordinal position.
* - By 1's based index position.
*
* **Usage Example**
*
* ```JavaScript
* const { recognizeChoices } = require('botbuilder-choices');
*
* const choices = ['red', 'green', 'blue'];
* const utterance = context.activity.text;
* const results = recognizeChoices(utterance, choices);
* if (results.length == 1) {
* await context.sendActivity(`I like ${results[0].resolution.value} too!`);
* } else if (results.length > 1) {
* const ambiguous = results.map((r) => r.resolution.value);
* await context.sendActivity(ChoiceFactory.forChannel(context, ambiguous, `Which one?`));
* } else {
* await context.sendActivity(ChoiceFactory.forChannel(context, choices, `I didn't get that... Which color?`));
* }
* ```
* @param utterance The text or user utterance to search over. For an incoming 'message' activity you can simply use `context.activity.text`.
* @param choices List of choices to search over.
* @param options (Optional) options used to tweak the search that's performed.
*/
export function recognizeChoices(utterance: string, choices: (string|Choice)[], options?: FindChoicesOptions): ModelResult<FoundChoice>[] {
function matchChoiceByIndex(match: ModelResult<any>) {
try {
const index = parseInt(match.resolution.value) - 1;
if (index >= 0 && index < list.length) {
const choice = list[index];
matched.push({
start: match.start,
end: match.end,
typeName: 'choice',
text: match.text,
resolution: {
value: choice.value,
index: index,
score: 1.0
}
});
}
} catch (e) { }
}
// Normalize choices
const list: Choice[] = (choices || []).map((choice, index) => typeof choice === 'string' ? { value: choice } : choice).filter((choice) => choice);
// Try finding choices by text search first
// - We only want to use a single strategy for returning results to avoid issues where utterances
// like the "the third one" or "the red one" or "the first division book" would miss-recognize as
// a numerical index or ordinal as well.
const locale = options && options.locale ? options.locale : 'en-us';
let matched = findChoices(utterance, list, options);
if (matched.length === 0) {
// Next try finding by ordinal
const ordinals = Recognizers.recognizeOrdinal(utterance, locale);
if (ordinals.length > 0) {
ordinals.forEach(matchChoiceByIndex);
} else {
// Finally try by numerical index
Recognizers.recognizeNumber(utterance, locale).forEach(matchChoiceByIndex);
}
// Sort any found matches by their position within the utterance.
// - The results from findChoices() are already properly sorted so we just need this
// for ordinal & numerical lookups.
matched = matched.sort((a,b) => a.start - b.start);
}
return matched;
}