551 lines
20 KiB
Dart
551 lines
20 KiB
Dart
import 'package:flutter_speed_dial/flutter_speed_dial.dart';
|
|
import 'package:go_router/go_router.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_alert.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_mzansi_calendar_services.dart';
|
|
import 'package:mzansi_innovation_hub/mih_services/mih_validation_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_package_components/mih_button.dart';
|
|
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_date_field.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_tool_body.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_package_window.dart';
|
|
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_text_form_field.dart';
|
|
import 'package:mzansi_innovation_hub/mih_components/mih_package_components/mih_time_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_objects/appointment.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_packages/calendar/builder/build_appointment_list.dart';
|
|
import 'package:flutter/material.dart';
|
|
import '../../../main.dart';
|
|
|
|
import '../../../mih_components/mih_package_components/mih_calendar.dart';
|
|
import '../../../mih_components/mih_pop_up_messages/mih_loading_circle.dart';
|
|
import '../../../mih_config/mih_env.dart';
|
|
import '../../../mih_components/mih_objects/app_user.dart';
|
|
|
|
class Appointments extends StatefulWidget {
|
|
final AppUser signedInUser;
|
|
final Business? business;
|
|
final BusinessUser? businessUser;
|
|
final bool personalSelected;
|
|
|
|
const Appointments({
|
|
super.key,
|
|
required this.signedInUser,
|
|
required this.business,
|
|
required this.businessUser,
|
|
required this.personalSelected,
|
|
});
|
|
|
|
@override
|
|
State<Appointments> createState() => _PatientAccessRequestState();
|
|
}
|
|
|
|
class _PatientAccessRequestState extends State<Appointments> {
|
|
TextEditingController selectedAppointmentDateController =
|
|
TextEditingController();
|
|
final TextEditingController _appointmentTitleController =
|
|
TextEditingController();
|
|
final TextEditingController _appointmentDescriptionIDController =
|
|
TextEditingController();
|
|
final TextEditingController _appointmentDateController =
|
|
TextEditingController();
|
|
final TextEditingController _appointmentTimeController =
|
|
TextEditingController();
|
|
String baseUrl = AppEnviroment.baseApiUrl;
|
|
|
|
String selectedDay = DateTime.now().toString().split(" ")[0];
|
|
|
|
late Future<List<Appointment>> personalAppointmentResults;
|
|
late Future<List<Appointment>> businessAppointmentResults;
|
|
late Future<List<Appointment>> appointmentResults;
|
|
|
|
final _formKey = GlobalKey<FormState>();
|
|
|
|
Widget displayAppointmentList(List<Appointment> appointmentList) {
|
|
if (appointmentList.isNotEmpty) {
|
|
return Expanded(
|
|
child: BuildAppointmentList(
|
|
appointmentList: appointmentList,
|
|
signedInUser: widget.signedInUser,
|
|
business: widget.business,
|
|
businessUser: widget.businessUser,
|
|
personalSelected: widget.personalSelected,
|
|
inWaitingRoom: false,
|
|
titleController: _appointmentTitleController,
|
|
descriptionIDController: _appointmentDescriptionIDController,
|
|
patientIdController: null,
|
|
dateController: _appointmentDateController,
|
|
timeController: _appointmentTimeController,
|
|
),
|
|
);
|
|
}
|
|
return Expanded(
|
|
child: Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 10.0),
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
children: [
|
|
const SizedBox(height: 50),
|
|
Icon(
|
|
MihIcons.calendar,
|
|
size: 165,
|
|
color: MihColors.getSecondaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
),
|
|
const SizedBox(height: 10),
|
|
Text(
|
|
"No appointments for $selectedDay",
|
|
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(
|
|
Icons.menu,
|
|
size: 20,
|
|
color: MihColors.getSecondaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
),
|
|
),
|
|
TextSpan(
|
|
text:
|
|
" to add an appointment or select a different date"),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
void addAppointmentWindow(double width) {
|
|
showDialog(
|
|
context: context,
|
|
barrierDismissible: false,
|
|
builder: (context) {
|
|
return MihPackageWindow(
|
|
fullscreen: false,
|
|
windowTitle: "Add Appointment",
|
|
onWindowTapClose: () {
|
|
Navigator.of(context).pop();
|
|
_appointmentDateController.clear();
|
|
_appointmentTimeController.clear();
|
|
_appointmentTitleController.clear();
|
|
_appointmentDescriptionIDController.clear();
|
|
},
|
|
windowBody: Padding(
|
|
padding:
|
|
MzansiInnovationHub.of(context)!.theme.screenType == "desktop"
|
|
? EdgeInsets.symmetric(horizontal: width * 0.05)
|
|
: const EdgeInsets.symmetric(horizontal: 0),
|
|
child: Column(
|
|
children: [
|
|
MihForm(
|
|
formKey: _formKey,
|
|
formFields: [
|
|
MihTextFormField(
|
|
fillColor: MihColors.getSecondaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
inputColor: MihColors.getPrimaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
controller: _appointmentTitleController,
|
|
multiLineInput: false,
|
|
requiredText: true,
|
|
hintText: "Appointment Title",
|
|
validator: (value) {
|
|
return MihValidationServices().isEmpty(value);
|
|
},
|
|
),
|
|
const SizedBox(height: 10),
|
|
MihDateField(
|
|
controller: _appointmentDateController,
|
|
labelText: "Date",
|
|
required: true,
|
|
validator: (value) {
|
|
return MihValidationServices().isEmpty(value);
|
|
},
|
|
),
|
|
const SizedBox(height: 10),
|
|
MihTimeField(
|
|
controller: _appointmentTimeController,
|
|
labelText: "Time",
|
|
required: true,
|
|
validator: (value) {
|
|
return MihValidationServices().isEmpty(value);
|
|
},
|
|
),
|
|
const SizedBox(height: 10),
|
|
MihTextFormField(
|
|
fillColor: MihColors.getSecondaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
inputColor: MihColors.getPrimaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
controller: _appointmentDescriptionIDController,
|
|
multiLineInput: true,
|
|
height: 250,
|
|
requiredText: true,
|
|
hintText: "Appointment Description",
|
|
validator: (value) {
|
|
return MihValidationServices().isEmpty(value);
|
|
},
|
|
),
|
|
const SizedBox(height: 20),
|
|
Center(
|
|
child: MihButton(
|
|
onPressed: () {
|
|
if (_formKey.currentState!.validate()) {
|
|
addAppointmentCall();
|
|
} else {
|
|
MihAlertServices().formNotFilledCompletely(context);
|
|
}
|
|
},
|
|
buttonColor: MihColors.getGreenColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
width: 300,
|
|
child: Text(
|
|
"Add",
|
|
style: TextStyle(
|
|
color: MihColors.getPrimaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode ==
|
|
"Dark"),
|
|
fontSize: 20,
|
|
fontWeight: FontWeight.bold,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
bool isAppointmentInputValid() {
|
|
if (_appointmentTitleController.text.isEmpty ||
|
|
_appointmentDescriptionIDController.text.isEmpty ||
|
|
_appointmentDateController.text.isEmpty ||
|
|
_appointmentTimeController.text.isEmpty) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
Future<void> addAppointmentCall() async {
|
|
if (isAppointmentInputValid()) {
|
|
int statusCode;
|
|
if (widget.personalSelected == false) {
|
|
statusCode = await MihMzansiCalendarApis.addBusinessAppointment(
|
|
widget.signedInUser,
|
|
widget.business!,
|
|
widget.businessUser!,
|
|
false,
|
|
_appointmentTitleController.text,
|
|
_appointmentDescriptionIDController.text,
|
|
_appointmentDateController.text,
|
|
_appointmentTimeController.text,
|
|
context,
|
|
);
|
|
} else {
|
|
statusCode = await MihMzansiCalendarApis.addPersonalAppointment(
|
|
widget.signedInUser,
|
|
_appointmentTitleController.text,
|
|
_appointmentDescriptionIDController.text,
|
|
_appointmentDateController.text,
|
|
_appointmentTimeController.text,
|
|
context,
|
|
);
|
|
}
|
|
if (statusCode == 201) {
|
|
context.pop();
|
|
successPopUp("Successfully Added Appointment",
|
|
"You appointment has been successfully added to your calendar.");
|
|
setState(() {
|
|
if (widget.personalSelected) {
|
|
appointmentResults = MihMzansiCalendarApis.getPersonalAppointments(
|
|
widget.signedInUser.app_id,
|
|
selectedDay,
|
|
);
|
|
} else {
|
|
appointmentResults = MihMzansiCalendarApis.getBusinessAppointments(
|
|
widget.business!.business_id,
|
|
false,
|
|
selectedDay,
|
|
);
|
|
}
|
|
});
|
|
} else {
|
|
internetConnectionPopUp();
|
|
}
|
|
} else {
|
|
showDialog(
|
|
context: context,
|
|
builder: (context) {
|
|
return const MIHErrorMessage(errorType: "Input Error");
|
|
},
|
|
);
|
|
}
|
|
checkforchange();
|
|
}
|
|
|
|
void successPopUp(String title, String message) {
|
|
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: title,
|
|
alertBody: Column(
|
|
children: [
|
|
Text(
|
|
message,
|
|
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.pop();
|
|
setState(() {
|
|
_appointmentDateController.clear();
|
|
_appointmentTimeController.clear();
|
|
_appointmentTitleController.clear();
|
|
_appointmentDescriptionIDController.clear();
|
|
});
|
|
},
|
|
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"),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
void internetConnectionPopUp() {
|
|
showDialog(
|
|
context: context,
|
|
builder: (context) {
|
|
return const MIHErrorMessage(
|
|
errorType: "Internet Connection",
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
String getTitle() {
|
|
if (widget.personalSelected == false) {
|
|
return "Business Appointments";
|
|
} else {
|
|
return "Personal Appointments";
|
|
}
|
|
}
|
|
|
|
void checkforchange() {
|
|
setState(() {
|
|
if (widget.personalSelected == false) {
|
|
appointmentResults = MihMzansiCalendarApis.getBusinessAppointments(
|
|
widget.business!.business_id,
|
|
false,
|
|
selectedDay,
|
|
);
|
|
} else {
|
|
appointmentResults = MihMzansiCalendarApis.getPersonalAppointments(
|
|
widget.signedInUser.app_id,
|
|
selectedDay,
|
|
);
|
|
}
|
|
});
|
|
}
|
|
|
|
Widget getBody(double width) {
|
|
return Stack(
|
|
children: [
|
|
MihSingleChildScroll(
|
|
child: Column(
|
|
children: [
|
|
MIHCalendar(
|
|
calendarWidth: 500,
|
|
rowHeight: 35,
|
|
setDate: (value) {
|
|
setState(() {
|
|
selectedDay = value;
|
|
selectedAppointmentDateController.text = selectedDay;
|
|
});
|
|
}),
|
|
// Divider(
|
|
// color: MihColors.getSecondaryColor(MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
// ),
|
|
Row(
|
|
mainAxisSize: MainAxisSize.max,
|
|
children: [
|
|
FutureBuilder(
|
|
future: appointmentResults,
|
|
builder: (context, snapshot) {
|
|
if (snapshot.connectionState ==
|
|
ConnectionState.waiting) {
|
|
return const Expanded(
|
|
child: Center(child: Mihloadingcircle()));
|
|
} else if (snapshot.connectionState ==
|
|
ConnectionState.done &&
|
|
snapshot.hasData) {
|
|
return displayAppointmentList(snapshot.requireData);
|
|
} else {
|
|
return Center(
|
|
child: Text(
|
|
"Error pulling appointments",
|
|
style: TextStyle(
|
|
fontSize: 25,
|
|
color: MihColors.getRedColor(
|
|
MzansiInnovationHub.of(context)!
|
|
.theme
|
|
.mode ==
|
|
"Dark")),
|
|
textAlign: TextAlign.center,
|
|
),
|
|
);
|
|
}
|
|
}),
|
|
],
|
|
)
|
|
],
|
|
),
|
|
),
|
|
Positioned(
|
|
right: 10,
|
|
bottom: 10,
|
|
child: MihFloatingMenu(
|
|
icon: Icons.add,
|
|
animatedIcon: AnimatedIcons.menu_close,
|
|
children: [
|
|
SpeedDialChild(
|
|
child: Icon(
|
|
Icons.add,
|
|
color: MihColors.getPrimaryColor(
|
|
MzansiInnovationHub.of(context)!.theme.mode == "Dark"),
|
|
),
|
|
label: "Add Appointment",
|
|
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: () {
|
|
addAppointmentWindow(width);
|
|
},
|
|
)
|
|
],
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
selectedAppointmentDateController.dispose();
|
|
_appointmentDateController.dispose();
|
|
_appointmentTimeController.dispose();
|
|
_appointmentTitleController.dispose();
|
|
_appointmentDescriptionIDController.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
void initState() {
|
|
selectedAppointmentDateController.addListener(checkforchange);
|
|
setState(() {
|
|
if (widget.personalSelected == false) {
|
|
appointmentResults = MihMzansiCalendarApis.getBusinessAppointments(
|
|
widget.business!.business_id,
|
|
false,
|
|
selectedDay,
|
|
);
|
|
} else {
|
|
appointmentResults = MihMzansiCalendarApis.getPersonalAppointments(
|
|
widget.signedInUser.app_id,
|
|
selectedDay,
|
|
);
|
|
}
|
|
});
|
|
super.initState();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
double screenWidth = MediaQuery.of(context).size.width;
|
|
return MihPackageToolBody(
|
|
borderOn: false,
|
|
bodyItem: getBody(screenWidth),
|
|
);
|
|
}
|
|
}
|