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,52 +204,79 @@ class _MihBusinessHomeState extends State<MihBusinessHome>
Widget getBody(double width, double height) { Widget getBody(double width, double height) {
return MihSingleChildScroll( return MihSingleChildScroll(
child: Column( child: KeyboardListener(
children: [ focusNode: _focusNode,
const SizedBox(height: 10), autofocus: true,
Padding( onKeyEvent: (event) async {
padding: EdgeInsets.symmetric(horizontal: width / 20), if (event is KeyDownEvent &&
child: MihSearchBar( event.logicalKey == LogicalKeyboardKey.enter) {
controller: searchController, Navigator.of(context).pushNamed(
hintText: "Ask Mzansi", '/mzansi-ai',
prefixIcon: Icons.search, arguments: MzansiAiArguments(
prefixAltIcon: MihIcons.mzansiAi, widget.signedInUser,
fillColor: searchController.text.isEmpty ? null : searchController.text,
MzanziInnovationHub.of(context)!.theme.secondaryColor(), ),
hintColor: MzanziInnovationHub.of(context)!.theme.primaryColor(), );
onPrefixIconTap: () { searchController.clear();
print("Search Text: ${searchController.text}"); }
}, },
searchFocusNode: _searchFocusNode, child: Column(
), children: [
), const SizedBox(height: 10),
const SizedBox(height: 10), Padding(
ValueListenableBuilder( padding: EdgeInsets.symmetric(horizontal: width / 20),
valueListenable: searchPackageName, child: MihSearchBar(
builder: (context, value, child) { controller: searchController,
List<Widget> filteredPackages = value hintText: "Ask Mzansi",
.where((package) => package.keys.first prefixIcon: Icons.search,
.toLowerCase() prefixAltIcon: MihIcons.mzansiAi,
.contains(searchController.text.toLowerCase())) fillColor:
.map((package) => package.values.first) MzanziInnovationHub.of(context)!.theme.secondaryColor(),
.toList(); hintColor:
return GridView.builder( MzanziInnovationHub.of(context)!.theme.primaryColor(),
physics: const NeverScrollableScrollPhysics(), onPrefixIconTap: () {
shrinkWrap: true, Navigator.of(context).pushNamed(
padding: getPadding(width, height), '/mzansi-ai',
// shrinkWrap: true, arguments: MzansiAiArguments(
itemCount: filteredPackages.length, widget.signedInUser,
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( searchController.text.isEmpty
maxCrossAxisExtent: packageSize, ? null
crossAxisSpacing: 5, : searchController.text,
), ),
itemBuilder: (context, index) { );
return filteredPackages[index]; searchController.clear();
}, },
); searchFocusNode: _searchFocusNode,
}, ),
), ),
], const SizedBox(height: 10),
ValueListenableBuilder(
valueListenable: searchPackageName,
builder: (context, value, child) {
List<Widget> filteredPackages = value
.where((package) => package.keys.first
.toLowerCase()
.contains(searchController.text.toLowerCase()))
.map((package) => package.values.first)
.toList();
return GridView.builder(
physics: const NeverScrollableScrollPhysics(),
shrinkWrap: true,
padding: getPadding(width, height),
// shrinkWrap: true,
itemCount: filteredPackages.length,
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(
maxCrossAxisExtent: packageSize,
crossAxisSpacing: 5,
),
itemBuilder: (context, index) {
return filteredPackages[index];
},
);
},
),
],
),
), ),
); );
} }

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,53 +247,80 @@ class _MihPersonalHomeState extends State<MihPersonalHome>
Widget getBody(double width, double height) { Widget getBody(double width, double height) {
return MihSingleChildScroll( return MihSingleChildScroll(
child: Column( child: KeyboardListener(
children: [ focusNode: _focusNode,
const SizedBox(height: 10), autofocus: true,
Padding( onKeyEvent: (event) async {
padding: EdgeInsets.symmetric(horizontal: width / 20), if (event is KeyDownEvent &&
child: MihSearchBar( event.logicalKey == LogicalKeyboardKey.enter) {
controller: searchController, Navigator.of(context).pushNamed(
hintText: "Ask Mzansi", '/mzansi-ai',
prefixIcon: Icons.search, arguments: MzansiAiArguments(
prefixAltIcon: MihIcons.mzansiAi, widget.signedInUser,
fillColor: searchController.text.isEmpty ? null : searchController.text,
MzanziInnovationHub.of(context)!.theme.secondaryColor(), ),
hintColor: MzanziInnovationHub.of(context)!.theme.primaryColor(), );
onPrefixIconTap: () { searchController.clear();
print("Search Text: ${searchController.text}"); }
}, },
searchFocusNode: _searchFocusNode, child: Column(
), children: [
), const SizedBox(height: 10),
const SizedBox(height: 10), Padding(
ValueListenableBuilder( padding: EdgeInsets.symmetric(horizontal: width / 20),
valueListenable: searchPackageName, child: MihSearchBar(
builder: (context, value, child) { controller: searchController,
List<Widget> filteredPackages = value hintText: "Ask Mzansi",
.where((package) => package.keys.first prefixIcon: Icons.search,
.toLowerCase() prefixAltIcon: MihIcons.mzansiAi,
.contains(searchController.text.toLowerCase())) fillColor:
.map((package) => package.values.first) MzanziInnovationHub.of(context)!.theme.secondaryColor(),
.toList(); hintColor:
return GridView.builder( MzanziInnovationHub.of(context)!.theme.primaryColor(),
physics: const NeverScrollableScrollPhysics(), onPrefixIconTap: () {
shrinkWrap: true, Navigator.of(context).pushNamed(
padding: getPadding(width, height), '/mzansi-ai',
// shrinkWrap: true, arguments: MzansiAiArguments(
itemCount: filteredPackages.length, widget.signedInUser,
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( searchController.text.isEmpty
maxCrossAxisExtent: packageSize, ? null
crossAxisSpacing: 5, : searchController.text,
), ),
itemBuilder: (context, index) { );
return filteredPackages[index]; searchController.clear();
// return personalPackages[index];
}, },
); searchFocusNode: _searchFocusNode,
}, ),
), ),
], const SizedBox(height: 10),
ValueListenableBuilder(
valueListenable: searchPackageName,
builder: (context, value, child) {
List<Widget> filteredPackages = value
.where((package) => package.keys.first
.toLowerCase()
.contains(searchController.text.toLowerCase()))
.map((package) => package.values.first)
.toList();
return GridView.builder(
physics: const NeverScrollableScrollPhysics(),
shrinkWrap: true,
padding: getPadding(width, height),
// shrinkWrap: true,
itemCount: filteredPackages.length,
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(
maxCrossAxisExtent: packageSize,
crossAxisSpacing: 5,
),
itemBuilder: (context, index) {
return filteredPackages[index];
// return personalPackages[index];
},
);
},
),
],
),
), ),
); );
} }

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,
), ),
); );
} }