Ask Mzansi Activates chat

This commit is contained in:
2025-06-02 10:33:34 +02:00
parent 42177b2528
commit e9b97a4e11
9 changed files with 197 additions and 100 deletions

View File

@@ -124,6 +124,10 @@ class _MihSearchBarState extends State<MihSearchBar> {
child: TextField( child: TextField(
controller: widget.controller, // Assign the controller controller: widget.controller, // Assign the controller
focusNode: widget.searchFocusNode, focusNode: widget.searchFocusNode,
onSubmitted: (value) {
widget.onPrefixIconTap
?.call(); // Call the prefix icon tap handler
},
style: TextStyle( style: TextStyle(
color: widget.hintColor, color: widget.hintColor,
fontWeight: FontWeight.w600, fontWeight: FontWeight.w600,

View File

@@ -207,3 +207,13 @@ class WalletArguments {
this.index, this.index,
); );
} }
class MzansiAiArguments {
final AppUser signedInUser;
final String? startUpQuestion;
MzansiAiArguments(
this.signedInUser,
this.startUpQuestion,
);
}

View File

@@ -229,7 +229,10 @@ class _MIHHomeLegacyState extends State<MIHHomeLegacy> {
onTap: () { onTap: () {
Navigator.of(context).pushNamed( Navigator.of(context).pushNamed(
'/mzansi-ai', '/mzansi-ai',
arguments: widget.signedInUser, arguments: MzansiAiArguments(
widget.signedInUser,
"",
),
); );
}, },
tileName: "Mzansi AI", tileName: "Mzansi AI",
@@ -464,7 +467,10 @@ class _MIHHomeLegacyState extends State<MIHHomeLegacy> {
onTap: () { onTap: () {
Navigator.of(context).pushNamed( Navigator.of(context).pushNamed(
'/mzansi-ai', '/mzansi-ai',
arguments: widget.signedInUser, arguments: MzansiAiArguments(
widget.signedInUser,
"",
),
); );
}, },
tileName: "Mzansi AI", tileName: "Mzansi AI",

View File

@@ -1,3 +1,4 @@
import 'package:flutter/services.dart';
import 'package:mzansi_innovation_hub/main.dart'; import 'package:mzansi_innovation_hub/main.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_layout/mih_single_child_scroll.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_layout/mih_single_child_scroll.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
@@ -45,6 +46,7 @@ class _MihBusinessHomeState extends State<MihBusinessHome>
late final AnimationController _marqueeController; late final AnimationController _marqueeController;
late final ScrollController _scrollController; late final ScrollController _scrollController;
final FocusNode _searchFocusNode = FocusNode(); final FocusNode _searchFocusNode = FocusNode();
final FocusNode _focusNode = FocusNode();
final String maintenanceMsg = final String maintenanceMsg =
"\tHeads up! We're doing maintenance on Thur, 15 May 2025 at 10 PM (CAT). MIH may be unavailable briefly."; "\tHeads up! We're doing maintenance on Thur, 15 May 2025 at 10 PM (CAT). MIH may be unavailable briefly.";
@@ -202,6 +204,22 @@ class _MihBusinessHomeState extends State<MihBusinessHome>
Widget getBody(double width, double height) { Widget getBody(double width, double height) {
return MihSingleChildScroll( return MihSingleChildScroll(
child: KeyboardListener(
focusNode: _focusNode,
autofocus: true,
onKeyEvent: (event) async {
if (event is KeyDownEvent &&
event.logicalKey == LogicalKeyboardKey.enter) {
Navigator.of(context).pushNamed(
'/mzansi-ai',
arguments: MzansiAiArguments(
widget.signedInUser,
searchController.text.isEmpty ? null : searchController.text,
),
);
searchController.clear();
}
},
child: Column( child: Column(
children: [ children: [
const SizedBox(height: 10), const SizedBox(height: 10),
@@ -214,9 +232,19 @@ class _MihBusinessHomeState extends State<MihBusinessHome>
prefixAltIcon: MihIcons.mzansiAi, prefixAltIcon: MihIcons.mzansiAi,
fillColor: fillColor:
MzanziInnovationHub.of(context)!.theme.secondaryColor(), MzanziInnovationHub.of(context)!.theme.secondaryColor(),
hintColor: MzanziInnovationHub.of(context)!.theme.primaryColor(), hintColor:
MzanziInnovationHub.of(context)!.theme.primaryColor(),
onPrefixIconTap: () { onPrefixIconTap: () {
print("Search Text: ${searchController.text}"); Navigator.of(context).pushNamed(
'/mzansi-ai',
arguments: MzansiAiArguments(
widget.signedInUser,
searchController.text.isEmpty
? null
: searchController.text,
),
);
searchController.clear();
}, },
searchFocusNode: _searchFocusNode, searchFocusNode: _searchFocusNode,
), ),
@@ -249,6 +277,7 @@ class _MihBusinessHomeState extends State<MihBusinessHome>
), ),
], ],
), ),
),
); );
} }
} }

View File

@@ -1,3 +1,4 @@
import 'package:flutter/services.dart';
import 'package:mzansi_innovation_hub/main.dart'; import 'package:mzansi_innovation_hub/main.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_layout/mih_single_child_scroll.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_layout/mih_single_child_scroll.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
@@ -53,6 +54,7 @@ class _MihPersonalHomeState extends State<MihPersonalHome>
late final AnimationController _marqueeController; late final AnimationController _marqueeController;
late final ScrollController _scrollController; late final ScrollController _scrollController;
final FocusNode _searchFocusNode = FocusNode(); final FocusNode _searchFocusNode = FocusNode();
final FocusNode _focusNode = FocusNode();
final String maintenanceMsg = final String maintenanceMsg =
"\tHeads up! We're doing maintenance on Thur, 15 May 2025 at 10 PM (CAT). MIH may be unavailable briefly."; "\tHeads up! We're doing maintenance on Thur, 15 May 2025 at 10 PM (CAT). MIH may be unavailable briefly.";
@@ -245,6 +247,22 @@ class _MihPersonalHomeState extends State<MihPersonalHome>
Widget getBody(double width, double height) { Widget getBody(double width, double height) {
return MihSingleChildScroll( return MihSingleChildScroll(
child: KeyboardListener(
focusNode: _focusNode,
autofocus: true,
onKeyEvent: (event) async {
if (event is KeyDownEvent &&
event.logicalKey == LogicalKeyboardKey.enter) {
Navigator.of(context).pushNamed(
'/mzansi-ai',
arguments: MzansiAiArguments(
widget.signedInUser,
searchController.text.isEmpty ? null : searchController.text,
),
);
searchController.clear();
}
},
child: Column( child: Column(
children: [ children: [
const SizedBox(height: 10), const SizedBox(height: 10),
@@ -257,9 +275,19 @@ class _MihPersonalHomeState extends State<MihPersonalHome>
prefixAltIcon: MihIcons.mzansiAi, prefixAltIcon: MihIcons.mzansiAi,
fillColor: fillColor:
MzanziInnovationHub.of(context)!.theme.secondaryColor(), MzanziInnovationHub.of(context)!.theme.secondaryColor(),
hintColor: MzanziInnovationHub.of(context)!.theme.primaryColor(), hintColor:
MzanziInnovationHub.of(context)!.theme.primaryColor(),
onPrefixIconTap: () { onPrefixIconTap: () {
print("Search Text: ${searchController.text}"); Navigator.of(context).pushNamed(
'/mzansi-ai',
arguments: MzansiAiArguments(
widget.signedInUser,
searchController.text.isEmpty
? null
: searchController.text,
),
);
searchController.clear();
}, },
searchFocusNode: _searchFocusNode, searchFocusNode: _searchFocusNode,
), ),
@@ -293,6 +321,7 @@ class _MihPersonalHomeState extends State<MihPersonalHome>
), ),
], ],
), ),
),
); );
} }
} }

View File

@@ -1,15 +1,19 @@
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_action.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_action.dart';
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tools.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tools.dart';
import 'package:mzansi_innovation_hub/mih_objects/app_user.dart'; import 'package:mzansi_innovation_hub/mih_objects/arguments.dart';
import 'package:mzansi_innovation_hub/mih_packages/mzansi_ai/package_tools/ai_chat.dart'; import 'package:mzansi_innovation_hub/mih_packages/mzansi_ai/package_tools/ai_chat.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
class MzansiAi extends StatefulWidget { class MzansiAi extends StatefulWidget {
final AppUser signedInUser; // final AppUser signedInUser;
// final String? startUpQuestion;
final MzansiAiArguments arguments;
const MzansiAi({ const MzansiAi({
super.key, super.key,
required this.signedInUser, required this.arguments,
// required this.signedInUser,
// this.startUpQuestion,
}); });
@override @override
@@ -46,7 +50,10 @@ class _MzansiAiState extends State<MzansiAi> {
List<Widget> getToolBody() { List<Widget> getToolBody() {
List<Widget> toolBodies = [ List<Widget> toolBodies = [
AiChat(signedInUser: widget.signedInUser), AiChat(
signedInUser: widget.arguments.signedInUser,
startUpQuestion: widget.arguments.startUpQuestion,
),
]; ];
return toolBodies; return toolBodies;
} }

View File

@@ -3,6 +3,7 @@ import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart'; import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
import 'package:mzansi_innovation_hub/mih_objects/app_user.dart'; import 'package:mzansi_innovation_hub/mih_objects/app_user.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:mzansi_innovation_hub/mih_objects/arguments.dart';
class MzansiAiTile extends StatefulWidget { class MzansiAiTile extends StatefulWidget {
final AppUser signedInUser; final AppUser signedInUser;
@@ -25,7 +26,10 @@ class _MzansiAiTileState extends State<MzansiAiTile> {
onTap: () { onTap: () {
Navigator.of(context).pushNamed( Navigator.of(context).pushNamed(
'/mzansi-ai', '/mzansi-ai',
arguments: widget.signedInUser, arguments: MzansiAiArguments(
widget.signedInUser,
"",
),
); );
}, },
appName: "Mzansi AI", appName: "Mzansi AI",

View File

@@ -21,9 +21,11 @@ import 'package:uuid/uuid.dart';
class AiChat extends StatefulWidget { class AiChat extends StatefulWidget {
final AppUser signedInUser; final AppUser signedInUser;
final String? startUpQuestion;
const AiChat({ const AiChat({
super.key, super.key,
required this.signedInUser, required this.signedInUser,
this.startUpQuestion,
}); });
@override @override
@@ -593,6 +595,12 @@ class _AiChatState extends State<AiChat> {
_loadMessages(); _loadMessages();
initTTS(); initTTS();
_ttsVoiceController.addListener(voiceSelected); _ttsVoiceController.addListener(voiceSelected);
if (widget.startUpQuestion != null && widget.startUpQuestion!.isNotEmpty) {
final partialText = types.PartialText(text: widget.startUpQuestion!);
WidgetsBinding.instance.addPostFrameCallback((_) {
_handleSendPressed(partialText);
});
}
} }
@override @override

View File

@@ -331,11 +331,11 @@ class RouteGenerator {
//Mzansi AI //Mzansi AI
case '/mzansi-ai': case '/mzansi-ai':
if (args is AppUser) { if (args is MzansiAiArguments) {
return MaterialPageRoute( return MaterialPageRoute(
settings: settings, settings: settings,
builder: (_) => MzansiAi( builder: (_) => MzansiAi(
signedInUser: args, arguments: args,
), ),
); );
} }