Как получить результат адаптивной карты? - PullRequest
0 голосов
/ 18 февраля 2019

Я пытаюсь получить результат моей AdaptiveCard.Мой бот использует водяные логи.В одном Waterfallstep я представляю пользователю количество комнат со временем и датой.Затем пользователь может выбрать комнату.Я попробовал это, как показано ниже.К сожалению, activity остается нулевым.Как получить результат адаптивной карты

private async Task<DialogTurnResult> AfterChoice(WaterfallStepContext step, CancellationToken cancellationToken)
{
    if (step.Result is Activity activity && activity.Value != null && ((dynamic)activity.Value).chosenRoom is JValue chosenRoom)
    {
        dynamic requestedBooking = JsonConvert.DeserializeObject<ExpandoObject>((string)chosenRoom.Value);
        this.roomemail = requestedBooking.roomEmail;
        return await step.EndDialogAsync();
    }
    else
    {
        return await step.BeginDialogAsync(whatever, cancellationToken: cancellationToken);
    }
}

Как получить выбор пользователя?

1 Ответ

0 голосов
/ 19 февраля 2019

Адаптивные карты отправляют свои результаты отправки немного иначе, чем обычный текст пользователя.Когда пользователь печатает в чате и отправляет обычное сообщение, оно заканчивается на Context.Activity.Text.Когда пользователь заполняет ввод на адаптивной карте, он заканчивается на Context.Activity.Value, который является объектом, в котором имена ключей - id на вашей карте, а значения - значения полей в адаптивной карте.

Например, json:

{
    "type": "AdaptiveCard",
    "body": [
        {
            "type": "TextBlock",
            "text": "Test Adaptive Card"
        },
        {
            "type": "ColumnSet",
            "columns": [
                {
                    "type": "Column",
                    "items": [
                        {
                            "type": "TextBlock",
                            "text": "Text:"
                        }
                    ],
                    "width": 20
                },
                {
                    "type": "Column",
                    "items": [
                        {
                            "type": "Input.Text",
                            "id": "userText",
                            "placeholder": "Enter Some Text"
                        }
                    ],
                    "width": 80
                }
            ]
        }
    ],
    "actions": [
        {
            "type": "Action.Submit",
            "title": "Submit"
        }
    ],
    "$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "version": "1.0"
}

.. создает карту, которая выглядит следующим образом:

Test Adaptive Card

Если пользователь вводит «Testing Testing 123» в текстовое поле и нажимает «Отправить», Context.Activity будет выглядеть примерно так:

{ type: 'message',
  value: { userText: 'Testing Testing 123' },
  from: { id: 'xxxxxxxx-05d4-478a-9daa-9b18c79bb66b', name: 'User' },
  locale: '',
  channelData: { postback: true },
  channelId: 'emulator',
  conversation: { id: 'xxxxxxxx-182b-11e9-be61-091ac0e3a4ac|livechat' },
  id: 'xxxxxxxx-182b-11e9-ad8e-63b45e3ebfa7',
  localTimestamp: 2019-01-14T18:39:21.000Z,
  recipient: { id: '1', name: 'Bot', role: 'bot' },
  timestamp: 2019-01-14T18:39:21.773Z,
  serviceUrl: 'http://localhost:58453' }

Представление пользователя можно увидеть в Context.Activity.Value.userText.

Обратите внимание, что отправка адаптивной карты отправляется как postBack, что означает, что данные отправки не отображаются в окне чата как часть разговора - они остаются на адаптивной карте.

Использование адаптивных карт с Waterfall Dialogs

Ваш вопрос не совсем связан с этим, но, поскольку вы можете попытаться это сделать, я подумал, что важно включить его в свойответ.

Собственно, адаптивные карты не работают как подсказки.При появлении приглашения оно будет отображаться и ждать ввода пользователя, прежде чем продолжить.Но с адаптивными картами (даже если они содержат поле ввода и кнопку отправки), в адаптивной карте нет кода, который заставил бы диалоговое окно «Водопад» ожидать ввода данных пользователем, прежде чем продолжить диалог.

Итак,если вы используете адаптивную карту, которая принимает пользовательский ввод, вы, как правило, хотите обрабатывать все, что пользователь отправляет, вне контекста диалога водопада.

При этом, если вы хотите использовать адаптивную карту какчасть диалога водопада, есть обходной путь.По сути, вы:

  1. Отображение адаптивной карты
  2. Отображение текстовой подсказки
  3. Преобразование ввода адаптивной карты пользователя во ввод текстовой подсказки

В вашем классе Waterfall Dialog (шаги 1 и 2):

private async Task<DialogTurnResult> DisplayCardAsync(WaterfallStepContext stepContext, CancellationToken cancellationToken)
{
    // Create the Adaptive Card
    var cardPath = Path.Combine(".", "AdaptiveCard.json");
    var cardJson = File.ReadAllText(cardPath);
    var cardAttachment = new Attachment()
    {
        ContentType = "application/vnd.microsoft.card.adaptive",
        Content = JsonConvert.DeserializeObject(cardJson),
    };

    // Create the text prompt
    var opts = new PromptOptions
    {
        Prompt = new Activity
        {
            Attachments = new List<Attachment>() { cardAttachment },
            Type = ActivityTypes.Message,
            Text = "waiting for user input...", // You can comment this out if you don't want to display any text. Still works.
        }
    };

    // Display a Text Prompt and wait for input
    return await stepContext.PromptAsync(nameof(TextPrompt), opts);
}

private async Task<DialogTurnResult> HandleResponseAsync(WaterfallStepContext stepContext, CancellationToken cancellationToken)
{
    // Do something with step.result
    // Adaptive Card submissions are objects, so you likely need to JObject.Parse(step.result)
    await stepContext.Context.SendActivityAsync($"INPUT: {stepContext.Result}");
    return await stepContext.NextAsync();
}

В вашем классе основного бота (<your-bot>.cs) (шаг 3):

var activity = turnContext.Activity;

if (string.IsNullOrWhiteSpace(activity.Text) && activity.Value != null)
{
    activity.Text = JsonConvert.SerializeObject(activity.Value);
}
...