Compare commits
272 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e5ce03e396 | |||
| c67529dbac | |||
| 84cb6b2e83 | |||
| 6e07a55885 | |||
| 8fb31695a8 | |||
| 4fafa35888 | |||
| 0cf9634c5d | |||
| 787a8057b2 | |||
| 5f911d51f9 | |||
| 8da29792b4 | |||
| 5e003a4d71 | |||
| fcf1bbbb15 | |||
| ff7f363983 | |||
| 843997e58c | |||
| 3778ebb261 | |||
| b1487839a7 | |||
| 221030eff3 | |||
| 5135629b33 | |||
| 281ea863e8 | |||
| 1c0dd6d328 | |||
| 07d4ba4afa | |||
| 6ad6b6ccbd | |||
| ce2575035f | |||
| baea2c9fdb | |||
| 27639cb964 | |||
| 1143d11054 | |||
| 213f3d418d | |||
| e33a62b909 | |||
| ebab9bae52 | |||
| 82c25c5406 | |||
| 3f0fc08a5c | |||
| f137ea41ac | |||
| a7effa3576 | |||
| 74341a9cc6 | |||
| c5267c0540 | |||
| d4ba3aaa03 | |||
| 103ccdc022 | |||
| f8a722eb50 | |||
| fdb28080e3 | |||
| 8a384921c5 | |||
| 4b47bf5288 | |||
| 141611b84d | |||
| a29d0afeb8 | |||
| eb93714022 | |||
| 071612a521 | |||
| 726a60ad25 | |||
| b897986c1f | |||
| 7d4d7fc713 | |||
| 91075255f4 | |||
| 5c2f19dcc4 | |||
| 58aebbeabe | |||
| 670480910b | |||
| 45c1b247d3 | |||
|
|
ecd24670e9 | ||
|
|
ab99518ff2 | ||
|
|
6ac6aff59e | ||
| 33ca7b613e | |||
|
|
e83b0b50ae | ||
|
|
bcd19de256 | ||
|
|
ae36879ec5 | ||
|
|
eda609cf7e | ||
|
|
e1744b09e3 | ||
| a3845df8bd | |||
| d7eebe152b | |||
| d8a807b43e | |||
| 5172017a54 | |||
| 535924691e | |||
| ca8e0f56ac | |||
| 63071325cb | |||
| 800cd635c2 | |||
| f46ce36861 | |||
| b69fd92b19 | |||
| 9e19dc0fa4 | |||
| 38c40e2dfe | |||
| 02bbc32d23 | |||
| 5b052a1fa9 | |||
| d5349d981c | |||
| d6c28b631a | |||
| 74c5276c94 | |||
| d5e349d218 | |||
| 44527c8f10 | |||
| fbb9d8573c | |||
| 010fc0bc74 | |||
| 45ac3f03e6 | |||
| eea3248525 | |||
| b945a34ad4 | |||
| e3ac1be71c | |||
| 3a955e67ef | |||
| 777043e2ca | |||
| 6a8b9c6902 | |||
| 07360dd308 | |||
| 7c59e2a5c8 | |||
| 0b57e10532 | |||
| 5681c6d73b | |||
| f44ff6443c | |||
| fe5b61f6bc | |||
| e7729a8ce8 | |||
| 227a2f7ae7 | |||
| f26c1eb01a | |||
| ca0f13a6df | |||
| 021a25f50c | |||
| 0a9f0c000e | |||
| 3ff670886c | |||
| a6d5e4ad35 | |||
| 456dff6402 | |||
| d3fdc83373 | |||
| 8704d4dd64 | |||
| 37920466ac | |||
| 56d54a3711 | |||
| aef501cd25 | |||
| 4f5761271c | |||
| 252e120b99 | |||
| b5c26c3e43 | |||
| 74be2fc559 | |||
| b519b99a91 | |||
| 72261af7b9 | |||
| ad96725478 | |||
| ef4c3102a9 | |||
| 9a75bcc810 | |||
| 586e67b369 | |||
| e8cae1a894 | |||
| cd8115c597 | |||
| 730c5d2bdf | |||
| cea8ccab5a | |||
| 4f168c5b0e | |||
| 99f8b1a3f9 | |||
| 30c06261c8 | |||
| c843c0a55d | |||
| 94ac83db9e | |||
| a3b8da5357 | |||
| 47bc23c029 | |||
| c16d8b6e91 | |||
| a6fe4499d0 | |||
| 004c2397c5 | |||
| d64193d1f8 | |||
| e330875c6f | |||
| 3d2addf1d6 | |||
| 3b6e1d22ec | |||
| ff913c0c54 | |||
| c45c933277 | |||
| 89d6999abf | |||
| 969ecf8fdc | |||
| ee7d3881e6 | |||
| cc3f18f7e2 | |||
| 08bfe1d417 | |||
| 3593011d6d | |||
| 5ae11f6625 | |||
| a61bca7d81 | |||
| fb3b033f45 | |||
| 91a2d0dd1e | |||
| 04c643561b | |||
| 526bd40255 | |||
| 1889f6cada | |||
| c1dd9a60f2 | |||
| ec869c54c3 | |||
| b440c0a42b | |||
| baa826c52c | |||
| 6b530529e3 | |||
| a633b52949 | |||
| c103d1979c | |||
| 82c01cb7bf | |||
| 43bc552698 | |||
| bb6a8e3090 | |||
| 565e9199d4 | |||
| c5de46040d | |||
| 934b1b2301 | |||
| 302152449f | |||
| 0f591bd111 | |||
|
|
ea04c000cf | ||
| a59d2bf336 | |||
| d5fcad8fa5 | |||
| 609828a26c | |||
| d0a5c6c858 | |||
| 4a293c0aa4 | |||
| d51d13a685 | |||
| bf6bda04a6 | |||
| b749dc62d9 | |||
| b82a5a72c7 | |||
| 56ab6bb950 | |||
| bb54605ddb | |||
| c4b3a12213 | |||
| 43d715f4f8 | |||
| a9d6ca1baa | |||
| dfcfced28f | |||
| d0474fed21 | |||
| f228307859 | |||
| ab9cbce41c | |||
| 70413cc294 | |||
| a4d510f023 | |||
| e99388711d | |||
| 00cd5488e3 | |||
| b69a52a5a8 | |||
| f5c05d7431 | |||
| 0403d8c4b5 | |||
| fb7bf4ad65 | |||
| 3417299989 | |||
| 66887d23f8 | |||
| f064730dcf | |||
| a49505c153 | |||
| 9d898d8dff | |||
| c517075197 | |||
| 2dc3396fcc | |||
| d318b17b18 | |||
| 4221ae73b9 | |||
| b49e93825e | |||
| 790bf0e4f8 | |||
| a0bd74b703 | |||
| 96daf77b45 | |||
| 77ffd95fb6 | |||
| 24cc7f64b5 | |||
| d76512c11f | |||
| e1e7542a6d | |||
| 121cde211f | |||
| b79decf04f | |||
| a26b826729 | |||
| cc3d1ae430 | |||
| 329e450f5b | |||
| 0a03b1c60c | |||
| 652d4bb2fd | |||
| 5903fcd31a | |||
| b0825fe7cb | |||
| 8a2debcaa8 | |||
| 609c8d46db | |||
| 1387fb1af5 | |||
| 1f73aa3b87 | |||
| 47ca6d7311 | |||
| 64443a5076 | |||
| 20d5bc4004 | |||
| 9257786191 | |||
| 66db154b02 | |||
| 5770f6c353 | |||
| c5713cf6e0 | |||
| 4fe544b35f | |||
| 6542f1c399 | |||
|
|
963a708ab8 | ||
|
|
2f0b11e5ba | ||
| d9b2598bca | |||
| 0c15bd3e90 | |||
| 0daffb5041 | |||
| 9485213d6e | |||
| 9a8f157451 | |||
| a6f6cdaace | |||
| effbac2c91 | |||
| f8de726959 | |||
| 01220144c3 | |||
| 3f93cc9a7a | |||
| 83a16bad79 | |||
| d2f93db580 | |||
| a3c1963d30 | |||
| 8e6ef25954 | |||
| 580fa027f2 | |||
| 9346cccd20 | |||
| 51f2435c33 | |||
| ba5772f3ac | |||
| 662d20617d | |||
| e6d1cf0268 | |||
| ed7c5a2269 | |||
| 98beb20827 | |||
| 76a54ce969 | |||
| c53f30035d | |||
| ec01e2212b | |||
| f840418caf | |||
| 76d24a8e2b | |||
| 821cbc0a0d | |||
| ea25598144 | |||
| 70fc306789 | |||
| ba46f38497 | |||
| 67692967c3 | |||
| cc51afadb3 | |||
| d9fb9dd758 | |||
| d75da5389a | |||
|
|
efaf9b3574 |
16
.gitignore
vendored
@@ -1,16 +1,14 @@
|
|||||||
# *database/auto.cnf
|
mih_minio/
|
||||||
# *database/binlog.index
|
mih_db/
|
||||||
# *database/mysql.sock
|
mih_git/
|
||||||
File_Storage
|
mih_nginx/
|
||||||
database/
|
mih_monitor/
|
||||||
|
mih_wp/
|
||||||
certbot/
|
certbot/
|
||||||
Firebase-emulator/
|
Firebase-emulator/
|
||||||
Mzansi_Mail/
|
Mzansi_Mail/
|
||||||
# database/ibdata1
|
|
||||||
# database/mysql.ibd
|
|
||||||
# database/undo*
|
|
||||||
# database/#innodb_redo/#ib_redo*
|
|
||||||
.venv
|
.venv
|
||||||
google-chrome-stable_current_amd64.deb
|
google-chrome-stable_current_amd64.deb
|
||||||
.env
|
.env
|
||||||
Frontend/android/app/.cxx/
|
Frontend/android/app/.cxx/
|
||||||
|
.DS_Store
|
||||||
17
.vscode/launch.json
vendored
@@ -6,14 +6,25 @@
|
|||||||
"configurations": [
|
"configurations": [
|
||||||
{
|
{
|
||||||
"name": "Debug",
|
"name": "Debug",
|
||||||
"cwd": "Frontend",
|
"cwd": "mih_ui",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "dart",
|
"type": "dart",
|
||||||
"program": "lib/main_dev.dart"
|
"program": "lib/main_dev.dart"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug (web)",
|
||||||
|
"cwd": "mih_ui",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "dart",
|
||||||
|
"program": "lib/main_dev.dart",
|
||||||
|
"args": [
|
||||||
|
"--web-port",
|
||||||
|
"1995"
|
||||||
|
]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "Profile",
|
"name": "Profile",
|
||||||
"cwd": "Frontend",
|
"cwd": "mih_ui",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "dart",
|
"type": "dart",
|
||||||
"flutterMode": "profile",
|
"flutterMode": "profile",
|
||||||
@@ -21,7 +32,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Release",
|
"name": "Release",
|
||||||
"cwd": "Frontend",
|
"cwd": "mih_ui",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "dart",
|
"type": "dart",
|
||||||
"flutterMode": "release",
|
"flutterMode": "release",
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
# This file tracks properties of this Flutter project.
|
|
||||||
# Used by Flutter tool to assess capabilities and perform upgrades etc.
|
|
||||||
#
|
|
||||||
# This file should be version controlled and should not be manually edited.
|
|
||||||
|
|
||||||
version:
|
|
||||||
revision: "8defaa71a77c16e8547abdbfad2053ce3a6e2d5b"
|
|
||||||
channel: "stable"
|
|
||||||
|
|
||||||
project_type: app
|
|
||||||
|
|
||||||
# Tracks metadata for the flutter migrate command
|
|
||||||
migration:
|
|
||||||
platforms:
|
|
||||||
- platform: root
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: android
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: ios
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: linux
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: macos
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: web
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
- platform: windows
|
|
||||||
create_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
base_revision: 8defaa71a77c16e8547abdbfad2053ce3a6e2d5b
|
|
||||||
|
|
||||||
# User provided section
|
|
||||||
|
|
||||||
# List of Local paths (relative to this file) that should be
|
|
||||||
# ignored by the migrate tool.
|
|
||||||
#
|
|
||||||
# Files that are not part of the templates will be ignored by default.
|
|
||||||
unmanaged_files:
|
|
||||||
- 'lib/main.dart'
|
|
||||||
- 'ios/Runner.xcodeproj/project.pbxproj'
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
# Install Operating system and dependencies
|
|
||||||
#FROM ubuntu:22.04
|
|
||||||
FROM debian:latest AS build-env
|
|
||||||
|
|
||||||
#ENV DEBIAN_FRONTEND=noninteractive
|
|
||||||
|
|
||||||
RUN apt-get update --fix-missing
|
|
||||||
|
|
||||||
RUN apt-get install -y curl git wget unzip gdb libstdc++6 libglu1-mesa fonts-droid-fallback
|
|
||||||
# RUN apt-get install -y curl git wget unzip libgconf-2-4 gdb libstdc++6 libglu1-mesa fonts-droid-fallback
|
|
||||||
RUN apt-get install python3 -y
|
|
||||||
|
|
||||||
# download Flutter SDK from Flutter Github repo
|
|
||||||
RUN git clone -b flutter-3.32-candidate.0 https://github.com/flutter/flutter.git /usr/local/flutter
|
|
||||||
# RUN git clone -b stable https://github.com/flutter/flutter.git /usr/local/flutter
|
|
||||||
|
|
||||||
# Set flutter environment path
|
|
||||||
ENV PATH="/usr/local/flutter/bin:/usr/local/flutter/bin/cache/dart-sdk/bin:${PATH}"
|
|
||||||
#ENV PATH "$PATH:/home/developer/flutter/bin"
|
|
||||||
|
|
||||||
RUN flutter doctor -v
|
|
||||||
|
|
||||||
# Enable flutter web
|
|
||||||
RUN flutter channel flutter-3.32-candidate.0
|
|
||||||
# RUN flutter channel stable
|
|
||||||
RUN flutter config --enable-web
|
|
||||||
|
|
||||||
# Copy files to container and build
|
|
||||||
RUN mkdir /app/
|
|
||||||
COPY . /app/
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
# RUN flutter upgrade
|
|
||||||
RUN flutter build web --release -t ./lib/main_prod.dart
|
|
||||||
# RUN flutter build web --release --web-renderer canvaskit -t ./lib/main_prod.dart
|
|
||||||
|
|
||||||
|
|
||||||
# RUN cd ..
|
|
||||||
|
|
||||||
# WORKDIR /app/build/web/
|
|
||||||
|
|
||||||
EXPOSE 83
|
|
||||||
|
|
||||||
RUN ["chmod", "+x", "/app/server/server.sh"]
|
|
||||||
|
|
||||||
ENTRYPOINT [ "/app/server/server.sh"]
|
|
||||||
|
|
||||||
# RUN ["python3", "-u", "/app/server/MIH_web_server.py"]
|
|
||||||
|
Before Width: | Height: | Size: 9.9 KiB |
|
Before Width: | Height: | Size: 6.1 KiB |
|
Before Width: | Height: | Size: 14 KiB |
|
Before Width: | Height: | Size: 24 KiB |
|
Before Width: | Height: | Size: 34 KiB |
|
Before Width: | Height: | Size: 3.9 KiB |
|
Before Width: | Height: | Size: 2.5 KiB |
|
Before Width: | Height: | Size: 5.4 KiB |
|
Before Width: | Height: | Size: 8.6 KiB |
|
Before Width: | Height: | Size: 12 KiB |
@@ -1,4 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<resources>
|
|
||||||
<color name="ic_launcher_background">#3A4454</color>
|
|
||||||
</resources>
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
# flutter pub run flutter_launcher_icons
|
|
||||||
flutter_launcher_icons:
|
|
||||||
image_path: "lib/mih_components/mih_package_components/assets/images/app_icon/mih_app_icon.png"
|
|
||||||
|
|
||||||
android: "launcher_icon"
|
|
||||||
# image_path_android: "assets/icon/icon.png"
|
|
||||||
min_sdk_android: 21 # android min sdk min:16, default 21
|
|
||||||
adaptive_icon_background: "#3A4454"
|
|
||||||
adaptive_icon_foreground: "lib/mih_components/mih_package_components/assets/images/app_icon/mih_app_icon.png"
|
|
||||||
# adaptive_icon_monochrome: "assets/icon/monochrome.png"
|
|
||||||
|
|
||||||
ios: true
|
|
||||||
image_path_ios: "lib/mih_components/mih_package_components/assets/images/app_icon/mih_app_icon.png"
|
|
||||||
remove_alpha_channel_ios: true
|
|
||||||
# image_path_ios_dark_transparent: "assets/icon/icon_dark.png"
|
|
||||||
# image_path_ios_tinted_grayscale: "assets/icon/icon_tinted.png"
|
|
||||||
# desaturate_tinted_to_grayscale_ios: true
|
|
||||||
|
|
||||||
web:
|
|
||||||
generate: true
|
|
||||||
image_path: "lib/mih_components/mih_package_components/assets/images/app_icon/circle_logo.png"
|
|
||||||
background_color: "#3A4454"
|
|
||||||
theme_color: "#3A4454"
|
|
||||||
|
|
||||||
windows:
|
|
||||||
generate: true
|
|
||||||
image_path: "lib/mih_components/mih_package_components/assets/images/app_icon/circle_logo.png"
|
|
||||||
icon_size: 48 # min:48, max:256, default: 48
|
|
||||||
|
|
||||||
macos:
|
|
||||||
generate: true
|
|
||||||
image_path: "lib/mih_components/mih_package_components/assets/images/app_icon/circle_logo.png"
|
|
||||||
|
Before Width: | Height: | Size: 126 KiB |
|
Before Width: | Height: | Size: 1.1 KiB |
|
Before Width: | Height: | Size: 2.0 KiB |
|
Before Width: | Height: | Size: 3.1 KiB |
|
Before Width: | Height: | Size: 1.5 KiB |
|
Before Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 4.8 KiB |
|
Before Width: | Height: | Size: 2.0 KiB |
|
Before Width: | Height: | Size: 4.3 KiB |
|
Before Width: | Height: | Size: 6.9 KiB |
|
Before Width: | Height: | Size: 2.6 KiB |
|
Before Width: | Height: | Size: 5.6 KiB |
|
Before Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 6.4 KiB |
|
Before Width: | Height: | Size: 6.9 KiB |
|
Before Width: | Height: | Size: 11 KiB |
|
Before Width: | Height: | Size: 3.9 KiB |
|
Before Width: | Height: | Size: 8.6 KiB |
|
Before Width: | Height: | Size: 4.1 KiB |
|
Before Width: | Height: | Size: 9.1 KiB |
|
Before Width: | Height: | Size: 10 KiB |
@@ -1,144 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/about_mih_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_access_controlls_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_authentication_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_banner_ad_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_calculator_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_calendar_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_mine_sweeper_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_ai_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_directory_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_wallet_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/patient_manager_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
import 'package:upgrader/upgrader.dart';
|
|
||||||
import 'mih_config/mih_env.dart';
|
|
||||||
import 'mih_config/mih_theme.dart';
|
|
||||||
|
|
||||||
class MzansiInnovationHub extends StatefulWidget {
|
|
||||||
final GoRouter router;
|
|
||||||
const MzansiInnovationHub({
|
|
||||||
super.key,
|
|
||||||
required this.router,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MzansiInnovationHub> createState() => _MzansiInnovationHubState();
|
|
||||||
|
|
||||||
// ignore: library_private_types_in_public_api
|
|
||||||
static _MzansiInnovationHubState? of(BuildContext context) =>
|
|
||||||
context.findAncestorStateOfType<_MzansiInnovationHubState>();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MzansiInnovationHubState extends State<MzansiInnovationHub> {
|
|
||||||
late MihTheme theme;
|
|
||||||
|
|
||||||
Color getPrimany() {
|
|
||||||
return MihColors.getPrimaryColor(theme.mode == "Dark");
|
|
||||||
}
|
|
||||||
|
|
||||||
String getTitle() {
|
|
||||||
if (AppEnviroment.getEnv() == "Dev") {
|
|
||||||
return "Dev | MIH App: Mzansi Innovation Hub";
|
|
||||||
} else {
|
|
||||||
return "MIH App: Mzansi Innovation Hub";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void changeTheme(ThemeMode themeMode) {
|
|
||||||
setState(() {
|
|
||||||
if (themeMode == ThemeMode.light) {
|
|
||||||
setState(() {
|
|
||||||
theme.mode = "Light";
|
|
||||||
});
|
|
||||||
} else if (themeMode == ThemeMode.dark) {
|
|
||||||
setState(() {
|
|
||||||
theme.mode = "Dark";
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
theme.mode = "Dark";
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
theme = MihTheme();
|
|
||||||
super.initState();
|
|
||||||
theme.mode = "Dark";
|
|
||||||
theme.platform = Theme.of(context).platform;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
double width = MediaQuery.sizeOf(context).width;
|
|
||||||
theme.setScreenType(width);
|
|
||||||
precacheImage(theme.loadingImage(), context);
|
|
||||||
return MultiProvider(
|
|
||||||
providers: [
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihAuthenticationProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MzansiProfileProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MzansiWalletProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MzansiAiProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MzansiDirectoryProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihBannerAdProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihCalculatorProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihAccessControllsProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihCalendarProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => AboutMihProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => MihMineSweeperProvider(),
|
|
||||||
),
|
|
||||||
ChangeNotifierProvider(
|
|
||||||
create: (context) => PatientManagerProvider(),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
child: MaterialApp.router(
|
|
||||||
title: getTitle(),
|
|
||||||
themeMode: ThemeMode.dark,
|
|
||||||
theme: theme.getThemeData(),
|
|
||||||
darkTheme: theme.getThemeData(),
|
|
||||||
debugShowCheckedModeBanner: false,
|
|
||||||
routerConfig: widget.router,
|
|
||||||
builder: (context, child) {
|
|
||||||
if (child == null) {
|
|
||||||
return const Scaffold(
|
|
||||||
body: Center(
|
|
||||||
child: CircularProgressIndicator(),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return UpgradeAlert(
|
|
||||||
navigatorKey: widget.router.routerDelegate.navigatorKey,
|
|
||||||
child: child,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MIHAction extends StatefulWidget {
|
|
||||||
final void Function()? onTap;
|
|
||||||
final double iconSize;
|
|
||||||
final Widget icon;
|
|
||||||
const MIHAction({
|
|
||||||
super.key,
|
|
||||||
required this.icon,
|
|
||||||
required this.iconSize,
|
|
||||||
required this.onTap,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHAction> createState() => _MIHActionState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHActionState extends State<MIHAction> {
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return IconButton(
|
|
||||||
iconSize: widget.iconSize,
|
|
||||||
padding: const EdgeInsets.all(0),
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
icon: widget.icon,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
|
|
||||||
class MIHBody extends StatefulWidget {
|
|
||||||
final bool borderOn;
|
|
||||||
final List<Widget> bodyItems;
|
|
||||||
const MIHBody({
|
|
||||||
super.key,
|
|
||||||
required this.borderOn,
|
|
||||||
required this.bodyItems,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHBody> createState() => _MIHBodyState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHBodyState extends State<MIHBody> {
|
|
||||||
//double paddingSize = 10;
|
|
||||||
|
|
||||||
double getHorizontalPaddingSize(Size screenSize) {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return 10;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// mobile
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return 10;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
double getVerticalPaddingSize(Size screenSize) {
|
|
||||||
// mobile
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return 10;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Decoration? getBoader() {
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 3.0),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
Size screenSize = MediaQuery.sizeOf(context);
|
|
||||||
return Padding(
|
|
||||||
padding: EdgeInsets.only(
|
|
||||||
left: getHorizontalPaddingSize(screenSize),
|
|
||||||
right: getHorizontalPaddingSize(screenSize),
|
|
||||||
bottom: getVerticalPaddingSize(screenSize),
|
|
||||||
top: 0,
|
|
||||||
),
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.only(
|
|
||||||
left: 10,
|
|
||||||
right: 10,
|
|
||||||
bottom: 10,
|
|
||||||
top: getVerticalPaddingSize(screenSize),
|
|
||||||
),
|
|
||||||
width: screenSize.width,
|
|
||||||
height: screenSize.height,
|
|
||||||
decoration: getBoader(),
|
|
||||||
child: ScrollConfiguration(
|
|
||||||
behavior: ScrollConfiguration.of(context).copyWith(scrollbars: false),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: widget.bodyItems,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MIHHeader extends StatefulWidget {
|
|
||||||
final MainAxisAlignment headerAlignment;
|
|
||||||
final List<Widget> headerItems;
|
|
||||||
const MIHHeader({
|
|
||||||
super.key,
|
|
||||||
required this.headerAlignment,
|
|
||||||
required this.headerItems,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHHeader> createState() => _MIHHeaderState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHHeaderState extends State<MIHHeader> {
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return SizedBox(
|
|
||||||
height: 50,
|
|
||||||
child: Row(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
mainAxisAlignment: widget.headerAlignment,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: widget.headerItems,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,152 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import '../../mih_packages/mih_home/components/mih_app_drawer.dart';
|
|
||||||
import 'mih_body.dart';
|
|
||||||
import 'mih_header.dart';
|
|
||||||
|
|
||||||
class MIHLayoutBuilder extends StatefulWidget {
|
|
||||||
final Widget actionButton;
|
|
||||||
final Widget? secondaryActionButton;
|
|
||||||
final MIHHeader header;
|
|
||||||
final MIHBody body;
|
|
||||||
final MIHAppDrawer? actionDrawer;
|
|
||||||
final Widget? secondaryActionDrawer;
|
|
||||||
final Widget? bottomNavBar;
|
|
||||||
final bool pullDownToRefresh;
|
|
||||||
final Future<void> Function() onPullDown;
|
|
||||||
//final String type;
|
|
||||||
const MIHLayoutBuilder({
|
|
||||||
super.key,
|
|
||||||
required this.actionButton,
|
|
||||||
required this.header,
|
|
||||||
required this.secondaryActionButton,
|
|
||||||
required this.body,
|
|
||||||
required this.actionDrawer,
|
|
||||||
required this.secondaryActionDrawer,
|
|
||||||
required this.bottomNavBar,
|
|
||||||
required this.pullDownToRefresh,
|
|
||||||
required this.onPullDown,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHLayoutBuilder> createState() => _MIHLayoutBuilderState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHLayoutBuilderState extends State<MIHLayoutBuilder> {
|
|
||||||
List<Widget> getList() {
|
|
||||||
List<Widget> temp = [];
|
|
||||||
temp.add(widget.header);
|
|
||||||
temp.add(widget.body);
|
|
||||||
return temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
// openTheDrawer() {
|
|
||||||
// _scaffoldKey.currentState!.openEndDrawer();
|
|
||||||
// }
|
|
||||||
|
|
||||||
Widget getLayoutHeader() {
|
|
||||||
List<Widget> temp = [];
|
|
||||||
temp.add(widget.actionButton);
|
|
||||||
temp.add(Flexible(child: widget.header));
|
|
||||||
if (widget.secondaryActionButton != null) {
|
|
||||||
temp.add(widget.secondaryActionButton!);
|
|
||||||
} else {
|
|
||||||
//print(widget.header.headerItems.length);
|
|
||||||
if (widget.header.headerItems.length == 1) {
|
|
||||||
temp.add(const SizedBox(
|
|
||||||
width: 50,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: temp,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody(double width, double height) {
|
|
||||||
if (widget.pullDownToRefresh == true) {
|
|
||||||
return SafeArea(
|
|
||||||
child: LayoutBuilder(builder: (context, BoxConstraints constraints) {
|
|
||||||
double newheight = constraints.maxHeight;
|
|
||||||
//print(newheight);
|
|
||||||
return RefreshIndicator(
|
|
||||||
onRefresh: widget.onPullDown,
|
|
||||||
child: ListView.builder(
|
|
||||||
itemCount: 1,
|
|
||||||
itemBuilder: (BuildContext context, int index) {
|
|
||||||
return SafeArea(
|
|
||||||
child: SizedBox(
|
|
||||||
width: width,
|
|
||||||
height: newheight,
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
getLayoutHeader(),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Expanded(child: widget.body),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
// child: SafeArea(
|
|
||||||
// child: SizedBox(
|
|
||||||
// width: width,
|
|
||||||
// height: height,
|
|
||||||
// child: Column(
|
|
||||||
// mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
// children: [
|
|
||||||
// getLayoutHeader(),
|
|
||||||
// Expanded(child: widget.body),
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return SafeArea(
|
|
||||||
child: SizedBox(
|
|
||||||
width: width,
|
|
||||||
height: height,
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
getLayoutHeader(),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Expanded(child: widget.body),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
Size screenSize = MediaQuery.sizeOf(context);
|
|
||||||
return Scaffold(
|
|
||||||
//drawerEnableOpenDragGesture: true,
|
|
||||||
drawer: widget.actionDrawer,
|
|
||||||
endDrawer: widget.secondaryActionDrawer,
|
|
||||||
body: getBody(screenSize.width, screenSize.height),
|
|
||||||
bottomNavigationBar: widget.bottomNavBar,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_window.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_yt_video_player.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MIHTile extends StatefulWidget {
|
|
||||||
final String tileName;
|
|
||||||
final String? videoID;
|
|
||||||
final Widget tileIcon;
|
|
||||||
final void Function() onTap;
|
|
||||||
// final Widget tileIcon;
|
|
||||||
final Color p;
|
|
||||||
final Color s;
|
|
||||||
|
|
||||||
const MIHTile({
|
|
||||||
super.key,
|
|
||||||
required this.onTap,
|
|
||||||
required this.tileName,
|
|
||||||
this.videoID,
|
|
||||||
required this.tileIcon,
|
|
||||||
required this.p,
|
|
||||||
required this.s,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHTile> createState() => _MIHTileState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHTileState extends State<MIHTile> {
|
|
||||||
late Color mainC;
|
|
||||||
late Color secondC;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
mainC = widget.p;
|
|
||||||
secondC = widget.s;
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
void displayHint() {
|
|
||||||
if (widget.videoID != null) {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageWindow(
|
|
||||||
fullscreen: false,
|
|
||||||
windowTitle: widget.tileName,
|
|
||||||
onWindowTapClose: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
windowBody: Column(
|
|
||||||
children: [
|
|
||||||
MIHYTVideoPlayer(videoYTLink: widget.videoID!),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
// print(
|
|
||||||
// "Tile Name: ${widget.tileName}\nTitle Type: ${widget.tileIcon.runtimeType.toString()}");
|
|
||||||
return FittedBox(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
AnimatedContainer(
|
|
||||||
//alignment: Alignment.center,
|
|
||||||
width: 250,
|
|
||||||
height: 250,
|
|
||||||
duration: const Duration(seconds: 2),
|
|
||||||
child: Material(
|
|
||||||
color: mainC,
|
|
||||||
// shadowColor:
|
|
||||||
// MihColors.getSecondaryColor(MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// elevation: 5,
|
|
||||||
borderRadius: BorderRadius.circular(80),
|
|
||||||
child: InkWell(
|
|
||||||
borderRadius: BorderRadius.circular(80),
|
|
||||||
// ho
|
|
||||||
onTap: widget.onTap,
|
|
||||||
onLongPress: () {
|
|
||||||
displayHint();
|
|
||||||
},
|
|
||||||
// hoverDuration: ,
|
|
||||||
splashColor: MihColors.getHighlightColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
highlightColor: MihColors.getHighlightColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
child: widget.tileIcon,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
SizedBox(
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
widget.tileName,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
softWrap: true,
|
|
||||||
overflow: TextOverflow.visible,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 40.0,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,164 +0,0 @@
|
|||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.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_alert.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/Example/package_tools/package_tool_one.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/Example/package_tools/package_tool_two.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/arguments.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class PackageTest extends StatefulWidget {
|
|
||||||
// final AppUser user;
|
|
||||||
// final Business business;
|
|
||||||
final TestArguments arguments;
|
|
||||||
const PackageTest({
|
|
||||||
super.key,
|
|
||||||
required this.arguments,
|
|
||||||
// required this.user,
|
|
||||||
// required this.business,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<PackageTest> createState() => _PackageTestState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _PackageTestState extends State<PackageTest> {
|
|
||||||
int _selcetedIndex = 0;
|
|
||||||
|
|
||||||
MihPackageAction getAction() {
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
iconSize: 35,
|
|
||||||
onTap: () {
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
extra: true,
|
|
||||||
);
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
// Navigator.of(context).pop();
|
|
||||||
// Navigator.of(context).popAndPushNamed(
|
|
||||||
// '/',
|
|
||||||
// arguments: AuthArguments(true, false),
|
|
||||||
// );
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools() {
|
|
||||||
Map<Widget, void Function()?> temp = Map();
|
|
||||||
temp[const Icon(Icons.inbox)] = () {
|
|
||||||
setState(() {
|
|
||||||
_selcetedIndex = 0;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
temp[const Icon(Icons.outbond)] = () {
|
|
||||||
setState(() {
|
|
||||||
_selcetedIndex = 1;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: _selcetedIndex,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void showAlert() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: 100,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
alertTitle: "Oops! Looks like some fields are missing.",
|
|
||||||
alertBody: Column(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"We noticed that some required fields are still empty. To ensure your request is processed smoothly, please fill out all the highlighted fields before submitting the form again.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
RichText(
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
children: <TextSpan>[
|
|
||||||
TextSpan(
|
|
||||||
text: "Here's a quick tip: ",
|
|
||||||
style: TextStyle(
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"))),
|
|
||||||
const TextSpan(text: "Look for fields with an asterisk ("),
|
|
||||||
TextSpan(
|
|
||||||
text: "*",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"))),
|
|
||||||
const TextSpan(
|
|
||||||
text: ") next to them, as these are mandatory."),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody() {
|
|
||||||
List<Widget> toolBodies = [
|
|
||||||
PackageToolOne(
|
|
||||||
user: widget.arguments.user,
|
|
||||||
business: widget.arguments.business,
|
|
||||||
),
|
|
||||||
const PackageToolTwo(),
|
|
||||||
];
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
List<String> toolTitles = [
|
|
||||||
"Tool One",
|
|
||||||
"Tool Two",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(),
|
|
||||||
appBody: getToolBody(),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
selectedbodyIndex: _selcetedIndex,
|
|
||||||
onIndexChange: (newValue) {
|
|
||||||
setState(() {
|
|
||||||
_selcetedIndex = newValue;
|
|
||||||
});
|
|
||||||
print("Index: $_selcetedIndex");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/app_user.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/arguments.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/business.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tile.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class TestPackageTile extends StatefulWidget {
|
|
||||||
final AppUser signedInUser;
|
|
||||||
final Business? business;
|
|
||||||
final double packageSize;
|
|
||||||
const TestPackageTile({
|
|
||||||
super.key,
|
|
||||||
required this.signedInUser,
|
|
||||||
required this.business,
|
|
||||||
required this.packageSize,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<TestPackageTile> createState() => _TestPackageTileState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _TestPackageTileState extends State<TestPackageTile> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageTile(
|
|
||||||
onTap: () {
|
|
||||||
context.goNamed(
|
|
||||||
'testPackage',
|
|
||||||
extra: TestArguments(
|
|
||||||
widget.signedInUser,
|
|
||||||
widget.business,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
// Navigator.of(context).pushNamed(
|
|
||||||
// '/package-dev',
|
|
||||||
// arguments: TestArguments(
|
|
||||||
// widget.signedInUser,
|
|
||||||
// widget.business,
|
|
||||||
// ),
|
|
||||||
// );
|
|
||||||
},
|
|
||||||
appName: "Test",
|
|
||||||
appIcon: Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
iconSize: widget.packageSize,
|
|
||||||
primaryColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
secondaryColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
Before Width: | Height: | Size: 30 KiB |
|
Before Width: | Height: | Size: 42 KiB |
|
Before Width: | Height: | Size: 34 KiB |
|
Before Width: | Height: | Size: 41 KiB |
|
Before Width: | Height: | Size: 41 KiB |
@@ -1,68 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihButton extends StatelessWidget {
|
|
||||||
final void Function()? onPressed;
|
|
||||||
final void Function()? onLongPressed;
|
|
||||||
final Color buttonColor;
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final double? borderRadius;
|
|
||||||
final double? elevation; // 0 = flat, higher = more shadow
|
|
||||||
final Widget child;
|
|
||||||
|
|
||||||
const MihButton({
|
|
||||||
super.key,
|
|
||||||
required this.onPressed,
|
|
||||||
this.onLongPressed,
|
|
||||||
required this.buttonColor,
|
|
||||||
this.width,
|
|
||||||
this.height,
|
|
||||||
this.borderRadius,
|
|
||||||
this.elevation,
|
|
||||||
required this.child,
|
|
||||||
});
|
|
||||||
Color _darkerColor(Color color, [double amount = .1]) {
|
|
||||||
final hsl = HSLColor.fromColor(color);
|
|
||||||
final hslDark = hsl.withLightness((hsl.lightness - amount).clamp(0.0, 1.0));
|
|
||||||
return hslDark.toColor();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final Color effectiveButtonColor = onPressed == null
|
|
||||||
? buttonColor.withValues(alpha: 0.6) // Example disabled color
|
|
||||||
: buttonColor;
|
|
||||||
final Color rippleColor = _darkerColor(effectiveButtonColor, 0.1);
|
|
||||||
final double radius = borderRadius ?? 25.0;
|
|
||||||
final double effectiveElevation =
|
|
||||||
onPressed == null ? 0.0 : (elevation ?? 4.0);
|
|
||||||
return MouseRegion(
|
|
||||||
cursor: onPressed == null
|
|
||||||
? SystemMouseCursors.basic
|
|
||||||
: SystemMouseCursors.click,
|
|
||||||
child: Material(
|
|
||||||
color: effectiveButtonColor,
|
|
||||||
borderRadius: BorderRadius.circular(radius),
|
|
||||||
elevation: effectiveElevation,
|
|
||||||
shadowColor: Colors.black,
|
|
||||||
child: InkWell(
|
|
||||||
borderRadius: BorderRadius.circular(radius),
|
|
||||||
splashColor: rippleColor,
|
|
||||||
highlightColor: rippleColor.withValues(alpha: 0.2),
|
|
||||||
hoverColor: rippleColor.withValues(alpha: 0.3),
|
|
||||||
onTap: onPressed,
|
|
||||||
onLongPress: onLongPressed,
|
|
||||||
child: Container(
|
|
||||||
width: width,
|
|
||||||
height: height,
|
|
||||||
padding: (width == null || height == null)
|
|
||||||
? const EdgeInsets.symmetric(horizontal: 24, vertical: 12)
|
|
||||||
: null,
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: child,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
import 'dart:io';
|
|
||||||
|
|
||||||
import 'package:file_picker/file_picker.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihCircleAvatar extends StatefulWidget {
|
|
||||||
final ImageProvider<Object>? imageFile;
|
|
||||||
final double width;
|
|
||||||
final bool editable;
|
|
||||||
final TextEditingController? fileNameController;
|
|
||||||
final onChange;
|
|
||||||
final PlatformFile? userSelectedfile;
|
|
||||||
final Color frameColor;
|
|
||||||
final Color? backgroundColor;
|
|
||||||
const MihCircleAvatar({
|
|
||||||
super.key,
|
|
||||||
required this.imageFile,
|
|
||||||
required this.width,
|
|
||||||
required this.editable,
|
|
||||||
required this.fileNameController,
|
|
||||||
required this.userSelectedfile,
|
|
||||||
required this.frameColor,
|
|
||||||
required this.backgroundColor,
|
|
||||||
required this.onChange,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihCircleAvatar> createState() => _MihCircleAvatarState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihCircleAvatarState extends State<MihCircleAvatar> {
|
|
||||||
late ImageProvider<Object>? imagePreview;
|
|
||||||
|
|
||||||
ImageProvider<Object>? getAvatar() {
|
|
||||||
// Color dark = const Color(0XFF3A4454);
|
|
||||||
if (widget.imageFile == null) {
|
|
||||||
return null;
|
|
||||||
// if (widget.backgroundColor == dark) {
|
|
||||||
// print("here in light icon");
|
|
||||||
// return const AssetImage(
|
|
||||||
// 'lib/mih_components/mih_package_components/assets/images/i-dont-know-light.png');
|
|
||||||
// } else {
|
|
||||||
// print("here in dark icon");
|
|
||||||
// return const AssetImage(
|
|
||||||
// 'lib/mih_components/mih_package_components/assets/images/i-dont-know-dark.png');
|
|
||||||
// }
|
|
||||||
} else {
|
|
||||||
return widget.imageFile;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
setState(() {
|
|
||||||
imagePreview = getAvatar();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
alignment: Alignment.center,
|
|
||||||
width: widget.width,
|
|
||||||
height: widget.width,
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
children: [
|
|
||||||
Visibility(
|
|
||||||
visible: imagePreview != null,
|
|
||||||
child: Positioned(
|
|
||||||
right: widget.width * 0.03,
|
|
||||||
child: CircleAvatar(
|
|
||||||
radius: widget.width / 2.2,
|
|
||||||
backgroundColor: widget.backgroundColor,
|
|
||||||
backgroundImage: imagePreview,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: imagePreview != null,
|
|
||||||
child: Icon(
|
|
||||||
size: widget.width,
|
|
||||||
MihIcons.mihRing,
|
|
||||||
color: widget.frameColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: imagePreview == null,
|
|
||||||
child: Icon(
|
|
||||||
MihIcons.iDontKnow,
|
|
||||||
size: widget.width,
|
|
||||||
color: widget.frameColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: widget.editable,
|
|
||||||
child: Positioned(
|
|
||||||
bottom: 0,
|
|
||||||
right: 0,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor: WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onPressed: () async {
|
|
||||||
try {
|
|
||||||
FilePickerResult? result =
|
|
||||||
await FilePicker.platform.pickFiles(
|
|
||||||
type: FileType.image,
|
|
||||||
);
|
|
||||||
// print("Here 1");
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.getPlatform() ==
|
|
||||||
"Web") {
|
|
||||||
// print("Here 2");
|
|
||||||
if (result == null) return;
|
|
||||||
// print("Here 3");
|
|
||||||
PlatformFile? selectedFile = result.files.first;
|
|
||||||
setState(() {
|
|
||||||
// print("Here 4");
|
|
||||||
widget.onChange(selectedFile);
|
|
||||||
// print("Here 5");
|
|
||||||
imagePreview = MemoryImage(selectedFile.bytes!);
|
|
||||||
});
|
|
||||||
|
|
||||||
setState(() {
|
|
||||||
widget.fileNameController!.text = selectedFile.name;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
if (result != null) {
|
|
||||||
File file = File(result.files.single.path!);
|
|
||||||
PlatformFile? androidFile = PlatformFile(
|
|
||||||
path: file.path,
|
|
||||||
name: file.path.split('/').last,
|
|
||||||
size: file.lengthSync(),
|
|
||||||
bytes: await file.readAsBytes(), // Read file bytes
|
|
||||||
//extension: fileExtension,
|
|
||||||
);
|
|
||||||
setState(() {
|
|
||||||
widget.onChange(androidFile);
|
|
||||||
imagePreview = FileImage(file);
|
|
||||||
});
|
|
||||||
|
|
||||||
setState(() {
|
|
||||||
widget.fileNameController!.text =
|
|
||||||
file.path.split('/').last;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
print("here in else");
|
|
||||||
// User canceled the picker
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
print("Here Error: $e");
|
|
||||||
}
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.edit,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,207 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihDateField extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final String labelText;
|
|
||||||
final bool required;
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final double? borderRadius;
|
|
||||||
final double? elevation;
|
|
||||||
final FormFieldValidator<String>? validator;
|
|
||||||
const MihDateField({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
required this.labelText,
|
|
||||||
required this.required,
|
|
||||||
this.width,
|
|
||||||
this.height,
|
|
||||||
this.borderRadius,
|
|
||||||
this.elevation,
|
|
||||||
this.validator,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihDateField> createState() => _MihDateFieldState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihDateFieldState extends State<MihDateField> {
|
|
||||||
FormFieldState<String>? _formFieldState;
|
|
||||||
|
|
||||||
Future<void> _selectDate(BuildContext context) async {
|
|
||||||
DateTime? picked = await showDatePicker(
|
|
||||||
context: context,
|
|
||||||
initialDate: widget.controller.text.isNotEmpty
|
|
||||||
? DateTime.tryParse(widget.controller.text) ?? DateTime.now()
|
|
||||||
: DateTime.now(),
|
|
||||||
firstDate: DateTime(2000),
|
|
||||||
lastDate: DateTime(2100),
|
|
||||||
);
|
|
||||||
if (picked != null) {
|
|
||||||
widget.controller.text = picked.toString().split(" ")[0];
|
|
||||||
_formFieldState?.didChange(widget.controller.text);
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Center(
|
|
||||||
child: SizedBox(
|
|
||||||
width: widget.width,
|
|
||||||
height: widget.height,
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.labelText,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (!widget.required)
|
|
||||||
Text(
|
|
||||||
"(Optional)",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
FormField<String>(
|
|
||||||
initialValue: widget.controller.text,
|
|
||||||
validator: widget.validator,
|
|
||||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
|
||||||
builder: (field) {
|
|
||||||
_formFieldState = field;
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Material(
|
|
||||||
elevation: widget.elevation ?? 4.0,
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(widget.borderRadius ?? 8.0),
|
|
||||||
child: TextFormField(
|
|
||||||
controller: widget.controller,
|
|
||||||
readOnly: true,
|
|
||||||
onTap: () => _selectDate(context),
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
),
|
|
||||||
decoration: InputDecoration(
|
|
||||||
suffixIcon: Icon(
|
|
||||||
Icons.calendar_today,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
errorStyle: const TextStyle(height: 0, fontSize: 0),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 10.0, vertical: 8.0),
|
|
||||||
filled: true,
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
border: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: field.hasError
|
|
||||||
? BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 2.0,
|
|
||||||
)
|
|
||||||
: BorderSide.none,
|
|
||||||
),
|
|
||||||
enabledBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide.none,
|
|
||||||
),
|
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: field.hasError
|
|
||||||
? MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark")
|
|
||||||
: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
errorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
focusedErrorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onChanged: (value) {
|
|
||||||
field.didChange(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (field.hasError)
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(left: 8.0, top: 4.0),
|
|
||||||
child: Text(
|
|
||||||
field.errorText ?? '',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,248 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihDropdownField extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final String hintText;
|
|
||||||
final bool requiredText;
|
|
||||||
final List<String> dropdownOptions;
|
|
||||||
final bool editable;
|
|
||||||
final bool enableSearch;
|
|
||||||
final FormFieldValidator<String>? validator;
|
|
||||||
final Function(String?)? onSelected;
|
|
||||||
|
|
||||||
const MihDropdownField({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
required this.hintText,
|
|
||||||
required this.dropdownOptions,
|
|
||||||
required this.requiredText,
|
|
||||||
required this.editable,
|
|
||||||
required this.enableSearch,
|
|
||||||
this.validator,
|
|
||||||
this.onSelected,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihDropdownField> createState() => _MihDropdownFieldState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihDropdownFieldState extends State<MihDropdownField> {
|
|
||||||
late List<DropdownMenuEntry<String>> menu;
|
|
||||||
|
|
||||||
List<DropdownMenuEntry<String>> buildMenuOptions(List<String> options) {
|
|
||||||
List<DropdownMenuEntry<String>> menuList = [];
|
|
||||||
for (final i in options) {
|
|
||||||
menuList.add(DropdownMenuEntry(
|
|
||||||
value: i,
|
|
||||||
label: i,
|
|
||||||
style: ButtonStyle(
|
|
||||||
foregroundColor: WidgetStatePropertyAll(MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark")),
|
|
||||||
),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
return menuList;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void didChangeDependencies() {
|
|
||||||
super.didChangeDependencies();
|
|
||||||
menu = buildMenuOptions(widget.dropdownOptions);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
menu = widget.dropdownOptions
|
|
||||||
.map((e) => DropdownMenuEntry(value: e, label: e))
|
|
||||||
.toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.hintText,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (!widget.requiredText)
|
|
||||||
Text(
|
|
||||||
"(Optional)",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
FormField<String>(
|
|
||||||
validator: widget.validator,
|
|
||||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
|
||||||
initialValue: widget.controller.text,
|
|
||||||
builder: (field) {
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Theme(
|
|
||||||
data: Theme.of(context).copyWith(
|
|
||||||
textSelectionTheme: TextSelectionThemeData(
|
|
||||||
cursorColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
selectionColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")
|
|
||||||
.withValues(alpha: 0.3),
|
|
||||||
selectionHandleColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: DropdownMenu(
|
|
||||||
controller: widget.controller,
|
|
||||||
dropdownMenuEntries: menu,
|
|
||||||
enableSearch: widget.enableSearch,
|
|
||||||
enableFilter: widget.enableSearch,
|
|
||||||
enabled: widget.editable,
|
|
||||||
textInputAction: widget.enableSearch
|
|
||||||
? TextInputAction.search
|
|
||||||
: TextInputAction.none,
|
|
||||||
requestFocusOnTap: widget.enableSearch,
|
|
||||||
menuHeight: 400,
|
|
||||||
expandedInsets: EdgeInsets.zero,
|
|
||||||
textStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
),
|
|
||||||
trailingIcon: Icon(
|
|
||||||
Icons.arrow_drop_down,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
selectedTrailingIcon: Icon(
|
|
||||||
Icons.arrow_drop_up,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
leadingIcon: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
widget.controller.clear();
|
|
||||||
field.didChange('');
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.delete_outline_rounded,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onSelected: (String? selectedValue) {
|
|
||||||
field.didChange(selectedValue);
|
|
||||||
widget.onSelected?.call(selectedValue);
|
|
||||||
},
|
|
||||||
menuStyle: MenuStyle(
|
|
||||||
backgroundColor: WidgetStatePropertyAll(
|
|
||||||
MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")),
|
|
||||||
side: WidgetStatePropertyAll(
|
|
||||||
BorderSide(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 1.0),
|
|
||||||
),
|
|
||||||
shape: WidgetStatePropertyAll(
|
|
||||||
RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
10), // Increase for more roundness
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
inputDecorationTheme: InputDecorationTheme(
|
|
||||||
errorStyle: const TextStyle(height: 0, fontSize: 0),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 10.0, vertical: 8.0),
|
|
||||||
filled: true,
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
enabledBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
borderSide: BorderSide.none,
|
|
||||||
),
|
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: field.hasError
|
|
||||||
? MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")
|
|
||||||
: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
errorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
focusedErrorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (field.hasError)
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(left: 8.0, top: 4.0),
|
|
||||||
child: Text(
|
|
||||||
field.errorText ?? '',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_speed_dial/flutter_speed_dial.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihFloatingMenu extends StatefulWidget {
|
|
||||||
final IconData? icon;
|
|
||||||
final double? iconSize;
|
|
||||||
final AnimatedIconData? animatedIcon;
|
|
||||||
final SpeedDialDirection? direction;
|
|
||||||
final List<SpeedDialChild> children;
|
|
||||||
const MihFloatingMenu({
|
|
||||||
super.key,
|
|
||||||
this.icon,
|
|
||||||
this.iconSize,
|
|
||||||
this.animatedIcon,
|
|
||||||
this.direction,
|
|
||||||
required this.children,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihFloatingMenu> createState() => _MihFloatingMenuState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihFloatingMenuState extends State<MihFloatingMenu> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return SpeedDial(
|
|
||||||
icon: widget.icon,
|
|
||||||
buttonSize: Size(widget.iconSize ?? 56.0, widget.iconSize ?? 56.0),
|
|
||||||
animatedIcon: widget.animatedIcon,
|
|
||||||
direction: widget.direction ?? SpeedDialDirection.up,
|
|
||||||
activeIcon: Icons.close,
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
activeBackgroundColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
foregroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
overlayColor: Colors.black,
|
|
||||||
overlayOpacity: 0.5,
|
|
||||||
children: widget.children,
|
|
||||||
onOpen: () => debugPrint('OPENING DIAL'),
|
|
||||||
onClose: () => debugPrint('DIAL CLOSED'),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihForm extends StatefulWidget {
|
|
||||||
final GlobalKey<FormState> formKey;
|
|
||||||
final List<Widget> formFields;
|
|
||||||
const MihForm({
|
|
||||||
super.key,
|
|
||||||
required this.formKey,
|
|
||||||
required this.formFields,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihForm> createState() => _MihFormState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihFormState extends State<MihForm> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Form(
|
|
||||||
key: widget.formKey,
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: widget.formFields,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
import 'package:flutter/widgets.dart'; // You need this import for IconData
|
|
||||||
|
|
||||||
class MihIcons {
|
|
||||||
MihIcons._(); // This makes the class non-instantiable (good practice for utility classes)
|
|
||||||
|
|
||||||
// This MUST match the 'family' name you specify in pubspec.yaml
|
|
||||||
static const _mihFontFam = 'MihIcons';
|
|
||||||
// Set to your package name ONLY if this font is part of a separate package you created
|
|
||||||
static const String? _mihFontPkg = null;
|
|
||||||
|
|
||||||
// IconData constants based on your style.css file
|
|
||||||
// Note: We convert the hex code from CSS (\eXXX) to an integer (0xeXXX)
|
|
||||||
|
|
||||||
static const IconData mineSweeper =
|
|
||||||
IconData(0xe900, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData mzansiDirectory =
|
|
||||||
IconData(0xe901, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData personalProfile =
|
|
||||||
IconData(0xe902, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData aboutMih =
|
|
||||||
IconData(0xe903, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData accessControl =
|
|
||||||
IconData(0xe904, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData businessProfile =
|
|
||||||
IconData(0xe905, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData businessSetup =
|
|
||||||
IconData(0xe906, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData calculator =
|
|
||||||
IconData(0xe907, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData calendar =
|
|
||||||
IconData(0xe908, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData iDontKnow =
|
|
||||||
IconData(0xe909, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData mihLogo =
|
|
||||||
IconData(0xe90a, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData mihRing =
|
|
||||||
IconData(0xe90b, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData mzansiAi =
|
|
||||||
IconData(0xe90c, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData mzansiWallet =
|
|
||||||
IconData(0xe90d, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData notifications =
|
|
||||||
IconData(0xe90e, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData patientManager =
|
|
||||||
IconData(0xe90f, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData patientProfile =
|
|
||||||
IconData(0xe910, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
|
|
||||||
static const IconData profileSetup =
|
|
||||||
IconData(0xe911, fontFamily: _mihFontFam, fontPackage: _mihFontPkg);
|
|
||||||
}
|
|
||||||
@@ -1,238 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_text_form_field.dart';
|
|
||||||
|
|
||||||
class MihNumericStepper extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final Color fillColor;
|
|
||||||
final Color inputColor;
|
|
||||||
final String hintText;
|
|
||||||
final bool requiredText;
|
|
||||||
final double? width;
|
|
||||||
final int? minValue;
|
|
||||||
final int? maxValue;
|
|
||||||
final bool validationOn;
|
|
||||||
const MihNumericStepper({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
required this.fillColor,
|
|
||||||
required this.inputColor,
|
|
||||||
required this.hintText,
|
|
||||||
required this.requiredText,
|
|
||||||
this.width,
|
|
||||||
this.minValue,
|
|
||||||
this.maxValue,
|
|
||||||
required this.validationOn,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihNumericStepper> createState() => _MihNumericStepperState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihNumericStepperState extends State<MihNumericStepper> {
|
|
||||||
late int _currentValue;
|
|
||||||
late bool error;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
widget.controller.removeListener(_syncCurrentValue);
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_currentValue =
|
|
||||||
int.tryParse(widget.controller.text) ?? widget.minValue ?? 0;
|
|
||||||
widget.controller.text = _currentValue.toString();
|
|
||||||
int.tryParse(widget.controller.text) ?? widget.minValue ?? 0;
|
|
||||||
widget.controller.addListener(_syncCurrentValue);
|
|
||||||
// print("Current Value: $_currentValue");
|
|
||||||
}
|
|
||||||
|
|
||||||
void _syncCurrentValue() {
|
|
||||||
final newValue =
|
|
||||||
int.tryParse(widget.controller.text) ?? widget.minValue ?? 0;
|
|
||||||
if (newValue != _currentValue) {
|
|
||||||
setState(() {
|
|
||||||
_currentValue = newValue;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Column(
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.hintText,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 18,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
children: [
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
25), // Optional: rounds the corners
|
|
||||||
boxShadow: const [
|
|
||||||
BoxShadow(
|
|
||||||
color: Color.fromARGB(60, 0, 0,
|
|
||||||
0), // 0.2 opacity = 51 in alpha (255 * 0.2)
|
|
||||||
spreadRadius: -2,
|
|
||||||
blurRadius: 10,
|
|
||||||
offset: Offset(0, 5),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
left: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 40,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor: WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")),
|
|
||||||
),
|
|
||||||
color: widget.inputColor,
|
|
||||||
iconSize: 20,
|
|
||||||
onPressed: () {
|
|
||||||
print("Current Value: $_currentValue");
|
|
||||||
if (_currentValue >= (widget.minValue ?? 0)) {
|
|
||||||
setState(() {
|
|
||||||
widget.controller.text =
|
|
||||||
(_currentValue - 1).toString();
|
|
||||||
_currentValue =
|
|
||||||
int.tryParse(widget.controller.text)!;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
print("New Current Value: $_currentValue");
|
|
||||||
},
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.remove,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: _currentValue < (widget.minValue ?? 0) ||
|
|
||||||
(widget.maxValue != null &&
|
|
||||||
_currentValue > widget.maxValue!),
|
|
||||||
child: const SizedBox(
|
|
||||||
height: 21,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(width: 15),
|
|
||||||
Expanded(
|
|
||||||
child: MihTextFormField(
|
|
||||||
width: widget.width,
|
|
||||||
fillColor: widget.fillColor,
|
|
||||||
inputColor: widget.inputColor,
|
|
||||||
controller: widget.controller,
|
|
||||||
hintText: null,
|
|
||||||
requiredText: widget.requiredText,
|
|
||||||
readOnly: true,
|
|
||||||
numberMode: true,
|
|
||||||
textIputAlignment: TextAlign.center,
|
|
||||||
validator: (value) {
|
|
||||||
if (widget.validationOn) {
|
|
||||||
return MihValidationServices().validateNumber(
|
|
||||||
value, widget.minValue, widget.maxValue);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
25), // Optional: rounds the corners
|
|
||||||
boxShadow: const [
|
|
||||||
BoxShadow(
|
|
||||||
color: Color.fromARGB(60, 0, 0,
|
|
||||||
0), // 0.2 opacity = 51 in alpha (255 * 0.2)
|
|
||||||
spreadRadius: -2,
|
|
||||||
blurRadius: 10,
|
|
||||||
offset: Offset(0, 5),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
left: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 40,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor: WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")),
|
|
||||||
),
|
|
||||||
color: widget.inputColor,
|
|
||||||
iconSize: 20,
|
|
||||||
onPressed: () {
|
|
||||||
print("Current Value: $_currentValue");
|
|
||||||
if (widget.maxValue == null ||
|
|
||||||
_currentValue <= widget.maxValue!) {
|
|
||||||
setState(() {
|
|
||||||
widget.controller.text =
|
|
||||||
(_currentValue + 1).toString();
|
|
||||||
_currentValue =
|
|
||||||
int.tryParse(widget.controller.text)!;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
print("New Current Value: $_currentValue");
|
|
||||||
},
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.add,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: _currentValue < (widget.minValue ?? 0) ||
|
|
||||||
(widget.maxValue != null &&
|
|
||||||
_currentValue > widget.maxValue!),
|
|
||||||
child: const SizedBox(
|
|
||||||
height: 21,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,217 +0,0 @@
|
|||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:ken_logger/ken_logger.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_scack_bar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mih_home/components/mih_app_drawer.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tools.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
// ignore: must_be_immutable
|
|
||||||
class MihPackage extends StatefulWidget {
|
|
||||||
final Widget appActionButton;
|
|
||||||
final MihPackageTools appTools;
|
|
||||||
final List<Widget> appBody;
|
|
||||||
final List<String>? appToolTitles;
|
|
||||||
final MIHAppDrawer? actionDrawer;
|
|
||||||
int selectedbodyIndex;
|
|
||||||
final Function(int) onIndexChange;
|
|
||||||
MihPackage({
|
|
||||||
super.key,
|
|
||||||
required this.appActionButton,
|
|
||||||
required this.appTools,
|
|
||||||
required this.appBody,
|
|
||||||
this.appToolTitles,
|
|
||||||
this.actionDrawer,
|
|
||||||
required this.selectedbodyIndex,
|
|
||||||
required this.onIndexChange,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackage> createState() => _MihPackageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageState extends State<MihPackage>
|
|
||||||
with SingleTickerProviderStateMixin {
|
|
||||||
late PageController _pageController;
|
|
||||||
late AnimationController _animationController;
|
|
||||||
DateTime? lastPressedAt;
|
|
||||||
|
|
||||||
void unfocusAll() {
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _peakAnimation() async {
|
|
||||||
int currentPage = widget.selectedbodyIndex;
|
|
||||||
double peakOffset = _pageController.position.viewportDimension * 0.075;
|
|
||||||
double currentOffset =
|
|
||||||
_pageController.page! * _pageController.position.viewportDimension;
|
|
||||||
int nextPage =
|
|
||||||
currentPage + 1 < widget.appBody.length ? currentPage + 1 : currentPage;
|
|
||||||
if (nextPage != currentPage) {
|
|
||||||
await Future.delayed(const Duration(milliseconds: 100));
|
|
||||||
await _pageController.animateTo(
|
|
||||||
currentOffset + peakOffset,
|
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
curve: Curves.easeOut,
|
|
||||||
);
|
|
||||||
// await Future.delayed(const Duration(milliseconds: 100));
|
|
||||||
await _pageController.animateTo(
|
|
||||||
currentPage * _pageController.position.viewportDimension,
|
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
curve: Curves.easeIn,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
_pageController.dispose();
|
|
||||||
_animationController.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void didUpdateWidget(covariant MihPackage oldWidget) {
|
|
||||||
super.didUpdateWidget(oldWidget);
|
|
||||||
if (oldWidget.selectedbodyIndex != widget.selectedbodyIndex) {
|
|
||||||
_pageController.animateToPage(
|
|
||||||
widget.selectedbodyIndex,
|
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
curve: Curves.easeInOut,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_pageController = PageController(initialPage: widget.selectedbodyIndex);
|
|
||||||
_animationController = AnimationController(
|
|
||||||
vsync: this,
|
|
||||||
duration: const Duration(milliseconds: 400),
|
|
||||||
);
|
|
||||||
// if (!MzansiInnovationHub.of(context)!.theme.kIsWeb) {
|
|
||||||
// // Trigger the peak animation on start (or call this elsewhere)
|
|
||||||
// WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
||||||
// _peakAnimation();
|
|
||||||
// });
|
|
||||||
// }
|
|
||||||
if (!MzansiInnovationHub.of(context)!.theme.kIsWeb) {
|
|
||||||
// Trigger the peak animation only AFTER the route transition is complete
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
||||||
final ModalRoute? currentRoute = ModalRoute.of(context);
|
|
||||||
if (currentRoute != null) {
|
|
||||||
currentRoute.animation?.addStatusListener((status) {
|
|
||||||
if (status == AnimationStatus.completed && mounted) {
|
|
||||||
// Ensure the widget is still mounted and the animation is completed
|
|
||||||
_peakAnimation();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
Size screenSize = MediaQuery.of(context).size;
|
|
||||||
return GestureDetector(
|
|
||||||
onTap: unfocusAll,
|
|
||||||
child: PopScope(
|
|
||||||
canPop: false,
|
|
||||||
onPopInvokedWithResult: (bool didPop, Object? result) {
|
|
||||||
if (GoRouterState.of(context).name == 'mihHome' ||
|
|
||||||
GoRouterState.of(context).name == 'mihAuthentication') {
|
|
||||||
if (lastPressedAt == null ||
|
|
||||||
DateTime.now().difference(lastPressedAt!) >
|
|
||||||
const Duration(seconds: 2)) {
|
|
||||||
// First press: show a message and update the timestamp.
|
|
||||||
lastPressedAt = DateTime.now();
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
MihSnackBar(
|
|
||||||
child: Text("Press back again to exit"),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
// Second press within 2 seconds: exit the app.
|
|
||||||
KenLogger.warning('Exiting app...'); // Your custom logger
|
|
||||||
SystemChannels.platform.invokeMethod('SystemNavigator.pop');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
extra: true,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: Scaffold(
|
|
||||||
drawer: widget.actionDrawer,
|
|
||||||
body: SafeArea(
|
|
||||||
bottom: false,
|
|
||||||
minimum: EdgeInsets.only(bottom: 0),
|
|
||||||
child: Container(
|
|
||||||
width: screenSize.width,
|
|
||||||
height: screenSize.height,
|
|
||||||
//color: Colors.black,
|
|
||||||
padding: const EdgeInsets.only(top: 5),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
widget.appActionButton,
|
|
||||||
const SizedBox(
|
|
||||||
width: 10,
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: Container(
|
|
||||||
// alignment: Alignment.center,
|
|
||||||
// alignment: Alignment.centerRight,
|
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
// color: Colors.black,
|
|
||||||
child: FittedBox(
|
|
||||||
child: Text(
|
|
||||||
widget.appToolTitles != null
|
|
||||||
? widget
|
|
||||||
.appToolTitles![widget.selectedbodyIndex]
|
|
||||||
: "",
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 23,
|
|
||||||
fontWeight: FontWeight.w600,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 5),
|
|
||||||
widget.appTools,
|
|
||||||
const SizedBox(width: 5),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Expanded(
|
|
||||||
child: PageView.builder(
|
|
||||||
controller: _pageController,
|
|
||||||
itemCount: widget.appBody.length,
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return widget.appBody[index];
|
|
||||||
},
|
|
||||||
onPageChanged: (index) {
|
|
||||||
setState(() {
|
|
||||||
widget.selectedbodyIndex = index;
|
|
||||||
widget.onIndexChange(widget.selectedbodyIndex);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihPackageAction extends StatefulWidget {
|
|
||||||
final void Function()? onTap;
|
|
||||||
final double iconSize;
|
|
||||||
final Widget icon;
|
|
||||||
const MihPackageAction({
|
|
||||||
super.key,
|
|
||||||
required this.icon,
|
|
||||||
required this.iconSize,
|
|
||||||
required this.onTap,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageAction> createState() => _MihPackageActionState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageActionState extends State<MihPackageAction> {
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return IconButton(
|
|
||||||
iconSize: widget.iconSize,
|
|
||||||
padding: const EdgeInsets.all(0),
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
icon: widget.icon,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,117 +0,0 @@
|
|||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihPackageAlert extends StatefulWidget {
|
|
||||||
final Widget alertIcon;
|
|
||||||
final String alertTitle;
|
|
||||||
final Widget alertBody;
|
|
||||||
final Color alertColour;
|
|
||||||
const MihPackageAlert({
|
|
||||||
super.key,
|
|
||||||
required this.alertIcon,
|
|
||||||
required this.alertTitle,
|
|
||||||
required this.alertBody,
|
|
||||||
required this.alertColour,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageAlert> createState() => _MihPackageAlertState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageAlertState extends State<MihPackageAlert> {
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
late double popUpTitleSize;
|
|
||||||
late double popUpSubtitleSize;
|
|
||||||
late double popUpBodySize;
|
|
||||||
late double popUpIconSize;
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
Size? size;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = (size!.width / 4) * 2;
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 30.0;
|
|
||||||
popUpSubtitleSize = 20.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = size!.width - (size!.width * 0.1);
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 25.0;
|
|
||||||
popUpSubtitleSize = 18.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
size = MediaQuery.of(context).size;
|
|
||||||
checkScreenSize();
|
|
||||||
return Dialog(
|
|
||||||
child: Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(color: widget.alertColour, width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
widget.alertIcon,
|
|
||||||
//const SizedBox(height: 5),
|
|
||||||
Text(
|
|
||||||
widget.alertTitle,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.alertColour,
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
widget.alertBody,
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
context.pop();
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,234 +0,0 @@
|
|||||||
import 'package:app_settings/app_settings.dart';
|
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:local_auth/local_auth.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_alert.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_window.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_yt_video_player.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihPackageTile extends StatefulWidget {
|
|
||||||
final String appName;
|
|
||||||
final String? ytVideoID;
|
|
||||||
final Widget appIcon;
|
|
||||||
final void Function() onTap;
|
|
||||||
final double iconSize;
|
|
||||||
final Color primaryColor;
|
|
||||||
final Color secondaryColor;
|
|
||||||
final bool? authenticateUser;
|
|
||||||
const MihPackageTile({
|
|
||||||
super.key,
|
|
||||||
required this.onTap,
|
|
||||||
required this.appName,
|
|
||||||
this.ytVideoID,
|
|
||||||
required this.appIcon,
|
|
||||||
required this.iconSize,
|
|
||||||
required this.primaryColor,
|
|
||||||
required this.secondaryColor,
|
|
||||||
this.authenticateUser,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageTile> createState() => _MihPackageTileState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageTileState extends State<MihPackageTile> {
|
|
||||||
final LocalAuthentication _auth = LocalAuthentication();
|
|
||||||
|
|
||||||
void displayHint() {
|
|
||||||
if (widget.ytVideoID != null) {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageWindow(
|
|
||||||
fullscreen: false,
|
|
||||||
windowTitle: widget.appName,
|
|
||||||
// windowTools: const [],
|
|
||||||
onWindowTapClose: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
windowBody: MIHYTVideoPlayer(
|
|
||||||
videoYTLink: widget.ytVideoID!,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> isUserAuthenticated() async {
|
|
||||||
final bool canAuthWithBio = await _auth.canCheckBiometrics;
|
|
||||||
final bool canAuthenticate =
|
|
||||||
canAuthWithBio || await _auth.isDeviceSupported();
|
|
||||||
print("Auth Available: $canAuthenticate");
|
|
||||||
if (canAuthenticate) {
|
|
||||||
try {
|
|
||||||
final bool didBioAuth = await _auth.authenticate(
|
|
||||||
localizedReason: "Authenticate to access ${widget.appName}",
|
|
||||||
options: const AuthenticationOptions(
|
|
||||||
biometricOnly: false,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
if (didBioAuth) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
authErrorPopUp();
|
|
||||||
}
|
|
||||||
// print("Authenticated: $didBioAuth");
|
|
||||||
} catch (error) {
|
|
||||||
print("Auth Error: $error");
|
|
||||||
authErrorPopUp();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
print("Auth Error: No Biometrics Available");
|
|
||||||
authErrorPopUp();
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void authErrorPopUp() {
|
|
||||||
Widget alertpopUp = MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.fingerprint,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 100,
|
|
||||||
),
|
|
||||||
alertTitle: "Biometric Authentication Required",
|
|
||||||
alertBody: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"Hi there! To jump into the ${widget.appName} Package, you'll need to authenticate yourself with your devices biometrics, please set up biometric authentication (like fingerprint, face ID, pattern or pin) on your device first.\n\nIf you have already set up biometric authentication, press \"Authenticate now\" to try again or press \"Set Up Authentication\" to go to your device settings.",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 15,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
Wrap(
|
|
||||||
runAlignment: WrapAlignment.center,
|
|
||||||
crossAxisAlignment: WrapCrossAlignment.center,
|
|
||||||
spacing: 10,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
AppSettings.openAppSettings(
|
|
||||||
type: AppSettingsType.security,
|
|
||||||
);
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Set Up Authentication",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
authenticateUser();
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Authenticate Now",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return alertpopUp;
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> authenticateUser() async {
|
|
||||||
if (widget.authenticateUser != null &&
|
|
||||||
widget.authenticateUser! &&
|
|
||||||
!kIsWeb) {
|
|
||||||
if (await isUserAuthenticated()) {
|
|
||||||
widget.onTap();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
widget.onTap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Container(
|
|
||||||
// alignment: Alignment.topCenter,
|
|
||||||
// color: Colors.black,
|
|
||||||
// width: widget.iconSize,
|
|
||||||
// height: widget.iconSize + widget.iconSize / 3,
|
|
||||||
child: GestureDetector(
|
|
||||||
onTap: () async {
|
|
||||||
authenticateUser();
|
|
||||||
},
|
|
||||||
onLongPress: null, // Do this later
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Flexible(
|
|
||||||
flex: 3,
|
|
||||||
child: LayoutBuilder(
|
|
||||||
builder: (context, constraints) {
|
|
||||||
double iconHeight = constraints.maxWidth;
|
|
||||||
return Container(
|
|
||||||
width: iconHeight,
|
|
||||||
height: iconHeight,
|
|
||||||
child:
|
|
||||||
FittedBox(fit: BoxFit.fitHeight, child: widget.appIcon),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Flexible(
|
|
||||||
flex: 1,
|
|
||||||
child: FittedBox(
|
|
||||||
child: Text(
|
|
||||||
widget.appName,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
// softWrap: true,
|
|
||||||
// overflow: TextOverflow.visible,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20.0,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,95 +0,0 @@
|
|||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihPackageToolBody extends StatefulWidget {
|
|
||||||
final bool borderOn;
|
|
||||||
final Widget bodyItem;
|
|
||||||
final double? innerHorizontalPadding;
|
|
||||||
const MihPackageToolBody({
|
|
||||||
super.key,
|
|
||||||
required this.borderOn,
|
|
||||||
required this.bodyItem,
|
|
||||||
this.innerHorizontalPadding,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageToolBody> createState() => _MihPackageToolBodyState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageToolBodyState extends State<MihPackageToolBody> {
|
|
||||||
late double _innerBodyPadding;
|
|
||||||
double getHorizontalPaddingSize(Size screenSize) {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return widget.innerHorizontalPadding ?? 10;
|
|
||||||
} else {
|
|
||||||
return widget.innerHorizontalPadding ?? 0;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// mobile
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return widget.innerHorizontalPadding ?? 10;
|
|
||||||
} else {
|
|
||||||
return widget.innerHorizontalPadding ?? 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
double getVerticalPaddingSize(Size screenSize) {
|
|
||||||
// mobile
|
|
||||||
if (widget.borderOn) {
|
|
||||||
return 10;
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Decoration? getBoader() {
|
|
||||||
if (widget.borderOn) {
|
|
||||||
_innerBodyPadding = 10.0;
|
|
||||||
return BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 3.0),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
_innerBodyPadding = 0.0;
|
|
||||||
return BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 3.0),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
Size screenSize = MediaQuery.sizeOf(context);
|
|
||||||
|
|
||||||
return Padding(
|
|
||||||
padding: EdgeInsets.only(
|
|
||||||
left: getHorizontalPaddingSize(screenSize),
|
|
||||||
right: getHorizontalPaddingSize(screenSize),
|
|
||||||
bottom: getVerticalPaddingSize(screenSize),
|
|
||||||
top: 0,
|
|
||||||
),
|
|
||||||
child: Container(
|
|
||||||
height: screenSize.height,
|
|
||||||
decoration: getBoader(),
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.all(_innerBodyPadding),
|
|
||||||
child: widget.bodyItem,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
// ignore: must_be_immutable
|
|
||||||
class MihPackageTools extends StatefulWidget {
|
|
||||||
final Map<Widget, void Function()?> tools;
|
|
||||||
int selcetedIndex;
|
|
||||||
MihPackageTools({
|
|
||||||
super.key,
|
|
||||||
required this.tools,
|
|
||||||
required this.selcetedIndex,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageTools> createState() => _MihPackageToolsState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageToolsState extends State<MihPackageTools> {
|
|
||||||
List<Widget> getTools() {
|
|
||||||
List<Widget> temp = [];
|
|
||||||
int index = 0;
|
|
||||||
widget.tools.forEach((icon, onTap) {
|
|
||||||
temp.add(
|
|
||||||
Visibility(
|
|
||||||
visible: widget.selcetedIndex != index,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: onTap,
|
|
||||||
icon: icon,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
temp.add(
|
|
||||||
Visibility(
|
|
||||||
visible: widget.selcetedIndex == index,
|
|
||||||
child: IconButton.filled(
|
|
||||||
onPressed: onTap,
|
|
||||||
icon: icon,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
index += 1;
|
|
||||||
});
|
|
||||||
return temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.end,
|
|
||||||
children: getTools(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,217 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_speed_dial/flutter_speed_dial.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_single_child_scroll.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_floating_menu.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihPackageWindow extends StatefulWidget {
|
|
||||||
final String windowTitle;
|
|
||||||
final Widget windowBody;
|
|
||||||
final List<SpeedDialChild>? menuOptions;
|
|
||||||
final void Function() onWindowTapClose;
|
|
||||||
final bool fullscreen;
|
|
||||||
const MihPackageWindow({
|
|
||||||
super.key,
|
|
||||||
required this.fullscreen,
|
|
||||||
required this.windowTitle,
|
|
||||||
this.menuOptions,
|
|
||||||
required this.onWindowTapClose,
|
|
||||||
required this.windowBody,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPackageWindow> createState() => _MihPackageWindowState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPackageWindowState extends State<MihPackageWindow> {
|
|
||||||
late double windowTitleSize;
|
|
||||||
late double horizontralWindowPadding;
|
|
||||||
late double vertticalWindowPadding;
|
|
||||||
late double windowWidth;
|
|
||||||
late double windowHeight;
|
|
||||||
late double width;
|
|
||||||
late double height;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
// print("screen width: $width");
|
|
||||||
// print("screen height: $height");
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
windowTitleSize = 25;
|
|
||||||
horizontralWindowPadding = width / 7;
|
|
||||||
vertticalWindowPadding = 10;
|
|
||||||
windowWidth = width;
|
|
||||||
windowHeight = height;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
windowTitleSize = 20;
|
|
||||||
horizontralWindowPadding = 10;
|
|
||||||
vertticalWindowPadding = 10;
|
|
||||||
windowWidth = width;
|
|
||||||
windowHeight = height;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getHeader() {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(25), // Optional: rounds the corners
|
|
||||||
boxShadow: const [
|
|
||||||
BoxShadow(
|
|
||||||
color: Color.fromARGB(
|
|
||||||
60, 0, 0, 0), // 0.2 opacity = 51 in alpha (255 * 0.2)
|
|
||||||
spreadRadius: -2,
|
|
||||||
blurRadius: 10,
|
|
||||||
offset: Offset(0, 5),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
left: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 40,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor: WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")),
|
|
||||||
),
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
iconSize: 20,
|
|
||||||
onPressed: () {
|
|
||||||
widget.onWindowTapClose();
|
|
||||||
},
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.close,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Text(
|
|
||||||
widget.windowTitle,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: windowTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: (widget.menuOptions?.isNotEmpty ?? false),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
right: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 40,
|
|
||||||
child: MihFloatingMenu(
|
|
||||||
iconSize: 40,
|
|
||||||
animatedIcon: AnimatedIcons.menu_close,
|
|
||||||
direction: SpeedDialDirection.down,
|
|
||||||
children: widget.menuOptions != null ? widget.menuOptions! : [],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// If no menu, add a SizedBox to keep alignment
|
|
||||||
if (!(widget.menuOptions?.isNotEmpty ?? false))
|
|
||||||
const SizedBox(width: 40),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
var size = MediaQuery.of(context).size;
|
|
||||||
setState(() {
|
|
||||||
width = size.width;
|
|
||||||
height = size.height;
|
|
||||||
});
|
|
||||||
checkScreenSize();
|
|
||||||
return Dialog(
|
|
||||||
insetPadding: EdgeInsets.symmetric(
|
|
||||||
horizontal: horizontralWindowPadding,
|
|
||||||
vertical: vertticalWindowPadding,
|
|
||||||
),
|
|
||||||
insetAnimationCurve: Easing.emphasizedDecelerate,
|
|
||||||
insetAnimationDuration: Durations.short1,
|
|
||||||
child: Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: widget.fullscreen
|
|
||||||
? Column(
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
getHeader(),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Expanded(
|
|
||||||
child: SingleChildScrollView(child: widget.windowBody)),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
getHeader(),
|
|
||||||
const SizedBox(height: 5),
|
|
||||||
Flexible(
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.only(
|
|
||||||
left: 25,
|
|
||||||
right: 25,
|
|
||||||
bottom: vertticalWindowPadding,
|
|
||||||
),
|
|
||||||
child: ConstrainedBox(
|
|
||||||
constraints: BoxConstraints(
|
|
||||||
maxHeight: windowHeight * 0.85,
|
|
||||||
maxWidth: windowWidth * 0.85,
|
|
||||||
),
|
|
||||||
child: MihSingleChildScroll(child: widget.windowBody),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,109 +0,0 @@
|
|||||||
import 'package:file_picker/file_picker.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/app_user.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_circle_avatar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_file_services.dart';
|
|
||||||
|
|
||||||
class MihPersonalProfilePreview extends StatefulWidget {
|
|
||||||
final AppUser user;
|
|
||||||
const MihPersonalProfilePreview({
|
|
||||||
super.key,
|
|
||||||
required this.user,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihPersonalProfilePreview> createState() =>
|
|
||||||
_MihPersonalProfilePreviewState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihPersonalProfilePreviewState extends State<MihPersonalProfilePreview> {
|
|
||||||
late Future<String> futureImageUrl;
|
|
||||||
// String imageUrl = "";
|
|
||||||
PlatformFile? file;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
futureImageUrl =
|
|
||||||
MihFileApi.getMinioFileUrl(widget.user.pro_pic_path, context);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
double profilePictureWidth = 60;
|
|
||||||
return Row(
|
|
||||||
children: [
|
|
||||||
FutureBuilder(
|
|
||||||
future: futureImageUrl,
|
|
||||||
builder: (context, asyncSnapshot) {
|
|
||||||
if (asyncSnapshot.connectionState == ConnectionState.done &&
|
|
||||||
asyncSnapshot.hasData) {
|
|
||||||
if (asyncSnapshot.requireData != "") {
|
|
||||||
return MihCircleAvatar(
|
|
||||||
imageFile: NetworkImage(asyncSnapshot.requireData),
|
|
||||||
width: profilePictureWidth,
|
|
||||||
editable: false,
|
|
||||||
fileNameController: TextEditingController(),
|
|
||||||
userSelectedfile: file,
|
|
||||||
frameColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onChange: () {},
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return Icon(
|
|
||||||
MihIcons.iDontKnow,
|
|
||||||
size: profilePictureWidth,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return Icon(
|
|
||||||
MihIcons.mihRing,
|
|
||||||
size: profilePictureWidth,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
),
|
|
||||||
const SizedBox(width: 15),
|
|
||||||
Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.user.username.isNotEmpty
|
|
||||||
? widget.user.username
|
|
||||||
: "Username",
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 18,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
widget.user.fname.isNotEmpty
|
|
||||||
? "${widget.user.fname} ${widget.user.lname}"
|
|
||||||
: "Name Surname",
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 15,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
widget.user.type.toUpperCase(),
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 10,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,140 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihRadioOptions extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final String hintText;
|
|
||||||
final Color fillColor;
|
|
||||||
final Color secondaryFillColor;
|
|
||||||
final bool requiredText;
|
|
||||||
final List<String> radioOptions;
|
|
||||||
const MihRadioOptions({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
required this.hintText,
|
|
||||||
required this.fillColor,
|
|
||||||
required this.secondaryFillColor,
|
|
||||||
required this.requiredText,
|
|
||||||
required this.radioOptions,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihRadioOptions> createState() => _MihRadioOptionsState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihRadioOptionsState extends State<MihRadioOptions> {
|
|
||||||
// late String _currentSelection;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
if (widget.controller.text.isEmpty && widget.radioOptions.isNotEmpty) {
|
|
||||||
widget.controller.text = widget.radioOptions[0];
|
|
||||||
}
|
|
||||||
// else{
|
|
||||||
// int index = widget.radioOptions
|
|
||||||
// .indexWhere((element) => element == option);
|
|
||||||
// _currentSelection = widget.radioOptions[index];
|
|
||||||
// widget.controller.text = option;
|
|
||||||
|
|
||||||
// }
|
|
||||||
// _currentSelection = widget.radioOptions[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
// The method to handle a change in selection.
|
|
||||||
void _onChanged(String? value) {
|
|
||||||
if (value != null) {
|
|
||||||
widget.controller.text = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget displayRadioOptions(String selection) {
|
|
||||||
return Material(
|
|
||||||
elevation: 4.0,
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
child: Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: widget.fillColor,
|
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
children: widget.radioOptions.map((option) {
|
|
||||||
return GestureDetector(
|
|
||||||
onTap: () {
|
|
||||||
_onChanged(option);
|
|
||||||
},
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Expanded(
|
|
||||||
child: Text(
|
|
||||||
option,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.secondaryFillColor,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
fontSize: 15,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Radio<String>(
|
|
||||||
value: option,
|
|
||||||
groupValue: selection,
|
|
||||||
onChanged: _onChanged,
|
|
||||||
activeColor: widget.secondaryFillColor,
|
|
||||||
fillColor: WidgetStateProperty.resolveWith<Color?>(
|
|
||||||
(Set<WidgetState> states) {
|
|
||||||
if (states.contains(WidgetState.selected)) {
|
|
||||||
return widget.secondaryFillColor; // Color when selected
|
|
||||||
}
|
|
||||||
return widget.secondaryFillColor;
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}).toList(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return AnimatedBuilder(
|
|
||||||
animation: widget.controller,
|
|
||||||
builder: (context, child) {
|
|
||||||
final currentSelection = widget.controller.text;
|
|
||||||
return Column(
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.hintText,
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: !widget.requiredText,
|
|
||||||
child: Text(
|
|
||||||
"(Optional)",
|
|
||||||
textAlign: TextAlign.right,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
displayRadioOptions(currentSelection),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
SnackBar MihSnackBar({
|
|
||||||
required Widget child,
|
|
||||||
}) {
|
|
||||||
return SnackBar(
|
|
||||||
content: child,
|
|
||||||
shape: StadiumBorder(),
|
|
||||||
behavior: SnackBarBehavior.floating,
|
|
||||||
duration: Duration(seconds: 2),
|
|
||||||
width: null,
|
|
||||||
action: SnackBarAction(
|
|
||||||
label: "Dismiss",
|
|
||||||
onPressed: () {},
|
|
||||||
),
|
|
||||||
// elevation: 30,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,185 +0,0 @@
|
|||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihSearchBar extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final String hintText;
|
|
||||||
final IconData prefixIcon;
|
|
||||||
final IconData? prefixAltIcon;
|
|
||||||
final List<Widget>? suffixTools;
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final Color fillColor;
|
|
||||||
final Color hintColor;
|
|
||||||
final void Function()? onPrefixIconTap;
|
|
||||||
final void Function()? onClearIconTap;
|
|
||||||
final double? elevation;
|
|
||||||
final FocusNode searchFocusNode;
|
|
||||||
|
|
||||||
const MihSearchBar({
|
|
||||||
Key? key,
|
|
||||||
required this.controller,
|
|
||||||
required this.hintText,
|
|
||||||
required this.prefixIcon,
|
|
||||||
this.prefixAltIcon,
|
|
||||||
this.suffixTools,
|
|
||||||
this.width,
|
|
||||||
this.height,
|
|
||||||
required this.fillColor,
|
|
||||||
required this.hintColor,
|
|
||||||
required this.onPrefixIconTap,
|
|
||||||
this.onClearIconTap,
|
|
||||||
this.elevation,
|
|
||||||
required this.searchFocusNode,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihSearchBar> createState() => _MihSearchBarState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihSearchBarState extends State<MihSearchBar> {
|
|
||||||
bool _showClearIcon = false;
|
|
||||||
|
|
||||||
Widget getPrefixIcon() {
|
|
||||||
if (_showClearIcon) {
|
|
||||||
// If the clear icon is shown and an alternative prefix icon is provided, use it
|
|
||||||
return widget.prefixAltIcon != null
|
|
||||||
? Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Icon(
|
|
||||||
widget.prefixAltIcon,
|
|
||||||
color: widget.hintColor,
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Icon(
|
|
||||||
Icons.search,
|
|
||||||
color: widget.hintColor,
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
); // Default to search icon if no alt icon
|
|
||||||
} else {
|
|
||||||
// Return the primary prefix icon or the alternative if provided
|
|
||||||
return Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Icon(
|
|
||||||
Icons.search,
|
|
||||||
color: widget.hintColor,
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
// 1. Add the listener to the controller
|
|
||||||
widget.controller.addListener(_updateClearIconVisibility);
|
|
||||||
// 2. Initialize the clear icon visibility based on the current text
|
|
||||||
_updateClearIconVisibility();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
widget.controller.removeListener(_updateClearIconVisibility);
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
void _updateClearIconVisibility() {
|
|
||||||
if (!mounted) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
final bool shouldShow = widget.controller.text.isNotEmpty;
|
|
||||||
// Only call setState if the visibility state actually changes
|
|
||||||
if (_showClearIcon != shouldShow) {
|
|
||||||
setState(() {
|
|
||||||
_showClearIcon = shouldShow;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Material(
|
|
||||||
elevation: widget.elevation ?? 4.0, // Use provided elevation or default
|
|
||||||
borderRadius: BorderRadius.circular(30.0),
|
|
||||||
color: widget.fillColor,
|
|
||||||
child: AnimatedContainer(
|
|
||||||
// Keep AnimatedContainer for width/height transitions
|
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
width: widget.width,
|
|
||||||
height: widget.height ?? 50,
|
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(30.0),
|
|
||||||
),
|
|
||||||
child: Theme(
|
|
||||||
data: Theme.of(context).copyWith(
|
|
||||||
textSelectionTheme: TextSelectionThemeData(
|
|
||||||
selectionColor: widget.hintColor.withValues(alpha: 0.3),
|
|
||||||
selectionHandleColor: widget.hintColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: TextField(
|
|
||||||
textAlignVertical: TextAlignVertical.center,
|
|
||||||
controller: widget.controller, // Assign the controller
|
|
||||||
focusNode: widget.searchFocusNode,
|
|
||||||
autocorrect: true,
|
|
||||||
spellCheckConfiguration: kIsWeb ? null : SpellCheckConfiguration(),
|
|
||||||
onSubmitted: (value) {
|
|
||||||
widget.onPrefixIconTap
|
|
||||||
?.call(); // Call the prefix icon tap handler
|
|
||||||
},
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.hintColor,
|
|
||||||
fontWeight: FontWeight.w600,
|
|
||||||
fontSize: 16,
|
|
||||||
),
|
|
||||||
cursorColor: widget.hintColor,
|
|
||||||
decoration: InputDecoration(
|
|
||||||
isDense: true,
|
|
||||||
hintText: widget.hintText,
|
|
||||||
hintStyle: TextStyle(
|
|
||||||
color: widget.hintColor,
|
|
||||||
fontWeight: FontWeight.w600,
|
|
||||||
fontSize: 16,
|
|
||||||
),
|
|
||||||
border: InputBorder.none,
|
|
||||||
contentPadding:
|
|
||||||
const EdgeInsets.symmetric(horizontal: 10.0, vertical: 15.0),
|
|
||||||
prefixIcon: GestureDetector(
|
|
||||||
onTap: widget.onPrefixIconTap,
|
|
||||||
child: getPrefixIcon(),
|
|
||||||
),
|
|
||||||
suffixIcon: Row(
|
|
||||||
// Use a Row for multiple suffix icons
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
// Optional suffix tools
|
|
||||||
if (widget.suffixTools != null) ...widget.suffixTools!,
|
|
||||||
// Clear Icon (conditionally visible)
|
|
||||||
if (_showClearIcon) // Only show if input is not empty
|
|
||||||
IconButton(
|
|
||||||
iconSize: 35,
|
|
||||||
icon: Icon(Icons.clear,
|
|
||||||
color: widget.hintColor), // Clear icon
|
|
||||||
onPressed: widget.onClearIconTap ??
|
|
||||||
() {
|
|
||||||
widget.controller.clear();
|
|
||||||
// No need for setState here, _updateClearIconVisibility will handle it
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihSingleChildScroll extends StatefulWidget {
|
|
||||||
final Widget child;
|
|
||||||
const MihSingleChildScroll({
|
|
||||||
super.key,
|
|
||||||
required this.child,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihSingleChildScroll> createState() => _MihSingleChildScrollState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihSingleChildScrollState extends State<MihSingleChildScroll> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return SafeArea(
|
|
||||||
bottom: false,
|
|
||||||
minimum: EdgeInsets.only(bottom: 5),
|
|
||||||
child: ScrollConfiguration(
|
|
||||||
behavior: ScrollConfiguration.of(context).copyWith(scrollbars: false),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: widget.child,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,312 +0,0 @@
|
|||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihTextFormField extends StatefulWidget {
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final Color fillColor;
|
|
||||||
final Color inputColor;
|
|
||||||
final TextEditingController controller;
|
|
||||||
final bool? hasError;
|
|
||||||
final String? hintText;
|
|
||||||
final double? borderRadius;
|
|
||||||
final bool? multiLineInput;
|
|
||||||
final bool? readOnly;
|
|
||||||
final bool? passwordMode;
|
|
||||||
final bool? numberMode;
|
|
||||||
final bool requiredText;
|
|
||||||
final FormFieldValidator<String>? validator;
|
|
||||||
final List<String>? autofillHints;
|
|
||||||
final double? elevation;
|
|
||||||
final TextAlign? textIputAlignment;
|
|
||||||
|
|
||||||
const MihTextFormField({
|
|
||||||
Key? key,
|
|
||||||
this.width,
|
|
||||||
this.height,
|
|
||||||
required this.fillColor,
|
|
||||||
required this.inputColor,
|
|
||||||
required this.controller,
|
|
||||||
this.hasError,
|
|
||||||
required this.hintText,
|
|
||||||
required this.requiredText,
|
|
||||||
this.borderRadius,
|
|
||||||
this.multiLineInput,
|
|
||||||
this.readOnly,
|
|
||||||
this.passwordMode,
|
|
||||||
this.numberMode,
|
|
||||||
this.validator,
|
|
||||||
this.autofillHints,
|
|
||||||
this.elevation,
|
|
||||||
this.textIputAlignment,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihTextFormField> createState() => _MihTextFormFieldState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihTextFormFieldState extends State<MihTextFormField> {
|
|
||||||
late bool _obscureText;
|
|
||||||
FormFieldState<String>? _formFieldState;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_obscureText = widget.passwordMode ?? false;
|
|
||||||
widget.controller.addListener(_onControllerTextChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void didUpdateWidget(covariant MihTextFormField oldWidget) {
|
|
||||||
super.didUpdateWidget(oldWidget);
|
|
||||||
// If the controller itself changes, remove listener from old and add to new
|
|
||||||
if (widget.controller != oldWidget.controller) {
|
|
||||||
oldWidget.controller.removeListener(_onControllerTextChanged);
|
|
||||||
widget.controller.addListener(_onControllerTextChanged);
|
|
||||||
// Immediately update form field state if controller changed and has value
|
|
||||||
_formFieldState?.didChange(widget.controller.text);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void _onControllerTextChanged() {
|
|
||||||
// Only update the FormField's value if it's not already the same
|
|
||||||
// and if the formFieldState is available.
|
|
||||||
if (_formFieldState != null &&
|
|
||||||
_formFieldState!.value != widget.controller.text) {
|
|
||||||
_formFieldState!.didChange(widget.controller.text);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
widget.controller.removeListener(_onControllerTextChanged);
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final isMultiline = widget.multiLineInput == true;
|
|
||||||
return Center(
|
|
||||||
child: SizedBox(
|
|
||||||
width: widget.width,
|
|
||||||
// height: widget.height,
|
|
||||||
height: isMultiline ? null : widget.height,
|
|
||||||
child: Theme(
|
|
||||||
data: Theme.of(context).copyWith(
|
|
||||||
textSelectionTheme: TextSelectionThemeData(
|
|
||||||
selectionColor: widget.inputColor.withValues(alpha: 0.3),
|
|
||||||
selectionHandleColor: widget.inputColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Visibility(
|
|
||||||
visible: widget.hintText != null,
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.hintText ?? "",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: !widget.requiredText,
|
|
||||||
child: Text(
|
|
||||||
"(Optional)",
|
|
||||||
textAlign: TextAlign.right,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
FormField<String>(
|
|
||||||
initialValue: widget.controller.text,
|
|
||||||
validator: widget.validator,
|
|
||||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
|
||||||
builder: (field) {
|
|
||||||
_formFieldState = field;
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment:
|
|
||||||
CrossAxisAlignment.start, // <-- Add this line
|
|
||||||
children: [
|
|
||||||
Material(
|
|
||||||
elevation: widget.elevation ?? 4.0,
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(widget.borderRadius ?? 8.0),
|
|
||||||
child: SizedBox(
|
|
||||||
height: widget.height != null
|
|
||||||
? widget.height! - 30
|
|
||||||
: null,
|
|
||||||
child: TextFormField(
|
|
||||||
controller: widget.controller,
|
|
||||||
cursorColor: widget.inputColor,
|
|
||||||
autofillHints: widget.autofillHints,
|
|
||||||
autocorrect: true,
|
|
||||||
spellCheckConfiguration: (kIsWeb ||
|
|
||||||
widget.passwordMode == true ||
|
|
||||||
widget.numberMode == true)
|
|
||||||
? null
|
|
||||||
: SpellCheckConfiguration(),
|
|
||||||
textAlign:
|
|
||||||
widget.textIputAlignment ?? TextAlign.start,
|
|
||||||
textAlignVertical: widget.multiLineInput == true
|
|
||||||
? TextAlignVertical.top
|
|
||||||
: TextAlignVertical.center,
|
|
||||||
obscureText: widget.passwordMode == true
|
|
||||||
? _obscureText
|
|
||||||
: false,
|
|
||||||
expands: widget.passwordMode == true
|
|
||||||
? false
|
|
||||||
: (widget.multiLineInput ?? false),
|
|
||||||
maxLines: widget.passwordMode == true ? 1 : null,
|
|
||||||
readOnly: widget.readOnly ?? false,
|
|
||||||
keyboardType: widget.numberMode == true
|
|
||||||
? const TextInputType.numberWithOptions(
|
|
||||||
decimal: true)
|
|
||||||
: null,
|
|
||||||
inputFormatters: widget.numberMode == true
|
|
||||||
? [
|
|
||||||
FilteringTextInputFormatter.allow(
|
|
||||||
RegExp(r'^\d*\.?\d*'))
|
|
||||||
]
|
|
||||||
: null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: widget.inputColor,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
),
|
|
||||||
decoration: InputDecoration(
|
|
||||||
suffixIcon: widget.passwordMode == true
|
|
||||||
? FocusScope(
|
|
||||||
canRequestFocus: false,
|
|
||||||
child: IconButton(
|
|
||||||
icon: Icon(
|
|
||||||
_obscureText
|
|
||||||
? Icons.visibility_off
|
|
||||||
: Icons.visibility,
|
|
||||||
color: widget.inputColor,
|
|
||||||
),
|
|
||||||
onPressed: () {
|
|
||||||
setState(() {
|
|
||||||
_obscureText = !_obscureText;
|
|
||||||
});
|
|
||||||
},
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: null,
|
|
||||||
errorStyle: const TextStyle(
|
|
||||||
height: 0, fontSize: 0), // <-- Add this line
|
|
||||||
contentPadding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 10.0, vertical: 8.0),
|
|
||||||
filled: true,
|
|
||||||
fillColor: widget.fillColor,
|
|
||||||
border: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: field.hasError
|
|
||||||
? BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 2.0,
|
|
||||||
)
|
|
||||||
: BorderSide.none,
|
|
||||||
),
|
|
||||||
enabledBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide.none,
|
|
||||||
),
|
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: field.hasError
|
|
||||||
? MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark")
|
|
||||||
: widget.inputColor,
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
errorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
focusedErrorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onChanged: (value) {
|
|
||||||
field.didChange(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (field.hasError)
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding:
|
|
||||||
const EdgeInsets.only(left: 8.0, top: 4.0),
|
|
||||||
child: Text(
|
|
||||||
field.errorText ?? '',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,217 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihTimeField extends StatefulWidget {
|
|
||||||
final TextEditingController controller;
|
|
||||||
final String labelText;
|
|
||||||
final bool required;
|
|
||||||
final double? width;
|
|
||||||
final double? height;
|
|
||||||
final double? borderRadius;
|
|
||||||
final double? elevation;
|
|
||||||
final FormFieldValidator<String>? validator;
|
|
||||||
|
|
||||||
const MihTimeField({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
required this.labelText,
|
|
||||||
required this.required,
|
|
||||||
this.width,
|
|
||||||
this.height,
|
|
||||||
this.borderRadius,
|
|
||||||
this.elevation,
|
|
||||||
this.validator,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihTimeField> createState() => _MihTimeFieldState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihTimeFieldState extends State<MihTimeField> {
|
|
||||||
FormFieldState<String>? _formFieldState;
|
|
||||||
|
|
||||||
Future<void> _selectTime(BuildContext context) async {
|
|
||||||
TimeOfDay? picked = await showTimePicker(
|
|
||||||
context: context,
|
|
||||||
initialTime: widget.controller.text.isNotEmpty
|
|
||||||
? TimeOfDay(
|
|
||||||
hour: int.tryParse(widget.controller.text.split(":")[0]) ?? 0,
|
|
||||||
minute: int.tryParse(widget.controller.text.split(":")[1]) ?? 0,
|
|
||||||
)
|
|
||||||
: TimeOfDay.now(),
|
|
||||||
builder: (context, child) {
|
|
||||||
return MediaQuery(
|
|
||||||
data: MediaQuery.of(context).copyWith(alwaysUse24HourFormat: true),
|
|
||||||
child: child as Widget,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
if (picked != null) {
|
|
||||||
final hours = picked.hour.toString().padLeft(2, '0');
|
|
||||||
final minutes = picked.minute.toString().padLeft(2, '0');
|
|
||||||
widget.controller.text = "$hours:$minutes";
|
|
||||||
_formFieldState?.didChange(widget.controller.text);
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Center(
|
|
||||||
child: SizedBox(
|
|
||||||
width: widget.width,
|
|
||||||
height: widget.height,
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
widget.labelText,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (!widget.required)
|
|
||||||
Text(
|
|
||||||
"(Optional)",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
FormField<String>(
|
|
||||||
initialValue: widget.controller.text,
|
|
||||||
validator: widget.validator,
|
|
||||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
|
||||||
builder: (field) {
|
|
||||||
_formFieldState = field;
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Material(
|
|
||||||
elevation: widget.elevation ?? 4.0,
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(widget.borderRadius ?? 8.0),
|
|
||||||
child: TextFormField(
|
|
||||||
controller: widget.controller,
|
|
||||||
readOnly: true,
|
|
||||||
onTap: () => _selectTime(context),
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
),
|
|
||||||
decoration: InputDecoration(
|
|
||||||
suffixIcon: Icon(
|
|
||||||
Icons.access_time,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
errorStyle: const TextStyle(height: 0, fontSize: 0),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 10.0, vertical: 8.0),
|
|
||||||
filled: true,
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
border: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: field.hasError
|
|
||||||
? BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 2.0,
|
|
||||||
)
|
|
||||||
: BorderSide.none,
|
|
||||||
),
|
|
||||||
enabledBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide.none,
|
|
||||||
),
|
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: field.hasError
|
|
||||||
? MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark")
|
|
||||||
: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
errorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
focusedErrorBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
widget.borderRadius ?? 8.0),
|
|
||||||
borderSide: BorderSide(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onChanged: (value) {
|
|
||||||
field.didChange(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (field.hasError)
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(left: 8.0, top: 4.0),
|
|
||||||
child: Text(
|
|
||||||
field.errorText ?? '',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihToggle extends StatefulWidget {
|
|
||||||
final String hintText;
|
|
||||||
final bool initialPostion;
|
|
||||||
final Color fillColor;
|
|
||||||
final Color secondaryFillColor;
|
|
||||||
final bool? readOnly;
|
|
||||||
final double? elevation;
|
|
||||||
final void Function(bool) onChange;
|
|
||||||
const MihToggle({
|
|
||||||
super.key,
|
|
||||||
required this.hintText,
|
|
||||||
required this.initialPostion,
|
|
||||||
required this.fillColor,
|
|
||||||
required this.secondaryFillColor,
|
|
||||||
this.readOnly,
|
|
||||||
this.elevation,
|
|
||||||
required this.onChange,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihToggle> createState() => _MihToggleState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihToggleState extends State<MihToggle> {
|
|
||||||
late bool togglePosition;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void didUpdateWidget(covariant MihToggle oldWidget) {
|
|
||||||
super.didUpdateWidget(oldWidget);
|
|
||||||
if (widget.initialPostion != oldWidget.initialPostion) {
|
|
||||||
setState(() {
|
|
||||||
togglePosition = widget.initialPostion;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
togglePosition = widget.initialPostion;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Expanded(
|
|
||||||
child: Text(
|
|
||||||
widget.hintText,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: widget.fillColor,
|
|
||||||
fontSize: 18,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
// Material(
|
|
||||||
// elevation: widget.elevation ?? 0.01,
|
|
||||||
// shadowColor: widget.secondaryFillColor.withOpacity(0.5),
|
|
||||||
// color: Colors.transparent,
|
|
||||||
// shape: StadiumBorder(),
|
|
||||||
Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Colors.transparent,
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
30), // Adjust the border radius to match the toggle
|
|
||||||
boxShadow: [
|
|
||||||
BoxShadow(
|
|
||||||
color: Colors.black.withOpacity(0.15),
|
|
||||||
offset: Offset(
|
|
||||||
0, widget.elevation ?? 10), // Adjust the vertical offset
|
|
||||||
blurRadius: widget.elevation ?? 10,
|
|
||||||
spreadRadius: 0,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Switch(
|
|
||||||
value: togglePosition,
|
|
||||||
trackOutlineColor: WidgetStateProperty.resolveWith<Color?>(
|
|
||||||
(states) {
|
|
||||||
if (widget.readOnly == true) {
|
|
||||||
return Colors.grey;
|
|
||||||
}
|
|
||||||
if (states.contains(WidgetState.selected)) {
|
|
||||||
return MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"); // Outline color when active
|
|
||||||
}
|
|
||||||
return MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"); // Outline color when active
|
|
||||||
},
|
|
||||||
),
|
|
||||||
activeColor: widget.readOnly == true
|
|
||||||
? Colors.grey
|
|
||||||
: widget.secondaryFillColor,
|
|
||||||
activeTrackColor: widget.readOnly == true
|
|
||||||
? Colors.grey.shade400
|
|
||||||
: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
inactiveThumbColor: widget.readOnly == true
|
|
||||||
? Colors.grey
|
|
||||||
: widget.secondaryFillColor,
|
|
||||||
inactiveTrackColor: widget.readOnly == true
|
|
||||||
? Colors.grey.shade400
|
|
||||||
: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// activeColor: widget.secondaryFillColor,
|
|
||||||
// activeTrackColor: widget.fillColor,
|
|
||||||
// inactiveThumbColor: widget.fillColor,
|
|
||||||
// inactiveTrackColor: widget.secondaryFillColor,
|
|
||||||
// onChanged: widget.readOnly != true ? widget.onChange : null,
|
|
||||||
onChanged: widget.readOnly != true
|
|
||||||
? (newValue) {
|
|
||||||
setState(() {
|
|
||||||
togglePosition = newValue; // Update internal state
|
|
||||||
});
|
|
||||||
widget.onChange(newValue); // Call the parent's onChange
|
|
||||||
}
|
|
||||||
: null,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,530 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
|
|
||||||
class MIHDeleteMessage extends StatefulWidget {
|
|
||||||
final String deleteType;
|
|
||||||
final void Function() onTap;
|
|
||||||
const MIHDeleteMessage({
|
|
||||||
super.key,
|
|
||||||
required this.deleteType,
|
|
||||||
required this.onTap,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHDeleteMessage> createState() => _MIHDeleteMessageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHDeleteMessageState extends State<MIHDeleteMessage> {
|
|
||||||
var messageTypes = <String, Widget>{};
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
late double popUpTitleSize;
|
|
||||||
late double popUpSubtitleSize;
|
|
||||||
late double popUpBodySize;
|
|
||||||
late double popUpIconSize;
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
late double width;
|
|
||||||
late double height;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = (width / 4) * 2;
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 25.0;
|
|
||||||
popUpSubtitleSize = 20.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = width - (width * 0.1);
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 20.0;
|
|
||||||
popUpSubtitleSize = 18.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDeleteNote() {
|
|
||||||
messageTypes["Note"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Are you sure you want to delete this?",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This note will be deleted permanently. Are you certain you want to delete it?",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
MihButton(
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
buttonColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Delete",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setFileNote() {
|
|
||||||
messageTypes["File"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Are you sure you want to delete this?",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This file will be deleted permanently. Are you certain you want to delete it?",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
MihButton(
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
buttonColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Delete",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDeleteEmployee() {
|
|
||||||
messageTypes["Employee"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Are you sure you want to delete this?",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This team member will be deleted permanently from the business profile. Are you certain you want to delete it?",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
|
|
||||||
MihButton(
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
buttonColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Delete",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDeleteAppointment() {
|
|
||||||
messageTypes["Appointment"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Are you sure you want to delete this?",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This appointment will be deleted permanently from your calendar. Are you certain you want to delete it?",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
MihButton(
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
buttonColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Delete",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDeleteLoyaltyCard() {
|
|
||||||
messageTypes["Loyalty Card"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Are you sure you want to delete this?",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This Card will be deleted permanently from your Mzansi Wallet. Are you certain you want to delete it?",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
MihButton(
|
|
||||||
onPressed: widget.onTap,
|
|
||||||
buttonColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Delete",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getDeleteMessage(String type) {
|
|
||||||
return messageTypes[type];
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
var size = MediaQuery.of(context).size;
|
|
||||||
setState(() {
|
|
||||||
width = size.width;
|
|
||||||
height = size.height;
|
|
||||||
});
|
|
||||||
checkScreenSize();
|
|
||||||
setDeleteNote();
|
|
||||||
setFileNote();
|
|
||||||
setDeleteEmployee();
|
|
||||||
setDeleteAppointment();
|
|
||||||
setDeleteLoyaltyCard();
|
|
||||||
//print(size);
|
|
||||||
// setState(() {
|
|
||||||
// width = size.width;
|
|
||||||
// height = size.height;
|
|
||||||
// });
|
|
||||||
return Dialog(child: getDeleteMessage(widget.deleteType));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,909 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
|
|
||||||
class MIHErrorMessage extends StatefulWidget {
|
|
||||||
final String errorType;
|
|
||||||
const MIHErrorMessage({
|
|
||||||
super.key,
|
|
||||||
required this.errorType,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHErrorMessage> createState() => _MIHErrorMessageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHErrorMessageState extends State<MIHErrorMessage> {
|
|
||||||
var messageTypes = <String, Widget>{};
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
late double popUpTitleSize;
|
|
||||||
late double popUpSubtitleSize;
|
|
||||||
late double popUpBodySize;
|
|
||||||
late double popUpIconSize;
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
Size? size;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = (size!.width / 4) * 2;
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 25.0;
|
|
||||||
popUpSubtitleSize = 20.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = size!.width - (size!.width * 0.1);
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 20.0;
|
|
||||||
popUpSubtitleSize = 18.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setInputError() {
|
|
||||||
messageTypes["Input Error"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 5),
|
|
||||||
Text(
|
|
||||||
"Oops! Looks like some fields are missing.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"We noticed that some required fields are still empty. To ensure your request is processed smoothly, please fill out all the highlighted fields before submitting the form again.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
RichText(
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
children: <TextSpan>[
|
|
||||||
TextSpan(
|
|
||||||
text: "Here's a quick tip: ",
|
|
||||||
style: TextStyle(
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"))),
|
|
||||||
const TextSpan(
|
|
||||||
text: "Look for fields with an asterisk ("),
|
|
||||||
TextSpan(
|
|
||||||
text: "*",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"))),
|
|
||||||
const TextSpan(
|
|
||||||
text: ") next to them, as these are mandatory."),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setPasswordRequirementsError() {
|
|
||||||
messageTypes["Password Requirements"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Password Doesn't Meet Requirements",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Oops! Your password doesn't quite meet our standards. To keep your account secure, please make sure your password meets the following requirements",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
RichText(
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 15.0,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
children: <TextSpan>[
|
|
||||||
TextSpan(
|
|
||||||
text: "Requirements:\n",
|
|
||||||
style: TextStyle(
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"))),
|
|
||||||
const TextSpan(
|
|
||||||
text: "1) Contailes at least 8 characters\n",
|
|
||||||
),
|
|
||||||
const TextSpan(
|
|
||||||
text: "2) Contains at least 1 uppercase letter (A-Z)\n",
|
|
||||||
),
|
|
||||||
const TextSpan(
|
|
||||||
text: "3) Contains at least 1 lowercase letter (a-z)\n",
|
|
||||||
),
|
|
||||||
const TextSpan(
|
|
||||||
text: "4) Contains at least 1 number (0-9)\n",
|
|
||||||
),
|
|
||||||
const TextSpan(
|
|
||||||
text:
|
|
||||||
"5) Contains at least 1 special character (!@#\$%^&*)\n",
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setInvalidUsernameError() {
|
|
||||||
messageTypes["Invalid Username"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Let's Fix That Username",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Let's create a great username for you! Just a few quick tips:\n• Your username should start with a letter.\n• You can use letters, numbers, and/ or underscores.\n• Keep it between 6 and 20 characters.\n• Avoid special characters like @, #, or \$.\"",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setInvalidEmailError() {
|
|
||||||
messageTypes["Invalid Email"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Oops! Invalid Email",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Looks like there's a little hiccup with that email address. Please double-check that you've entered it correctly, including the \"@\" symbol and a domain (like example@email.com).",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setUserExistsError() {
|
|
||||||
messageTypes["User Exists"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Email Already Exists",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"An account is already registered with this email address. Please try logging in or use a different email.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Here are some things to keep in mind:",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpSubtitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"1) Are you sure you're using the correct email address associated with your account?\n2) Is your caps lock key on? Passwords are case-sensitive.\n3) If you've forgotten your password, no worries! Click on \"Forgot Password?\" to reset it.",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15.0,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setPwdMatchError() {
|
|
||||||
messageTypes["Password Match"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Passwords Don't Match",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"The password and confirm password fields do not match. Please make sure they are identical.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Here are some things to keep in mind:",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpSubtitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"1) Are you sure you're using the correct email address associated with your account?\n2) Is your caps lock key on? Passwords are case-sensitive.\n3) If you've forgotten your password, no worries! Click on \"Forgot Password?\" to reset it.",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setinvalidCredError() {
|
|
||||||
messageTypes["Invalid Credentials"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Uh oh! Login attempt unsuccessful.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"The email address or password you entered doesn't seem to match our records. Please double-check your information and try again.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Here are some things to keep in mind:",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpSubtitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"1) Are you sure you're using the correct email address associated with your account?\n2) Is your caps lock key on? Passwords are case-sensitive.\n3) If you've forgotten your password, no worries! Click on \"Forgot Password?\" to reset it.",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setInternetError() {
|
|
||||||
messageTypes["Internet Connection"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Internet Connection Lost!",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"We seem to be having some trouble connecting you to the internet. This could be due to a temporary outage or an issue with your device's connection.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Here are a few things you can try:",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpSubtitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"1) Check your Wi-Fi signal strength or try connecting to a different network.\n2) Restart your device (computer, phone, etc.) and your router/modem.\n3) If you're using cellular data, ensure you have a strong signal and haven't reached your data limit.",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setLocationError() {
|
|
||||||
messageTypes["Location Denied"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Location Services Not Enabled",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"To get the most out of MIH, we need your location. Please go to the site settings of the app and enable location services. Once you do that, we can start showing you relevant information based on your location.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// const SizedBox(height: 15),
|
|
||||||
// Text(
|
|
||||||
// "Here are a few things you can try:",
|
|
||||||
// style: TextStyle(
|
|
||||||
// color: MihColors.getRedColor(MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// fontSize: popUpSubtitleSize,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// const SizedBox(height: 10),
|
|
||||||
// Text(
|
|
||||||
// "1) Check your Wi-Fi signal strength or try connecting to a different network.\n2) Restart your device (computer, phone, etc.) and your router/modem.\n3) If you're using cellular data, ensure you have a strong signal and haven't reached your data limit.",
|
|
||||||
// textAlign: TextAlign.left,
|
|
||||||
// style: TextStyle(
|
|
||||||
// color:
|
|
||||||
// MihColors.getSecondaryColor(MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// fontSize: popUpBodySize,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getErrorMessage(String type) {
|
|
||||||
return messageTypes[type];
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
size = MediaQuery.of(context).size;
|
|
||||||
checkScreenSize();
|
|
||||||
setInputError();
|
|
||||||
setinvalidCredError();
|
|
||||||
setInternetError();
|
|
||||||
setUserExistsError();
|
|
||||||
setPwdMatchError();
|
|
||||||
setPasswordRequirementsError();
|
|
||||||
setInvalidEmailError();
|
|
||||||
setInvalidUsernameError();
|
|
||||||
setLocationError();
|
|
||||||
//print(size);
|
|
||||||
// setState(() {
|
|
||||||
// width = size.width;
|
|
||||||
// height = size.height;
|
|
||||||
// });
|
|
||||||
return Dialog(child: getErrorMessage(widget.errorType));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,97 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
import 'package:gif_view/gif_view.dart';
|
|
||||||
|
|
||||||
class Mihloadingcircle extends StatefulWidget {
|
|
||||||
final String? message;
|
|
||||||
const Mihloadingcircle({super.key, this.message});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<Mihloadingcircle> createState() => _MihloadingcircleState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihloadingcircleState extends State<Mihloadingcircle> {
|
|
||||||
// final GifController _controller = GifController();
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
|
|
||||||
late double width;
|
|
||||||
late double height;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = 250;
|
|
||||||
popUpheight = 250;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = 250;
|
|
||||||
popUpheight = 250;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
//_controller.animateTo(26);
|
|
||||||
super.initState();
|
|
||||||
checkScreenSize();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
// _controller.dispose();
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Dialog(
|
|
||||||
child: IntrinsicWidth(
|
|
||||||
child: IntrinsicHeight(
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
// width: 250,
|
|
||||||
// height: 275,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
GifView.asset(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.loadingImageLocation(),
|
|
||||||
height: 200,
|
|
||||||
width: 200,
|
|
||||||
frameRate: 30,
|
|
||||||
),
|
|
||||||
widget.message != null
|
|
||||||
? Text(
|
|
||||||
widget.message!,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: SizedBox(),
|
|
||||||
],
|
|
||||||
)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,150 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
|
|
||||||
class MIHSuccessMessage extends StatefulWidget {
|
|
||||||
final String successType;
|
|
||||||
final String successMessage;
|
|
||||||
const MIHSuccessMessage({
|
|
||||||
super.key,
|
|
||||||
required this.successType,
|
|
||||||
required this.successMessage,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHSuccessMessage> createState() => _MIHSuccessMessageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHSuccessMessageState extends State<MIHSuccessMessage> {
|
|
||||||
var messageTypes = <String, Widget>{};
|
|
||||||
late String message;
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
late double popUpTitleSize;
|
|
||||||
late double popUpSubtitleSize;
|
|
||||||
late double popUpBodySize;
|
|
||||||
late double popUpIconSize;
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
late Size? size;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = (size!.width / 4) * 2;
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 25.0;
|
|
||||||
popUpSubtitleSize = 20.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = size!.width - (size!.width * 0.1);
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 20.0;
|
|
||||||
popUpSubtitleSize = 18.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setSuccessmessage() {
|
|
||||||
messageTypes["Success"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.check_circle_outline_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Success!",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Center(
|
|
||||||
child: Text(
|
|
||||||
message,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Dismiss",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getSuccessMessage(String type) {
|
|
||||||
return messageTypes[type];
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
size = MediaQuery.of(context).size;
|
|
||||||
checkScreenSize();
|
|
||||||
message = widget.successMessage;
|
|
||||||
setSuccessmessage();
|
|
||||||
return Dialog(child: getSuccessMessage(widget.successType));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,519 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import '../../main.dart';
|
|
||||||
|
|
||||||
class MIHWarningMessage extends StatefulWidget {
|
|
||||||
final String warningType;
|
|
||||||
const MIHWarningMessage({
|
|
||||||
super.key,
|
|
||||||
required this.warningType,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MIHWarningMessage> createState() => _MIHDeleteMessageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MIHDeleteMessageState extends State<MIHWarningMessage> {
|
|
||||||
var messageTypes = <String, Widget>{};
|
|
||||||
late double popUpWidth;
|
|
||||||
late double? popUpheight;
|
|
||||||
late double popUpTitleSize;
|
|
||||||
late double popUpSubtitleSize;
|
|
||||||
late double popUpBodySize;
|
|
||||||
late double popUpIconSize;
|
|
||||||
late double popUpPaddingSize;
|
|
||||||
late double width;
|
|
||||||
late double height;
|
|
||||||
|
|
||||||
void checkScreenSize() {
|
|
||||||
if (MzansiInnovationHub.of(context)!.theme.screenType == "desktop") {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = (width / 4) * 2;
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 25.0;
|
|
||||||
popUpSubtitleSize = 20.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 25.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
popUpWidth = width - (width * 0.1);
|
|
||||||
popUpheight = null;
|
|
||||||
popUpTitleSize = 20.0;
|
|
||||||
popUpSubtitleSize = 18.0;
|
|
||||||
popUpBodySize = 15;
|
|
||||||
popUpPaddingSize = 15.0;
|
|
||||||
popUpIconSize = 100;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setNoAccess() {
|
|
||||||
messageTypes["No Access"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Access Pending",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Your access request is currently being reviewed.\nOnce approved, you'll be able to view patient data.\nPlease follow up with the patient to approve your access request.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setNoPatientAccess() {
|
|
||||||
messageTypes["No Patient Access"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Patient Profile Access Needed",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"To proceed with booking an appointment, you must have access to the patient's profile. This will allow you to view their medical history and other relevant information. Please contact the appropriate administrator to request access.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setAppointmentCanelled() {
|
|
||||||
messageTypes["Appointment Canelled"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Appointment Cancelled",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This appointment has been canceled. As a result, you no longer have access to the patient's profile. If you would like to view the patient's profile again, please book a new appointment.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setAccessCanelled() {
|
|
||||||
messageTypes["Access Cancelled"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Access Cancelled",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"This appointment has been canceled. As a result, access has been cancelled and the doctor no longer have access to the patient's profile. If you would like them to view the patient's profile again, please book a new appointment through them.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setAccessDeclined() {
|
|
||||||
messageTypes["Access Declined"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Access Declined",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Your request to access the patient's profile has been denied. Please contact the patient directly to inquire about the reason for this restriction.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void setExpiredAccess() {
|
|
||||||
messageTypes["Expired Access"] = Stack(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(popUpPaddingSize),
|
|
||||||
width: popUpWidth,
|
|
||||||
height: popUpheight,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 5.0),
|
|
||||||
),
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: popUpIconSize,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"Access Expired",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpTitleSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Text(
|
|
||||||
"You no longer have access to this patient profile. The authorized access period has ended. Access to a patients profile is limited to 7 days from appointment date.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: popUpBodySize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: 5,
|
|
||||||
right: 5,
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.close,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 35,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getDeleteMessage(String type) {
|
|
||||||
return messageTypes[type];
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
var size = MediaQuery.of(context).size;
|
|
||||||
setState(() {
|
|
||||||
width = size.width;
|
|
||||||
height = size.height;
|
|
||||||
});
|
|
||||||
checkScreenSize();
|
|
||||||
setNoAccess();
|
|
||||||
setExpiredAccess();
|
|
||||||
setAccessDeclined();
|
|
||||||
setAppointmentCanelled();
|
|
||||||
setAccessCanelled();
|
|
||||||
//print(size);
|
|
||||||
// setState(() {
|
|
||||||
// width = size.width;
|
|
||||||
// height = size.height;
|
|
||||||
// });
|
|
||||||
return Dialog(child: getDeleteMessage(widget.warningType));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:google_mobile_ads/google_mobile_ads.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_env.dart';
|
|
||||||
|
|
||||||
class MihBannerAdProvider extends ChangeNotifier {
|
|
||||||
BannerAd? bannerAd;
|
|
||||||
final adUnitId = AppEnviroment.bannerAdUnitId;
|
|
||||||
bool isBannerAdLoaded = false;
|
|
||||||
String errorMessage = '';
|
|
||||||
|
|
||||||
MihBannerAdProvider({
|
|
||||||
this.bannerAd,
|
|
||||||
this.isBannerAdLoaded = false,
|
|
||||||
this.errorMessage = '',
|
|
||||||
});
|
|
||||||
|
|
||||||
void reset() {
|
|
||||||
bannerAd = null;
|
|
||||||
isBannerAdLoaded = false;
|
|
||||||
errorMessage = "";
|
|
||||||
notifyListeners();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
bannerAd?.dispose();
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
void loadBannerAd() {
|
|
||||||
if (bannerAd != null) {
|
|
||||||
bannerAd!.dispose();
|
|
||||||
bannerAd = null;
|
|
||||||
isBannerAdLoaded = false;
|
|
||||||
}
|
|
||||||
bannerAd = BannerAd(
|
|
||||||
adUnitId: adUnitId,
|
|
||||||
request: const AdRequest(),
|
|
||||||
size: AdSize.banner,
|
|
||||||
listener: BannerAdListener(
|
|
||||||
onAdLoaded: (ad) {
|
|
||||||
debugPrint('$ad loaded.');
|
|
||||||
isBannerAdLoaded = true;
|
|
||||||
notifyListeners();
|
|
||||||
},
|
|
||||||
onAdFailedToLoad: (ad, err) {
|
|
||||||
debugPrint('BannerAd failed to load: $err');
|
|
||||||
errorMessage =
|
|
||||||
'Failed to load ad- Message: ${err.message} Code :${err.code}';
|
|
||||||
ad.dispose(); // Dispose the ad to free resources
|
|
||||||
isBannerAdLoaded = false; // ⬅️ Explicitly set to false
|
|
||||||
bannerAd = null; // ⬅️ Explicitly set to null
|
|
||||||
notifyListeners();
|
|
||||||
},
|
|
||||||
onAdOpened: (Ad ad) => debugPrint('$ad opened.'),
|
|
||||||
onAdClosed: (Ad ad) => debugPrint('$ad closed.'),
|
|
||||||
onAdImpression: (Ad ad) => debugPrint('$ad impression.'),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
bannerAd!.load();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MzansiAiProvider extends ChangeNotifier {
|
|
||||||
int toolIndex;
|
|
||||||
String? startUpQuestion;
|
|
||||||
|
|
||||||
MzansiAiProvider({
|
|
||||||
this.toolIndex = 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
void reset() {
|
|
||||||
toolIndex = 0;
|
|
||||||
startUpQuestion = null;
|
|
||||||
notifyListeners();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setToolIndex(int index) {
|
|
||||||
toolIndex = index;
|
|
||||||
notifyListeners();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setStartUpQuestion(String? question) {
|
|
||||||
startUpQuestion = question;
|
|
||||||
notifyListeners();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,123 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
class MihColors {
|
|
||||||
static Color getPrimaryColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0XFF3A4454);
|
|
||||||
} else {
|
|
||||||
return const Color(0XFFbedcfe);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getSecondaryColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0XFFbedcfe);
|
|
||||||
} else {
|
|
||||||
return const Color(0XFF3A4454);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getHighlightColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0XFF9bc7fa);
|
|
||||||
} else {
|
|
||||||
return const Color(0XFF354866);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getGreyColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0XFFc8c8c8);
|
|
||||||
} else {
|
|
||||||
return const Color(0XFF747474);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getGreenColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xff8ae290);
|
|
||||||
} else {
|
|
||||||
return const Color(0xFF41B349);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getRedColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffD87E8B);
|
|
||||||
} else {
|
|
||||||
return const Color(0xffbb3d4f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getPinkColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffdaa2e9);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xffdaa2e9);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getOrangeColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffd69d7d);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xFFBD7145);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getYellowColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xfff4e467);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xffd4af37);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getBluishPurpleColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xff6e7dcc);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xFF5567C0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getPurpleColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffb682e7);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xFF9857D4);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getGoldColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xFFD4AF37);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xffFFD700);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getSilverColor(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffC0C0C0);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xFFA6A6A6);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Color getBronze(bool darkMode) {
|
|
||||||
if (darkMode == true) {
|
|
||||||
return const Color(0xffB1560F);
|
|
||||||
} else {
|
|
||||||
// Add a different shade of pink for light mode
|
|
||||||
return const Color(0xFFCD7F32);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,253 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import "package:universal_html/html.dart" as html;
|
|
||||||
|
|
||||||
class MihTheme {
|
|
||||||
// late int _mainColor;
|
|
||||||
// late int _secondColor;
|
|
||||||
//late int _errColor;
|
|
||||||
//late int _succColor;
|
|
||||||
// late int _mesColor;
|
|
||||||
late String mode;
|
|
||||||
late String screenType;
|
|
||||||
late AssetImage loading;
|
|
||||||
late String loadingAssetText;
|
|
||||||
late TargetPlatform platform;
|
|
||||||
bool kIsWeb = const bool.fromEnvironment('dart.library.js_util');
|
|
||||||
String latestVersion = "1.2.2";
|
|
||||||
// Options:-
|
|
||||||
// f3f9d2 = Cream
|
|
||||||
// f0f0c9 = cream2
|
|
||||||
// caffd0 = light green
|
|
||||||
// B0F2B4 = light grean 2 *
|
|
||||||
// 85bda6 = light green 3
|
|
||||||
// 70f8ba = green
|
|
||||||
// F7F3EA = white
|
|
||||||
// a63446 = red
|
|
||||||
//747474
|
|
||||||
|
|
||||||
MihTheme() {
|
|
||||||
mode = "Dark";
|
|
||||||
//_errColor = 0xffD87E8B;
|
|
||||||
//_succColor = 0xffB0F2B4;
|
|
||||||
//_mesColor = 0xffc8c8c8d9;
|
|
||||||
}
|
|
||||||
|
|
||||||
ThemeData getData(bool bool) {
|
|
||||||
return ThemeData(
|
|
||||||
fontFamily: 'Segoe UI',
|
|
||||||
scaffoldBackgroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
// pageTransitionsTheme: PageTransitionsTheme(
|
|
||||||
// builders: Map<TargetPlatform, PageTransitionsBuilder>.fromIterable(
|
|
||||||
// TargetPlatform.values,
|
|
||||||
// value: (dynamic _) => const FadeUpwardsPageTransitionsBuilder(),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
colorScheme: ColorScheme(
|
|
||||||
brightness: getBritness(),
|
|
||||||
primary: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
onPrimary: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
secondary: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
onSecondary: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
error: MihColors.getRedColor(mode == "Dark"),
|
|
||||||
onError: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
surface: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
onSurface: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
),
|
|
||||||
datePickerTheme: DatePickerThemeData(
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
headerBackgroundColor: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
headerForegroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
),
|
|
||||||
appBarTheme: AppBarTheme(
|
|
||||||
color: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
foregroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
titleTextStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
floatingActionButtonTheme: FloatingActionButtonThemeData(
|
|
||||||
backgroundColor: MihColors.getSecondaryColor(mode == "Dark"),
|
|
||||||
foregroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
extendedTextStyle:
|
|
||||||
TextStyle(color: MihColors.getPrimaryColor(mode == "Dark")),
|
|
||||||
),
|
|
||||||
drawerTheme: DrawerThemeData(
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(mode == "Dark"),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
String getPlatform() {
|
|
||||||
// if (isPwa()) {
|
|
||||||
// if (platform == TargetPlatform.android) {
|
|
||||||
// return "Android";
|
|
||||||
// } else if (platform == TargetPlatform.iOS) {
|
|
||||||
// return "iOS";
|
|
||||||
// }
|
|
||||||
// } else
|
|
||||||
if (kIsWeb) {
|
|
||||||
return "Web";
|
|
||||||
} else if (!kIsWeb) {
|
|
||||||
if (platform == TargetPlatform.android) {
|
|
||||||
return "Android";
|
|
||||||
} else if (platform == TargetPlatform.iOS) {
|
|
||||||
return "iOS";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return "Other";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isPwa() {
|
|
||||||
return html.window.matchMedia('(display-mode: standalone)').matches;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setMode(String m) {
|
|
||||||
mode;
|
|
||||||
}
|
|
||||||
|
|
||||||
String getLatestVersion() {
|
|
||||||
return latestVersion;
|
|
||||||
}
|
|
||||||
|
|
||||||
ThemeData getThemeData() {
|
|
||||||
return getData(mode == "Dark");
|
|
||||||
}
|
|
||||||
|
|
||||||
ThemeData darkMode() {
|
|
||||||
return getData(mode == "Dark");
|
|
||||||
}
|
|
||||||
|
|
||||||
ThemeData lightMode() {
|
|
||||||
return getData(mode == "Dark");
|
|
||||||
}
|
|
||||||
|
|
||||||
Brightness getBritness() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
return Brightness.dark;
|
|
||||||
} else {
|
|
||||||
return Brightness.light;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Color messageTextColor() {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// _mesColor = 0XFFc8c8c8;
|
|
||||||
// } else {
|
|
||||||
// _mesColor = 0XFF747474;
|
|
||||||
// }
|
|
||||||
// return Color(_mesColor);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// Color errorColor() {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// return const Color(0xffD87E8B);
|
|
||||||
// } else {
|
|
||||||
// return const Color(0xffbb3d4f);
|
|
||||||
// }
|
|
||||||
// //return Color(_errColor);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// Color highlightColor() {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// return const Color(0XFF9bc7fa);
|
|
||||||
// } else {
|
|
||||||
// return const Color(0XFF354866);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
// Color successColor() {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// return const Color(0xffB0F2B4);
|
|
||||||
// } else {
|
|
||||||
// return const Color(0xff56a95b);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
AssetImage loadingImage() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
loading = const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_light.gif',
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
loading = const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_dark.gif',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return loading;
|
|
||||||
}
|
|
||||||
|
|
||||||
AssetImage altLoadingImage() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
loading = const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_dark.gif',
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
loading = const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_light.gif',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return loading;
|
|
||||||
}
|
|
||||||
|
|
||||||
String loadingImageLocation() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
loadingAssetText =
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_light.gif';
|
|
||||||
} else {
|
|
||||||
loadingAssetText =
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_dark.gif';
|
|
||||||
}
|
|
||||||
return loadingAssetText;
|
|
||||||
}
|
|
||||||
|
|
||||||
String altLoadingImageLocation() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
loadingAssetText =
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_dark.gif';
|
|
||||||
} else {
|
|
||||||
loadingAssetText =
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/loading_light.gif';
|
|
||||||
}
|
|
||||||
return loadingAssetText;
|
|
||||||
}
|
|
||||||
|
|
||||||
AssetImage aiLogoImage() {
|
|
||||||
if (mode == "Dark") {
|
|
||||||
return const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/mzansi_ai-dark.png',
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return const AssetImage(
|
|
||||||
'lib/mih_components/mih_package_components/assets/images/mzansi_ai-light.png',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setScreenType(double width) {
|
|
||||||
if (width <= 800) {
|
|
||||||
screenType = "mobile";
|
|
||||||
} else {
|
|
||||||
screenType = "desktop";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Color MihColors.getPrimaryColor(mode == "Dark") {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// _mainColor = 0XFF3A4454;
|
|
||||||
// } else {
|
|
||||||
// _mainColor = 0XFFbedcfe;
|
|
||||||
// }
|
|
||||||
// return Color(_mainColor);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// Color MihColors.getSecondaryColor(mode == "Dark") {
|
|
||||||
// if (mode == "Dark") {
|
|
||||||
// _secondColor = 0XFFbedcfe;
|
|
||||||
// } else {
|
|
||||||
// _secondColor = 0XFF3A4454;
|
|
||||||
// }
|
|
||||||
// return Color(_secondColor);
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
@@ -1,287 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_single_child_scroll.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
|
||||||
|
|
||||||
class MihAttributes extends StatefulWidget {
|
|
||||||
const MihAttributes({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihAttributes> createState() => _MihAttributesState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihAttributesState extends State<MihAttributes> {
|
|
||||||
Future<void> launchUrlLink(Uri linkUrl) async {
|
|
||||||
if (!await launchUrl(linkUrl)) {
|
|
||||||
throw Exception('Could not launch $linkUrl');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
TableRow displayIcon(IconData icon, String creator, String link) {
|
|
||||||
return TableRow(
|
|
||||||
children: [
|
|
||||||
TableCell(
|
|
||||||
verticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
child: SizedBox(
|
|
||||||
height: 150,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 15.0),
|
|
||||||
child: FittedBox(
|
|
||||||
child: Center(
|
|
||||||
child: Icon(
|
|
||||||
icon,
|
|
||||||
// size: 125,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TableCell(
|
|
||||||
verticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(15.0),
|
|
||||||
child: Center(
|
|
||||||
child: Text(
|
|
||||||
creator,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TableCell(
|
|
||||||
verticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 15.0),
|
|
||||||
child: MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
launchUrlLink(
|
|
||||||
Uri.parse(
|
|
||||||
link,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 100,
|
|
||||||
child: Text(
|
|
||||||
"Visit",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
innerHorizontalPadding: 10,
|
|
||||||
bodyItem: getBody(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody() {
|
|
||||||
String message =
|
|
||||||
"Some APIs, Icons and Assets used in MIH were sourced from third party providers.\n";
|
|
||||||
message +=
|
|
||||||
"We are grateful to the talented creators for providing these resources.\n";
|
|
||||||
message +=
|
|
||||||
"As per the terms for free use for these third party providers, the following assets require attribution";
|
|
||||||
|
|
||||||
return MihSingleChildScroll(
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
MihIcons.mihLogo,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 165,
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
SelectableText(
|
|
||||||
message,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
SizedBox(
|
|
||||||
width: 700,
|
|
||||||
child: Table(
|
|
||||||
defaultVerticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
columnWidths: const {
|
|
||||||
0: FlexColumnWidth(1),
|
|
||||||
1: FlexColumnWidth(1),
|
|
||||||
2: FlexColumnWidth(1),
|
|
||||||
},
|
|
||||||
children: [
|
|
||||||
const TableRow(
|
|
||||||
children: [
|
|
||||||
TableCell(
|
|
||||||
verticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 15.0),
|
|
||||||
child: Center(
|
|
||||||
child: Text(
|
|
||||||
"Resources",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TableCell(
|
|
||||||
verticalAlignment: TableCellVerticalAlignment.middle,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 15.0),
|
|
||||||
child: Center(
|
|
||||||
child: Text(
|
|
||||||
"Creator",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TableCell(
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(bottom: 15.0),
|
|
||||||
child: Center(
|
|
||||||
child: Text(
|
|
||||||
"Link",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
displayIcon(MihIcons.mihRing, "Tarah Meth",
|
|
||||||
"https://www.linkedin.com/in/tarah-meth-3b6309254/"),
|
|
||||||
displayIcon(MihIcons.mihLogo, "Tarah Meth",
|
|
||||||
"https://www.linkedin.com/in/tarah-meth-3b6309254/"),
|
|
||||||
displayIcon(MihIcons.mzansiAi, "Ollama", "https://ollama.com/"),
|
|
||||||
displayIcon(MihIcons.mzansiWallet, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/wallet-passes-app_3884407?term=wallet&page=1&position=21&origin=search&related_id=3884407"),
|
|
||||||
displayIcon(MihIcons.patientProfile, "RaftelDesign",
|
|
||||||
"https://www.flaticon.com/free-icon/patient_2376100?term=medication&page=1&position=6&origin=search&related_id=2376100"),
|
|
||||||
displayIcon(MihIcons.patientProfile, "Srip",
|
|
||||||
"https://www.flaticon.com/free-icon/hospital_1233930?term=medical+snake&page=1&position=7&origin=search&related_id=1233930"),
|
|
||||||
displayIcon(MihIcons.calendar, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/calendar_2278049?term=calendar&page=1&position=5&origin=search&related_id=2278049"),
|
|
||||||
displayIcon(MihIcons.calculator, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/calculator_2374409?term=calculator&page=1&position=20&origin=search&related_id=2374409"),
|
|
||||||
displayIcon(MihIcons.aboutMih, "Chanut",
|
|
||||||
"https://www.flaticon.com/free-icon/info_151776?term=about&page=1&position=8&origin=search&related_id=151776"),
|
|
||||||
displayIcon(MihIcons.personalProfile, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/user_1077063?term=profile&page=1&position=6&origin=search&related_id=1077063"),
|
|
||||||
displayIcon(MihIcons.businessProfile, "Gravisio",
|
|
||||||
"https://www.flaticon.com/free-icon/contractor_11813336?term=company+profile&page=1&position=2&origin=search&related_id=11813336"),
|
|
||||||
displayIcon(MihIcons.patientManager, "Vector Tank",
|
|
||||||
"https://www.flaticon.com/free-icon/doctor_10215061?term=doctor&page=1&position=73&origin=search&related_id=10215061"),
|
|
||||||
displayIcon(MihIcons.profileSetup, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/add-user_748137?term=profile+add&page=1&position=1&origin=search&related_id=748137"),
|
|
||||||
displayIcon(MihIcons.businessSetup, "kerismaker",
|
|
||||||
"https://www.flaticon.com/free-icon/business_13569850?term=company+add&page=1&position=25&origin=search&related_id=13569850"),
|
|
||||||
displayIcon(MihIcons.calculator, "fawazahmed0",
|
|
||||||
"https://github.com/fawazahmed0/exchange-api"),
|
|
||||||
displayIcon(MihIcons.iDontKnow, "Freepik",
|
|
||||||
"https://www.flaticon.com/free-icon/i-dont-know_5359909?term=i+dont+know&page=1&position=7&origin=search&related_id=5359909"),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// SizedBox(
|
|
||||||
// width: 500,
|
|
||||||
// child: Column(
|
|
||||||
// children: [
|
|
||||||
// const SizedBox(
|
|
||||||
// width: double.infinity,
|
|
||||||
// child: Row(
|
|
||||||
// mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
||||||
// mainAxisSize: MainAxisSize.max,
|
|
||||||
// children: [
|
|
||||||
// Flexible(
|
|
||||||
// child: Text(
|
|
||||||
// "Icon",
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontSize: 25,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// Flexible(
|
|
||||||
// child: Text(
|
|
||||||
// "Creator",
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontSize: 25,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// Flexible(
|
|
||||||
// child: Text(
|
|
||||||
// "Link",
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontSize: 25,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// const Padding(
|
|
||||||
// padding: EdgeInsets.symmetric(vertical: 10.0),
|
|
||||||
// child: Divider(),
|
|
||||||
// ),
|
|
||||||
// displayIcon(MihIcons.mihLogo, "Tarah Meth",
|
|
||||||
// "https://app.mzansi-innovation-hub.co.za/"),
|
|
||||||
// const SizedBox(height: 10),
|
|
||||||
// displayIcon(MihIcons.mihLogo, "Test",
|
|
||||||
// "https://www.flaticon.com/free-icons/mih"),
|
|
||||||
// const SizedBox(height: 10),
|
|
||||||
// displayIcon(MihIcons.mihLogo, "Test",
|
|
||||||
// "https://www.flaticon.com/free-icons/mih"),
|
|
||||||
// const SizedBox(height: 10),
|
|
||||||
// displayIcon(MihIcons.mihLogo, "Test",
|
|
||||||
// "https://www.flaticon.com/free-icons/mih"),
|
|
||||||
// const SizedBox(height: 10),
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// )
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,960 +0,0 @@
|
|||||||
import 'package:flutter_speed_dial/flutter_speed_dial.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_business_details_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_install_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_single_child_scroll.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_layout/mih_tile.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tool_body.dart';
|
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_floating_menu.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_user_services.dart';
|
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
|
||||||
import 'package:redacted/redacted.dart';
|
|
||||||
import 'package:share_plus/share_plus.dart';
|
|
||||||
|
|
||||||
class MihInfo extends StatefulWidget {
|
|
||||||
const MihInfo({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihInfo> createState() => _MihInfoState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihInfoState extends State<MihInfo> {
|
|
||||||
late Future<int> _futureUserCount;
|
|
||||||
late Future<int> _futureBusinessCount;
|
|
||||||
final Uri _tiktokUrl =
|
|
||||||
Uri.parse('https://www.tiktok.com/@mzansi.innovation.hub');
|
|
||||||
final Uri _whatsappUrl =
|
|
||||||
Uri.parse('https://whatsapp.com/channel/0029Vax3INCIyPtMn8KgeM2F');
|
|
||||||
final Uri _twitch = Uri.parse('https://www.twitch.tv/mzansi_innovation_hub');
|
|
||||||
final Uri _kick = Uri.parse('https://kick.com/mzansi-innovation-hub');
|
|
||||||
final Uri _threadsUrl =
|
|
||||||
Uri.parse('https://www.threads.net/@mzansi.innovation.hub');
|
|
||||||
final Uri _instagramUrl =
|
|
||||||
Uri.parse('https://www.instagram.com/mzansi.innovation.hub');
|
|
||||||
final Uri _youtubeUrl =
|
|
||||||
Uri.parse('https://www.youtube.com/@mzansiinnovationhub');
|
|
||||||
final Uri _xUrl = Uri.parse('https://x.com/mzansi_inno_hub');
|
|
||||||
final Uri _linkedinUrl =
|
|
||||||
Uri.parse('https://www.linkedin.com/company/mzansi-innovation-hub/');
|
|
||||||
final Uri _facebookUrl =
|
|
||||||
Uri.parse('https://www.facebook.com/profile.php?id=61565345762136');
|
|
||||||
final Uri _redditUrl =
|
|
||||||
Uri.parse('https://www.reddit.com/r/Mzani_Innovation_Hub/');
|
|
||||||
|
|
||||||
Widget founderBio() {
|
|
||||||
String bio = "";
|
|
||||||
bio += "BSc Computer Science & Information Systems\n";
|
|
||||||
bio += "(University of the Western Cape)\n";
|
|
||||||
bio +=
|
|
||||||
"6 Year of banking experience with one of the big 5 banks of South Africa.";
|
|
||||||
return Wrap(
|
|
||||||
alignment: WrapAlignment.center,
|
|
||||||
crossAxisAlignment: WrapCrossAlignment.center,
|
|
||||||
spacing: 10,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 300,
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
fit: StackFit.loose,
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(left: 4.0),
|
|
||||||
child: CircleAvatar(
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
backgroundImage: const AssetImage(
|
|
||||||
"lib/mih_components/mih_package_components/assets/images/founder.jpg"),
|
|
||||||
//'https://media.licdn.com/dms/image/D4D03AQGd1-QhjtWWpA/profile-displayphoto-shrink_400_400/0/1671698053061?e=2147483647&v=beta&t=a3dJI5yxs5-KeXjj10LcNCFuC9IOfa8nNn3k_Qyr0CA'),
|
|
||||||
radius: 75,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Icon(
|
|
||||||
MihIcons.mihRing,
|
|
||||||
size: 165,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(
|
|
||||||
width: 400,
|
|
||||||
child: Text(
|
|
||||||
bio,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
//fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 17,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget founderTitle() {
|
|
||||||
String heading = "Yasien Meth (Founder & CEO)";
|
|
||||||
return Column(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
heading,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 25,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget ourVision() {
|
|
||||||
String heading = "Our Vision";
|
|
||||||
String vision =
|
|
||||||
"Digitizing Mzansi one process at a time. Discover essential Mzansi apps to streamline your personal and professional life. Simplify your daily tasks with our user-friendly solutions.";
|
|
||||||
|
|
||||||
return SizedBox(
|
|
||||||
width: 500,
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
heading,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 25,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
vision,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
//fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 17,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget ourMission() {
|
|
||||||
String heading = "Our Mission";
|
|
||||||
String mission =
|
|
||||||
"Bridge the digital divide in Mzansi, ensuring that everyone can benefit from the power of technology. We empower lives by providing simple, elegant solutions that elevate daily experiences. With our user-friendly approach, we're making the digital world accessible to all, ensuring no one is left behind in the digital revolution.";
|
|
||||||
|
|
||||||
return SizedBox(
|
|
||||||
width: 500,
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
heading,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 25,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
mission,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
//fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 17,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget mihSocials() {
|
|
||||||
String heading = "Follow Our Journey";
|
|
||||||
return Column(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
heading,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 25,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
SizedBox(
|
|
||||||
width: 500,
|
|
||||||
height: 600,
|
|
||||||
child: GridView.builder(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
// left: width / 10,
|
|
||||||
// right: width / 10,
|
|
||||||
// //bottom: height / 5,
|
|
||||||
// top: 20,
|
|
||||||
),
|
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
// shrinkWrap: true,
|
|
||||||
itemCount: getSocialsList().length,
|
|
||||||
gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
|
|
||||||
mainAxisSpacing: 15, maxCrossAxisExtent: 150),
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return getSocialsList()[index];
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getSocialsList() {
|
|
||||||
List<Widget> socials = [];
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_youtubeUrl);
|
|
||||||
},
|
|
||||||
tileName: "YouTube",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.youtube,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 175,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_tiktokUrl);
|
|
||||||
},
|
|
||||||
tileName: "TikTok",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.tiktok,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_twitch);
|
|
||||||
},
|
|
||||||
tileName: "Twitch",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.twitch,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_threadsUrl);
|
|
||||||
},
|
|
||||||
tileName: "Threads",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.threads,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_whatsappUrl);
|
|
||||||
},
|
|
||||||
tileName: "Whatsapp",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.whatsapp,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_instagramUrl);
|
|
||||||
},
|
|
||||||
tileName: "Instagram",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.instagram,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_xUrl);
|
|
||||||
},
|
|
||||||
tileName: "X",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.xTwitter,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_linkedinUrl);
|
|
||||||
},
|
|
||||||
tileName: "LinkedIn",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.linkedin,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_facebookUrl);
|
|
||||||
},
|
|
||||||
tileName: "FaceBook",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.facebook,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_redditUrl);
|
|
||||||
},
|
|
||||||
tileName: "Reddit",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: FaIcon(
|
|
||||||
FontAwesomeIcons.reddit,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
size: 200,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
socials.add(MIHTile(
|
|
||||||
onTap: () {
|
|
||||||
launchSocialUrl(_kick);
|
|
||||||
},
|
|
||||||
tileName: "Kick",
|
|
||||||
tileIcon: Center(
|
|
||||||
child: Text(
|
|
||||||
"KICK",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 100,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// FaIcon(
|
|
||||||
// FontAwesomeIcons.tv,
|
|
||||||
// color: MihColors.getPrimaryColor(MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// size: 200,
|
|
||||||
// ),
|
|
||||||
),
|
|
||||||
p: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
s: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
));
|
|
||||||
//==================================================================
|
|
||||||
return socials;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> launchSocialUrl(Uri linkUrl) async {
|
|
||||||
if (!await launchUrl(linkUrl)) {
|
|
||||||
throw Exception('Could not launch $linkUrl');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getInstallButtonText() {
|
|
||||||
final isWebAndroid =
|
|
||||||
kIsWeb && (defaultTargetPlatform == TargetPlatform.android);
|
|
||||||
final isWebIos = kIsWeb && (defaultTargetPlatform == TargetPlatform.iOS);
|
|
||||||
String btnText = "";
|
|
||||||
IconData platformIcon;
|
|
||||||
if (isWebAndroid) {
|
|
||||||
btnText = "Install MIH";
|
|
||||||
platformIcon = FontAwesomeIcons.googlePlay;
|
|
||||||
} else if (isWebIos) {
|
|
||||||
btnText = "Install MIH";
|
|
||||||
platformIcon = FontAwesomeIcons.appStoreIos;
|
|
||||||
} else if (MzansiInnovationHub.of(context)!.theme.getPlatform() ==
|
|
||||||
"Android") {
|
|
||||||
btnText = "Update MIH";
|
|
||||||
platformIcon = FontAwesomeIcons.googlePlay;
|
|
||||||
} else if (MzansiInnovationHub.of(context)!.theme.getPlatform() == "iOS") {
|
|
||||||
btnText = "Update MIH";
|
|
||||||
platformIcon = FontAwesomeIcons.appStoreIos;
|
|
||||||
} else {
|
|
||||||
btnText = "Install MIH";
|
|
||||||
platformIcon = FontAwesomeIcons.globe;
|
|
||||||
}
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
FaIcon(
|
|
||||||
platformIcon,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(
|
|
||||||
btnText,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void shareMIHLink(BuildContext context, String message, String link) {
|
|
||||||
String shareText = "$message: $link";
|
|
||||||
SharePlus.instance.share(
|
|
||||||
ShareParams(text: shareText),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget displayBusinessCount() {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
FutureBuilder<int>(
|
|
||||||
future: _futureBusinessCount,
|
|
||||||
builder: (context, snapshot) {
|
|
||||||
bool isLoading = true;
|
|
||||||
String userCount = "⚠️";
|
|
||||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
|
||||||
isLoading = true;
|
|
||||||
} else if (snapshot.connectionState == ConnectionState.done &&
|
|
||||||
snapshot.hasError) {
|
|
||||||
isLoading = false;
|
|
||||||
} else if (snapshot.connectionState == ConnectionState.done &&
|
|
||||||
snapshot.hasData) {
|
|
||||||
isLoading = false;
|
|
||||||
userCount = snapshot.data.toString();
|
|
||||||
} else {
|
|
||||||
isLoading = true;
|
|
||||||
}
|
|
||||||
return SizedBox(
|
|
||||||
child: Text(
|
|
||||||
userCount,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 23,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
).redacted(
|
|
||||||
context: context,
|
|
||||||
redact: isLoading,
|
|
||||||
configuration: RedactedConfiguration(
|
|
||||||
defaultBorderRadius: BorderRadius.circular(5),
|
|
||||||
redactedColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(
|
|
||||||
"Businesses",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
fontSize: 20,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget displayUserCount() {
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
FutureBuilder<int>(
|
|
||||||
future: _futureUserCount,
|
|
||||||
builder: (context, snapshot) {
|
|
||||||
bool isLoading = true;
|
|
||||||
String userCount = "⚠️";
|
|
||||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
|
||||||
isLoading = true;
|
|
||||||
} else if (snapshot.connectionState == ConnectionState.done &&
|
|
||||||
snapshot.hasError) {
|
|
||||||
isLoading = false;
|
|
||||||
} else if (snapshot.connectionState == ConnectionState.done &&
|
|
||||||
snapshot.hasData) {
|
|
||||||
isLoading = false;
|
|
||||||
userCount = snapshot.data.toString();
|
|
||||||
} else {
|
|
||||||
isLoading = true;
|
|
||||||
}
|
|
||||||
return SizedBox(
|
|
||||||
child: Text(
|
|
||||||
userCount,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 23,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
).redacted(
|
|
||||||
context: context,
|
|
||||||
redact: isLoading,
|
|
||||||
configuration: RedactedConfiguration(
|
|
||||||
defaultBorderRadius: BorderRadius.circular(5),
|
|
||||||
redactedColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(
|
|
||||||
"People",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
fontSize: 20,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_futureUserCount = MihUserServices().fetchUserCount();
|
|
||||||
_futureBusinessCount = MihBusinessDetailsServices().fetchBusinessCount();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
innerHorizontalPadding: 10,
|
|
||||||
bodyItem: getBody(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody() {
|
|
||||||
return Stack(
|
|
||||||
children: [
|
|
||||||
MihSingleChildScroll(
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 165,
|
|
||||||
child: FittedBox(
|
|
||||||
child: Icon(
|
|
||||||
MihIcons.mihLogo,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
const Text(
|
|
||||||
"Mzansi Innovation Hub",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 30,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
"MIH App Version: ${MzansiInnovationHub.of(context)!.theme.getLatestVersion()}",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
fontSize: 15,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// ===================== Divider
|
|
||||||
// Padding(
|
|
||||||
// padding: EdgeInsets.symmetric(
|
|
||||||
// vertical: 10.0,
|
|
||||||
// horizontal: 25,
|
|
||||||
// ),
|
|
||||||
// child: Divider(
|
|
||||||
// thickness: 1,
|
|
||||||
// color: MihColors.getGreyColor(
|
|
||||||
// MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
// Text(
|
|
||||||
// "The MIH Community",
|
|
||||||
// textAlign: TextAlign.center,
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// fontSize: 22,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
Wrap(
|
|
||||||
alignment: WrapAlignment.spaceAround,
|
|
||||||
crossAxisAlignment: WrapCrossAlignment.center,
|
|
||||||
spacing: 25,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
displayUserCount(),
|
|
||||||
displayBusinessCount(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
"The MIH Community",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 22,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// ===================== Divider
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.symmetric(
|
|
||||||
vertical: 10.0,
|
|
||||||
horizontal: 25,
|
|
||||||
),
|
|
||||||
child: Divider(
|
|
||||||
thickness: 1,
|
|
||||||
color: MihColors.getGreyColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// const SizedBox(
|
|
||||||
// height: 10,
|
|
||||||
// ),
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 25.0),
|
|
||||||
child: Wrap(
|
|
||||||
alignment: WrapAlignment.start,
|
|
||||||
crossAxisAlignment: WrapCrossAlignment.start,
|
|
||||||
spacing: 10,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
ourVision(),
|
|
||||||
ourMission(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(
|
|
||||||
height: 25,
|
|
||||||
),
|
|
||||||
Wrap(
|
|
||||||
alignment: WrapAlignment.center,
|
|
||||||
crossAxisAlignment: WrapCrossAlignment.center,
|
|
||||||
spacing: 10,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
MihInstallServices().installMihTrigger(context);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: getInstallButtonText(),
|
|
||||||
),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
launchSocialUrl(
|
|
||||||
Uri.parse(
|
|
||||||
"https://www.youtube.com/playlist?list=PLuT35kJIui0H5kXjxNOZlHoOPZbQLr4qh",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
FaIcon(
|
|
||||||
FontAwesomeIcons.youtube,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(
|
|
||||||
"MIH Beginners Guide",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
launchSocialUrl(
|
|
||||||
Uri.parse(
|
|
||||||
"https://patreon.com/MzansiInnovationHub?utm_medium=unknown&utm_source=join_link&utm_campaign=creatorshare_creator&utm_content=copyLink",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
FaIcon(
|
|
||||||
FontAwesomeIcons.patreon,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Text(
|
|
||||||
"Support Our Journey",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]),
|
|
||||||
const SizedBox(
|
|
||||||
height: 10,
|
|
||||||
),
|
|
||||||
// ===================== Divider
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.symmetric(
|
|
||||||
vertical: 10.0,
|
|
||||||
horizontal: 25,
|
|
||||||
),
|
|
||||||
child: Divider(
|
|
||||||
thickness: 1,
|
|
||||||
color: MihColors.getGreyColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
// spacing: 10,
|
|
||||||
// runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
founderTitle(),
|
|
||||||
founderBio(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
// ===================== Divider
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.symmetric(
|
|
||||||
vertical: 10.0,
|
|
||||||
horizontal: 25,
|
|
||||||
),
|
|
||||||
child: Divider(
|
|
||||||
thickness: 1,
|
|
||||||
color: MihColors.getGreyColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
mihSocials(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
right: 10,
|
|
||||||
bottom: 10,
|
|
||||||
child: MihFloatingMenu(
|
|
||||||
icon: Icons.share,
|
|
||||||
children: [
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.vpn_lock,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
label: "MIH - Web",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onTap: () {
|
|
||||||
shareMIHLink(
|
|
||||||
context,
|
|
||||||
"Check out the MIH app on the Web",
|
|
||||||
"https://app.mzansi-innovation-hub.co.za/",
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.apple,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
label: "MIH - iOS",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onTap: () {
|
|
||||||
shareMIHLink(
|
|
||||||
context,
|
|
||||||
"Check out the MIH app on the App Store",
|
|
||||||
"https://apps.apple.com/za/app/mzansi-innovation-hub/id6743310890",
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.android,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
label: "MIH - Android",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onTap: () {
|
|
||||||
shareMIHLink(
|
|
||||||
context,
|
|
||||||
"Check out the MIH app on the Play Store",
|
|
||||||
"https://play.google.com/store/apps/details?id=za.co.mzansiinnovationhub.mih",
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
import 'package:go_router/go_router.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_tools.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_access_controlls_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/access_review/package_tools/mih_access_requests.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MihAccess extends StatefulWidget {
|
|
||||||
const MihAccess({
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihAccess> createState() => _MihAccessState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihAccessState extends State<MihAccess> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(),
|
|
||||||
appBody: getToolBody(),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
selectedbodyIndex: context.watch<MihAccessControllsProvider>().toolIndex,
|
|
||||||
onIndexChange: (newValue) {
|
|
||||||
context.read<MihAccessControllsProvider>().setToolIndex(newValue);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageAction getAction() {
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
iconSize: 35,
|
|
||||||
onTap: () {
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
);
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools() {
|
|
||||||
Map<Widget, void Function()?> temp = {};
|
|
||||||
temp[const Icon(Icons.people)] = () {
|
|
||||||
context.read<MihAccessControllsProvider>().setToolIndex(0);
|
|
||||||
};
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: context.watch<MihAccessControllsProvider>().toolIndex,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody() {
|
|
||||||
List<Widget> toolBodies = [
|
|
||||||
MihAccessRequest(),
|
|
||||||
];
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
List<String> toolTitles = [
|
|
||||||
"Access",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tile.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihCalculatorTile extends StatefulWidget {
|
|
||||||
final double packageSize;
|
|
||||||
|
|
||||||
const MihCalculatorTile({
|
|
||||||
super.key,
|
|
||||||
required this.packageSize,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihCalculatorTile> createState() => _MihCalculatorTileState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihCalculatorTileState extends State<MihCalculatorTile> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageTile(
|
|
||||||
onTap: () {
|
|
||||||
context.goNamed(
|
|
||||||
"mihCalculator",
|
|
||||||
);
|
|
||||||
},
|
|
||||||
appName: "Calculator",
|
|
||||||
appIcon: Icon(
|
|
||||||
MihIcons.calculator,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// size: widget.packageSize,
|
|
||||||
),
|
|
||||||
iconSize: widget.packageSize,
|
|
||||||
primaryColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
secondaryColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
import 'package:go_router/go_router.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_tools.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_calendar_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/calendar/package_tools/appointments.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MzansiCalendar extends StatefulWidget {
|
|
||||||
const MzansiCalendar({
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MzansiCalendar> createState() => _MzansiCalendarState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MzansiCalendarState extends State<MzansiCalendar> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(),
|
|
||||||
appBody: getToolBody(),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
selectedbodyIndex: context.watch<MihCalendarProvider>().toolIndex,
|
|
||||||
onIndexChange: (newIndex) {
|
|
||||||
context.read<MihCalendarProvider>().setToolIndex(newIndex);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageAction getAction() {
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
iconSize: 35,
|
|
||||||
onTap: () {
|
|
||||||
context.read<MihCalendarProvider>().resetSelectedDay();
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
);
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools() {
|
|
||||||
Map<Widget, void Function()?> temp = {};
|
|
||||||
temp[const Icon(Icons.calendar_month)] = () {
|
|
||||||
context.read<MihCalendarProvider>().setToolIndex(0);
|
|
||||||
};
|
|
||||||
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: context.watch<MihCalendarProvider>().toolIndex,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody() {
|
|
||||||
List<Widget> toolBodies = [
|
|
||||||
//appointment here
|
|
||||||
Appointments(),
|
|
||||||
];
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
MzansiProfileProvider mzansiProfileProvider =
|
|
||||||
context.read<MzansiProfileProvider>();
|
|
||||||
List<String> toolTitles = [
|
|
||||||
mzansiProfileProvider.personalHome == true ? "Personal" : "Business",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,295 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_form.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_alert.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_text_form_field.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_error_message.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_alert_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_authentication_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
|
|
||||||
class MihForgotPassword extends StatefulWidget {
|
|
||||||
const MihForgotPassword({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihForgotPassword> createState() => _MihForgotPasswordState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihForgotPasswordState extends State<MihForgotPassword> {
|
|
||||||
final emailController = TextEditingController();
|
|
||||||
bool successfulForgotPassword = false;
|
|
||||||
final _formKey = GlobalKey<FormState>();
|
|
||||||
final FocusNode _focusNode = FocusNode();
|
|
||||||
bool acceptWarning = false;
|
|
||||||
|
|
||||||
Future<void> submitPasswodReset() async {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const Mihloadingcircle();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
try {
|
|
||||||
var resetPassEmailSent = await MihAuthenticationServices()
|
|
||||||
.forgotPassword(emailController.text);
|
|
||||||
context.pop();
|
|
||||||
if (resetPassEmailSent) {
|
|
||||||
setState(() {
|
|
||||||
successfulForgotPassword = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} on Exception {
|
|
||||||
//loginError();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void prePassResteWarning() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: 100,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
alertTitle: "Password Reset Confirmation",
|
|
||||||
alertBody: Column(
|
|
||||||
//mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 25.0),
|
|
||||||
child: Text(
|
|
||||||
"Before you reset your password, please be aware that you'll receive an email with a link to confirm your identity and set a new password. Make sure to check your inbox, including spam or junk folders. If you don't receive the email within a few minutes, please try resending the reset request.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15.0,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
setState(() {
|
|
||||||
acceptWarning = true;
|
|
||||||
});
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
validateInput();
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Continue",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void resetLinkSentSuccessfully() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.check_circle_outline_rounded,
|
|
||||||
size: 150,
|
|
||||||
color: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
alertTitle: "Successfully Sent Reset Link",
|
|
||||||
alertBody: Column(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"We've sent a password reset link to your email address. Please check your inbox, including spam or junk folders.\n\nOnce you find the email, click on the link to reset your password.\n\nIf you don't receive the email within a few minutes, please try resending the reset request.\n\nThe reset link will expire after 2 hours",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
Center(
|
|
||||||
child: MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
extra: true,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
elevation: 10,
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Dismiss",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
// return const MIHSuccessMessage(
|
|
||||||
// successType: "Success",
|
|
||||||
// successMessage:
|
|
||||||
// "We've sent a password reset link to your email address. Please check your inbox, including spam or junk folders.\n\nOnce you find the email, click on the link to reset your password.\n\nIf you don't receive the email within a few minutes, please try resending the reset request.\n\nThe reset link will expire after 2 hours");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void validateInput() async {
|
|
||||||
if (emailController.text.isEmpty) {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const MIHErrorMessage(errorType: "Input Error");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
await submitPasswodReset();
|
|
||||||
if (successfulForgotPassword) {
|
|
||||||
// Navigator.of(context).pushNamedAndRemoveUntil('/', (route) => false);
|
|
||||||
resetLinkSentSuccessfully();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
double screenWidth = MediaQuery.of(context).size.width;
|
|
||||||
return MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
bodyItem: getBody(screenWidth),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody(double width) {
|
|
||||||
return KeyboardListener(
|
|
||||||
focusNode: _focusNode,
|
|
||||||
autofocus: true,
|
|
||||||
onKeyEvent: (event) async {
|
|
||||||
if (event is KeyDownEvent &&
|
|
||||||
event.logicalKey == LogicalKeyboardKey.enter) {
|
|
||||||
validateInput();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: SafeArea(
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
physics: const BouncingScrollPhysics(),
|
|
||||||
child: Padding(
|
|
||||||
padding: MzansiInnovationHub.of(context)!.theme.screenType ==
|
|
||||||
"desktop"
|
|
||||||
? EdgeInsets.symmetric(vertical: 25, horizontal: width * 0.2)
|
|
||||||
: EdgeInsets.symmetric(vertical: 25, horizontal: width * 0.075),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
//logo
|
|
||||||
Icon(
|
|
||||||
Icons.lock,
|
|
||||||
size: 100,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
//Heading
|
|
||||||
Text(
|
|
||||||
'Forgot Password',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
MihForm(
|
|
||||||
formKey: _formKey,
|
|
||||||
formFields: [
|
|
||||||
MihTextFormField(
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
inputColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
controller: emailController,
|
|
||||||
multiLineInput: false,
|
|
||||||
requiredText: true,
|
|
||||||
hintText: "Email",
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().validateEmail(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
Align(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
if (_formKey.currentState!.validate()) {
|
|
||||||
prePassResteWarning();
|
|
||||||
} else {
|
|
||||||
MihAlertServices().formNotFilledCompletely(context);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Reset Password",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,266 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_form.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_alert.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_text_form_field.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_error_message.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_alert_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_authentication_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
|
|
||||||
class MihResetPassword extends StatefulWidget {
|
|
||||||
final String token;
|
|
||||||
const MihResetPassword({
|
|
||||||
super.key,
|
|
||||||
required this.token,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihResetPassword> createState() => _MihResetPasswordState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihResetPasswordState extends State<MihResetPassword> {
|
|
||||||
final passwordController = TextEditingController();
|
|
||||||
final confirmPasswordController = TextEditingController();
|
|
||||||
final FocusNode _focusNode = FocusNode();
|
|
||||||
final _formKey = GlobalKey<FormState>();
|
|
||||||
|
|
||||||
void submitFormInput() async {
|
|
||||||
if (passwordController.text != confirmPasswordController.text) {
|
|
||||||
passwordError();
|
|
||||||
} else {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const Mihloadingcircle();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
bool successfulResetPassword = await MihAuthenticationServices()
|
|
||||||
.resetPassword(widget.token, passwordController.text);
|
|
||||||
context.pop();
|
|
||||||
if (successfulResetPassword) {
|
|
||||||
resetSuccessfully();
|
|
||||||
} else {
|
|
||||||
loginError();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void resetSuccessfully() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.check_circle_outline_rounded,
|
|
||||||
size: 150,
|
|
||||||
color: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
alertTitle: "Successfully Reset Password",
|
|
||||||
alertBody: Column(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"Great news! Your password reset is complete. You can now log in to Mzansi Innovation Hub using your new password.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
Center(
|
|
||||||
child: MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
extra: true,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
elevation: 10,
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Dismiss",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
// return const MIHSuccessMessage(
|
|
||||||
// successType: "Success",
|
|
||||||
// successMessage:
|
|
||||||
// "We've sent a password reset link to your email address. Please check your inbox, including spam or junk folders.\n\nOnce you find the email, click on the link to reset your password.\n\nIf you don't receive the email within a few minutes, please try resending the reset request.\n\nThe reset link will expire after 2 hours");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void loginError() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const MIHErrorMessage(errorType: "Invalid Credentials");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void passwordError() {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const MIHErrorMessage(errorType: "Password Match");
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
double screenWidth = MediaQuery.of(context).size.width;
|
|
||||||
return MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
bodyItem: getBody(screenWidth),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody(double width) {
|
|
||||||
return KeyboardListener(
|
|
||||||
focusNode: _focusNode,
|
|
||||||
autofocus: true,
|
|
||||||
onKeyEvent: (event) async {
|
|
||||||
if (event is KeyDownEvent &&
|
|
||||||
event.logicalKey == LogicalKeyboardKey.enter) {
|
|
||||||
if (_formKey.currentState!.validate()) {
|
|
||||||
submitFormInput();
|
|
||||||
} else {
|
|
||||||
MihAlertServices().formNotFilledCompletely(context);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: SafeArea(
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
physics: const BouncingScrollPhysics(),
|
|
||||||
child: Padding(
|
|
||||||
padding:
|
|
||||||
MzansiInnovationHub.of(context)!.theme.screenType == "desktop"
|
|
||||||
? EdgeInsets.symmetric(horizontal: width * 0.2)
|
|
||||||
: EdgeInsets.symmetric(horizontal: width * 0.075),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text("Token: ${widget.token}"), // For testing purposes only
|
|
||||||
//logo
|
|
||||||
Icon(
|
|
||||||
Icons.lock,
|
|
||||||
size: 100,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
//Heading
|
|
||||||
Text(
|
|
||||||
'Reset Password',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
MihForm(
|
|
||||||
formKey: _formKey,
|
|
||||||
formFields: [
|
|
||||||
MihTextFormField(
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
inputColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
controller: passwordController,
|
|
||||||
multiLineInput: false,
|
|
||||||
requiredText: true,
|
|
||||||
hintText: "Password",
|
|
||||||
passwordMode: true,
|
|
||||||
autofillHints: const [AutofillHints.password],
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().validatePassword(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
MihTextFormField(
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
inputColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
controller: confirmPasswordController,
|
|
||||||
multiLineInput: false,
|
|
||||||
requiredText: true,
|
|
||||||
hintText: "Confirm Password",
|
|
||||||
passwordMode: true,
|
|
||||||
autofillHints: const [AutofillHints.password],
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().validatePassword(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
//spacer
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
// sign in button
|
|
||||||
Center(
|
|
||||||
child: MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
if (_formKey.currentState!.validate()) {
|
|
||||||
submitFormInput();
|
|
||||||
} else {
|
|
||||||
MihAlertServices().formNotFilledCompletely(context);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Reset Password",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,558 +0,0 @@
|
|||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:ken_logger/ken_logger.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/business.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/business_user.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/user_consent.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_button.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_alert.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_circle_avatar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/app_user.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_window.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_scack_bar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/about_mih_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_env.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mih_home/components/mih_app_drawer.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mih_home/package_tools/mih_business_home.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mih_home/package_tools/mih_personal_home.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_business_details_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_file_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_my_business_user_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_user_consent_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_user_services.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MihHome extends StatefulWidget {
|
|
||||||
const MihHome({
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihHome> createState() => _MihHomeState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihHomeState extends State<MihHome> {
|
|
||||||
DateTime latestPrivacyPolicyDate = DateTime.parse("2024-12-01");
|
|
||||||
DateTime latestTermOfServiceDate = DateTime.parse("2024-12-01");
|
|
||||||
bool _isLoadingInitialData = true;
|
|
||||||
|
|
||||||
Future<void> _loadInitialData() async {
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
_isLoadingInitialData = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
MzansiProfileProvider mzansiProfileProvider =
|
|
||||||
context.read<MzansiProfileProvider>();
|
|
||||||
// Note: getUserData sets user and userProfilePicUrl in the provider
|
|
||||||
if (mzansiProfileProvider.user == null) {
|
|
||||||
await getUserData();
|
|
||||||
}
|
|
||||||
// Note: getUserConsentStatus sets userConsent in the provider
|
|
||||||
if (mzansiProfileProvider.userConsent == null) {
|
|
||||||
await getUserConsentStatus();
|
|
||||||
}
|
|
||||||
// 1. Get Business Data
|
|
||||||
if (mzansiProfileProvider.user != null &&
|
|
||||||
mzansiProfileProvider.user!.type == "business" &&
|
|
||||||
mzansiProfileProvider.business == null) {
|
|
||||||
KenLogger.success(mzansiProfileProvider.business == null
|
|
||||||
? "Business is null, fetching business data..."
|
|
||||||
: "Business data already loaded.");
|
|
||||||
await getBusinessData();
|
|
||||||
}
|
|
||||||
// 2. Set state after all data is loaded
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
_isLoadingInitialData = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> getBusinessData() async {
|
|
||||||
AppUser? user = context.read<MzansiProfileProvider>().user;
|
|
||||||
String logoUrl;
|
|
||||||
String signatureUrl;
|
|
||||||
Business? responseBusiness =
|
|
||||||
await MihBusinessDetailsServices().getBusinessDetailsByUser(context);
|
|
||||||
// if (responseBusiness == null && user!.type == "business") {
|
|
||||||
// if (mounted) {
|
|
||||||
// context.goNamed(
|
|
||||||
// 'businessProfileSetup',
|
|
||||||
// extra: user,
|
|
||||||
// );
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
if (responseBusiness != null && user!.type == "business") {
|
|
||||||
// Get Business
|
|
||||||
// Business Profile Set Up aleary
|
|
||||||
logoUrl = await MihFileApi.getMinioFileUrl(
|
|
||||||
context.read<MzansiProfileProvider>().business!.logo_path,
|
|
||||||
context,
|
|
||||||
);
|
|
||||||
context.read<MzansiProfileProvider>().setBusinessProfilePicUrl(logoUrl);
|
|
||||||
// Get Business User
|
|
||||||
await MihMyBusinessUserServices().getBusinessUser(context);
|
|
||||||
signatureUrl = await MihFileApi.getMinioFileUrl(
|
|
||||||
context.read<MzansiProfileProvider>().businessUser!.sig_path,
|
|
||||||
context,
|
|
||||||
);
|
|
||||||
context
|
|
||||||
.read<MzansiProfileProvider>()
|
|
||||||
.setBusinessUserSignatureUrl(signatureUrl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool showPolicyWindow(UserConsent? userConsent) {
|
|
||||||
if (userConsent == null) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
if (userConsent.privacy_policy_accepted
|
|
||||||
.isAfter(latestPrivacyPolicyDate) &&
|
|
||||||
userConsent.terms_of_services_accepted
|
|
||||||
.isAfter(latestTermOfServiceDate)) {
|
|
||||||
return false;
|
|
||||||
} else {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void createOrUpdateAccpetance(MzansiProfileProvider mzansiProfileProvider) {
|
|
||||||
UserConsent? userConsent = mzansiProfileProvider.userConsent;
|
|
||||||
userConsent != null
|
|
||||||
? MihUserConsentServices()
|
|
||||||
.updateUserConsentStatus(
|
|
||||||
DateTime.now().toIso8601String(),
|
|
||||||
DateTime.now().toIso8601String(),
|
|
||||||
mzansiProfileProvider,
|
|
||||||
context,
|
|
||||||
)
|
|
||||||
.then((value) {
|
|
||||||
if (value == 200) {
|
|
||||||
context.goNamed("mihHome");
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
MihSnackBar(
|
|
||||||
child: Text("Thank you for accepting our Policies"),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
MihSnackBar(
|
|
||||||
child: Text("There was an error, please try again later"),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
: MihUserConsentServices()
|
|
||||||
.insertUserConsentStatus(
|
|
||||||
DateTime.now().toIso8601String(),
|
|
||||||
DateTime.now().toIso8601String(),
|
|
||||||
mzansiProfileProvider,
|
|
||||||
context,
|
|
||||||
)
|
|
||||||
.then((value) {
|
|
||||||
if (value == 201) {
|
|
||||||
context.goNamed("mihHome");
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
MihSnackBar(
|
|
||||||
child: Text("Thank you for accepting our Policies"),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
MihSnackBar(
|
|
||||||
child: Text("There was an error, please try again later"),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> getUserData() async {
|
|
||||||
if (!mounted) return;
|
|
||||||
String url;
|
|
||||||
await MihUserServices().getUserDetails(
|
|
||||||
context,
|
|
||||||
);
|
|
||||||
if (!mounted) return;
|
|
||||||
url = await MihFileApi.getMinioFileUrl(
|
|
||||||
context.read<MzansiProfileProvider>().user!.pro_pic_path,
|
|
||||||
context,
|
|
||||||
);
|
|
||||||
context.read<MzansiProfileProvider>().setUserProfilePicUrl(url);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> getUserConsentStatus() async {
|
|
||||||
if (!mounted) return;
|
|
||||||
await MihUserConsentServices().getUserConsentStatus(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_loadInitialData();
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
List<String> toolTitles = [
|
|
||||||
"Personal",
|
|
||||||
"Business",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Consumer<MzansiProfileProvider>(
|
|
||||||
builder: (BuildContext context,
|
|
||||||
MzansiProfileProvider mzansiProfileProvider, Widget? child) {
|
|
||||||
if (_isLoadingInitialData) {
|
|
||||||
return Scaffold(
|
|
||||||
body: Center(
|
|
||||||
child: Mihloadingcircle(),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
// bool showConsentWindow =
|
|
||||||
// showPolicyWindow(mzansiProfileProvider.userConsent);
|
|
||||||
return Stack(
|
|
||||||
children: [
|
|
||||||
RefreshIndicator(
|
|
||||||
onRefresh: () async {
|
|
||||||
await _loadInitialData();
|
|
||||||
},
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
physics: const AlwaysScrollableScrollPhysics(),
|
|
||||||
child: SizedBox(
|
|
||||||
height: MediaQuery.of(context).size.height,
|
|
||||||
child: MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(mzansiProfileProvider,
|
|
||||||
mzansiProfileProvider.user!.type != "personal"),
|
|
||||||
appBody: getToolBody(mzansiProfileProvider),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
actionDrawer: getActionDrawer(),
|
|
||||||
selectedbodyIndex:
|
|
||||||
mzansiProfileProvider.personalHome ? 0 : 1,
|
|
||||||
onIndexChange: (newValue) {
|
|
||||||
mzansiProfileProvider.setPersonalHome(newValue == 0);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Visibility(
|
|
||||||
visible: showPolicyWindow(mzansiProfileProvider.userConsent),
|
|
||||||
child: Container(
|
|
||||||
color: Colors.black.withValues(alpha: 0.5),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
MihPackageWindow(
|
|
||||||
fullscreen: false,
|
|
||||||
windowTitle: "Privacy Policy & Terms Of Service Alert!",
|
|
||||||
onWindowTapClose: () {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return MihPackageAlert(
|
|
||||||
alertIcon: Icon(
|
|
||||||
Icons.warning_amber_rounded,
|
|
||||||
size: 100,
|
|
||||||
color: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
alertTitle:
|
|
||||||
"Oops, Looks like you missed a step!",
|
|
||||||
alertBody: Text(
|
|
||||||
"We're excited for you to keep using the MIH app! Before you do, please take a moment to accept our Privacy Policy and Terms of Service. Thanks for helping us keep your experience great!",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
alertColour: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
windowBody: Column(
|
|
||||||
children: [
|
|
||||||
Icon(
|
|
||||||
Icons.policy,
|
|
||||||
size: 150,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Welcome to the MIH App",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
fontSize: 30,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"To keep using the MIH app, please take a moment to review and accept our Policies. Our agreements helps us keep things running smoothly and securely.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark",
|
|
||||||
),
|
|
||||||
fontSize: 15,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
Center(
|
|
||||||
child: Wrap(
|
|
||||||
alignment: WrapAlignment.center,
|
|
||||||
spacing: 10,
|
|
||||||
runSpacing: 10,
|
|
||||||
children: [
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
WidgetsBinding.instance
|
|
||||||
.addPostFrameCallback((_) async {
|
|
||||||
context
|
|
||||||
.read<AboutMihProvider>()
|
|
||||||
.setToolIndex(1);
|
|
||||||
});
|
|
||||||
context.goNamed("aboutMih",
|
|
||||||
extra:
|
|
||||||
mzansiProfileProvider.personalHome);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getOrangeColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
elevation: 10,
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Privacy Policy",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
WidgetsBinding.instance
|
|
||||||
.addPostFrameCallback((_) async {
|
|
||||||
context
|
|
||||||
.read<AboutMihProvider>()
|
|
||||||
.setToolIndex(2);
|
|
||||||
});
|
|
||||||
context.goNamed("aboutMih",
|
|
||||||
extra:
|
|
||||||
mzansiProfileProvider.personalHome);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getYellowColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
elevation: 10,
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Terms of Service",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
MihButton(
|
|
||||||
onPressed: () {
|
|
||||||
DateTime now = DateTime.now();
|
|
||||||
KenLogger.success("Date Time Now: $now");
|
|
||||||
createOrUpdateAccpetance(
|
|
||||||
mzansiProfileProvider);
|
|
||||||
},
|
|
||||||
buttonColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
elevation: 10,
|
|
||||||
width: 300,
|
|
||||||
child: Text(
|
|
||||||
"Accept",
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getAction() {
|
|
||||||
return Builder(builder: (context) {
|
|
||||||
return Consumer<MzansiProfileProvider>(
|
|
||||||
builder: (BuildContext context,
|
|
||||||
MzansiProfileProvider mzansiProfileProvider, Widget? child) {
|
|
||||||
ImageProvider<Object>? currentImage;
|
|
||||||
String imageKey;
|
|
||||||
if (mzansiProfileProvider.personalHome) {
|
|
||||||
currentImage = mzansiProfileProvider.userProfilePicture;
|
|
||||||
imageKey = 'user_${mzansiProfileProvider.userProfilePicUrl}';
|
|
||||||
} else {
|
|
||||||
currentImage = mzansiProfileProvider.businessProfilePicture;
|
|
||||||
imageKey =
|
|
||||||
'business_${mzansiProfileProvider.businessProfilePicUrl}';
|
|
||||||
}
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: Padding(
|
|
||||||
padding: const EdgeInsets.only(left: 5.0),
|
|
||||||
child: MihCircleAvatar(
|
|
||||||
key: Key(imageKey),
|
|
||||||
imageFile: currentImage,
|
|
||||||
width: 50,
|
|
||||||
editable: false,
|
|
||||||
fileNameController: null,
|
|
||||||
userSelectedfile: null,
|
|
||||||
// frameColor: frameColor,
|
|
||||||
frameColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onChange: (_) {},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
iconSize: 45,
|
|
||||||
onTap: () {
|
|
||||||
Scaffold.of(context).openDrawer();
|
|
||||||
FocusScope.of(context)
|
|
||||||
.requestFocus(FocusNode()); // Fully unfocus all fields
|
|
||||||
// FocusScope.of(context).unfocus(); // Unfocus any text fields
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
MIHAppDrawer getActionDrawer() {
|
|
||||||
return MIHAppDrawer();
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools(
|
|
||||||
MzansiProfileProvider mzansiProfileProvider, bool isBusinessUser) {
|
|
||||||
Map<Widget, void Function()?> temp = {};
|
|
||||||
temp[const Icon(Icons.person)] = () {
|
|
||||||
setState(() {
|
|
||||||
mzansiProfileProvider.setPersonalHome(true);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
if (isBusinessUser) {
|
|
||||||
temp[const Icon(Icons.business_center)] = () {
|
|
||||||
setState(() {
|
|
||||||
mzansiProfileProvider.setPersonalHome(false);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: mzansiProfileProvider.personalHome ? 0 : 1,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody(MzansiProfileProvider mzansiProfileProvider) {
|
|
||||||
List<Widget> toolBodies = [];
|
|
||||||
AppUser? user = mzansiProfileProvider.user;
|
|
||||||
Business? business = mzansiProfileProvider.business;
|
|
||||||
BusinessUser? businessUser = mzansiProfileProvider.businessUser;
|
|
||||||
String userProfilePictureUrl =
|
|
||||||
mzansiProfileProvider.userProfilePicUrl ?? "";
|
|
||||||
toolBodies.add(
|
|
||||||
MihPersonalHome(
|
|
||||||
signedInUser: user!,
|
|
||||||
personalSelected: mzansiProfileProvider.personalHome,
|
|
||||||
business: business,
|
|
||||||
businessUser: businessUser,
|
|
||||||
propicFile: userProfilePictureUrl != ""
|
|
||||||
? NetworkImage(userProfilePictureUrl)
|
|
||||||
: null,
|
|
||||||
isDevActive: AppEnviroment.getEnv() == "Dev",
|
|
||||||
isUserNew: user.username == "",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
if (user.type != "personal") {
|
|
||||||
toolBodies.add(
|
|
||||||
MihBusinessHome(
|
|
||||||
isLoading: _isLoadingInitialData,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_circle_avatar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_mine_sweeper_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class BuildMinesweeperLeaderboardList extends StatefulWidget {
|
|
||||||
const BuildMinesweeperLeaderboardList({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<BuildMinesweeperLeaderboardList> createState() =>
|
|
||||||
_BuildMinesweeperLeaderboardListState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _BuildMinesweeperLeaderboardListState
|
|
||||||
extends State<BuildMinesweeperLeaderboardList> {
|
|
||||||
Color getMedalColor(int index) {
|
|
||||||
switch (index) {
|
|
||||||
case (0):
|
|
||||||
return MihColors.getGoldColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark");
|
|
||||||
case (1):
|
|
||||||
return MihColors.getSilverColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark");
|
|
||||||
case (2):
|
|
||||||
return MihColors.getBronze(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark");
|
|
||||||
default:
|
|
||||||
return MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final double width = MediaQuery.sizeOf(context).width;
|
|
||||||
return Consumer2<MzansiProfileProvider, MihMineSweeperProvider>(
|
|
||||||
builder: (BuildContext context, MzansiProfileProvider profileProvider,
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, Widget? child) {
|
|
||||||
return ListView.separated(
|
|
||||||
shrinkWrap: true,
|
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
separatorBuilder: (BuildContext context, index) {
|
|
||||||
return Divider(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
itemCount: mineSweeperProvider.leaderboard!.length,
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return Padding(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: width / 20),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"#${index + 1}",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
color: getMedalColor(index),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
MihCircleAvatar(
|
|
||||||
key: UniqueKey(),
|
|
||||||
imageFile:
|
|
||||||
mineSweeperProvider.leaderboardUserPictures.isNotEmpty
|
|
||||||
? mineSweeperProvider.leaderboardUserPictures[index]
|
|
||||||
: null,
|
|
||||||
width: 80,
|
|
||||||
editable: false,
|
|
||||||
fileNameController: null,
|
|
||||||
userSelectedfile: null,
|
|
||||||
frameColor: getMedalColor(index),
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onChange: () {},
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"${mineSweeperProvider.leaderboard![index].username}${profileProvider.user!.username == mineSweeperProvider.leaderboard![index].username ? " (You)" : ""}",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: getMedalColor(index),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
"Score: ${mineSweeperProvider.leaderboard![index].game_score}\nTime: ${mineSweeperProvider.leaderboard![index].game_time}",
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
color: getMedalColor(index),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,101 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
||||||
import 'package:go_router/go_router.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_tools.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_banner_ad_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_mine_sweeper_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/package_tools/mih_mine_sweeper_leader_board.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/package_tools/mine_sweeper_game.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/package_tools/mine_sweeper_quick_start_guide.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/package_tools/my_score_board.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MihMineSweeper extends StatefulWidget {
|
|
||||||
const MihMineSweeper({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihMineSweeper> createState() => _MihMineSweeperState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihMineSweeperState extends State<MihMineSweeper> {
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) async {
|
|
||||||
context.read<MihBannerAdProvider>().loadBannerAd();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
appBody: getToolBody(),
|
|
||||||
selectedbodyIndex: context.watch<MihMineSweeperProvider>().toolIndex,
|
|
||||||
onIndexChange: (newIndex) {
|
|
||||||
context.read<MihMineSweeperProvider>().setToolIndex(newIndex);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageAction getAction() {
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
iconSize: 35,
|
|
||||||
onTap: () {
|
|
||||||
MihMineSweeperProvider mineSweeperProvider =
|
|
||||||
context.read<MihMineSweeperProvider>();
|
|
||||||
mineSweeperProvider.setToolIndex(0);
|
|
||||||
mineSweeperProvider.setDifficulty("Easy");
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
);
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools() {
|
|
||||||
Map<Widget, void Function()?> temp = {};
|
|
||||||
temp[const Icon(FontAwesomeIcons.bomb)] = () {
|
|
||||||
context.read<MihMineSweeperProvider>().setToolIndex(0);
|
|
||||||
};
|
|
||||||
temp[const Icon(Icons.leaderboard_rounded)] = () {
|
|
||||||
context.read<MihMineSweeperProvider>().setToolIndex(1);
|
|
||||||
};
|
|
||||||
temp[const Icon(Icons.perm_identity_rounded)] = () {
|
|
||||||
context.read<MihMineSweeperProvider>().setToolIndex(2);
|
|
||||||
};
|
|
||||||
temp[const Icon(Icons.rule_rounded)] = () {
|
|
||||||
context.read<MihMineSweeperProvider>().setToolIndex(3);
|
|
||||||
};
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: context.watch<MihMineSweeperProvider>().toolIndex,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
List<String> toolTitles = [
|
|
||||||
"Minesweeper",
|
|
||||||
"Leader Board",
|
|
||||||
"My Scores",
|
|
||||||
"Guide",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody() {
|
|
||||||
List<Widget> toolBodies = [
|
|
||||||
const MineSweeperGame(),
|
|
||||||
const MihMineSweeperLeaderBoard(),
|
|
||||||
const MyScoreBoard(),
|
|
||||||
const MineSweeperQuickStartGuide(),
|
|
||||||
];
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.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_package_tile.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
|
|
||||||
class MihMineSweeperTile extends StatefulWidget {
|
|
||||||
final bool personalSelected;
|
|
||||||
final double packageSize;
|
|
||||||
const MihMineSweeperTile({
|
|
||||||
super.key,
|
|
||||||
required this.personalSelected,
|
|
||||||
required this.packageSize,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihMineSweeperTile> createState() => _MihMineSweeperTileState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihMineSweeperTileState extends State<MihMineSweeperTile> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageTile(
|
|
||||||
onTap: () {
|
|
||||||
context.goNamed(
|
|
||||||
"mihMinesweeper",
|
|
||||||
);
|
|
||||||
},
|
|
||||||
appName: "Minesweeper",
|
|
||||||
appIcon: Icon(
|
|
||||||
MihIcons.mineSweeper,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
// size: widget.packageSize,
|
|
||||||
),
|
|
||||||
iconSize: widget.packageSize,
|
|
||||||
primaryColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
secondaryColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,198 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
||||||
import 'package:ken_logger/ken_logger.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_dropdwn_field.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_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_mine_sweeper_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/builders/build_minesweeper_leaderboard_list.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_file_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_minesweeper_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MihMineSweeperLeaderBoard extends StatefulWidget {
|
|
||||||
const MihMineSweeperLeaderBoard({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MihMineSweeperLeaderBoard> createState() =>
|
|
||||||
_MihMineSweeperLeaderBoardState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihMineSweeperLeaderBoardState extends State<MihMineSweeperLeaderBoard> {
|
|
||||||
TextEditingController filterController = TextEditingController();
|
|
||||||
|
|
||||||
Future<void> initialiseLeaderboard() async {
|
|
||||||
MihMineSweeperProvider mineSweeperProvider =
|
|
||||||
context.read<MihMineSweeperProvider>();
|
|
||||||
filterController.text = mineSweeperProvider.difficulty;
|
|
||||||
KenLogger.success("getting data");
|
|
||||||
await MihMinesweeperServices().getTop20Leaderboard(mineSweeperProvider);
|
|
||||||
List<ImageProvider<Object>?> userPictures = [];
|
|
||||||
String userPicUrl = "";
|
|
||||||
for (final ranking in mineSweeperProvider.leaderboard!) {
|
|
||||||
userPicUrl = await MihFileApi.getMinioFileUrl(ranking.proPicUrl, context);
|
|
||||||
userPictures.add(NetworkImage(userPicUrl));
|
|
||||||
}
|
|
||||||
mineSweeperProvider.setLeaderboardUserPictures(
|
|
||||||
leaderboardUserPictures: userPictures);
|
|
||||||
}
|
|
||||||
|
|
||||||
void refreshLeaderBoard(
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, String difficulty) {
|
|
||||||
mineSweeperProvider.setDifficulty(difficulty);
|
|
||||||
mineSweeperProvider.setLeaderboard(leaderboard: null);
|
|
||||||
mineSweeperProvider.setMyScoreboard(myScoreboard: null);
|
|
||||||
initialiseLeaderboard();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) async {
|
|
||||||
await initialiseLeaderboard();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final double width = MediaQuery.sizeOf(context).width;
|
|
||||||
return Consumer<MihMineSweeperProvider>(
|
|
||||||
builder: (BuildContext context,
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, Widget? child) {
|
|
||||||
return RefreshIndicator(
|
|
||||||
onRefresh: () async {
|
|
||||||
refreshLeaderBoard(mineSweeperProvider, filterController.text);
|
|
||||||
},
|
|
||||||
child: MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
bodyItem: getBody(width),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody(double width) {
|
|
||||||
return Consumer<MihMineSweeperProvider>(
|
|
||||||
builder: (BuildContext context,
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, Widget? child) {
|
|
||||||
if (mineSweeperProvider.leaderboard == null) {
|
|
||||||
return Center(
|
|
||||||
child: Mihloadingcircle(),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return SingleChildScrollView(
|
|
||||||
physics: const AlwaysScrollableScrollPhysics(),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: width / 20),
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
Flexible(
|
|
||||||
child: MihDropdownField(
|
|
||||||
controller: filterController,
|
|
||||||
hintText: "Leaderboards",
|
|
||||||
dropdownOptions: const [
|
|
||||||
"Very Easy",
|
|
||||||
"Easy",
|
|
||||||
"Intermediate",
|
|
||||||
"Hard",
|
|
||||||
],
|
|
||||||
requiredText: true,
|
|
||||||
editable: true,
|
|
||||||
enableSearch: false,
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().isEmpty(value);
|
|
||||||
},
|
|
||||||
onSelected: (selection) {
|
|
||||||
refreshLeaderBoard(mineSweeperProvider, selection!);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
mineSweeperProvider.leaderboard!.isEmpty
|
|
||||||
? Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
const SizedBox(height: 50),
|
|
||||||
Icon(
|
|
||||||
MihIcons.mineSweeper,
|
|
||||||
size: 165,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Be the first on the leaderboard.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
overflow: TextOverflow.visible,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
Center(
|
|
||||||
child: RichText(
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
children: [
|
|
||||||
TextSpan(text: "Press "),
|
|
||||||
WidgetSpan(
|
|
||||||
alignment: PlaceholderAlignment.middle,
|
|
||||||
child: Icon(
|
|
||||||
FontAwesomeIcons.bomb,
|
|
||||||
size: 20,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TextSpan(text: " and start a new game"),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: BuildMinesweeperLeaderboardList(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,209 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
||||||
import 'package:ken_logger/ken_logger.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_circle_avatar.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_dropdwn_field.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_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mih_mine_sweeper_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mine_sweeper/builders/build_my_scoreboard_list.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_minesweeper_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MyScoreBoard extends StatefulWidget {
|
|
||||||
const MyScoreBoard({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MyScoreBoard> createState() => _MihMineSweeperLeaderBoardState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MihMineSweeperLeaderBoardState extends State<MyScoreBoard> {
|
|
||||||
TextEditingController filterController = TextEditingController();
|
|
||||||
|
|
||||||
Future<void> initialiseLeaderboard() async {
|
|
||||||
MzansiProfileProvider profileProvider =
|
|
||||||
context.read<MzansiProfileProvider>();
|
|
||||||
MihMineSweeperProvider mineSweeperProvider =
|
|
||||||
context.read<MihMineSweeperProvider>();
|
|
||||||
filterController.text = mineSweeperProvider.difficulty;
|
|
||||||
KenLogger.success("getting data");
|
|
||||||
await MihMinesweeperServices()
|
|
||||||
.getMyScoreboard(profileProvider, mineSweeperProvider);
|
|
||||||
KenLogger.success("${mineSweeperProvider.myScoreboard}");
|
|
||||||
}
|
|
||||||
|
|
||||||
void refreshLeaderBoard(
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, String difficulty) {
|
|
||||||
mineSweeperProvider.setDifficulty(difficulty);
|
|
||||||
mineSweeperProvider.setLeaderboard(leaderboard: null);
|
|
||||||
mineSweeperProvider.setMyScoreboard(myScoreboard: null);
|
|
||||||
initialiseLeaderboard();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) async {
|
|
||||||
await initialiseLeaderboard();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final double width = MediaQuery.sizeOf(context).width;
|
|
||||||
return Consumer<MihMineSweeperProvider>(
|
|
||||||
builder: (BuildContext context,
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, Widget? child) {
|
|
||||||
return RefreshIndicator(
|
|
||||||
onRefresh: () async {
|
|
||||||
refreshLeaderBoard(mineSweeperProvider, filterController.text);
|
|
||||||
},
|
|
||||||
child: MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
bodyItem: getBody(width),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody(double width) {
|
|
||||||
return Consumer2<MzansiProfileProvider, MihMineSweeperProvider>(
|
|
||||||
builder: (BuildContext context, MzansiProfileProvider profileProvider,
|
|
||||||
MihMineSweeperProvider mineSweeperProvider, Widget? child) {
|
|
||||||
if (mineSweeperProvider.myScoreboard == null) {
|
|
||||||
return Center(
|
|
||||||
child: Mihloadingcircle(),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return SingleChildScrollView(
|
|
||||||
physics: const AlwaysScrollableScrollPhysics(),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Center(
|
|
||||||
child: MihCircleAvatar(
|
|
||||||
imageFile: profileProvider.userProfilePicture,
|
|
||||||
width: 150,
|
|
||||||
editable: false,
|
|
||||||
fileNameController: null,
|
|
||||||
userSelectedfile: null,
|
|
||||||
frameColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onChange: (selectedImage) {},
|
|
||||||
key: ValueKey(profileProvider.userProfilePicUrl),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: width / 20),
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
Flexible(
|
|
||||||
child: MihDropdownField(
|
|
||||||
controller: filterController,
|
|
||||||
hintText: "Scoreboards",
|
|
||||||
dropdownOptions: const [
|
|
||||||
"Very Easy",
|
|
||||||
"Easy",
|
|
||||||
"Intermediate",
|
|
||||||
"Hard",
|
|
||||||
],
|
|
||||||
requiredText: true,
|
|
||||||
editable: true,
|
|
||||||
enableSearch: false,
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().isEmpty(value);
|
|
||||||
},
|
|
||||||
onSelected: (selection) {
|
|
||||||
refreshLeaderBoard(mineSweeperProvider, selection!);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
mineSweeperProvider.myScoreboard!.isEmpty
|
|
||||||
? Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
const SizedBox(height: 50),
|
|
||||||
Icon(
|
|
||||||
MihIcons.mineSweeper,
|
|
||||||
size: 165,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"You have played and ${mineSweeperProvider.difficulty} yet.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
overflow: TextOverflow.visible,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
Center(
|
|
||||||
child: RichText(
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
children: [
|
|
||||||
TextSpan(text: "Press "),
|
|
||||||
WidgetSpan(
|
|
||||||
alignment: PlaceholderAlignment.middle,
|
|
||||||
child: Icon(
|
|
||||||
FontAwesomeIcons.bomb,
|
|
||||||
size: 20,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TextSpan(text: " and start a new game"),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: BuildMyScoreBoardList(),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,78 +0,0 @@
|
|||||||
import 'package:go_router/go_router.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_tools.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_ai_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_packages/mzansi_ai/package_tools/ai_chat.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class MzansiAi extends StatefulWidget {
|
|
||||||
const MzansiAi({
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<MzansiAi> createState() => _MzansiAiState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _MzansiAiState extends State<MzansiAi> {
|
|
||||||
MihPackageAction getAction() {
|
|
||||||
return MihPackageAction(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
iconSize: 35,
|
|
||||||
onTap: () {
|
|
||||||
context.read<MzansiAiProvider>().setStartUpQuestion(null);
|
|
||||||
context.goNamed(
|
|
||||||
'mihHome',
|
|
||||||
);
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
MihPackageTools getTools() {
|
|
||||||
Map<Widget, void Function()?> temp = {};
|
|
||||||
temp[const Icon(Icons.chat)] = () {
|
|
||||||
context.read<MzansiAiProvider>().setToolIndex(0);
|
|
||||||
};
|
|
||||||
|
|
||||||
return MihPackageTools(
|
|
||||||
tools: temp,
|
|
||||||
selcetedIndex: context.watch<MzansiAiProvider>().toolIndex,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Widget> getToolBody() {
|
|
||||||
List<Widget> toolBodies = [
|
|
||||||
AiChat(),
|
|
||||||
];
|
|
||||||
return toolBodies;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> getToolTitle() {
|
|
||||||
List<String> toolTitles = [
|
|
||||||
"Ask Mzansi",
|
|
||||||
];
|
|
||||||
return toolTitles;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackage(
|
|
||||||
appActionButton: getAction(),
|
|
||||||
appTools: getTools(),
|
|
||||||
appBody: getToolBody(),
|
|
||||||
appToolTitles: getToolTitle(),
|
|
||||||
selectedbodyIndex: context.watch<MzansiAiProvider>().toolIndex,
|
|
||||||
onIndexChange: (newValue) {
|
|
||||||
context.read<MzansiAiProvider>().setToolIndex(newValue);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,887 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
import 'dart:convert';
|
|
||||||
import 'package:flutter_speed_dial/flutter_speed_dial.dart';
|
|
||||||
import 'package:gpt_markdown/gpt_markdown.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_icons.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_ai_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_profile_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_services/mih_validation_services.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_dropdwn_field.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_numeric_stepper.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_tool_body.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_package_window.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_floating_menu.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_radio_options.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_env.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_chat_ui/flutter_chat_ui.dart';
|
|
||||||
import 'package:flutter_chat_types/flutter_chat_types.dart' as types;
|
|
||||||
import 'package:flutter/services.dart' show rootBundle;
|
|
||||||
import 'package:flutter_tts/flutter_tts.dart';
|
|
||||||
import 'package:ollama_dart/ollama_dart.dart' as ollama;
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
import 'package:uuid/uuid.dart';
|
|
||||||
|
|
||||||
class AiChat extends StatefulWidget {
|
|
||||||
const AiChat({
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<AiChat> createState() => _AiChatState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _AiChatState extends State<AiChat> {
|
|
||||||
final TextEditingController _modelController = TextEditingController();
|
|
||||||
final TextEditingController _fontSizeController = TextEditingController();
|
|
||||||
final TextEditingController _ttsVoiceController = TextEditingController();
|
|
||||||
final ValueNotifier<bool> _showModelOptions = ValueNotifier(false);
|
|
||||||
FlutterTts _flutterTts = FlutterTts();
|
|
||||||
final ValueNotifier<String> _ttsVoiceName = ValueNotifier("");
|
|
||||||
// bool _ttsOn = false;
|
|
||||||
String? textStream;
|
|
||||||
List<Map> _voices = [];
|
|
||||||
List<String> _voicesString = [];
|
|
||||||
List<types.Message> _messages = [];
|
|
||||||
late types.User _user;
|
|
||||||
late types.User _mihAI;
|
|
||||||
String systemPromt = "0";
|
|
||||||
bool _aiThinking = false;
|
|
||||||
final client = ollama.OllamaClient(
|
|
||||||
baseUrl: "${AppEnviroment.baseAiUrl}/api",
|
|
||||||
);
|
|
||||||
List<ollama.Message> _chatHistory = [];
|
|
||||||
double _chatFrontSize = 15;
|
|
||||||
|
|
||||||
String getModel() {
|
|
||||||
return AppEnviroment.getEnv() == "Prod" ? 'gemma3n:e4b' : "gemma3:1b";
|
|
||||||
}
|
|
||||||
|
|
||||||
String setSystemPromt() {
|
|
||||||
String temp = "";
|
|
||||||
temp +=
|
|
||||||
"You are Mzansi AI, a helpful and friendly AI assistant running on the 'MIH App'.\n";
|
|
||||||
temp +=
|
|
||||||
"The MIH App was created by 'Mzansi Innovation Hub', a South African-based startup company.";
|
|
||||||
temp +=
|
|
||||||
"Your primary purpose is to assist users by answering general questions and helping with creative writing tasks or any other task a user might have for you.\n";
|
|
||||||
temp +=
|
|
||||||
"Maintain a casual and friendly tone, but always remain professional.\n";
|
|
||||||
temp +=
|
|
||||||
"Strive for a balance between being empathetic and delivering factual information accurately.\n";
|
|
||||||
temp +=
|
|
||||||
"You may use lighthearted or playful language if the context is appropriate and enhances the user experience.\n";
|
|
||||||
temp += "You operate within the knowledge domain of the 'MIH App'.\n";
|
|
||||||
temp += "Here is a description of the MIH App and its features:\n";
|
|
||||||
temp +=
|
|
||||||
"MIH App Description: MIH is the first super app of Mzansi, designed to streamline both personal and business life. It's an all-in-one platform for managing professional profiles, teams, appointments, and quick calculations. \n";
|
|
||||||
temp += "Key Features:\n";
|
|
||||||
temp +=
|
|
||||||
"- Mzansi Profile: Central hub for managing personal and business information, including business team details.";
|
|
||||||
temp += "- Mzansi Wallet: Digitally store loyalty cards.\n";
|
|
||||||
temp +=
|
|
||||||
"- Patient Manager (For Medical Practices): Seamless patient appointment scheduling and data management.\n";
|
|
||||||
temp +=
|
|
||||||
"- Mzansi AI: Your friendly AI assistant for quick answers and support (that's you!).\n";
|
|
||||||
temp +=
|
|
||||||
"- Mzansi Directory: A place to search and find out more about the people and businesses across Mzansi.\n";
|
|
||||||
temp +=
|
|
||||||
"- Calendar: Integrated calendar for managing personal and business appointments.\n";
|
|
||||||
temp +=
|
|
||||||
"- Calculator: Simple calculator with tip and forex calculation functionality.\n";
|
|
||||||
temp += "- MIH Access: Manage and view profile access security.\n";
|
|
||||||
temp +=
|
|
||||||
"- MIH Minesweeper: The first game from MIH! It's the classic brain-teaser ready to entertain you no matter where you are.\n";
|
|
||||||
temp += "**Core Rules and Guidelines:**\n";
|
|
||||||
temp +=
|
|
||||||
"- **Accuracy First:** Always prioritize providing correct information.\n";
|
|
||||||
temp +=
|
|
||||||
"- **Uncertainty Handling:** If you are unsure about an answer, politely respond with: 'Please bear with us as we are still learning and do not have all the answers.'\n";
|
|
||||||
temp +=
|
|
||||||
"- **Response Length:** Aim to keep responses under 250 words. If a more comprehensive answer is required, exceed this limit but offer to elaborate further (e.g., 'Would you like me to elaborate on this topic?').\n";
|
|
||||||
temp +=
|
|
||||||
"- **Language & Safety:** Never use offensive language or generate harmful content. If a user presses for information that is inappropriate or out of bounds, clearly state why you cannot provide it (e.g., 'I cannot assist with that request as it goes against my safety guidelines.').\n";
|
|
||||||
temp +=
|
|
||||||
"- **Out-of-Scope Questions:** - If a question is unclear, ask the user to rephrase or clarify it. - If a question is entirely out of your scope and you cannot provide a useful answer, admit you don't know. - If a user is unhappy with your response or needs further assistance beyond your capabilities, suggest they visit the 'Mzansi Innovation Hub Social Media Pages' for more direct support. Do not provide specific links, just refer to the pages generally.\n";
|
|
||||||
temp +=
|
|
||||||
"- **Target Audience:** Adapt your explanations to beginners and intermediate users, but be prepared for more complex questions from expert users. Ensure your language is clear and easy to understand.\n";
|
|
||||||
return temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
void _addMessage(types.Message message) {
|
|
||||||
setState(() {
|
|
||||||
_messages.insert(0, message);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void _loadMessages() async {
|
|
||||||
final response = await rootBundle.loadString('assets/messages.json');
|
|
||||||
final messages = (jsonDecode(response) as List)
|
|
||||||
.map((e) => types.Message.fromJson(e as Map<String, dynamic>))
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
setState(() {
|
|
||||||
_messages = messages;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void _handleSendPressed(types.PartialText message) {
|
|
||||||
FocusScope.of(context).unfocus();
|
|
||||||
final textMessage = types.TextMessage(
|
|
||||||
author: _user,
|
|
||||||
createdAt: DateTime.now().millisecondsSinceEpoch,
|
|
||||||
id: const Uuid().v4(),
|
|
||||||
text: message.text,
|
|
||||||
);
|
|
||||||
//Add user prompt to history
|
|
||||||
setState(() {
|
|
||||||
_chatHistory.add(
|
|
||||||
ollama.Message(
|
|
||||||
role: ollama.MessageRole.user,
|
|
||||||
content: message.text,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
_addMessage(textMessage);
|
|
||||||
|
|
||||||
_handleMessageBack(message.text);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _handleMessageBack(String userMessage) async {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const Mihloadingcircle();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
Stream<String> aiChatStream =
|
|
||||||
_generateChatCompletionWithHistoryStream(userMessage, client);
|
|
||||||
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
barrierDismissible: false,
|
|
||||||
builder: (context) {
|
|
||||||
return responseWindow(aiChatStream);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget responseWindow(
|
|
||||||
Stream<String> aiChatStream,
|
|
||||||
) {
|
|
||||||
return StreamBuilder(
|
|
||||||
stream: aiChatStream,
|
|
||||||
builder: (context, snapshot) {
|
|
||||||
if (snapshot.hasData) {
|
|
||||||
textStream = snapshot.requireData;
|
|
||||||
// print("Text: $textStream");
|
|
||||||
// _speakText(textStream!);
|
|
||||||
return MihPackageWindow(
|
|
||||||
fullscreen: false,
|
|
||||||
windowTitle: 'Mzansi AI Thoughts',
|
|
||||||
menuOptions: _aiThinking == true
|
|
||||||
? null
|
|
||||||
: [
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.volume_up,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
label: "Read Aloud",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
onTap: () {
|
|
||||||
_speakText(snapshot.requireData);
|
|
||||||
},
|
|
||||||
)
|
|
||||||
],
|
|
||||||
onWindowTapClose: () {
|
|
||||||
_captureAIResponse(snapshot.requireData);
|
|
||||||
_flutterTts.stop();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
windowBody: SizedBox(
|
|
||||||
width: double.infinity,
|
|
||||||
// color: Colors.black,
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
SelectionArea(
|
|
||||||
child: GptMarkdown(
|
|
||||||
snapshot.requireData,
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
fontSize: _chatFrontSize,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return MihPackageWindow(
|
|
||||||
fullscreen: false,
|
|
||||||
windowTitle: 'Mzansi AI Thoughts',
|
|
||||||
// windowTools: [],
|
|
||||||
onWindowTapClose: () {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
windowBody: Mihloadingcircle(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _captureAIResponse(String responseMessage) {
|
|
||||||
types.TextMessage textMessage;
|
|
||||||
setState(() {
|
|
||||||
_chatHistory.add(
|
|
||||||
ollama.Message(
|
|
||||||
role: ollama.MessageRole.assistant,
|
|
||||||
content: responseMessage,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
textMessage = types.TextMessage(
|
|
||||||
author: _mihAI,
|
|
||||||
createdAt: DateTime.now().millisecondsSinceEpoch,
|
|
||||||
id: const Uuid().v4(),
|
|
||||||
|
|
||||||
text: responseMessage
|
|
||||||
.replaceAll("<think>\n\n", "**Thinking:**\n")
|
|
||||||
.replaceAll("<think>\n", "**Thinking:**\n")
|
|
||||||
.replaceAll("</think>\n\n", "\n**Answer:**\n"), //message.text,
|
|
||||||
);
|
|
||||||
|
|
||||||
_addMessage(textMessage);
|
|
||||||
}
|
|
||||||
|
|
||||||
Stream<String> _generateChatCompletionWithHistoryStream(
|
|
||||||
String userMessage,
|
|
||||||
final ollama.OllamaClient client,
|
|
||||||
) async* {
|
|
||||||
final aiStream = client.generateChatCompletionStream(
|
|
||||||
request: ollama.GenerateChatCompletionRequest(
|
|
||||||
model: _modelController.text,
|
|
||||||
messages: _chatHistory,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
String text = '';
|
|
||||||
setState(() {
|
|
||||||
_aiThinking = true;
|
|
||||||
});
|
|
||||||
await for (final res in aiStream) {
|
|
||||||
text += (res.message.content);
|
|
||||||
yield text;
|
|
||||||
}
|
|
||||||
setState(() {
|
|
||||||
_aiThinking = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void _resetChat() {
|
|
||||||
setState(() {
|
|
||||||
_messages = [];
|
|
||||||
_chatHistory = [];
|
|
||||||
_loadMessages();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
ChatTheme getChatTheme() {
|
|
||||||
return DarkChatTheme(
|
|
||||||
backgroundColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
inputBackgroundColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
inputTextColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
inputTextCursorColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
primaryColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
secondaryColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
errorColor: MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
sentMessageBodyTextStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: _chatFrontSize,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
fontFamily: 'Segoe UI',
|
|
||||||
),
|
|
||||||
receivedMessageBodyTextStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: _chatFrontSize,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
fontFamily: 'Segoe UI',
|
|
||||||
),
|
|
||||||
emptyChatPlaceholderTextStyle: TextStyle(
|
|
||||||
color: MihColors.getGreyColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontSize: _chatFrontSize,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
fontFamily: 'Segoe UI',
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _getSettings() {
|
|
||||||
return ValueListenableBuilder(
|
|
||||||
valueListenable: _showModelOptions,
|
|
||||||
builder: (BuildContext context, bool value, Widget? child) {
|
|
||||||
return Visibility(
|
|
||||||
visible: value,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(10.0),
|
|
||||||
child: Align(
|
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
child: FittedBox(
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.all(10.0),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
borderRadius: BorderRadius.circular(25.0),
|
|
||||||
border: Border.all(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
width: 3.0),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"Settings",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 300,
|
|
||||||
child: MihRadioOptions(
|
|
||||||
controller: _modelController,
|
|
||||||
hintText: "AI Model",
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
secondaryFillColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
requiredText: true,
|
|
||||||
radioOptions: [getModel()],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 230,
|
|
||||||
child: MihDropdownField(
|
|
||||||
controller: _ttsVoiceController,
|
|
||||||
hintText: "AI Voice",
|
|
||||||
dropdownOptions: _voicesString,
|
|
||||||
editable: true,
|
|
||||||
enableSearch: true,
|
|
||||||
requiredText: true,
|
|
||||||
validator: (value) {
|
|
||||||
return MihValidationServices().isEmpty(value);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
25), // Optional: rounds the corners
|
|
||||||
boxShadow: const [
|
|
||||||
BoxShadow(
|
|
||||||
color: Color.fromARGB(60, 0, 0,
|
|
||||||
0), // 0.2 opacity = 51 in alpha (255 * 0.2)
|
|
||||||
spreadRadius: -2,
|
|
||||||
blurRadius: 10,
|
|
||||||
offset: Offset(0, 5),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
left: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 50,
|
|
||||||
height: 50,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor:
|
|
||||||
WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark")),
|
|
||||||
),
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!
|
|
||||||
.theme
|
|
||||||
.mode ==
|
|
||||||
"Dark"),
|
|
||||||
iconSize: 25,
|
|
||||||
onPressed: () {
|
|
||||||
print("Start TTS now");
|
|
||||||
_speakText(
|
|
||||||
"This is the sample of the Mzansi A.I Voice.");
|
|
||||||
},
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.volume_up,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 300,
|
|
||||||
child: MihNumericStepper(
|
|
||||||
controller: _fontSizeController,
|
|
||||||
fillColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
inputColor: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark"),
|
|
||||||
hintText: "Font Size",
|
|
||||||
requiredText: true,
|
|
||||||
minValue: 1,
|
|
||||||
// maxValue: 5,
|
|
||||||
validationOn: true,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const SizedBox(height: 15),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void loadingPopUp(BuildContext context) {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (context) {
|
|
||||||
return const Mihloadingcircle();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _speakText(String text) async {
|
|
||||||
try {
|
|
||||||
loadingPopUp(context);
|
|
||||||
await _flutterTts.stop(); // Stop any ongoing speech
|
|
||||||
await _flutterTts.speak(text).then((value) {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
}); // Speak the new text
|
|
||||||
} catch (e) {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
print("TTS Error: $e");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setTtsVoice(String voiceName) {
|
|
||||||
_flutterTts.setVoice(
|
|
||||||
{
|
|
||||||
"name": voiceName,
|
|
||||||
"locale": _voices
|
|
||||||
.where((_voice) => _voice["name"].contains(voiceName))
|
|
||||||
.first["locale"]
|
|
||||||
},
|
|
||||||
);
|
|
||||||
_ttsVoiceController.text = voiceName;
|
|
||||||
}
|
|
||||||
|
|
||||||
void voiceSelected() {
|
|
||||||
if (_ttsVoiceController.text.isNotEmpty) {
|
|
||||||
_ttsVoiceName.value = _ttsVoiceController.text;
|
|
||||||
// print(
|
|
||||||
// "======================================== Voice Set ========================================");
|
|
||||||
setTtsVoice(_ttsVoiceController.text);
|
|
||||||
} else {
|
|
||||||
_ttsVoiceName.value = "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void fontSizeChanged() {
|
|
||||||
setState(() {
|
|
||||||
_chatFrontSize = double.parse(_fontSizeController.text);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
super.dispose();
|
|
||||||
_modelController.dispose();
|
|
||||||
_fontSizeController.dispose();
|
|
||||||
_ttsVoiceController.dispose();
|
|
||||||
_ttsVoiceController.removeListener(voiceSelected);
|
|
||||||
_fontSizeController.removeListener(fontSizeChanged);
|
|
||||||
client.endSession();
|
|
||||||
_flutterTts.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
void initTTS() {
|
|
||||||
_flutterTts.setVolume(1);
|
|
||||||
_fontSizeController.addListener(fontSizeChanged);
|
|
||||||
// _flutterTts.setSpeechRate(0.6);
|
|
||||||
// _flutterTts.setPitch(1.0);
|
|
||||||
_flutterTts.getVoices.then(
|
|
||||||
(data) {
|
|
||||||
try {
|
|
||||||
_voices = List<Map>.from(data);
|
|
||||||
|
|
||||||
setState(() {
|
|
||||||
_voices = _voices
|
|
||||||
.where(
|
|
||||||
(_voice) => _voice["name"].toLowerCase().contains("en-us"))
|
|
||||||
.toList();
|
|
||||||
_voicesString =
|
|
||||||
_voices.map((_voice) => _voice["name"] as String).toList();
|
|
||||||
_voicesString.sort();
|
|
||||||
// print(
|
|
||||||
// "=================== Voices ===================\n$_voicesString");
|
|
||||||
|
|
||||||
setTtsVoice(_voicesString.first);
|
|
||||||
});
|
|
||||||
} catch (e) {
|
|
||||||
print(e);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
MzansiAiProvider mzansiAiProvider = context.read<MzansiAiProvider>();
|
|
||||||
MzansiProfileProvider mzansiProfileProvider =
|
|
||||||
context.read<MzansiProfileProvider>();
|
|
||||||
_user = types.User(
|
|
||||||
firstName: mzansiProfileProvider.user!.fname,
|
|
||||||
id: mzansiProfileProvider
|
|
||||||
.user!.app_id, //'82091008-a484-4a89-ae75-a22bf8d6f3ac',
|
|
||||||
);
|
|
||||||
_mihAI = types.User(
|
|
||||||
firstName: "Mzansi AI",
|
|
||||||
id: const Uuid().v4(),
|
|
||||||
);
|
|
||||||
_modelController.text = getModel();
|
|
||||||
_fontSizeController.text = _chatFrontSize.ceil().toString();
|
|
||||||
systemPromt = setSystemPromt();
|
|
||||||
_chatHistory.add(
|
|
||||||
ollama.Message(
|
|
||||||
role: ollama.MessageRole.system,
|
|
||||||
content: systemPromt,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
initTTS();
|
|
||||||
_ttsVoiceController.addListener(voiceSelected);
|
|
||||||
if (mzansiAiProvider.startUpQuestion != null &&
|
|
||||||
mzansiAiProvider.startUpQuestion!.isNotEmpty) {
|
|
||||||
final partialText =
|
|
||||||
types.PartialText(text: mzansiAiProvider.startUpQuestion!);
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
||||||
_handleSendPressed(partialText);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return MihPackageToolBody(
|
|
||||||
borderOn: false,
|
|
||||||
bodyItem: getBody(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget getBody() {
|
|
||||||
return Stack(
|
|
||||||
children: [
|
|
||||||
Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
||||||
mainAxisSize: MainAxisSize.max,
|
|
||||||
children: [
|
|
||||||
_getSettings(),
|
|
||||||
Expanded(
|
|
||||||
child: GestureDetector(
|
|
||||||
onTap: () {
|
|
||||||
if (_showModelOptions.value == true) {
|
|
||||||
setState(() {
|
|
||||||
_showModelOptions.value = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: Chat(
|
|
||||||
messages: _messages,
|
|
||||||
emptyState: noMessagescDisplay(),
|
|
||||||
// onAttachmentPressed: _handleAttachmentPressed,
|
|
||||||
// onMessageTap: _handleMessageTap,
|
|
||||||
// onPreviewDataFetched: _handlePreviewDataFetched,
|
|
||||||
onSendPressed: _handleSendPressed,
|
|
||||||
showUserAvatars: false,
|
|
||||||
showUserNames: false,
|
|
||||||
user: _user,
|
|
||||||
theme: getChatTheme(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
left: 15,
|
|
||||||
top: 15,
|
|
||||||
child: Visibility(
|
|
||||||
visible: _showModelOptions.value == true,
|
|
||||||
child: Container(
|
|
||||||
// color: Colors.white,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(25), // Optional: rounds the corners
|
|
||||||
boxShadow: const [
|
|
||||||
BoxShadow(
|
|
||||||
color: Color.fromARGB(
|
|
||||||
60, 0, 0, 0), // 0.2 opacity = 51 in alpha (255 * 0.2)
|
|
||||||
spreadRadius: -2,
|
|
||||||
blurRadius: 10,
|
|
||||||
offset: Offset(0, 5),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
top: 2.0,
|
|
||||||
left: 5.0,
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 40,
|
|
||||||
child: IconButton.filled(
|
|
||||||
style: ButtonStyle(
|
|
||||||
backgroundColor: WidgetStateProperty.all<Color>(
|
|
||||||
MihColors.getRedColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
||||||
"Dark")),
|
|
||||||
),
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
iconSize: 20,
|
|
||||||
onPressed: () {
|
|
||||||
if (_showModelOptions.value == true) {
|
|
||||||
setState(() {
|
|
||||||
_showModelOptions.value = false;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
_showModelOptions.value = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.close,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
// IconButton.filled(
|
|
||||||
// iconSize: 20,
|
|
||||||
// onPressed: () {
|
|
||||||
// if (_showModelOptions.value == true) {
|
|
||||||
// setState(() {
|
|
||||||
// _showModelOptions.value = false;
|
|
||||||
// });
|
|
||||||
// } else {
|
|
||||||
// setState(() {
|
|
||||||
// _showModelOptions.value = true;
|
|
||||||
// });
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
// icon: const Icon(
|
|
||||||
// Icons.settings,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
right: 10,
|
|
||||||
bottom: 80,
|
|
||||||
child: MihFloatingMenu(
|
|
||||||
animatedIcon: AnimatedIcons.menu_close,
|
|
||||||
children: [
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.refresh,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
label: "New Chat",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onTap: () {
|
|
||||||
_resetChat();
|
|
||||||
},
|
|
||||||
),
|
|
||||||
SpeedDialChild(
|
|
||||||
child: Icon(
|
|
||||||
Icons.settings,
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
label: "Settings",
|
|
||||||
labelBackgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
labelStyle: TextStyle(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
backgroundColor: MihColors.getGreenColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
onTap: () {
|
|
||||||
if (_showModelOptions.value == true) {
|
|
||||||
setState(() {
|
|
||||||
_showModelOptions.value = false;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
setState(() {
|
|
||||||
_showModelOptions.value = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? noMessagescDisplay() {
|
|
||||||
return Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
// const SizedBox(height: 50),
|
|
||||||
Icon(
|
|
||||||
MihIcons.mzansiAi,
|
|
||||||
size: 165,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 10),
|
|
||||||
Text(
|
|
||||||
"Mzansi AI is here to help",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
overflow: TextOverflow.visible,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 25,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 25),
|
|
||||||
Center(
|
|
||||||
child: RichText(
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
text: TextSpan(
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
),
|
|
||||||
children: [
|
|
||||||
TextSpan(
|
|
||||||
text:
|
|
||||||
"Send us a message and we'll try our best to assist you"),
|
|
||||||
// WidgetSpan(
|
|
||||||
// alignment: PlaceholderAlignment.middle,
|
|
||||||
// child: Icon(
|
|
||||||
// Icons.menu,
|
|
||||||
// size: 20,
|
|
||||||
// color: MzansiInnovationHub.of(context)!
|
|
||||||
// .theme
|
|
||||||
// .secondaryColor(),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// TextSpan(text: " to add your first loyalty card."),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/business.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_business_profile_preview.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_directory_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class BuildBusinessSearchResultsList extends StatefulWidget {
|
|
||||||
final List<Business> businessList;
|
|
||||||
const BuildBusinessSearchResultsList({
|
|
||||||
super.key,
|
|
||||||
required this.businessList,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<BuildBusinessSearchResultsList> createState() =>
|
|
||||||
_BuildBusinessSearchResultsListState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _BuildBusinessSearchResultsListState
|
|
||||||
extends State<BuildBusinessSearchResultsList> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Consumer<MzansiDirectoryProvider>(
|
|
||||||
builder: (BuildContext context, MzansiDirectoryProvider directoryProvider,
|
|
||||||
Widget? child) {
|
|
||||||
return ListView.separated(
|
|
||||||
shrinkWrap: true,
|
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
itemCount: widget.businessList.length,
|
|
||||||
separatorBuilder: (BuildContext context, index) {
|
|
||||||
return Divider(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return Material(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
child: InkWell(
|
|
||||||
onTap: () {
|
|
||||||
directoryProvider.setSelectedBusiness(
|
|
||||||
business: widget.businessList[index],
|
|
||||||
);
|
|
||||||
context.pushNamed(
|
|
||||||
'businessProfileView',
|
|
||||||
);
|
|
||||||
},
|
|
||||||
splashColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark")
|
|
||||||
.withOpacity(0.2),
|
|
||||||
borderRadius: BorderRadius.circular(15),
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsetsGeometry.symmetric(
|
|
||||||
// vertical: 5,
|
|
||||||
horizontal: 25,
|
|
||||||
),
|
|
||||||
child: MihBusinessProfilePreview(
|
|
||||||
business: widget.businessList[index],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/main.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_objects/app_user.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_personal_profile_preview.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_components/mih_providers/mzansi_directory_provider.dart';
|
|
||||||
import 'package:mzansi_innovation_hub/mih_config/mih_colors.dart';
|
|
||||||
import 'package:provider/provider.dart';
|
|
||||||
|
|
||||||
class BuildUserSearchResultsList extends StatefulWidget {
|
|
||||||
final List<AppUser> userList;
|
|
||||||
const BuildUserSearchResultsList({
|
|
||||||
super.key,
|
|
||||||
required this.userList,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<BuildUserSearchResultsList> createState() =>
|
|
||||||
_BuildUserSearchResultsListState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _BuildUserSearchResultsListState
|
|
||||||
extends State<BuildUserSearchResultsList> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Consumer<MzansiDirectoryProvider>(
|
|
||||||
builder: (BuildContext context, MzansiDirectoryProvider directoryProvider,
|
|
||||||
Widget? child) {
|
|
||||||
return ListView.separated(
|
|
||||||
shrinkWrap: true,
|
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
itemCount: widget.userList.length,
|
|
||||||
separatorBuilder: (BuildContext context, index) {
|
|
||||||
return Divider(
|
|
||||||
color: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return Material(
|
|
||||||
color: MihColors.getPrimaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
||||||
child: InkWell(
|
|
||||||
onTap: () {
|
|
||||||
directoryProvider.setSelectedUser(
|
|
||||||
user: widget.userList[index]);
|
|
||||||
context.pushNamed(
|
|
||||||
'mzansiProfileView',
|
|
||||||
);
|
|
||||||
},
|
|
||||||
splashColor: MihColors.getSecondaryColor(
|
|
||||||
MzansiInnovationHub.of(context)!.theme.mode == "Dark")
|
|
||||||
.withOpacity(0.2),
|
|
||||||
borderRadius: BorderRadius.circular(15),
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsetsGeometry.symmetric(
|
|
||||||
// vertical: 5,
|
|
||||||
horizontal: 25,
|
|
||||||
),
|
|
||||||
child:
|
|
||||||
MihPersonalProfilePreview(user: widget.userList[index]),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||