diff --git a/skills/src/csharp/experimental/eventskill/Content/EventCard.1.0.json b/skills/src/csharp/experimental/eventskill/Content/EventCard.1.0.json new file mode 100644 index 0000000000..74e5cd74d3 --- /dev/null +++ b/skills/src/csharp/experimental/eventskill/Content/EventCard.1.0.json @@ -0,0 +1,77 @@ +{ + "type": "AdaptiveCard", + "id": "EventCard", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "verticalContentAlignment": "Center", + "items": [ + { + "type": "TextBlock", + "id": "title", + "size": "Large", + "weight": "Bolder", + "color": "Dark", + "text": "Local Event" + } + ], + "width": "stretch" + } + ] + } + ] + }, + { + "type": "Image", + "url": "{ImageUrl}" + }, + { + "type": "TextBlock", + "text": "**{Title}**", + "size": "Medium", + "weight": "Bolder" + }, + { + "type": "Container", + "items": [ + { + "type": "TextBlock", + "text": "{StartDate}", + "spacing": "Small" + }, + { + "type": "TextBlock", + "text": "{Location}", + "spacing": "Small" + }, + { + "type": "TextBlock", + "text": "{Price}", + "spacing": "Small" + } + ] + }, + { + "type": "TextBlock", + "text": "Powered by **Eventbrite**", + "horizontalAlignment": "Right", + "size": "Small" + } + ], + "actions": [ + { + "type": "Action.OpenUrl", + "title": "Get Tickets", + "url": "{Url}" + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0", + "speak": "{Speak}" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/eventskill/Dialogs/EventDialogBase.cs b/skills/src/csharp/experimental/eventskill/Dialogs/EventDialogBase.cs index d753227d7d..fe4da46bcc 100644 --- a/skills/src/csharp/experimental/eventskill/Dialogs/EventDialogBase.cs +++ b/skills/src/csharp/experimental/eventskill/Dialogs/EventDialogBase.cs @@ -12,10 +12,12 @@ using Luis; using Microsoft.Bot.Builder; using Microsoft.Bot.Builder.Dialogs; +using Microsoft.Bot.Builder.Dialogs.Choices; using Microsoft.Bot.Builder.Skills; using Microsoft.Bot.Builder.Solutions.Authentication; using Microsoft.Bot.Builder.Solutions.Responses; using Microsoft.Bot.Builder.Solutions.Util; +using Microsoft.Bot.Connector; using Microsoft.Bot.Schema; namespace EventSkill.Dialogs @@ -175,5 +177,16 @@ protected async Task HandleDialogExceptions(WaterfallStepContext sc, Exception e var state = await StateAccessor.GetAsync(sc.Context); state.Clear(); } + + // Get card that renders for adaptive card 1.0 + protected string GetCardName(ITurnContext context, string name) + { + if (Channel.GetChannelId(context) == Channels.Msteams) + { + name += ".1.0"; + } + + return name; + } } } \ No newline at end of file diff --git a/skills/src/csharp/experimental/eventskill/Dialogs/FindEventsDialog.cs b/skills/src/csharp/experimental/eventskill/Dialogs/FindEventsDialog.cs index 3f0c8a4ed8..e1980d6a62 100644 --- a/skills/src/csharp/experimental/eventskill/Dialogs/FindEventsDialog.cs +++ b/skills/src/csharp/experimental/eventskill/Dialogs/FindEventsDialog.cs @@ -95,7 +95,7 @@ private async Task FindEvents(WaterfallStepContext sc, Cancell Url = item.Url }; - cards.Add(new Card("EventCard", eventCardData)); + cards.Add(new Card(GetCardName(sc.Context, "EventCard"), eventCardData)); } await sc.Context.SendActivityAsync(ResponseManager.GetCardResponse(FindEventsResponses.FoundEvents, cards, null)); diff --git a/skills/src/csharp/experimental/eventskill/Dialogs/MainDialog.cs b/skills/src/csharp/experimental/eventskill/Dialogs/MainDialog.cs index 7c715969ab..de027ccfa1 100644 --- a/skills/src/csharp/experimental/eventskill/Dialogs/MainDialog.cs +++ b/skills/src/csharp/experimental/eventskill/Dialogs/MainDialog.cs @@ -237,7 +237,7 @@ private async Task OnLogout(DialogContext dc) private async Task PopulateStateFromSemanticAction(ITurnContext context) { - // Example of populating local state with data passed through semanticAction out of Activity + // Populating local state with data passed through semanticAction out of Activity var activity = context.Activity; var semanticAction = activity.SemanticAction; if (semanticAction != null && semanticAction.Entities.ContainsKey("location")) diff --git a/skills/src/csharp/experimental/eventskill/EventSkill.csproj b/skills/src/csharp/experimental/eventskill/EventSkill.csproj index 1c26cd793d..ce6adedd1e 100644 --- a/skills/src/csharp/experimental/eventskill/EventSkill.csproj +++ b/skills/src/csharp/experimental/eventskill/EventSkill.csproj @@ -7,6 +7,7 @@ + @@ -24,6 +25,7 @@ + diff --git a/skills/src/csharp/experimental/eventskill/Models/EventSkillState.cs b/skills/src/csharp/experimental/eventskill/Models/EventSkillState.cs index 69e903fce3..987e2e7df4 100644 --- a/skills/src/csharp/experimental/eventskill/Models/EventSkillState.cs +++ b/skills/src/csharp/experimental/eventskill/Models/EventSkillState.cs @@ -11,7 +11,7 @@ public class EventSkillState public EventLuis LuisResult { get; set; } - public string CurrentCoordinates { get; internal set; } + public string CurrentCoordinates { get; set; } public void Clear() { diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.1.0.json new file mode 100644 index 0000000000..12b016f352 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.1.0.json @@ -0,0 +1,51 @@ +{ + "type": "AdaptiveCard", + "id": "FoodItemCard", + "body": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "{Quantity}", + "horizontalAlignment": "Center" + } + ], + "width": "auto" + }, + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "{Name}" + }, + { + "type": "TextBlock", + "text": "{SpecialRequest}", + "spacing": "None" + } + ], + "width": "stretch" + }, + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "${Price}", + "horizontalAlignment": "Right" + } + ], + "width": "auto" + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0", + "speak": "{Speak}" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.json b/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.json index 78478056d7..4f65ffebfd 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.json +++ b/skills/src/csharp/experimental/hospitalityskill/Content/FoodItemCard.json @@ -14,7 +14,7 @@ "horizontalAlignment": "Center" } ], - "width": "40px" + "width": 10 }, { "type": "Column", @@ -30,7 +30,7 @@ "spacing": "None" } ], - "width": "stretch" + "width": 70 }, { "type": "Column", @@ -41,7 +41,7 @@ "horizontalAlignment": "Right" } ], - "width": "auto" + "width": 20 } ] } diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.1.0.json new file mode 100644 index 0000000000..4f46f2d6f6 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.1.0.json @@ -0,0 +1,71 @@ +{ + "type": "AdaptiveCard", + "version": "1.0", + "id": "FoodOrderCard", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "verticalContentAlignment": "Center", + "items": [ + { + "type": "TextBlock", + "id": "title", + "size": "Large", + "weight": "Bolder", + "color": "Dark", + "text": "Current Order" + } + ], + "width": "stretch" + } + ] + } + ] + }, + { + "type": "Container", + "id": "items", + "items": [] + }, + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "Bill Total", + "horizontalAlignment": "Right" + } + ], + "width": "stretch" + }, + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "weight": "Bolder", + "text": "${BillTotal}", + "horizontalAlignment": "Right" + } + ], + "width": "auto" + } + ] + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.json b/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.json index 1ea3b7f4ab..259b64b5f2 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.json +++ b/skills/src/csharp/experimental/hospitalityskill/Content/FoodOrderCard.json @@ -60,7 +60,7 @@ "horizontalAlignment": "Center" } ], - "width": "40px" + "width": 10 }, { "type": "Column", @@ -71,7 +71,7 @@ "text": "Item" } ], - "width": "stretch" + "width": 70 }, { "type": "Column", @@ -83,7 +83,7 @@ "horizontalAlignment": "Right" } ], - "width": "auto" + "width": 20 } ] } diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/MenuCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/MenuCard.1.0.json new file mode 100644 index 0000000000..5c682a695c --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/MenuCard.1.0.json @@ -0,0 +1,45 @@ +{ + "type": "AdaptiveCard", + "id": "MenuCard", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "verticalContentAlignment": "Center", + "items": [ + { + "type": "TextBlock", + "id": "title", + "size": "Large", + "weight": "Bolder", + "color": "Dark", + "text": "{Type} Menu" + } + ], + "width": "stretch" + } + ] + }, + { + "type": "TextBlock", + "text": "{TimeAvailable}", + "weight": "Bolder", + "color": "Dark" + } + ] + }, + { + "type": "Container", + "id": "items", + "items": [] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0", + "speak": "{Speak}" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/MenuItemCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/MenuItemCard.1.0.json new file mode 100644 index 0000000000..a12cbcda50 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/MenuItemCard.1.0.json @@ -0,0 +1,36 @@ +{ + "type": "AdaptiveCard", + "id": "MenuItemCard", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "width": "auto", + "items": [ + { + "type": "TextBlock", + "text": "{Name} {Price}", + "weight": "Bolder" + } + ] + } + ] + }, + { + "type": "TextBlock", + "text": "{Description}", + "wrap": true, + "horizontalAlignment": "Left", + "spacing": "None" + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.1.0.json new file mode 100644 index 0000000000..23ebb063d1 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.1.0.json @@ -0,0 +1,38 @@ +{ + "type": "AdaptiveCard", + "version": "1.0", + "id": "RequestItemCard", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "verticalContentAlignment": "Center", + "items": [ + { + "type": "TextBlock", + "id": "title", + "size": "Large", + "weight": "Bolder", + "color": "Dark", + "text": "Your Item Requests" + } + ], + "width": "stretch" + } + ] + } + ] + }, + { + "type": "Container", + "id": "items", + "items": [] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.json b/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.json index bf394ae35e..8980af0f1e 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.json +++ b/skills/src/csharp/experimental/hospitalityskill/Content/RequestItemCard.json @@ -60,7 +60,7 @@ "horizontalAlignment": "Center" } ], - "width": "40px" + "width": 10 }, { "type": "Column", @@ -71,7 +71,7 @@ "text": "Item" } ], - "width": "stretch" + "width": 90 } ] } diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/ReservationDetails.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/ReservationDetails.1.0.json new file mode 100644 index 0000000000..406759ec80 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/ReservationDetails.1.0.json @@ -0,0 +1,46 @@ +{ + "type": "AdaptiveCard", + "id": "ReservationDetails", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "TextBlock", + "id": "title", + "size": "Large", + "weight": "Bolder", + "color": "Dark", + "text": "{Title}" + } + ], + "width": "stretch" + }, + { + "type": "Container", + "id": "items", + "items": [ + { + "type": "FactSet", + "facts": [ + { + "title": "Check-in Date", + "value": "{CheckInDate}" + }, + { + "title": "Check-out Date", + "value": "{CheckOutDate}" + }, + { + "title": "Check-out Time", + "value": "{CheckOutTime}" + } + ] + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0", + "speak": "{Speak}" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.1.0.json b/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.1.0.json new file mode 100644 index 0000000000..c3720d8b64 --- /dev/null +++ b/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.1.0.json @@ -0,0 +1,34 @@ +{ + "type": "AdaptiveCard", + "version": "1.0", + "id": "RoomItemCard", + "body": [ + { + "type": "ColumnSet", + "columns": [ + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "{Quantity}", + "horizontalAlignment": "Center" + } + ], + "width": "auto" + }, + { + "type": "Column", + "items": [ + { + "type": "TextBlock", + "text": "{Item}" + } + ], + "width": "stretch" + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json" +} \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.json b/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.json index 729f33e310..dc4883acaa 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.json +++ b/skills/src/csharp/experimental/hospitalityskill/Content/RoomItemCard.json @@ -15,7 +15,7 @@ "horizontalAlignment": "Center" } ], - "width": "40px" + "width": 10 }, { "type": "Column", @@ -25,7 +25,7 @@ "text": "{Item}" } ], - "width": "stretch" + "width": 90 } ] } diff --git a/skills/src/csharp/experimental/hospitalityskill/Data/AvailableItems.json b/skills/src/csharp/experimental/hospitalityskill/Data/AvailableItems.json index 4e40df6f7c..f66478d641 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Data/AvailableItems.json +++ b/skills/src/csharp/experimental/hospitalityskill/Data/AvailableItems.json @@ -13,7 +13,7 @@ }, { "item": "Conditioner", - "names": [ "conditioner", "conditioners", "conditioner bottle", "conditioner bottles", "bottle of conditioner", "bottles of conditioner" ] + "names": [ "conditioner", "conditioners", "conditioner bottle", "conditioner bottles", "bottle of conditioner", "bottles of conditioner", "hair conditioner", "hair conditioners" ] }, { "item": "Body Wash", @@ -62,5 +62,17 @@ { "item": "Tissue Box", "names": [ "tissue box", "tissue boxes", "tissues", "tissue", "box of tissue", "box of tissues" ] + }, + { + "item": "Workout Clothes", + "names": [ "workout clothes", "work out clothes", "work out clothing", "workout clothing", "gym clothes", "exercise clothes" ] + }, + { + "item": "Running Shoes", + "names": [ "running shoes", "exercise shoes", "workout shoes", "workout shoes", "tennis shoes" ] + }, + { + "item": "Umbrella", + "names": [ "umbrella", "umbrellas" ] } ] diff --git a/skills/src/csharp/experimental/hospitalityskill/Data/RoomServiceMenu.json b/skills/src/csharp/experimental/hospitalityskill/Data/RoomServiceMenu.json index d233871fc5..13a6f41bac 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Data/RoomServiceMenu.json +++ b/skills/src/csharp/experimental/hospitalityskill/Data/RoomServiceMenu.json @@ -101,7 +101,7 @@ "name": "French Fries", "allNames": [ "french fries", "french fry" ], "glutenFree": true, - "vegetarian": true, + "vegetarian": true, "price": 6 }, { @@ -130,6 +130,21 @@ "vegetarian": true, "price": 7, "description": "Two scoops of vanilla ice cream" + }, + { + "name": "Coffee", + "allNames": [ "coffee", "cup of coffee", "coffees" ], + "glutenFree": true, + "vegetarian": true, + "price": 3, + "description": "Latte, mocha, cappuccino, espresso, and macchiato available." + }, + { + "name": "Water Bottle", + "allNames": [ "water bottle", "water bottles", "bottles of water", "bottle of water", "water", "waters" ], + "glutenFree": true, + "vegetarian": true, + "price": 3 } ] } diff --git a/skills/src/csharp/experimental/hospitalityskill/Deployment/Resources/LU/en/Hospitality.lu b/skills/src/csharp/experimental/hospitalityskill/Deployment/Resources/LU/en/Hospitality.lu index b5b30018d7..529cc208c4 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Deployment/Resources/LU/en/Hospitality.lu +++ b/skills/src/csharp/experimental/hospitalityskill/Deployment/Resources/LU/en/Hospitality.lu @@ -1,4 +1,4 @@ -> ! Automatically generated by [LUDown CLI](https://github.com/Microsoft/botbuilder-tools/tree/master/Ludown), Thu Aug 15 2019 15:40:54 GMT-0700 (Pacific Daylight Time) +> ! Automatically generated by [LUDown CLI](https://github.com/Microsoft/botbuilder-tools/tree/master/Ludown), Mon Aug 26 2019 12:38:00 GMT-0700 (Pacific Daylight Time) > ! Source LUIS JSON file: stdin @@ -192,6 +192,8 @@ - can i see a lunch menu - can i see a room service menu - can you bring me {FoodRequest=3 {Food=chocolate chip cookies}} +- can you please send me some food? +- can you send me somethink to drink? - do you have room service - i also want {FoodRequest=2 {Food=parfaits}} - i need {FoodRequest=1 more {Food=brownie}} @@ -202,6 +204,8 @@ - i need a {Food=scrambled egg burrito} - i need a {FoodRequest={Food=turkey sandwich} {SpecialRequest=with extra mayo}} now - i need more {Food=orange juice} +- i need some food +- i need somthing to drink - i want {FoodRequest=2 {Food=cookies}}, a {FoodRequest={Food=cobb salad} {SpecialRequest=without chicken}} and {FoodRequest=3 {Food=salmons}} - i want {FoodRequest=2 things of {Food=sorbet}} and an {Food=earl grey tea} - i want a {FoodRequest={SpecialRequest=vegetarian} {Food=cobb salad}} @@ -211,6 +215,7 @@ - i want to order an {Food=egg white omelet} - i want to order from the all day menu - i want to order room service +- i'm hungry - order {FoodRequest=2 {Food=muffins}} and {FoodRequest=3 {Food=butter croissants}} - show me a dinner menu - what are the room service options for this hotel @@ -238,11 +243,11 @@ $PREBUILT:number > # Phrase list definitions $Items:phraseList interchangeable -- shampoo,conditioner,towel,hand towel,bath towel,pillow,pillows,pillow case,blanket,blankets,tissues,tissue box,lotion,moisturizer,body lotion,sheets,down pillow,feather pillow,bar of soap,dish soap,laundry detergent,soap,toothpaste,deodorant,towels,toothbrush,toilet paper,body wash,sunscreen,cushion,lip balm,perfume,toothbrushes,mouthwash,shaving cream,detergent,hand sanitizer,shower gel,hand soap,hand lotion,hand cream,liquid soap,bar soap,paper towels,baby wipes,purell,hair conditioner,laundry soap,fabric softener,facial tissue,kleenex,trash bags,hair dryer,detergent powder,cleaning supplies,garbage bags,toiletries,cutlery,cleaning products,bedding,duvet,comforter,comforters,bedspread,pillowcase,duvets,pillowcases,bedspreads,linens,bedsheets,bedlinen,bedclothes,bed sheets,bedsheet,bedcovers,washcloths,bathmats,bathmat,bed sheet,handtowel,washclothes,toiletpaper,facecloths,handtowels,soap/shampoo,papertowels,babywipes,loofas,kleenexes,washrag,dishsoap,waterbottles,papertowel,qtips,chap stick,q tips,q tip,ziplocs,qtip,band aids,zip loc,ziploc,cottonballs,swabs,cotton tipped,bandaids,ziplock,bandaid,band aid,bandages,zip lock,guaze,swab,neosporin,wipes,gauze,bandage,antiseptic,trashbag,ointment,tweezers,scissors,thermometer,nail clippers,safety pins,mirror,comb,nail file,brush,sewing kit,q-tips,razor,dental floss,hairspray,shower cap,slippers,hairbrush,hair brush,combs,bobby pins,brushes,razors,hairbrushes,deoderant,shampoo/conditioner,shaver,chapstick,antiperspirant,anti perspirant,deodarant,bodywash,facewash,face wash,face cream,eye cream,cup,mug,fork,knife,spoon,bowl,cups,spoons,plate,pot,glass,plates,bowls,pan,mugs,pots,dish +- shampoo,conditioner,towel,hand towel,bath towel,pillow,pillows,pillow case,blanket,blankets,tissues,tissue box,lotion,moisturizer,body lotion,sheets,down pillow,feather pillow,bar of soap,dish soap,laundry detergent,soap,toothpaste,deodorant,towels,toothbrush,toilet paper,body wash,sunscreen,cushion,lip balm,perfume,toothbrushes,mouthwash,shaving cream,detergent,hand sanitizer,shower gel,hand soap,hand lotion,hand cream,liquid soap,bar soap,paper towels,baby wipes,purell,hair conditioner,laundry soap,fabric softener,facial tissue,kleenex,trash bags,hair dryer,detergent powder,cleaning supplies,garbage bags,toiletries,cutlery,cleaning products,bedding,duvet,comforter,comforters,bedspread,pillowcase,duvets,pillowcases,bedspreads,linens,bedsheets,bedlinen,bedclothes,bed sheets,bedsheet,bedcovers,washcloths,bathmats,bathmat,bed sheet,handtowel,washclothes,toiletpaper,facecloths,handtowels,soap/shampoo,papertowels,babywipes,loofas,kleenexes,washrag,dishsoap,waterbottles,papertowel,qtips,chap stick,q tips,q tip,ziplocs,qtip,band aids,zip loc,ziploc,cottonballs,swabs,cotton tipped,bandaids,ziplock,bandaid,band aid,bandages,zip lock,guaze,swab,neosporin,wipes,gauze,bandage,antiseptic,trashbag,ointment,tweezers,scissors,thermometer,nail clippers,safety pins,mirror,comb,nail file,brush,sewing kit,q-tips,razor,dental floss,hairspray,shower cap,slippers,hairbrush,hair brush,combs,bobby pins,brushes,razors,hairbrushes,deoderant,shampoo/conditioner,shaver,chapstick,antiperspirant,anti perspirant,deodarant,bodywash,facewash,face wash,face cream,eye cream,mug,fork,knife,spoon,bowl,spoons,plate,pot,glass,plates,bowls,pan,mugs,pots,dish,workout clothes,work out clothes,running shoes,umbrella $Requests:phraseList interchangeable - vegetarian,gluten free,gluten friendly,without cheese on top,no tomatoes,with extra bacon,no meat,with extra ketchup,without meat,with extra ice,without nuts,with no lettuce or tomato,with the dressing on the side,on an english muffin,on gluten free bread,without jalapenos,with no mushrooms,without bacon,with milk,with sugar and milk,no dairy,dairy free,with extra cheese,without lettuce,without tomato,without chicken,with extra fruit,without fruit $Foods:phraseList interchangeable -- scrambled egg burrito,toasted bagel sandwich,butter croissant,yogurt parfait,cobb salad,chicken quesadilla,french fries,red pepper hummus,chocolate chip cookies,turkey sandwich,signature burger,garlic brushed salmon filet,smoked mac and cheese,lemon berry burst cake,ice cream,parfait,cookie,croissant,quesadilla,hummus,salmon +- scrambled egg burrito,toasted bagel sandwich,butter croissant,yogurt parfait,cobb salad,chicken quesadilla,french fries,red pepper hummus,chocolate chip cookies,turkey sandwich,signature burger,garlic brushed salmon filet,smoked mac and cheese,lemon berry burst cake,ice cream,parfait,cookie,croissant,quesadilla,hummus,salmon,coffee,cup of coffee > # List entities diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/ExtendStayDialog.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/ExtendStayDialog.cs index 801f58971c..7a6e174c2e 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/ExtendStayDialog.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/ExtendStayDialog.cs @@ -262,7 +262,7 @@ private async Task EndDialog(WaterfallStepContext sc, Cancella cardData.Title = string.Format(HospitalityStrings.UpdateReservation); // check out date moved confirmation - var reply = ResponseManager.GetCardResponse(ExtendStayResponses.ExtendStaySuccess, new Card("ReservationDetails", cardData), tokens); + var reply = ResponseManager.GetCardResponse(ExtendStayResponses.ExtendStaySuccess, new Card(GetCardName(sc.Context, "ReservationDetails"), cardData), tokens); await sc.Context.SendActivityAsync(reply); } diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/GetReservationDialog.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/GetReservationDialog.cs index 9b330e29c3..0746f7e8a0 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/GetReservationDialog.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/GetReservationDialog.cs @@ -42,7 +42,7 @@ private async Task ShowReservation(WaterfallStepContext sc, Ca cardData.Title = string.Format(HospitalityStrings.ReservationDetails); // send card with reservation details - var reply = ResponseManager.GetCardResponse(GetReservationResponses.ShowReservationDetails, new Card("ReservationDetails", cardData), null); + var reply = ResponseManager.GetCardResponse(GetReservationResponses.ShowReservationDetails, new Card(GetCardName(sc.Context, "ReservationDetails"), cardData), null); await sc.Context.SendActivityAsync(reply); return await sc.EndDialogAsync(); } diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/HospitalityDialogBase.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/HospitalityDialogBase.cs index 1cfd928272..1be2e38023 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/HospitalityDialogBase.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/HospitalityDialogBase.cs @@ -12,10 +12,12 @@ using Luis; using Microsoft.Bot.Builder; using Microsoft.Bot.Builder.Dialogs; +using Microsoft.Bot.Builder.Dialogs.Choices; using Microsoft.Bot.Builder.Skills; using Microsoft.Bot.Builder.Solutions.Authentication; using Microsoft.Bot.Builder.Solutions.Responses; using Microsoft.Bot.Builder.Solutions.Util; +using Microsoft.Bot.Connector; using Microsoft.Bot.Schema; namespace HospitalitySkill.Dialogs @@ -193,5 +195,16 @@ protected async Task HasCheckedOut(WaterfallStepContext sc, Ca return await sc.NextAsync(); } + + // Get card that renders for adaptive card 1.0 + protected string GetCardName(ITurnContext context, string name) + { + if (Channel.GetChannelId(context) == Channels.Msteams) + { + name += ".1.0"; + } + + return name; + } } } \ No newline at end of file diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/LateCheckOutDialog.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/LateCheckOutDialog.cs index 5fe80561a3..4bab980005 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/LateCheckOutDialog.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/LateCheckOutDialog.cs @@ -48,7 +48,7 @@ private async Task LateCheckOutPrompt(WaterfallStepContext sc, var cardData = userState.UserReservation; cardData.Title = string.Format(HospitalityStrings.ReservationDetails); - var reply = ResponseManager.GetCardResponse(LateCheckOutResponses.HasLateCheckOut, new Card("ReservationDetails", cardData), null); + var reply = ResponseManager.GetCardResponse(LateCheckOutResponses.HasLateCheckOut, new Card(GetCardName(sc.Context, "ReservationDetails"), cardData), null); await sc.Context.SendActivityAsync(reply); return await sc.EndDialogAsync(); @@ -112,7 +112,7 @@ private async Task EndDialog(WaterfallStepContext sc, Cancella cardData.Title = string.Format(HospitalityStrings.UpdateReservation); // check out time moved confirmation - var reply = ResponseManager.GetCardResponse(LateCheckOutResponses.MoveCheckOutSuccess, new Card("ReservationDetails", cardData), tokens); + var reply = ResponseManager.GetCardResponse(LateCheckOutResponses.MoveCheckOutSuccess, new Card(GetCardName(sc.Context, "ReservationDetails"), cardData), tokens); await sc.Context.SendActivityAsync(reply); } diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/RequestItemDialog.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/RequestItemDialog.cs index 1c8e008e4d..a6d7904175 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/RequestItemDialog.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/RequestItemDialog.cs @@ -181,13 +181,13 @@ private async Task EndDialog(WaterfallStepContext sc, Cancella Quantity = itemRequest.number == null ? 1 : (int)itemRequest.number[0] }; - roomItems.Add(new Card("RoomItemCard", roomItem)); + roomItems.Add(new Card(GetCardName(sc.Context, "RoomItemCard"), roomItem)); } await _hotelService.RequestItems(convState.ItemList); // if at least one item was available send this card reply - await sc.Context.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card("RequestItemCard"), null, "items", roomItems)); + await sc.Context.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card(GetCardName(sc.Context, "RequestItemCard")), null, "items", roomItems)); await sc.Context.SendActivityAsync(ResponseManager.GetResponse(RequestItemResponses.ItemsRequested)); } diff --git a/skills/src/csharp/experimental/hospitalityskill/Dialogs/RoomServiceDialog.cs b/skills/src/csharp/experimental/hospitalityskill/Dialogs/RoomServiceDialog.cs index 208db3e036..3fe086aa40 100644 --- a/skills/src/csharp/experimental/hospitalityskill/Dialogs/RoomServiceDialog.cs +++ b/skills/src/csharp/experimental/hospitalityskill/Dialogs/RoomServiceDialog.cs @@ -8,7 +8,10 @@ using HospitalitySkill.Services; using Microsoft.Bot.Builder; using Microsoft.Bot.Builder.Dialogs; +using Microsoft.Bot.Builder.Dialogs.Choices; using Microsoft.Bot.Builder.Solutions.Responses; +using Microsoft.Bot.Connector; +using Microsoft.Bot.Schema; using static Luis.HospitalityLuis._Entities; namespace HospitalitySkill.Dialogs @@ -57,9 +60,29 @@ private async Task MenuPrompt(WaterfallStepContext sc, Cancell // didn't order, prompt if 1 menu type not identified if (convState.FoodList.Count == 0 && string.IsNullOrWhiteSpace(menu?[0][0]) && menu?.Length != 1) { + var prompt = ResponseManager.GetResponse(RoomServiceResponses.MenuPrompt).Text; + + + var actions = new List() + { + new CardAction(type: ActionTypes.ImBack, title: "Breakfast", value: "Breakfast menu"), + new CardAction(type: ActionTypes.ImBack, title: "Lunch", value: "Lunch menu"), + new CardAction(type: ActionTypes.ImBack, title: "Dinner", value: "Dinner menu"), + new CardAction(type: ActionTypes.ImBack, title: "24 Hour", value: "24 hour menu") + }; + + var activity = MessageFactory.SuggestedActions(actions, prompt); + + // create hero card instead when channel does not support suggested actions + if (!Channel.SupportsSuggestedActions(sc.Context.Activity.ChannelId)) + { + var hero = new HeroCard(buttons: actions); + activity = MessageFactory.Attachment(hero.ToAttachment(), prompt); + } + return await sc.PromptAsync(DialogIds.MenuPrompt, new PromptOptions() { - Prompt = ResponseManager.GetResponse(RoomServiceResponses.MenuPrompt), + Prompt = (Activity)activity, RetryPrompt = ResponseManager.GetResponse(RoomServiceResponses.ChooseOneMenu) }); } @@ -93,11 +116,19 @@ private async Task ShowMenuCard(WaterfallStepContext sc, Cance List menuItems = new List(); foreach (var item in menu.Items) { - menuItems.Add(new Card("MenuItemCard", item)); + var cardName = GetCardName(sc.Context, "MenuItemCard"); + + // workaround for webchat not supporting hidden items on cards + if (Channel.GetChannelId(sc.Context) == Channels.Webchat) + { + cardName += ".1.0"; + } + + menuItems.Add(new Card(cardName, item)); } // show menu card - await sc.Context.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card("MenuCard", menu), null, "items", menuItems)); + await sc.Context.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card(GetCardName(sc.Context, "MenuCard"), menu), null, "items", menuItems)); // prompt for order return await sc.PromptAsync(DialogIds.FoodOrderPrompt, new PromptOptions() @@ -153,10 +184,17 @@ private async Task ValidateAddItems(PromptValidatorContext promptC private async Task ConfirmOrderPrompt(WaterfallStepContext sc, CancellationToken cancellationToken) { - return await sc.PromptAsync(DialogIds.ConfirmOrder, new PromptOptions() + var convState = await StateAccessor.GetAsync(sc.Context, () => new HospitalitySkillState()); + + if (convState.FoodList.Count > 0) { - Prompt = ResponseManager.GetResponse(RoomServiceResponses.ConfirmOrder) - }); + return await sc.PromptAsync(DialogIds.ConfirmOrder, new PromptOptions() + { + Prompt = ResponseManager.GetResponse(RoomServiceResponses.ConfirmOrder) + }); + } + + return await sc.NextAsync(false); } private async Task EndDialog(WaterfallStepContext sc, CancellationToken cancellationToken) @@ -205,7 +243,7 @@ private async Task ShowFoodOrder(ITurnContext turnContext) SpecialRequest = foodRequest.SpecialRequest == null ? null : foodRequest.SpecialRequest[0] }; - foodItems.Add(new Card("FoodItemCard", foodItemData)); + foodItems.Add(new Card(GetCardName(turnContext, "FoodItemCard"), foodItemData)); // add up bill totalFoodOrder.BillTotal += foodItemData.Price * foodItemData.Quantity; @@ -219,7 +257,7 @@ private async Task ShowFoodOrder(ITurnContext turnContext) if (convState.FoodList.Count > 0) { - await turnContext.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card("FoodOrderCard", totalFoodOrder), null, "items", foodItems)); + await turnContext.SendActivityAsync(ResponseManager.GetCardResponse(null, new Card(GetCardName(turnContext, "FoodOrderCard"), totalFoodOrder), null, "items", foodItems)); } } diff --git a/skills/src/csharp/experimental/hospitalityskill/HospitalitySkill.csproj b/skills/src/csharp/experimental/hospitalityskill/HospitalitySkill.csproj index e539f5fcda..d373575704 100644 --- a/skills/src/csharp/experimental/hospitalityskill/HospitalitySkill.csproj +++ b/skills/src/csharp/experimental/hospitalityskill/HospitalitySkill.csproj @@ -7,12 +7,19 @@ + + + + + + + @@ -37,12 +44,19 @@ + + + + + + + diff --git a/skills/src/csharp/experimental/musicskill/MusicSkill.csproj b/skills/src/csharp/experimental/musicskill/MusicSkill.csproj index b38adaf310..b2b92ccdb5 100644 --- a/skills/src/csharp/experimental/musicskill/MusicSkill.csproj +++ b/skills/src/csharp/experimental/musicskill/MusicSkill.csproj @@ -87,11 +87,6 @@ True deploy.ps1 - - True - True - deploy_cognitive_models.ps1 - True True @@ -107,11 +102,6 @@ True MainResponses.tt - - True - True - ResponseIdCollection.t4 - True True diff --git a/skills/src/csharp/experimental/newsskill/Dialogs/MainDialog.cs b/skills/src/csharp/experimental/newsskill/Dialogs/MainDialog.cs index 546f6159ca..08aebbd664 100644 --- a/skills/src/csharp/experimental/newsskill/Dialogs/MainDialog.cs +++ b/skills/src/csharp/experimental/newsskill/Dialogs/MainDialog.cs @@ -58,6 +58,9 @@ public MainDialog( { var state = await _stateAccessor.GetAsync(dc.Context, () => new NewsSkillState()); + // Populate state from SemanticAction as required + await PopulateStateFromSemanticAction(dc.Context); + // If dispatch result is general luis model _services.CognitiveModelSets["en"].LuisServices.TryGetValue("News", out var luisService); @@ -90,7 +93,7 @@ public MainDialog( turnResult = await dc.BeginDialogAsync(nameof(FavoriteTopicsDialog)); break; } - + case NewsLuis.Intent.FindArticles: { // send greeting response @@ -187,5 +190,19 @@ private async Task OnHelp(DialogContext dc) await _responder.ReplyWith(dc.Context, MainResponses.Help); return InterruptionAction.MessageSentToUser; } + + private async Task PopulateStateFromSemanticAction(ITurnContext context) + { + // Populating local state with data passed through semanticAction out of Activity + var activity = context.Activity; + var semanticAction = activity.SemanticAction; + if (semanticAction != null && semanticAction.Entities.ContainsKey("location")) + { + var location = semanticAction.Entities["location"]; + var locationObj = location.Properties["location"].ToString(); + var state = await _stateAccessor.GetAsync(context, () => new NewsSkillState()); + state.CurrentCoordinates = locationObj; + } + } } } \ No newline at end of file diff --git a/skills/src/csharp/experimental/newsskill/Dialogs/NewsDialogBase.cs b/skills/src/csharp/experimental/newsskill/Dialogs/NewsDialogBase.cs index c9b51fbb84..60b7534127 100644 --- a/skills/src/csharp/experimental/newsskill/Dialogs/NewsDialogBase.cs +++ b/skills/src/csharp/experimental/newsskill/Dialogs/NewsDialogBase.cs @@ -71,12 +71,18 @@ public async Task HandleDialogExceptions(WaterfallStepContext sc, Exc protected async Task GetMarket(WaterfallStepContext sc, CancellationToken cancellationToken) { var userState = await UserAccessor.GetAsync(sc.Context, () => new NewsSkillUserState()); + var convState = await ConvAccessor.GetAsync(sc.Context, () => new NewsSkillState()); // Check if there's already a location if (!string.IsNullOrWhiteSpace(userState.Market)) { return await sc.NextAsync(userState.Market); } + else if (!string.IsNullOrWhiteSpace(convState.CurrentCoordinates)) + { + // make maps service query with location coordinates instead of user input + return await sc.NextAsync(convState.CurrentCoordinates); + } // Prompt user for location return await sc.PromptAsync(nameof(TextPrompt), new PromptOptions() diff --git a/skills/src/csharp/experimental/newsskill/Models/NewsSkillState.cs b/skills/src/csharp/experimental/newsskill/Models/NewsSkillState.cs index 88657e0532..6c7fede1da 100644 --- a/skills/src/csharp/experimental/newsskill/Models/NewsSkillState.cs +++ b/skills/src/csharp/experimental/newsskill/Models/NewsSkillState.cs @@ -7,5 +7,7 @@ public NewsSkillState() } public Luis.NewsLuis LuisResult { get; set; } + + public string CurrentCoordinates { get; set; } } } diff --git a/skills/src/csharp/experimental/newsskill/manifestTemplate.json b/skills/src/csharp/experimental/newsskill/manifestTemplate.json index b4a86fb88f..693fc2af24 100644 --- a/skills/src/csharp/experimental/newsskill/manifestTemplate.json +++ b/skills/src/csharp/experimental/newsskill/manifestTemplate.json @@ -9,7 +9,14 @@ "id": "newsSkill_findArticles", "definition": { "description": "Find News articles.", - "slots": [], + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], "triggers": { "utteranceSources": [ { @@ -26,7 +33,14 @@ "id": "newsSkill_trendingArticles", "definition": { "description": "Show articles currently trending on social media.", - "slots": [], + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], "triggers": { "utteranceSources": [ { @@ -43,7 +57,14 @@ "id": "newsSkill_setFavoriteTopics", "definition": { "description": "Set the user's favorite news topic.", - "slots": [], + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], "triggers": { "utteranceSources": [ { @@ -60,7 +81,14 @@ "id": "newsSkill_showFavoriteTopics", "definition": { "description": "Show news articles of the user's currently set favorite topic.", - "slots": [], + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], "triggers": { "utteranceSources": [ { diff --git a/skills/src/csharp/experimental/weatherskill/Dialogs/MainDialog.cs b/skills/src/csharp/experimental/weatherskill/Dialogs/MainDialog.cs index a30966863d..ab67c44701 100644 --- a/skills/src/csharp/experimental/weatherskill/Dialogs/MainDialog.cs +++ b/skills/src/csharp/experimental/weatherskill/Dialogs/MainDialog.cs @@ -260,18 +260,15 @@ private async Task OnLogout(DialogContext dc) private async Task PopulateStateFromSkillContext(ITurnContext context) { - // If we have a SkillContext object populated from the SkillMiddleware we can retrieve requests slot (parameter) data - // and make available in local state as appropriate. - var skillContext = await _contextAccessor.GetAsync(context, () => new SkillContext()); - if (skillContext != null) + // Populating local state with data passed through semanticAction out of Activity + var activity = context.Activity; + var semanticAction = activity.SemanticAction; + if (semanticAction != null && semanticAction.Entities.ContainsKey("location")) { - // Example of populating local state with data passed through Skill Context - // if (skillContext.ContainsKey("Location")) - // { - // // Add to your local state - // var state = await _stateAccessor.GetAsync(context, () => new SkillState()); - // state.Location = skillContext["Location"]; - // } + var location = semanticAction.Entities["location"]; + var locationObj = location.Properties["location"].ToString(); + var state = await _stateAccessor.GetAsync(context, () => new SkillState()); + state.Geography = locationObj; } } diff --git a/skills/src/csharp/experimental/weatherskill/WeatherSkill.csproj b/skills/src/csharp/experimental/weatherskill/WeatherSkill.csproj index f5ec0d9c50..e4b6153b70 100644 --- a/skills/src/csharp/experimental/weatherskill/WeatherSkill.csproj +++ b/skills/src/csharp/experimental/weatherskill/WeatherSkill.csproj @@ -3,6 +3,7 @@ netcoreapp2.2 NU1701 + bf895f98-5182-4f36-a89d-d52cdde1a45c diff --git a/skills/src/csharp/experimental/weatherskill/manifestTemplate.json b/skills/src/csharp/experimental/weatherskill/manifestTemplate.json index 5ef406e0ca..b5fc87cc07 100644 --- a/skills/src/csharp/experimental/weatherskill/manifestTemplate.json +++ b/skills/src/csharp/experimental/weatherskill/manifestTemplate.json @@ -9,7 +9,14 @@ "id": "WeatherSkill_getForecast", "definition": { "description": "Showing the weather forecast.", - "slots": [], + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], "triggers": { "utteranceSources": [ { diff --git a/solutions/HospitalitySample/Content/NewUserGreeting.1.0.json b/solutions/HospitalitySample/Content/NewUserGreeting.1.0.json new file mode 100644 index 0000000000..a05e613192 --- /dev/null +++ b/solutions/HospitalitySample/Content/NewUserGreeting.1.0.json @@ -0,0 +1,45 @@ +{ + "type": "AdaptiveCard", + "id": "NewUserGreeting", + "backgroundImage": "", + "body": [ + { + "type": "Container", + "items": [ + { + "type": "Image", + "url": "", + "size": "Stretch" + } + ] + }, + { + "type": "Container", + "spacing": "None", + "backgroundImage": "", + "items": [ + { + "type": "TextBlock", + "id": "title", + "spacing": "Medium", + "size": "Large", + "weight": "Bolder", + "color": "Light", + "text": "Hi, I'm **your** Hospitality Assistant", + "wrap": true + }, + { + "type": "TextBlock", + "id": "body", + "size": "Medium", + "color": "Light", + "text": "Now that I'm up and running, explore the links here to learn what I can do.", + "wrap": true + } + ] + } + ], + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "version": "1.0", + "speak": "Hi, I'm **your** Hospitality Assistant. Now that I'm up and running, explore the links here to learn what I can do." +} \ No newline at end of file diff --git a/solutions/HospitalitySample/Content/NewUserGreeting.json b/solutions/HospitalitySample/Content/NewUserGreeting.json index 12c41afb01..8165dee179 100644 --- a/solutions/HospitalitySample/Content/NewUserGreeting.json +++ b/solutions/HospitalitySample/Content/NewUserGreeting.json @@ -42,23 +42,21 @@ "actions": [ { "type": "Action.Submit", - "title": "Get started", - "data": { - "action": "startOnboarding" - } + "title": "Send location", + "data": "\/event:{\"Name\": \"VA.Location\", \"Value\": \"47.639620,-122.130610\"}" }, { - "type": "Action.OpenUrl", - "title": "Documentation", - "url": "https://aka.ms/virtualassistantdocs" + "type": "Action.Submit", + "title": "Room service", + "data": "I want to order room service" }, { - "type": "Action.OpenUrl", - "title": "Skills", - "url": "https://aka.ms/botframeworkskills" + "type": "Action.Submit", + "title": "Today's weather", + "data": "What's the weather today?" } ], "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", "version": "1.0", - "speak": "Hi, I'm **your** Virtual Assistant. Now that I'm up and running, explore the links here to learn what I can do." + "speak": "Hi, I'm **your** Hospitality Assistant. Now that I'm up and running, explore the links here to learn what I can do." } \ No newline at end of file diff --git a/solutions/HospitalitySample/Deployment/Resources/QnA/en/hotel_FAQ.lu b/solutions/HospitalitySample/Deployment/Resources/QnA/en/hotel_FAQ.lu index d1f279f78d..895690ed33 100644 --- a/solutions/HospitalitySample/Deployment/Resources/QnA/en/hotel_FAQ.lu +++ b/solutions/HospitalitySample/Deployment/Resources/QnA/en/hotel_FAQ.lu @@ -1,4 +1,4 @@ -> ! Automatically generated by [LUDown CLI](https://github.com/Microsoft/botbuilder-tools/tree/master/Ludown), Tue Aug 06 2019 10:01:24 GMT-0700 (Pacific Daylight Time) +> ! Automatically generated by [LUDown CLI](https://github.com/Microsoft/botbuilder-tools/tree/master/Ludown), Mon Aug 26 2019 12:20:29 GMT-0700 (Pacific Daylight Time) > ! Source LUIS JSON file: Not Specified @@ -14,6 +14,7 @@ - Where is the fitness center? - When is the fitness center open - What hours is the gym open +- Where is the gymnasium? ```markdown Our fitness center is located on the 2nd floor, and is open 24 hours, 7 days a week.  @@ -73,27 +74,22 @@ We provide the following services and amenities: There are printers, copiers, and fax machines available in the 24-hour business center on the 1st floor. ``` -> Source: Editorial -## ? Where is breakfast served?  -- Where is the breakfast restaurant? - -```markdown -Breakfast is served in our restaurant on the 1st floor. -``` - > Source: Editorial ## ? What time is breakfast?  - When does the breakfast restaurant open? - When does breakfast close? ```markdown -Complimentary breakfast is served from 6:30am to 9:00am Monday through Friday, and from 7:00am to 9:30am on Saturday and Sunday. +Complimentary breakfast is served in our restaurant from 6:30am to 9:00am Monday through Friday, and from 7:00am to 9:30am on Saturday and Sunday. ``` > Source: Editorial ## ? Is there a pool? - Where is the pool? - When is the pool open? +- does this hotel have a pool +- I want to swim +- Please tell me where is the swimming pool ```markdown We have a seasonal outdoor pool on the 1st floor that is open Monday-Sunday from 7:00am to 11:00pm. diff --git a/solutions/HospitalitySample/HospitalitySample.csproj b/solutions/HospitalitySample/HospitalitySample.csproj index a685a4482e..95b5525322 100644 --- a/solutions/HospitalitySample/HospitalitySample.csproj +++ b/solutions/HospitalitySample/HospitalitySample.csproj @@ -78,6 +78,9 @@ + + Always + Always diff --git a/solutions/HospitalitySample/Responses/Main/MainResponses.cs b/solutions/HospitalitySample/Responses/Main/MainResponses.cs index ca8acbfd32..207d62e8ea 100644 --- a/solutions/HospitalitySample/Responses/Main/MainResponses.cs +++ b/solutions/HospitalitySample/Responses/Main/MainResponses.cs @@ -5,6 +5,7 @@ using System.IO; using AdaptiveCards; using Microsoft.Bot.Builder; +using Microsoft.Bot.Builder.Dialogs.Choices; using Microsoft.Bot.Builder.TemplateManager; using Microsoft.Bot.Schema; @@ -63,19 +64,25 @@ public static IMessageActivity BuildNewUserGreetingCard(ITurnContext turnContext { var introCard = File.ReadAllText(MainStrings.INTRO_PATH); var card = AdaptiveCard.FromJson(introCard).Card; + + // work around for adaptive card actions not working in Teams + if (!Channel.SupportsSuggestedActions(turnContext.Activity.ChannelId)) + { + introCard += File.ReadAllText(MainStrings.INTRO_PATH_TEAMS); + card = AdaptiveCard.FromJson(introCard).Card; + } + var attachment = new Attachment(AdaptiveCard.ContentType, content: card); var response = MessageFactory.Attachment(attachment, ssml: card.Speak, inputHint: InputHints.AcceptingInput); - - response.SuggestedActions = new SuggestedActions + if (Channel.SupportsSuggestedActions(turnContext.Activity.ChannelId)) { - Actions = new List() - { - new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_1, value: MainStrings.HELP_BTN_VALUE_1), - new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_2, value: MainStrings.HELP_BTN_VALUE_2), - new CardAction(type: ActionTypes.OpenUrl, title: MainStrings.HELP_BTN_TEXT_3, value: MainStrings.HELP_BTN_VALUE_3), - }, - }; + response.SuggestedActions = new SuggestedActions { Actions = GetCardActions() }; + } + else + { + response.Attachments.Add(new HeroCard(buttons: GetCardActions()).ToAttachment()); + } return response; } @@ -90,12 +97,7 @@ public static IMessageActivity BuildReturningUserGreetingCard(ITurnContext turnC response.SuggestedActions = new SuggestedActions { - Actions = new List() - { - new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_1, value: MainStrings.HELP_BTN_VALUE_1), - new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_2, value: MainStrings.HELP_BTN_VALUE_2), - new CardAction(type: ActionTypes.OpenUrl, title: MainStrings.HELP_BTN_TEXT_3, value: MainStrings.HELP_BTN_VALUE_3), - }, + Actions = GetCardActions() }; return response; @@ -107,21 +109,30 @@ public static IMessageActivity BuildHelpCard(ITurnContext turnContext, dynamic d { Title = MainStrings.HELP_TITLE, Text = MainStrings.HELP_TEXT, - }.ToAttachment(); + }; - var response = MessageFactory.Attachment(attachment, ssml: MainStrings.HELP_TEXT, inputHint: InputHints.AcceptingInput); + var response = MessageFactory.Attachment(attachment.ToAttachment(), ssml: MainStrings.HELP_TEXT, inputHint: InputHints.AcceptingInput); - response.SuggestedActions = new SuggestedActions + if (Channel.SupportsSuggestedActions(turnContext.Activity.ChannelId)) + { + response.SuggestedActions = new SuggestedActions { Actions = GetCardActions() }; + } + else { - Actions = new List() + response.Attachments.Add(new HeroCard(buttons: GetCardActions()).ToAttachment()); + } + + return response; + } + + private static List GetCardActions() + { + return new List() { new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_1, value: MainStrings.HELP_BTN_VALUE_1), new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_2, value: MainStrings.HELP_BTN_VALUE_2), - new CardAction(type: ActionTypes.OpenUrl, title: MainStrings.HELP_BTN_TEXT_3, value: MainStrings.HELP_BTN_VALUE_3), - }, - }; - - return response; + new CardAction(type: ActionTypes.ImBack, title: MainStrings.HELP_BTN_TEXT_3, value: MainStrings.HELP_BTN_VALUE_3), + }; } public class ResponseIds diff --git a/solutions/HospitalitySample/Responses/Main/MainStrings.Designer.cs b/solutions/HospitalitySample/Responses/Main/MainStrings.Designer.cs index 11329d5715..fd8611fc22 100644 --- a/solutions/HospitalitySample/Responses/Main/MainStrings.Designer.cs +++ b/solutions/HospitalitySample/Responses/Main/MainStrings.Designer.cs @@ -8,11 +8,10 @@ // //------------------------------------------------------------------------------ -namespace HospitalitySample.Responses.Main -{ +namespace HospitalitySample.Responses.Main { using System; - - + + /// /// A strongly-typed resource class, for looking up localized strings, etc. /// @@ -23,235 +22,202 @@ namespace HospitalitySample.Responses.Main [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "15.0.0.0")] [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] - public class MainStrings - { - + public class MainStrings { + private static global::System.Resources.ResourceManager resourceMan; - + private static global::System.Globalization.CultureInfo resourceCulture; - + [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] - internal MainStrings() - { + internal MainStrings() { } - + /// /// Returns the cached ResourceManager instance used by this class. /// [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] - public static global::System.Resources.ResourceManager ResourceManager - { - get - { - if (object.ReferenceEquals(resourceMan, null)) - { + public static global::System.Resources.ResourceManager ResourceManager { + get { + if (object.ReferenceEquals(resourceMan, null)) { global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("HospitalitySample.Responses.Main.MainStrings", typeof(MainStrings).Assembly); resourceMan = temp; } return resourceMan; } } - + /// /// Overrides the current thread's CurrentUICulture property for all /// resource lookups using this strongly typed resource class. /// [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] - public static global::System.Globalization.CultureInfo Culture - { - get - { + public static global::System.Globalization.CultureInfo Culture { + get { return resourceCulture; } - set - { + set { resourceCulture = value; } } - + /// /// Looks up a localized string similar to Ok, let's start over.. /// - public static string CANCELLED - { - get - { + public static string CANCELLED { + get { return ResourceManager.GetString("CANCELLED", resourceCulture); } } - + /// /// Looks up a localized string similar to What else can I help you with?. /// - public static string COMPLETED - { - get - { + public static string COMPLETED { + get { return ResourceManager.GetString("COMPLETED", resourceCulture); } } - + /// /// Looks up a localized string similar to I'm sorry, I'm not able to help with that.. /// - public static string CONFUSED - { - get - { + public static string CONFUSED { + get { return ResourceManager.GetString("CONFUSED", resourceCulture); } } - + /// /// Looks up a localized string similar to Sorry, it looks like something went wrong.. /// - public static string ERROR - { - get - { + public static string ERROR { + get { return ResourceManager.GetString("ERROR", resourceCulture); } } - + /// /// Looks up a localized string similar to Hi there!. /// - public static string GREETING - { - get - { + public static string GREETING { + get { return ResourceManager.GetString("GREETING", resourceCulture); } } - + /// /// Looks up a localized string similar to Hey, {0}!. /// - public static string GREETING_WITH_NAME - { - get - { + public static string GREETING_WITH_NAME { + get { return ResourceManager.GetString("GREETING_WITH_NAME", resourceCulture); } } - + /// - /// Looks up a localized string similar to Test LUIS. + /// Looks up a localized string similar to Extend Your Stay. /// - public static string HELP_BTN_TEXT_1 - { - get - { + public static string HELP_BTN_TEXT_1 { + get { return ResourceManager.GetString("HELP_BTN_TEXT_1", resourceCulture); } } - + /// - /// Looks up a localized string similar to Test QnA Maker. + /// Looks up a localized string similar to Find Local Events. /// - public static string HELP_BTN_TEXT_2 - { - get - { + public static string HELP_BTN_TEXT_2 { + get { return ResourceManager.GetString("HELP_BTN_TEXT_2", resourceCulture); } } - + /// - /// Looks up a localized string similar to Learn more. + /// Looks up a localized string similar to Recent News. /// - public static string HELP_BTN_TEXT_3 - { - get - { + public static string HELP_BTN_TEXT_3 { + get { return ResourceManager.GetString("HELP_BTN_TEXT_3", resourceCulture); } } - + /// - /// Looks up a localized string similar to Talk to a human. + /// Looks up a localized string similar to Can I extend my stay?. /// - public static string HELP_BTN_VALUE_1 - { - get - { + public static string HELP_BTN_VALUE_1 { + get { return ResourceManager.GetString("HELP_BTN_VALUE_1", resourceCulture); } } - + /// - /// Looks up a localized string similar to What is a Virtual Assistant?. + /// Looks up a localized string similar to Find me something to do nearby. /// - public static string HELP_BTN_VALUE_2 - { - get - { + public static string HELP_BTN_VALUE_2 { + get { return ResourceManager.GetString("HELP_BTN_VALUE_2", resourceCulture); } } - + /// - /// Looks up a localized string similar to https://docs.microsoft.com/en-us/azure/bot-service/?view=azure-bot-service-4.0. + /// Looks up a localized string similar to What's the news?. /// - public static string HELP_BTN_VALUE_3 - { - get - { + public static string HELP_BTN_VALUE_3 { + get { return ResourceManager.GetString("HELP_BTN_VALUE_3", resourceCulture); } } - + /// - /// Looks up a localized string similar to This card can be used to display information to help your user interact with your bot. The buttons below can be used for sample queries or links to external sites.. + /// Looks up a localized string similar to Try asking about the weather or ordering room service! Or try using the buttons below.. /// - public static string HELP_TEXT - { - get - { + public static string HELP_TEXT { + get { return ResourceManager.GetString("HELP_TEXT", resourceCulture); } } - + /// - /// Looks up a localized string similar to HELP. + /// Looks up a localized string similar to Help. /// - public static string HELP_TITLE - { - get - { + public static string HELP_TITLE { + get { return ResourceManager.GetString("HELP_TITLE", resourceCulture); } } - + /// /// Looks up a localized string similar to ./Content/NewUserGreeting.json. /// - public static string INTRO_PATH - { - get - { + public static string INTRO_PATH { + get { return ResourceManager.GetString("INTRO_PATH", resourceCulture); } } - + + /// + /// Looks up a localized string similar to ./Content/NewUserGreeting.1.0.json. + /// + public static string INTRO_PATH_TEAMS { + get { + return ResourceManager.GetString("INTRO_PATH_TEAMS", resourceCulture); + } + } + /// /// Looks up a localized string similar to ./Content/ReturningUserGreeting.json. /// - public static string INTRO_RETURNING - { - get - { + public static string INTRO_RETURNING { + get { return ResourceManager.GetString("INTRO_RETURNING", resourceCulture); } } - + /// /// Looks up a localized string similar to Ok, you're signed out.. /// - public static string LOGOUT - { - get - { + public static string LOGOUT { + get { return ResourceManager.GetString("LOGOUT", resourceCulture); } } diff --git a/solutions/HospitalitySample/Responses/Main/MainStrings.resx b/solutions/HospitalitySample/Responses/Main/MainStrings.resx index c177d896c5..5bfdadcd82 100644 --- a/solutions/HospitalitySample/Responses/Main/MainStrings.resx +++ b/solutions/HospitalitySample/Responses/Main/MainStrings.resx @@ -136,32 +136,35 @@ Hey, {0}! - Test LUIS + Extend Your Stay - Test QnA Maker + Find Local Events - Learn more + Recent News - Talk to a human + Can I extend my stay? - What is a Virtual Assistant? + Find me something to do nearby - https://docs.microsoft.com/en-us/azure/bot-service/?view=azure-bot-service-4.0 + What's the news? - This card can be used to display information to help your user interact with your bot. The buttons below can be used for sample queries or links to external sites. + Try asking about the weather or ordering room service! Or try using the buttons below. - HELP + Help ./Content/NewUserGreeting.json + + ./Content/NewUserGreeting.1.0.json + ./Content/ReturningUserGreeting.json diff --git a/solutions/HospitalitySample/Services/DispatchLuis.cs b/solutions/HospitalitySample/Services/DispatchLuis.cs index 12f69a647b..332f77af39 100644 --- a/solutions/HospitalitySample/Services/DispatchLuis.cs +++ b/solutions/HospitalitySample/Services/DispatchLuis.cs @@ -1,5 +1,6 @@ // // Code generated by LUISGen +// Tool github: https://github.com/microsoft/botbuilder-tools // Changes may cause incorrect behavior and will be lost if the code is // regenerated. // @@ -20,9 +21,11 @@ public enum Intent { q_hotel_FAQ, restaurantBookingSkill, pointOfInterestSkill, - hospitalitySkill, - WeatherSkill, + eventSkill, newsSkill, + WeatherSkill, + BingSearchSkill, + hospitalitySkill, None }; public Dictionary Intents; @@ -32,18 +35,24 @@ public class _Entities // Simple entities public string[] KEYWORD; public string[] ADDRESS; - public string[] Item; public string[] topic; public string[] site; + public string[] Item; + + // Pattern.any + public string[] CelebrityNamePatten; + public string[] MovieTitlePatten; // Instance public class _Instance { public InstanceData[] KEYWORD; public InstanceData[] ADDRESS; - public InstanceData[] Item; public InstanceData[] topic; public InstanceData[] site; + public InstanceData[] Item; + public InstanceData[] CelebrityNamePatten; + public InstanceData[] MovieTitlePatten; } [JsonProperty("$instance")] public _Instance _instance; diff --git a/solutions/HospitalitySample/skills.json b/solutions/HospitalitySample/skills.json index ea80988b04..04e7f2b916 100644 --- a/solutions/HospitalitySample/skills.json +++ b/solutions/HospitalitySample/skills.json @@ -365,6 +365,101 @@ } } ] + }, + { + "id": "eventSkill", + "msaAppId": "", + "name": "Event Skill", + "endpoint": "", + "description": "The Event experimental skill uses the Eventbrite API to retrieve local event information.", + "authenticationConnections": [], + "actions": [ + { + "id": "EventSkill_getEvents", + "definition": { + "description": "Finds events happening in the area today.", + "slots": [ + { + "name": "location", + "types": [ + "string" + ] + } + ], + "triggers": { + "utteranceSources": [ + { + "locale": "en", + "source": [ + "Event#GetEvents" + ] + } + ] + } + } + } + ] + }, + { + "id": "BingSearchSkill", + "msaAppId": "", + "name": "BingSearchSkill", + "endpoint": "", + "description": "", + "authenticationConnections": [], + "actions": [ + { + "id": "BingSearchSkill_GetCelebrityInfo", + "definition": { + "description": "Get celebrity info.", + "slots": [], + "triggers": { + "utteranceSources": [ + { + "locale": "en", + "source": [ + "BingSearchSkill#GetCelebrityInfo" + ] + } + ] + } + } + }, + { + "id": "BingSearchSkill_SearchMovieInfo", + "definition": { + "description": "Search movie info.", + "slots": [], + "triggers": { + "utteranceSources": [ + { + "locale": "en", + "source": [ + "BingSearchSkill#SearchMovieInfo" + ] + } + ] + } + } + }, + { + "id": "BingSearchSkill_QueryQna", + "definition": { + "description": "Query Qna.", + "slots": [], + "triggers": { + "utteranceSources": [ + { + "locale": "en", + "source": [ + "BingSearchSkill#QueryQna" + ] + } + ] + } + } + } + ] } ] } \ No newline at end of file