mirror of
https://github.com/SyncrowIOT/web.git
synced 2025-07-10 07:07:19 +00:00
Refactor schedule components and update imports for garage door and water heater modules
This commit is contained in:
@ -4,8 +4,8 @@ import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
|||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_bloc.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_event.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_event.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/opening_clsoing_time_dialog_body.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/opening_clsoing_time_dialog_body.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/time_out_alarm_dialog_body.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/time_out_alarm_dialog_body.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/shared/sensors_widgets/presence_display_data.dart';
|
import 'package:syncrow_web/pages/device_managment/shared/sensors_widgets/presence_display_data.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/shared/toggle_widget.dart';
|
import 'package:syncrow_web/pages/device_managment/shared/toggle_widget.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
||||||
|
@ -1,14 +1,13 @@
|
|||||||
import 'package:flutter/cupertino.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_bloc.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/seconds_picker.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/seconds_picker.dart';
|
||||||
|
|
||||||
class OpeningAndClosingTimeDialogBody extends StatefulWidget {
|
class OpeningAndClosingTimeDialogBody extends StatefulWidget {
|
||||||
final ValueChanged<int> onDurationChanged;
|
final ValueChanged<int> onDurationChanged;
|
||||||
final GarageDoorBloc bloc;
|
final GarageDoorBloc bloc;
|
||||||
|
|
||||||
OpeningAndClosingTimeDialogBody({
|
const OpeningAndClosingTimeDialogBody({
|
||||||
required this.onDurationChanged,
|
required this.onDurationChanged,
|
||||||
required this.bloc,
|
required this.bloc,
|
||||||
});
|
});
|
@ -26,7 +26,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
Table(
|
Table(
|
||||||
border: TableBorder.all(
|
border: TableBorder.all(
|
||||||
color: ColorsManager.graysColor,
|
color: ColorsManager.graysColor,
|
||||||
borderRadius: const BorderRadius.only(topLeft: Radius.circular(20), topRight: Radius.circular(20)),
|
borderRadius: const BorderRadius.only(
|
||||||
|
topLeft: Radius.circular(20), topRight: Radius.circular(20)),
|
||||||
),
|
),
|
||||||
children: [
|
children: [
|
||||||
TableRow(
|
TableRow(
|
||||||
@ -50,17 +51,21 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
BlocBuilder<GarageDoorBloc, GarageDoorState>(
|
BlocBuilder<GarageDoorBloc, GarageDoorState>(
|
||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
if (state is ScheduleGarageLoadingState) {
|
if (state is ScheduleGarageLoadingState) {
|
||||||
return const SizedBox(height: 200, child: Center(child: CircularProgressIndicator()));
|
return const SizedBox(
|
||||||
|
height: 200,
|
||||||
|
child: Center(child: CircularProgressIndicator()));
|
||||||
}
|
}
|
||||||
if (state is GarageDoorLoadedState && state.status.schedules?.isEmpty == true) {
|
if (state is GarageDoorLoadedState &&
|
||||||
|
state.status.schedules?.isEmpty == true) {
|
||||||
return _buildEmptyState(context);
|
return _buildEmptyState(context);
|
||||||
} else if (state is GarageDoorLoadedState) {
|
} else if (state is GarageDoorLoadedState) {
|
||||||
return Container(
|
return Container(
|
||||||
height: 200,
|
height: 200,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
border: Border.all(color: ColorsManager.graysColor),
|
border: Border.all(color: ColorsManager.graysColor),
|
||||||
borderRadius:
|
borderRadius: const BorderRadius.only(
|
||||||
const BorderRadius.only(bottomLeft: Radius.circular(20), bottomRight: Radius.circular(20)),
|
bottomLeft: Radius.circular(20),
|
||||||
|
bottomRight: Radius.circular(20)),
|
||||||
),
|
),
|
||||||
child: _buildTableBody(state, context));
|
child: _buildTableBody(state, context));
|
||||||
}
|
}
|
||||||
@ -78,7 +83,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
height: 200,
|
height: 200,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
border: Border.all(color: ColorsManager.graysColor),
|
border: Border.all(color: ColorsManager.graysColor),
|
||||||
borderRadius: const BorderRadius.only(bottomLeft: Radius.circular(20), bottomRight: Radius.circular(20)),
|
borderRadius: const BorderRadius.only(
|
||||||
|
bottomLeft: Radius.circular(20), bottomRight: Radius.circular(20)),
|
||||||
),
|
),
|
||||||
child: Center(
|
child: Center(
|
||||||
child: Column(
|
child: Column(
|
||||||
@ -112,7 +118,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
if (state.status.schedules != null)
|
if (state.status.schedules != null)
|
||||||
for (int i = 0; i < state.status.schedules!.length; i++)
|
for (int i = 0; i < state.status.schedules!.length; i++)
|
||||||
_buildScheduleRow(state.status.schedules![i], i, context, state),
|
_buildScheduleRow(
|
||||||
|
state.status.schedules![i], i, context, state),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -134,7 +141,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
TableRow _buildScheduleRow(ScheduleModel schedule, int index, BuildContext context, GarageDoorLoadedState state) {
|
TableRow _buildScheduleRow(ScheduleModel schedule, int index,
|
||||||
|
BuildContext context, GarageDoorLoadedState state) {
|
||||||
return TableRow(
|
return TableRow(
|
||||||
children: [
|
children: [
|
||||||
Center(
|
Center(
|
||||||
@ -152,7 +160,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
width: 24,
|
width: 24,
|
||||||
height: 24,
|
height: 24,
|
||||||
child: schedule.enable
|
child: schedule.enable
|
||||||
? const Icon(Icons.radio_button_checked, color: ColorsManager.blueColor)
|
? const Icon(Icons.radio_button_checked,
|
||||||
|
color: ColorsManager.blueColor)
|
||||||
: const Icon(
|
: const Icon(
|
||||||
Icons.radio_button_unchecked,
|
Icons.radio_button_unchecked,
|
||||||
color: ColorsManager.grayColor,
|
color: ColorsManager.grayColor,
|
||||||
@ -160,7 +169,9 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Center(child: Text(_getSelectedDays(ScheduleModel.parseSelectedDays(schedule.days)))),
|
Center(
|
||||||
|
child: Text(_getSelectedDays(
|
||||||
|
ScheduleModel.parseSelectedDays(schedule.days)))),
|
||||||
Center(child: Text(formatIsoStringToTime(schedule.time, context))),
|
Center(child: Text(formatIsoStringToTime(schedule.time, context))),
|
||||||
Center(child: Text(schedule.function.value ? 'On' : 'Off')),
|
Center(child: Text(schedule.function.value ? 'On' : 'Off')),
|
||||||
Center(
|
Center(
|
||||||
@ -170,18 +181,24 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
TextButton(
|
TextButton(
|
||||||
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
GarageDoorDialogHelper.showAddGarageDoorScheduleDialog(context,
|
GarageDoorDialogHelper.showAddGarageDoorScheduleDialog(
|
||||||
schedule: schedule, index: index, isEdit: true);
|
context,
|
||||||
|
schedule: schedule,
|
||||||
|
index: index,
|
||||||
|
isEdit: true);
|
||||||
},
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
'Edit',
|
'Edit',
|
||||||
style: context.textTheme.bodySmall!.copyWith(color: ColorsManager.blueColor),
|
style: context.textTheme.bodySmall!
|
||||||
|
.copyWith(color: ColorsManager.blueColor),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
context.read<GarageDoorBloc>().add(DeleteGarageDoorScheduleEvent(
|
context
|
||||||
|
.read<GarageDoorBloc>()
|
||||||
|
.add(DeleteGarageDoorScheduleEvent(
|
||||||
index: index,
|
index: index,
|
||||||
scheduleId: schedule.scheduleId,
|
scheduleId: schedule.scheduleId,
|
||||||
deviceId: state.status.uuid,
|
deviceId: state.status.uuid,
|
||||||
@ -189,7 +206,8 @@ class ScheduleGarageTableWidget extends StatelessWidget {
|
|||||||
},
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
'Delete',
|
'Delete',
|
||||||
style: context.textTheme.bodySmall!.copyWith(color: ColorsManager.blueColor),
|
style: context.textTheme.bodySmall!
|
||||||
|
.copyWith(color: ColorsManager.blueColor),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
@ -1,7 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/schedule__garage_table.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/schedule__garage_table.dart';
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
@ -4,9 +4,9 @@ import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_
|
|||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/helper/garage_door_helper.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/helper/garage_door_helper.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/models/garage_door_model.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/models/garage_door_model.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/schedule_garage_header.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/schedule_garage_header.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/schedule_garage_managment_ui.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/schedule_garage_managment_ui.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/schedule_garage_mode_buttons.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/schedule_garage_mode_buttons.dart';
|
||||||
|
|
||||||
class BuildGarageDoorScheduleView extends StatefulWidget {
|
class BuildGarageDoorScheduleView extends StatefulWidget {
|
||||||
const BuildGarageDoorScheduleView({super.key, required this.status});
|
const BuildGarageDoorScheduleView({super.key, required this.status});
|
@ -5,7 +5,7 @@ import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_
|
|||||||
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/bloc/garage_door_state.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/helper/garage_door_helper.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/helper/garage_door_helper.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/models/garage_door_model.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/models/garage_door_model.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/garage_door/widgets/schedule_garage_view.dart';
|
import 'package:syncrow_web/pages/device_managment/garage_door/schedule_view/schedule_garage_view.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/shared/icon_name_status_container.dart';
|
import 'package:syncrow_web/pages/device_managment/shared/icon_name_status_container.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/shared/table/report_table.dart';
|
import 'package:syncrow_web/pages/device_managment/shared/table/report_table.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/shared/toggle_widget.dart';
|
import 'package:syncrow_web/pages/device_managment/shared/toggle_widget.dart';
|
||||||
|
@ -0,0 +1,557 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
import 'package:bloc/bloc.dart';
|
||||||
|
import 'package:equatable/equatable.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/all_devices/models/device_status.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_entry.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
import 'package:syncrow_web/services/control_device_service.dart';
|
||||||
|
import 'package:syncrow_web/services/devices_mang_api.dart';
|
||||||
|
part 'schedule_event.dart';
|
||||||
|
part 'schedule_state.dart';
|
||||||
|
|
||||||
|
class ScheduleBloc extends Bloc<ScheduleEvent, ScheduleState> {
|
||||||
|
final String deviceId;
|
||||||
|
|
||||||
|
ScheduleBloc({
|
||||||
|
required this.deviceId,
|
||||||
|
}) : super(ScheduleInitial()) {
|
||||||
|
on<ScheduleInitializeAddEvent>(_initializeAddSchedule);
|
||||||
|
on<ScheduleUpdateSelectedTimeEvent>(_updateSelectedTime);
|
||||||
|
on<ScheduleUpdateSelectedDayEvent>(_updateSelectedDay);
|
||||||
|
on<ScheduleUpdateFunctionOnEvent>(_updateFunctionOn);
|
||||||
|
on<ScheduleGetEvent>(_getSchedule);
|
||||||
|
on<ScheduleAddEvent>(_onAddSchedule);
|
||||||
|
on<ScheduleEditEvent>(_onEditSchedule);
|
||||||
|
on<ScheduleUpdateEntryEvent>(_onUpdateSchedule);
|
||||||
|
on<UpdateScheduleModeEvent>(_onUpdateScheduleMode);
|
||||||
|
on<UpdateCountdownTimeEvent>(_onUpdateCountdownTime);
|
||||||
|
on<UpdateInchingTimeEvent>(_onUpdateInchingTime);
|
||||||
|
on<StartScheduleEvent>(_onStartScheduleEvent);
|
||||||
|
on<StopScheduleEvent>(_onStopScheduleEvent);
|
||||||
|
on<ScheduleDecrementCountdownEvent>(_onDecrementCountdown);
|
||||||
|
on<ScheduleFetchStatusEvent>(_fetchStatus);
|
||||||
|
on<ScheduleDeleteEvent>(_onDeleteSchedule);
|
||||||
|
}
|
||||||
|
Timer? _countdownTimer;
|
||||||
|
Duration countdownRemaining = Duration.zero;
|
||||||
|
|
||||||
|
void _onStopScheduleEvent(
|
||||||
|
StopScheduleEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
_countdownTimer?.cancel();
|
||||||
|
|
||||||
|
if (event.mode == ScheduleModes.countdown) {
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
isCountdownActive: false,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
} else if (event.mode == ScheduleModes.inching) {
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
isInchingActive: false,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onUpdateScheduleMode(
|
||||||
|
UpdateScheduleModeEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
scheduleMode: event.scheduleMode,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onUpdateCountdownTime(
|
||||||
|
UpdateCountdownTimeEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
countdownHours: event.hours,
|
||||||
|
countdownMinutes: event.minutes,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onUpdateInchingTime(
|
||||||
|
UpdateInchingTimeEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
inchingHours: event.hours,
|
||||||
|
inchingMinutes: event.minutes,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _initializeAddSchedule(
|
||||||
|
ScheduleInitializeAddEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
selectedTime: event.selectedTime,
|
||||||
|
selectedDays: event.selectedDays ?? List.filled(7, false),
|
||||||
|
functionOn: event.functionOn ?? false,
|
||||||
|
isEditing: event.isEditing,
|
||||||
|
scheduleMode: event.scheduleMode,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
emit(ScheduleLoaded(
|
||||||
|
schedules: const [],
|
||||||
|
selectedTime: event.selectedTime,
|
||||||
|
selectedDays: event.selectedDays ?? List.filled(7, false),
|
||||||
|
functionOn: event.functionOn ?? false,
|
||||||
|
isEditing: event.isEditing,
|
||||||
|
deviceId: deviceId,
|
||||||
|
scheduleMode: event.scheduleMode,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
isCountdownActive: false,
|
||||||
|
isInchingActive: false,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateSelectedTime(
|
||||||
|
ScheduleUpdateSelectedTimeEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
selectedTime: event.selectedTime,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateSelectedDay(
|
||||||
|
ScheduleUpdateSelectedDayEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
final updatedDays = List<bool>.from(currentState.selectedDays);
|
||||||
|
updatedDays[event.index] = event.value;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
selectedDays: updatedDays,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateFunctionOn(
|
||||||
|
ScheduleUpdateFunctionOnEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
functionOn: event.isOn,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _getSchedule(
|
||||||
|
ScheduleGetEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
try {
|
||||||
|
emit(ScheduleLoading());
|
||||||
|
final schedules = await DevicesManagementApi().getDeviceSchedules(
|
||||||
|
deviceId,
|
||||||
|
event.category,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
schedules: schedules,
|
||||||
|
selectedTime: null,
|
||||||
|
selectedDays: List.filled(7, false),
|
||||||
|
functionOn: false,
|
||||||
|
isEditing: false,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
emit(ScheduleLoaded(
|
||||||
|
schedules: schedules,
|
||||||
|
selectedTime: null,
|
||||||
|
selectedDays: List.filled(7, false),
|
||||||
|
functionOn: false,
|
||||||
|
isEditing: false,
|
||||||
|
deviceId: deviceId,
|
||||||
|
scheduleMode: ScheduleModes.schedule,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
isCountdownActive: false,
|
||||||
|
isInchingActive: false,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to load schedules: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _onAddSchedule(
|
||||||
|
ScheduleAddEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
try {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final newSchedule = ScheduleEntry(
|
||||||
|
category: event.category,
|
||||||
|
time: event.time,
|
||||||
|
function: Status(code: 'switch_1', value: event.functionOn),
|
||||||
|
days: event.selectedDays);
|
||||||
|
final success = await DevicesManagementApi().addScheduleRecord(
|
||||||
|
newSchedule,
|
||||||
|
deviceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
add(const ScheduleGetEvent(category: 'switch_1'));
|
||||||
|
} else {
|
||||||
|
emit(const ScheduleError('Failed to add schedule'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to add schedule: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _onEditSchedule(
|
||||||
|
ScheduleEditEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
try {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final updatedSchedule = ScheduleEntry(
|
||||||
|
scheduleId: event.scheduleId,
|
||||||
|
category: event.category,
|
||||||
|
time: event.time,
|
||||||
|
function: Status(code: 'switch_1', value: event.functionOn),
|
||||||
|
days: event.selectedDays,
|
||||||
|
);
|
||||||
|
|
||||||
|
final success = await DevicesManagementApi().editScheduleRecord(
|
||||||
|
deviceId,
|
||||||
|
updatedSchedule,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
add(const ScheduleGetEvent(category: 'switch_1'));
|
||||||
|
} else {
|
||||||
|
emit(const ScheduleError('Failed to update schedule'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to update schedule: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _onUpdateSchedule(
|
||||||
|
ScheduleUpdateEntryEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
try {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
final updatedSchedules = currentState.schedules.map((schedule) {
|
||||||
|
if (schedule.scheduleId == event.scheduleId) {
|
||||||
|
return schedule.copyWith(
|
||||||
|
function: Status(code: 'switch_1', value: event.functionOn),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return schedule;
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
final success = await DevicesManagementApi().updateScheduleRecord(
|
||||||
|
enable: event.enable,
|
||||||
|
uuid: deviceId,
|
||||||
|
scheduleId: event.scheduleId,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
schedules: updatedSchedules,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
emit(const ScheduleError('Failed to update schedule status'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to update schedule: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _onDeleteSchedule(
|
||||||
|
ScheduleDeleteEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
try {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
final success = await DevicesManagementApi().deleteScheduleRecord(
|
||||||
|
deviceId,
|
||||||
|
event.scheduleId,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
final updatedSchedules = currentState.schedules
|
||||||
|
.where((s) => s.scheduleId != event.scheduleId)
|
||||||
|
.toList();
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
schedules: updatedSchedules,
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
inchingHours: 0,
|
||||||
|
inchingMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
emit(const ScheduleError('Failed to delete schedule'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to delete schedule: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Duration? _currentCountdown;
|
||||||
|
|
||||||
|
Future<void> _onStartScheduleEvent(
|
||||||
|
StartScheduleEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final totalSeconds =
|
||||||
|
Duration(hours: event.hours, minutes: event.minutes).inSeconds;
|
||||||
|
final code = event.mode == ScheduleModes.countdown
|
||||||
|
? 'countdown_1'
|
||||||
|
: 'switch_inching';
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
final duration = Duration(seconds: totalSeconds);
|
||||||
|
_currentCountdown = duration;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
countdownRemaining: duration,
|
||||||
|
schedules: currentState.schedules.map((schedule) {
|
||||||
|
if (schedule.function.code == code) {
|
||||||
|
return schedule.copyWith(
|
||||||
|
function: Status(code: code, value: totalSeconds),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return schedule;
|
||||||
|
}).toList(),
|
||||||
|
countdownHours: event.mode == ScheduleModes.countdown ? event.hours : 0,
|
||||||
|
));
|
||||||
|
|
||||||
|
final success = await RemoteControlDeviceService().controlDevice(
|
||||||
|
deviceUuid: deviceId,
|
||||||
|
status: Status(
|
||||||
|
code: code,
|
||||||
|
value: totalSeconds,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
if (code == 'countdown_1') {
|
||||||
|
final countdownDuration = Duration(seconds: totalSeconds);
|
||||||
|
|
||||||
|
emit(
|
||||||
|
currentState.copyWith(
|
||||||
|
countdownHours: countdownDuration.inHours,
|
||||||
|
countdownMinutes: countdownDuration.inMinutes % 60,
|
||||||
|
countdownRemaining: countdownDuration,
|
||||||
|
isCountdownActive: true,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (countdownDuration.inSeconds > 0) {
|
||||||
|
_startCountdownTimer(emit, countdownDuration);
|
||||||
|
} else {
|
||||||
|
_countdownTimer?.cancel();
|
||||||
|
emit(
|
||||||
|
currentState.copyWith(
|
||||||
|
countdownHours: 0,
|
||||||
|
countdownMinutes: 0,
|
||||||
|
countdownRemaining: Duration.zero,
|
||||||
|
isCountdownActive: false,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else if (code == 'switch_inching') {
|
||||||
|
final inchingDuration = Duration(seconds: totalSeconds);
|
||||||
|
emit(
|
||||||
|
currentState.copyWith(
|
||||||
|
inchingHours: inchingDuration.inHours,
|
||||||
|
inchingMinutes: inchingDuration.inMinutes % 60,
|
||||||
|
isInchingActive: true,
|
||||||
|
countdownRemaining: inchingDuration,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _startCountdownTimer(
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
Duration duration,
|
||||||
|
) {
|
||||||
|
_countdownTimer?.cancel();
|
||||||
|
_countdownTimer = Timer.periodic(const Duration(seconds: 1), (timer) {
|
||||||
|
if (_currentCountdown != null && _currentCountdown! > Duration.zero) {
|
||||||
|
_currentCountdown = _currentCountdown! - const Duration(seconds: 1);
|
||||||
|
countdownRemaining = _currentCountdown!;
|
||||||
|
add(const ScheduleDecrementCountdownEvent());
|
||||||
|
} else {
|
||||||
|
timer.cancel();
|
||||||
|
add(StopScheduleEvent(
|
||||||
|
mode: _currentCountdown == null
|
||||||
|
? ScheduleModes.countdown
|
||||||
|
: ScheduleModes.inching,
|
||||||
|
deviceId: deviceId,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onDecrementCountdown(
|
||||||
|
ScheduleDecrementCountdownEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
countdownRemaining: countdownRemaining,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> close() {
|
||||||
|
_countdownTimer?.cancel();
|
||||||
|
return super.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _fetchStatus(
|
||||||
|
ScheduleFetchStatusEvent event,
|
||||||
|
Emitter<ScheduleState> emit,
|
||||||
|
) async {
|
||||||
|
emit(ScheduleLoading());
|
||||||
|
|
||||||
|
try {
|
||||||
|
final status =
|
||||||
|
await DevicesManagementApi().getDeviceStatus(event.deviceId);
|
||||||
|
final deviceStatus =
|
||||||
|
WaterHeaterStatusModel.fromJson(event.deviceId, status.status);
|
||||||
|
|
||||||
|
final scheduleMode = deviceStatus.scheduleMode;
|
||||||
|
final isCountdown = scheduleMode == ScheduleModes.countdown;
|
||||||
|
final isInching = scheduleMode == ScheduleModes.inching;
|
||||||
|
|
||||||
|
Duration? countdownRemaining;
|
||||||
|
var isCountdownActive = false;
|
||||||
|
var isInchingActive = false;
|
||||||
|
|
||||||
|
if (isCountdown) {
|
||||||
|
countdownRemaining = Duration(
|
||||||
|
hours: deviceStatus.countdownHours,
|
||||||
|
minutes: deviceStatus.countdownMinutes,
|
||||||
|
);
|
||||||
|
isCountdownActive = countdownRemaining > Duration.zero;
|
||||||
|
} else if (isInching) {
|
||||||
|
isInchingActive = Duration(
|
||||||
|
hours: deviceStatus.inchingHours,
|
||||||
|
minutes: deviceStatus.inchingMinutes,
|
||||||
|
) >
|
||||||
|
Duration.zero;
|
||||||
|
}
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
final currentState = state as ScheduleLoaded;
|
||||||
|
emit(currentState.copyWith(
|
||||||
|
scheduleMode: scheduleMode,
|
||||||
|
countdownHours: deviceStatus.countdownHours,
|
||||||
|
countdownMinutes: deviceStatus.countdownMinutes,
|
||||||
|
inchingHours: deviceStatus.inchingHours,
|
||||||
|
inchingMinutes: deviceStatus.inchingMinutes,
|
||||||
|
isCountdownActive: isCountdownActive,
|
||||||
|
isInchingActive: isInchingActive,
|
||||||
|
countdownRemaining: countdownRemaining ?? Duration.zero,
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
emit(ScheduleLoaded(
|
||||||
|
schedules: const [],
|
||||||
|
selectedTime: null,
|
||||||
|
selectedDays: List.filled(7, false),
|
||||||
|
functionOn: false,
|
||||||
|
isEditing: false,
|
||||||
|
deviceId: deviceId,
|
||||||
|
scheduleMode: scheduleMode,
|
||||||
|
countdownHours: deviceStatus.countdownHours,
|
||||||
|
countdownMinutes: deviceStatus.countdownMinutes,
|
||||||
|
inchingHours: deviceStatus.inchingHours,
|
||||||
|
inchingMinutes: deviceStatus.inchingMinutes,
|
||||||
|
isCountdownActive: isCountdownActive,
|
||||||
|
isInchingActive: isInchingActive,
|
||||||
|
countdownRemaining: countdownRemaining ?? Duration.zero,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isCountdownActive && countdownRemaining != null) {
|
||||||
|
_startCountdownTimer(emit, countdownRemaining);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
emit(ScheduleError('Failed to fetch device status: $e'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,221 @@
|
|||||||
|
part of 'schedule_bloc.dart';
|
||||||
|
|
||||||
|
abstract class ScheduleEvent extends Equatable {
|
||||||
|
const ScheduleEvent();
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleInitializeAddEvent extends ScheduleEvent {
|
||||||
|
final bool isEditing;
|
||||||
|
final ScheduleModes scheduleMode;
|
||||||
|
final TimeOfDay? selectedTime;
|
||||||
|
final List<bool>? selectedDays;
|
||||||
|
final bool? functionOn;
|
||||||
|
|
||||||
|
const ScheduleInitializeAddEvent({
|
||||||
|
required this.isEditing,
|
||||||
|
required this.scheduleMode,
|
||||||
|
this.selectedTime,
|
||||||
|
this.selectedDays,
|
||||||
|
this.functionOn,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object?> get props => [
|
||||||
|
isEditing,
|
||||||
|
scheduleMode,
|
||||||
|
selectedTime,
|
||||||
|
selectedDays,
|
||||||
|
functionOn,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleUpdateSelectedTimeEvent extends ScheduleEvent {
|
||||||
|
final TimeOfDay selectedTime;
|
||||||
|
|
||||||
|
const ScheduleUpdateSelectedTimeEvent(this.selectedTime);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [selectedTime];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleUpdateSelectedDayEvent extends ScheduleEvent {
|
||||||
|
final int index;
|
||||||
|
final bool value;
|
||||||
|
|
||||||
|
const ScheduleUpdateSelectedDayEvent(this.index, this.value);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [index, value];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleUpdateFunctionOnEvent extends ScheduleEvent {
|
||||||
|
final bool isOn;
|
||||||
|
|
||||||
|
const ScheduleUpdateFunctionOnEvent(this.isOn);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [isOn];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleGetEvent extends ScheduleEvent {
|
||||||
|
final String category;
|
||||||
|
|
||||||
|
const ScheduleGetEvent({required this.category});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [category];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleAddEvent extends ScheduleEvent {
|
||||||
|
final String category;
|
||||||
|
final String time;
|
||||||
|
final List<String> selectedDays;
|
||||||
|
final bool functionOn;
|
||||||
|
|
||||||
|
const ScheduleAddEvent({
|
||||||
|
required this.category,
|
||||||
|
required this.time,
|
||||||
|
required this.selectedDays,
|
||||||
|
required this.functionOn,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [category, time, selectedDays, functionOn];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleEditEvent extends ScheduleEvent {
|
||||||
|
final String scheduleId;
|
||||||
|
final String category;
|
||||||
|
final String time;
|
||||||
|
final List<String> selectedDays;
|
||||||
|
final bool functionOn;
|
||||||
|
|
||||||
|
const ScheduleEditEvent({
|
||||||
|
required this.scheduleId,
|
||||||
|
required this.category,
|
||||||
|
required this.time,
|
||||||
|
required this.selectedDays,
|
||||||
|
required this.functionOn,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [
|
||||||
|
scheduleId,
|
||||||
|
category,
|
||||||
|
time,
|
||||||
|
selectedDays,
|
||||||
|
functionOn,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleDeleteEvent extends ScheduleEvent {
|
||||||
|
final String scheduleId;
|
||||||
|
|
||||||
|
const ScheduleDeleteEvent(this.scheduleId);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [scheduleId];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleUpdateEntryEvent extends ScheduleEvent {
|
||||||
|
final String scheduleId;
|
||||||
|
final bool functionOn;
|
||||||
|
final bool enable;
|
||||||
|
|
||||||
|
const ScheduleUpdateEntryEvent({
|
||||||
|
required this.scheduleId,
|
||||||
|
required this.functionOn,
|
||||||
|
required this.enable,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [scheduleId, functionOn, enable];
|
||||||
|
}
|
||||||
|
|
||||||
|
class UpdateScheduleModeEvent extends ScheduleEvent {
|
||||||
|
final ScheduleModes scheduleMode;
|
||||||
|
|
||||||
|
const UpdateScheduleModeEvent({required this.scheduleMode});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [scheduleMode];
|
||||||
|
}
|
||||||
|
|
||||||
|
class UpdateCountdownTimeEvent extends ScheduleEvent {
|
||||||
|
final int hours;
|
||||||
|
final int minutes;
|
||||||
|
|
||||||
|
const UpdateCountdownTimeEvent({
|
||||||
|
required this.hours,
|
||||||
|
required this.minutes,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [hours, minutes];
|
||||||
|
}
|
||||||
|
|
||||||
|
class UpdateInchingTimeEvent extends ScheduleEvent {
|
||||||
|
final int hours;
|
||||||
|
final int minutes;
|
||||||
|
|
||||||
|
const UpdateInchingTimeEvent({
|
||||||
|
required this.hours,
|
||||||
|
required this.minutes,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [hours, minutes];
|
||||||
|
}
|
||||||
|
|
||||||
|
class StartScheduleEvent extends ScheduleEvent {
|
||||||
|
final ScheduleModes mode;
|
||||||
|
final int hours;
|
||||||
|
final int minutes;
|
||||||
|
|
||||||
|
const StartScheduleEvent({
|
||||||
|
required this.mode,
|
||||||
|
required this.hours,
|
||||||
|
required this.minutes,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object?> get props => [mode, hours, minutes];
|
||||||
|
}
|
||||||
|
|
||||||
|
class StopScheduleEvent extends ScheduleEvent {
|
||||||
|
final ScheduleModes mode;
|
||||||
|
final String deviceId;
|
||||||
|
|
||||||
|
const StopScheduleEvent({
|
||||||
|
required this.mode,
|
||||||
|
required this.deviceId,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object?> get props => [mode, deviceId];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleDecrementCountdownEvent extends ScheduleEvent {
|
||||||
|
const ScheduleDecrementCountdownEvent();
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleFetchStatusEvent extends ScheduleEvent {
|
||||||
|
final String deviceId;
|
||||||
|
|
||||||
|
const ScheduleFetchStatusEvent(this.deviceId);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [deviceId];
|
||||||
|
}
|
||||||
|
|
||||||
|
class DeleteScheduleEvent extends ScheduleEvent {
|
||||||
|
final String scheduleId;
|
||||||
|
|
||||||
|
const DeleteScheduleEvent(this.scheduleId);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [scheduleId];
|
||||||
|
}
|
@ -0,0 +1,109 @@
|
|||||||
|
part of 'schedule_bloc.dart';
|
||||||
|
|
||||||
|
abstract class ScheduleState extends Equatable {
|
||||||
|
const ScheduleState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleInitial extends ScheduleState {
|
||||||
|
@override
|
||||||
|
List<Object> get props => [];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleLoading extends ScheduleState {
|
||||||
|
@override
|
||||||
|
List<Object> get props => [];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleLoaded extends ScheduleState {
|
||||||
|
final List<ScheduleModel> schedules;
|
||||||
|
final TimeOfDay? selectedTime;
|
||||||
|
final List<bool> selectedDays;
|
||||||
|
final bool functionOn;
|
||||||
|
final bool isEditing;
|
||||||
|
final String deviceId;
|
||||||
|
final int countdownHours;
|
||||||
|
final int countdownMinutes;
|
||||||
|
final bool isCountdownActive;
|
||||||
|
final int inchingHours;
|
||||||
|
final int inchingMinutes;
|
||||||
|
final bool isInchingActive;
|
||||||
|
final ScheduleModes scheduleMode;
|
||||||
|
final Duration? countdownRemaining;
|
||||||
|
|
||||||
|
const ScheduleLoaded({
|
||||||
|
required this.schedules,
|
||||||
|
this.selectedTime,
|
||||||
|
required this.selectedDays,
|
||||||
|
required this.functionOn,
|
||||||
|
required this.isEditing,
|
||||||
|
required this.deviceId,
|
||||||
|
this.countdownHours = 0,
|
||||||
|
this.countdownMinutes = 0,
|
||||||
|
this.isCountdownActive = false,
|
||||||
|
this.inchingHours = 0,
|
||||||
|
this.inchingMinutes = 0,
|
||||||
|
this.isInchingActive = false,
|
||||||
|
this.scheduleMode = ScheduleModes.countdown,
|
||||||
|
this.countdownRemaining,
|
||||||
|
});
|
||||||
|
|
||||||
|
ScheduleLoaded copyWith({
|
||||||
|
List<ScheduleModel>? schedules,
|
||||||
|
TimeOfDay? selectedTime,
|
||||||
|
List<bool>? selectedDays,
|
||||||
|
bool? functionOn,
|
||||||
|
bool? isEditing,
|
||||||
|
int? countdownHours,
|
||||||
|
int? countdownMinutes,
|
||||||
|
bool? isCountdownActive,
|
||||||
|
int? inchingHours,
|
||||||
|
int? inchingMinutes,
|
||||||
|
bool? isInchingActive,
|
||||||
|
ScheduleModes? scheduleMode,
|
||||||
|
Duration? countdownRemaining,
|
||||||
|
}) {
|
||||||
|
return ScheduleLoaded(
|
||||||
|
schedules: schedules ?? this.schedules,
|
||||||
|
selectedTime: selectedTime ?? this.selectedTime,
|
||||||
|
selectedDays: selectedDays ?? this.selectedDays,
|
||||||
|
functionOn: functionOn ?? this.functionOn,
|
||||||
|
isEditing: isEditing ?? this.isEditing,
|
||||||
|
deviceId: deviceId,
|
||||||
|
countdownHours: countdownHours ?? this.countdownHours,
|
||||||
|
countdownMinutes: countdownMinutes ?? this.countdownMinutes,
|
||||||
|
isCountdownActive: isCountdownActive ?? this.isCountdownActive,
|
||||||
|
inchingHours: inchingHours ?? this.inchingHours,
|
||||||
|
inchingMinutes: inchingMinutes ?? this.inchingMinutes,
|
||||||
|
isInchingActive: isInchingActive ?? this.isInchingActive,
|
||||||
|
scheduleMode: scheduleMode ?? this.scheduleMode,
|
||||||
|
countdownRemaining: countdownRemaining ?? this.countdownRemaining,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object?> get props => [
|
||||||
|
schedules,
|
||||||
|
selectedTime,
|
||||||
|
selectedDays,
|
||||||
|
functionOn,
|
||||||
|
isEditing,
|
||||||
|
deviceId,
|
||||||
|
countdownHours,
|
||||||
|
countdownMinutes,
|
||||||
|
isCountdownActive,
|
||||||
|
inchingHours,
|
||||||
|
inchingMinutes,
|
||||||
|
isInchingActive,
|
||||||
|
scheduleMode,
|
||||||
|
countdownRemaining,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
class ScheduleError extends ScheduleState {
|
||||||
|
final String error;
|
||||||
|
|
||||||
|
const ScheduleError(this.error);
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Object> get props => [error];
|
||||||
|
}
|
@ -1,8 +1,9 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
|
||||||
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
||||||
class CountdownModeButtons extends StatelessWidget {
|
class CountdownModeButtons extends StatelessWidget {
|
||||||
@ -38,14 +39,10 @@ class CountdownModeButtons extends StatelessWidget {
|
|||||||
? DefaultButton(
|
? DefaultButton(
|
||||||
height: 40,
|
height: 40,
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
context
|
context.read<ScheduleBloc>().add(
|
||||||
.read<WaterHeaterBloc>()
|
StopScheduleEvent(
|
||||||
.add(StopScheduleEvent(deviceId));
|
mode: ScheduleModes.countdown,
|
||||||
context.read<WaterHeaterBloc>().add(
|
|
||||||
ToggleWaterHeaterEvent(
|
|
||||||
deviceId: deviceId,
|
deviceId: deviceId,
|
||||||
code: 'countdown_1',
|
|
||||||
value: 0,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -55,12 +52,11 @@ class CountdownModeButtons extends StatelessWidget {
|
|||||||
: DefaultButton(
|
: DefaultButton(
|
||||||
height: 40,
|
height: 40,
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
context.read<WaterHeaterBloc>().add(
|
context.read<ScheduleBloc>().add(
|
||||||
ToggleWaterHeaterEvent(
|
StartScheduleEvent(
|
||||||
deviceId: deviceId,
|
mode: ScheduleModes.countdown,
|
||||||
code: 'countdown_1',
|
hours: hours,
|
||||||
value: Duration(hours: hours, minutes: minutes)
|
minutes: minutes,
|
||||||
.inSeconds,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
@ -0,0 +1,185 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
||||||
|
class CountdownInchingView extends StatefulWidget {
|
||||||
|
const CountdownInchingView({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CountdownInchingView> createState() => _CountdownInchingViewState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CountdownInchingViewState extends State<CountdownInchingView> {
|
||||||
|
late FixedExtentScrollController _hoursController;
|
||||||
|
late FixedExtentScrollController _minutesController;
|
||||||
|
|
||||||
|
int _lastHours = -1;
|
||||||
|
int _lastMinutes = -1;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_hoursController = FixedExtentScrollController();
|
||||||
|
_minutesController = FixedExtentScrollController();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_hoursController.dispose();
|
||||||
|
_minutesController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateControllers(int displayHours, int displayMinutes) {
|
||||||
|
if (_lastHours != displayHours) {
|
||||||
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
|
if (_hoursController.hasClients) {
|
||||||
|
_hoursController.jumpToItem(displayHours);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
_lastHours = displayHours;
|
||||||
|
}
|
||||||
|
if (_lastMinutes != displayMinutes) {
|
||||||
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
|
if (_minutesController.hasClients) {
|
||||||
|
_minutesController.jumpToItem(displayMinutes);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
_lastMinutes = displayMinutes;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return BlocBuilder<ScheduleBloc, ScheduleState>(
|
||||||
|
builder: (context, state) {
|
||||||
|
if (state is! ScheduleLoaded) return const SizedBox.shrink();
|
||||||
|
|
||||||
|
final isCountDown = state.scheduleMode == ScheduleModes.countdown;
|
||||||
|
final isActive =
|
||||||
|
isCountDown ? state.isCountdownActive : state.isInchingActive;
|
||||||
|
final displayHours = isActive && state.countdownRemaining != null
|
||||||
|
? state.countdownRemaining!.inHours
|
||||||
|
: (isCountDown ? state.countdownHours : state.inchingHours);
|
||||||
|
final displayMinutes = isActive && state.countdownRemaining != null
|
||||||
|
? state.countdownRemaining!.inMinutes.remainder(60)
|
||||||
|
: (isCountDown ? state.countdownMinutes : state.inchingMinutes);
|
||||||
|
|
||||||
|
_updateControllers(displayHours, displayMinutes);
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
isCountDown ? 'Countdown:' : 'Inching:',
|
||||||
|
style: context.textTheme.bodySmall!.copyWith(
|
||||||
|
fontSize: 13,
|
||||||
|
color: ColorsManager.grayColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Visibility(
|
||||||
|
visible: !isCountDown,
|
||||||
|
child: const Text(
|
||||||
|
'Once enabled this feature, each time the device is turned on, '
|
||||||
|
'it will automatically turn off after a preset time.',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
_buildPickerColumn(
|
||||||
|
context,
|
||||||
|
'h',
|
||||||
|
displayHours,
|
||||||
|
100,
|
||||||
|
_hoursController,
|
||||||
|
(value) {
|
||||||
|
if (!isActive) {
|
||||||
|
context.read<ScheduleBloc>().add(UpdateCountdownTimeEvent(
|
||||||
|
hours: value, minutes: displayMinutes));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isActive: isActive,
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
_buildPickerColumn(
|
||||||
|
context,
|
||||||
|
'm',
|
||||||
|
displayMinutes,
|
||||||
|
60,
|
||||||
|
_minutesController,
|
||||||
|
(value) {
|
||||||
|
if (!isActive) {
|
||||||
|
context.read<ScheduleBloc>().add(UpdateCountdownTimeEvent(
|
||||||
|
hours: displayHours, minutes: value));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isActive: isActive,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildPickerColumn(
|
||||||
|
BuildContext context,
|
||||||
|
String label,
|
||||||
|
int initialValue,
|
||||||
|
int itemCount,
|
||||||
|
FixedExtentScrollController controller,
|
||||||
|
ValueChanged<int> onSelected, {
|
||||||
|
required bool isActive,
|
||||||
|
}) {
|
||||||
|
return Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
height: 40,
|
||||||
|
width: 80,
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: ColorsManager.boxColor,
|
||||||
|
borderRadius: BorderRadius.circular(8),
|
||||||
|
),
|
||||||
|
child: ListWheelScrollView.useDelegate(
|
||||||
|
controller: controller,
|
||||||
|
itemExtent: 40.0,
|
||||||
|
physics: isActive
|
||||||
|
? const NeverScrollableScrollPhysics()
|
||||||
|
: const FixedExtentScrollPhysics(),
|
||||||
|
onSelectedItemChanged: isActive ? null : onSelected,
|
||||||
|
childDelegate: ListWheelChildBuilderDelegate(
|
||||||
|
builder: (context, index) {
|
||||||
|
return Center(
|
||||||
|
child: Text(
|
||||||
|
index.toString().padLeft(2, '0'),
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 24,
|
||||||
|
color: isActive ? ColorsManager.grayColor : Colors.black,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
childCount: itemCount,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Text(
|
||||||
|
label,
|
||||||
|
style: const TextStyle(
|
||||||
|
color: ColorsManager.grayColor,
|
||||||
|
fontSize: 18,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -1,8 +1,11 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
|
||||||
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart'
|
||||||
|
hide StopScheduleEvent;
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
||||||
class InchingModeButtons extends StatelessWidget {
|
class InchingModeButtons extends StatelessWidget {
|
||||||
@ -38,15 +41,9 @@ class InchingModeButtons extends StatelessWidget {
|
|||||||
? DefaultButton(
|
? DefaultButton(
|
||||||
height: 40,
|
height: 40,
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
context
|
context.read<ScheduleBloc>().add(
|
||||||
.read<WaterHeaterBloc>()
|
StopScheduleEvent(
|
||||||
.add(StopScheduleEvent(deviceId));
|
deviceId: deviceId, mode: ScheduleModes.inching),
|
||||||
context.read<WaterHeaterBloc>().add(
|
|
||||||
ToggleWaterHeaterEvent(
|
|
||||||
deviceId: deviceId,
|
|
||||||
code: 'switch_inching',
|
|
||||||
value: 0,
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
backgroundColor: Colors.red,
|
backgroundColor: Colors.red,
|
@ -0,0 +1,107 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/count_down_button.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/count_down_inching_view.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/inching_mode_buttons.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedule_header.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedule_managment_ui.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedule_mode_buttons.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedule_mode_selector.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/helper/add_schedule_dialog_helper.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
|
||||||
|
class BuildScheduleView extends StatelessWidget {
|
||||||
|
const BuildScheduleView({super.key, required this.deviceUuid});
|
||||||
|
final String deviceUuid;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return BlocProvider(
|
||||||
|
create: (_) => ScheduleBloc(
|
||||||
|
deviceId: deviceUuid,
|
||||||
|
)
|
||||||
|
..add(const ScheduleGetEvent(category: "switch_1"))
|
||||||
|
..add(ScheduleFetchStatusEvent(deviceUuid)),
|
||||||
|
child: Dialog(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
insetPadding: const EdgeInsets.all(20),
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
),
|
||||||
|
child: SizedBox(
|
||||||
|
width: 700,
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
child: Padding(
|
||||||
|
padding:
|
||||||
|
const EdgeInsets.symmetric(horizontal: 40.0, vertical: 20),
|
||||||
|
child: BlocBuilder<ScheduleBloc, ScheduleState>(
|
||||||
|
builder: (context, state) {
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
const ScheduleHeader(),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
ScheduleModeSelector(
|
||||||
|
currentMode: state.scheduleMode,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
if (state.scheduleMode == ScheduleModes.schedule)
|
||||||
|
ScheduleManagementUI(
|
||||||
|
deviceUuid: deviceUuid,
|
||||||
|
onAddSchedule: () async {
|
||||||
|
final entry = await ScheduleDialogHelper
|
||||||
|
.showAddScheduleDialog(
|
||||||
|
context,
|
||||||
|
schedule: null,
|
||||||
|
isEdit: false,
|
||||||
|
);
|
||||||
|
if (entry != null) {
|
||||||
|
context.read<ScheduleBloc>().add(
|
||||||
|
ScheduleAddEvent(
|
||||||
|
category: entry.category,
|
||||||
|
time: entry.time,
|
||||||
|
functionOn: entry.function.value,
|
||||||
|
selectedDays: entry.days,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
if (state.scheduleMode == ScheduleModes.countdown ||
|
||||||
|
state.scheduleMode == ScheduleModes.inching)
|
||||||
|
const CountdownInchingView(),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
if (state.scheduleMode == ScheduleModes.countdown)
|
||||||
|
CountdownModeButtons(
|
||||||
|
isActive: state.isCountdownActive,
|
||||||
|
deviceId: deviceUuid,
|
||||||
|
hours: state.countdownHours,
|
||||||
|
minutes: state.countdownMinutes,
|
||||||
|
),
|
||||||
|
if (state.scheduleMode == ScheduleModes.inching)
|
||||||
|
InchingModeButtons(
|
||||||
|
isActive: state.isInchingActive,
|
||||||
|
deviceId: deviceUuid,
|
||||||
|
hours: state.inchingHours,
|
||||||
|
minutes: state.inchingMinutes,
|
||||||
|
),
|
||||||
|
if (state.scheduleMode != ScheduleModes.countdown &&
|
||||||
|
state.scheduleMode != ScheduleModes.inching)
|
||||||
|
ScheduleModeButtons(
|
||||||
|
onSave: () => Navigator.pop(context),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return const Center(child: CircularProgressIndicator());
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -1,17 +1,16 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedule_table.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedule_table.dart';
|
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
||||||
class ScheduleManagementUI extends StatelessWidget {
|
class ScheduleManagementUI extends StatelessWidget {
|
||||||
final WaterHeaterDeviceStatusLoaded state;
|
final String deviceUuid;
|
||||||
final Function onAddSchedule;
|
final VoidCallback onAddSchedule;
|
||||||
|
|
||||||
const ScheduleManagementUI({
|
const ScheduleManagementUI({
|
||||||
super.key,
|
super.key,
|
||||||
required this.state,
|
required this.deviceUuid,
|
||||||
required this.onAddSchedule,
|
required this.onAddSchedule,
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -28,7 +27,7 @@ class ScheduleManagementUI extends StatelessWidget {
|
|||||||
padding: 2,
|
padding: 2,
|
||||||
backgroundColor: ColorsManager.graysColor,
|
backgroundColor: ColorsManager.graysColor,
|
||||||
borderRadius: 15,
|
borderRadius: 15,
|
||||||
onPressed: () => onAddSchedule(),
|
onPressed: onAddSchedule,
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
const Icon(Icons.add, color: ColorsManager.primaryColor),
|
const Icon(Icons.add, color: ColorsManager.primaryColor),
|
||||||
@ -43,7 +42,7 @@ class ScheduleManagementUI extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
ScheduleTableWidget(state: state),
|
ScheduleTableWidget(deviceUuid: deviceUuid),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
@ -0,0 +1,87 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
|
|
||||||
|
class ScheduleModeSelector extends StatelessWidget {
|
||||||
|
final ScheduleModes currentMode;
|
||||||
|
|
||||||
|
const ScheduleModeSelector({
|
||||||
|
super.key,
|
||||||
|
required this.currentMode,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final currentMode = context.select<ScheduleBloc, ScheduleModes>(
|
||||||
|
(bloc) => bloc.state is ScheduleLoaded &&
|
||||||
|
(bloc.state as ScheduleLoaded).scheduleMode != null
|
||||||
|
? (bloc.state as ScheduleLoaded).scheduleMode
|
||||||
|
: ScheduleModes.schedule,
|
||||||
|
);
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
'Type:',
|
||||||
|
style: context.textTheme.bodySmall!.copyWith(
|
||||||
|
fontSize: 13,
|
||||||
|
color: ColorsManager.grayColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
|
children: [
|
||||||
|
_buildRadioTile(
|
||||||
|
context, 'Countdown', ScheduleModes.countdown, currentMode),
|
||||||
|
_buildRadioTile(
|
||||||
|
context, 'Schedule', ScheduleModes.schedule, currentMode),
|
||||||
|
_buildRadioTile(
|
||||||
|
context, 'Circulate', ScheduleModes.circulate, currentMode),
|
||||||
|
_buildRadioTile(
|
||||||
|
context, 'Inching', ScheduleModes.inching, currentMode),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildRadioTile(
|
||||||
|
BuildContext context,
|
||||||
|
String label,
|
||||||
|
ScheduleModes mode,
|
||||||
|
ScheduleModes currentMode,
|
||||||
|
) {
|
||||||
|
return Flexible(
|
||||||
|
child: ListTile(
|
||||||
|
contentPadding: EdgeInsets.zero,
|
||||||
|
title: Text(
|
||||||
|
label,
|
||||||
|
style: context.textTheme.bodySmall!.copyWith(
|
||||||
|
fontSize: 13,
|
||||||
|
color: ColorsManager.blackColor,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
leading: Radio<ScheduleModes>(
|
||||||
|
value: mode,
|
||||||
|
groupValue: currentMode,
|
||||||
|
onChanged: (ScheduleModes? value) {
|
||||||
|
if (value != null) {
|
||||||
|
context.read<ScheduleBloc>().add(
|
||||||
|
UpdateScheduleModeEvent(scheduleMode: value),
|
||||||
|
);
|
||||||
|
if (value == ScheduleModes.schedule) {
|
||||||
|
context.read<ScheduleBloc>().add(
|
||||||
|
const ScheduleGetEvent(category: 'switch_1'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -1,23 +1,38 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:flutter_svg/flutter_svg.dart';
|
import 'package:flutter_svg/svg.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/schedule_device/bloc/schedule_bloc.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/helper/add_schedule_dialog_helper.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_entry.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
import 'package:syncrow_web/utils/constants/assets.dart';
|
import 'package:syncrow_web/utils/constants/assets.dart';
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
|
||||||
import 'package:syncrow_web/utils/format_date_time.dart';
|
import 'package:syncrow_web/utils/format_date_time.dart';
|
||||||
|
|
||||||
import '../helper/add_schedule_dialog_helper.dart';
|
|
||||||
|
|
||||||
class ScheduleTableWidget extends StatelessWidget {
|
class ScheduleTableWidget extends StatelessWidget {
|
||||||
final WaterHeaterDeviceStatusLoaded state;
|
final String deviceUuid;
|
||||||
|
final String category;
|
||||||
|
|
||||||
const ScheduleTableWidget({
|
const ScheduleTableWidget({
|
||||||
super.key,
|
super.key,
|
||||||
required this.state,
|
required this.deviceUuid,
|
||||||
|
this.category = 'switch_1',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return BlocProvider(
|
||||||
|
create: (_) => ScheduleBloc(
|
||||||
|
deviceId: deviceUuid,
|
||||||
|
)..add(ScheduleGetEvent(category: category)),
|
||||||
|
child: _ScheduleTableView(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ScheduleTableView extends StatelessWidget {
|
||||||
|
const _ScheduleTableView();
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Column(
|
return Column(
|
||||||
@ -47,17 +62,17 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
BlocBuilder<WaterHeaterBloc, WaterHeaterState>(
|
BlocBuilder<ScheduleBloc, ScheduleState>(
|
||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
if (state is ScheduleLoadingState) {
|
if (state is ScheduleLoading) {
|
||||||
return const SizedBox(
|
return const SizedBox(
|
||||||
height: 200,
|
height: 200,
|
||||||
child: Center(child: CircularProgressIndicator()));
|
child: Center(child: CircularProgressIndicator()));
|
||||||
}
|
}
|
||||||
if (state is WaterHeaterDeviceStatusLoaded &&
|
if (state is ScheduleLoaded && state.schedules.isEmpty) {
|
||||||
state.schedules.isEmpty) {
|
|
||||||
return _buildEmptyState(context);
|
return _buildEmptyState(context);
|
||||||
} else if (state is WaterHeaterDeviceStatusLoaded) {
|
}
|
||||||
|
if (state is ScheduleLoaded) {
|
||||||
return Container(
|
return Container(
|
||||||
height: 200,
|
height: 200,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
@ -66,11 +81,12 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
bottomLeft: Radius.circular(20),
|
bottomLeft: Radius.circular(20),
|
||||||
bottomRight: Radius.circular(20)),
|
bottomRight: Radius.circular(20)),
|
||||||
),
|
),
|
||||||
child: _buildTableBody(state, context));
|
child: _buildTableBody(state.schedules, context));
|
||||||
}
|
}
|
||||||
return const SizedBox(
|
if (state is ScheduleError) {
|
||||||
height: 200,
|
return Center(child: Text(state.error));
|
||||||
);
|
}
|
||||||
|
return const SizedBox(height: 200);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@ -95,10 +111,10 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
padding: const EdgeInsets.all(8.0),
|
padding: const EdgeInsets.all(8.0),
|
||||||
child: Text(
|
child: Text(
|
||||||
'No schedules added yet',
|
'No schedules added yet',
|
||||||
style: context.textTheme.bodySmall!.copyWith(
|
style: Theme.of(context).textTheme.bodySmall!.copyWith(
|
||||||
fontSize: 13,
|
fontSize: 13,
|
||||||
color: ColorsManager.grayColor,
|
color: ColorsManager.grayColor,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@ -107,8 +123,7 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildTableBody(
|
Widget _buildTableBody(List<ScheduleModel> schedules, BuildContext context) {
|
||||||
WaterHeaterDeviceStatusLoaded state, BuildContext context) {
|
|
||||||
return SizedBox(
|
return SizedBox(
|
||||||
height: 200,
|
height: 200,
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
@ -116,8 +131,8 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
border: TableBorder.all(color: ColorsManager.graysColor),
|
border: TableBorder.all(color: ColorsManager.graysColor),
|
||||||
defaultVerticalAlignment: TableCellVerticalAlignment.middle,
|
defaultVerticalAlignment: TableCellVerticalAlignment.middle,
|
||||||
children: [
|
children: [
|
||||||
for (int i = 0; i < state.schedules.length; i++)
|
for (int i = 0; i < schedules.length; i++)
|
||||||
_buildScheduleRow(state.schedules[i], i, context, state),
|
_buildScheduleRow(schedules[i], i, context),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -139,20 +154,23 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
TableRow _buildScheduleRow(ScheduleModel schedule, int index,
|
TableRow _buildScheduleRow(
|
||||||
BuildContext context, WaterHeaterDeviceStatusLoaded state) {
|
ScheduleModel schedule, int index, BuildContext context) {
|
||||||
return TableRow(
|
return TableRow(
|
||||||
children: [
|
children: [
|
||||||
Center(
|
Center(
|
||||||
child: GestureDetector(
|
child: GestureDetector(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEntryEvent(
|
///TODO: Implement toggle functionality
|
||||||
index: index,
|
|
||||||
enable: !schedule.enable,
|
// Toggle enabled state using ScheduleBloc
|
||||||
scheduleId: schedule.scheduleId,
|
// context.read<ScheduleBloc>().add(
|
||||||
deviceId: state.status.uuid,
|
// UpdateScheduleEvent(
|
||||||
functionOn: schedule.function.value,
|
// scheduleId: schedule.scheduleId,
|
||||||
));
|
// functionOn: schedule.function.value,
|
||||||
|
// enable: !schedule.enable,
|
||||||
|
// ),
|
||||||
|
// );
|
||||||
},
|
},
|
||||||
child: SizedBox(
|
child: SizedBox(
|
||||||
width: 24,
|
width: 24,
|
||||||
@ -179,26 +197,46 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
TextButton(
|
TextButton(
|
||||||
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
ScheduleDialogHelper.showAddScheduleDialog(context,
|
ScheduleDialogHelper.showAddScheduleDialog(
|
||||||
schedule: schedule, index: index, isEdit: true);
|
context,
|
||||||
|
schedule: ScheduleEntry.fromScheduleModel(schedule),
|
||||||
|
isEdit: true,
|
||||||
|
).then((updatedSchedule) {
|
||||||
|
print('updatedSchedule : $updatedSchedule');
|
||||||
|
if (updatedSchedule != null) {
|
||||||
|
context.read<ScheduleBloc>().add(
|
||||||
|
ScheduleEditEvent(
|
||||||
|
scheduleId: schedule.scheduleId,
|
||||||
|
category: schedule.category,
|
||||||
|
time: updatedSchedule.time,
|
||||||
|
functionOn: updatedSchedule.function.value,
|
||||||
|
selectedDays: updatedSchedule.days),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
},
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
'Edit',
|
'Edit',
|
||||||
style: context.textTheme.bodySmall!
|
style: Theme.of(context)
|
||||||
|
.textTheme
|
||||||
|
.bodySmall!
|
||||||
.copyWith(color: ColorsManager.blueColor),
|
.copyWith(color: ColorsManager.blueColor),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
TextButton(
|
TextButton(
|
||||||
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
style: TextButton.styleFrom(padding: EdgeInsets.zero),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
context.read<WaterHeaterBloc>().add(DeleteScheduleEvent(
|
context.read<ScheduleBloc>().add(
|
||||||
index: index,
|
DeleteScheduleEvent(
|
||||||
scheduleId: schedule.scheduleId,
|
schedule.scheduleId,
|
||||||
));
|
),
|
||||||
|
);
|
||||||
},
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
'Delete',
|
'Delete',
|
||||||
style: context.textTheme.bodySmall!
|
style: Theme.of(context)
|
||||||
|
.textTheme
|
||||||
|
.bodySmall!
|
||||||
.copyWith(color: ColorsManager.blueColor),
|
.copyWith(color: ColorsManager.blueColor),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -210,13 +248,15 @@ class ScheduleTableWidget extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
String _getSelectedDays(List<bool> selectedDays) {
|
String _getSelectedDays(List<bool> selectedDays) {
|
||||||
final days = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
// Use the same order as in ScheduleDialogHelper
|
||||||
List<String> selectedDaysStr = [];
|
const days = ScheduleDialogHelper.allDays;
|
||||||
for (int i = 0; i < selectedDays.length; i++) {
|
return selectedDays
|
||||||
if (selectedDays[i]) {
|
.asMap()
|
||||||
selectedDaysStr.add(days[i]);
|
.entries
|
||||||
}
|
.where((entry) => entry.value)
|
||||||
}
|
.map((entry) => days[entry.key])
|
||||||
return selectedDaysStr.join(', ');
|
.join(', ');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Removed allDays from here as it is now in ScheduleDialogHelper
|
||||||
}
|
}
|
@ -1,240 +1,210 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_entry.dart';
|
||||||
import 'package:syncrow_web/pages/common/buttons/default_button.dart';
|
import 'package:syncrow_web/pages/device_managment/all_devices/models/device_status.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
|
||||||
|
|
||||||
class ScheduleDialogHelper {
|
class ScheduleDialogHelper {
|
||||||
static void showAddScheduleDialog(BuildContext context, {ScheduleModel? schedule, int? index, bool? isEdit}) {
|
static const List<String> allDays = [
|
||||||
final bloc = context.read<WaterHeaterBloc>();
|
'Sun',
|
||||||
|
'Mon',
|
||||||
|
'Tue',
|
||||||
|
'Wed',
|
||||||
|
'Thu',
|
||||||
|
'Fri',
|
||||||
|
'Sat'
|
||||||
|
];
|
||||||
|
|
||||||
if (schedule == null) {
|
static Future<ScheduleEntry?> showAddScheduleDialog(
|
||||||
bloc.add((const UpdateSelectedTimeEvent(null)));
|
BuildContext context, {
|
||||||
bloc.add(InitializeAddScheduleEvent(
|
ScheduleEntry? schedule,
|
||||||
selectedTime: null,
|
bool isEdit = false,
|
||||||
selectedDays: List.filled(7, false),
|
}) {
|
||||||
functionOn: false,
|
final initialTime = schedule != null
|
||||||
isEditing: false,
|
? _convertStringToTimeOfDay(schedule.time)
|
||||||
));
|
: TimeOfDay.now();
|
||||||
} else {
|
final initialDays = schedule != null
|
||||||
final time = _convertStringToTimeOfDay(schedule.time);
|
? _convertDaysStringToBooleans(schedule.days)
|
||||||
final selectedDays = _convertDaysStringToBooleans(schedule.days);
|
: List.filled(7, false);
|
||||||
|
bool? functionOn = schedule?.function.value ?? true;
|
||||||
|
TimeOfDay selectedTime = initialTime;
|
||||||
|
List<bool> selectedDays = List.of(initialDays);
|
||||||
|
|
||||||
bloc.add(InitializeAddScheduleEvent(
|
return showDialog<ScheduleEntry>(
|
||||||
selectedTime: time,
|
|
||||||
selectedDays: selectedDays,
|
|
||||||
functionOn: schedule.function.value,
|
|
||||||
isEditing: true,
|
|
||||||
index: index,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
context: context,
|
||||||
builder: (ctx) {
|
builder: (ctx) {
|
||||||
return BlocProvider.value(
|
return StatefulBuilder(
|
||||||
value: bloc,
|
builder: (ctx, setState) {
|
||||||
child: BlocBuilder<WaterHeaterBloc, WaterHeaterState>(
|
return AlertDialog(
|
||||||
builder: (context, state) {
|
shape: RoundedRectangleBorder(
|
||||||
if (state is WaterHeaterDeviceStatusLoaded) {
|
borderRadius: BorderRadius.circular(20),
|
||||||
return AlertDialog(
|
),
|
||||||
shape: RoundedRectangleBorder(
|
content: Column(
|
||||||
borderRadius: BorderRadius.circular(20),
|
mainAxisSize: MainAxisSize.min,
|
||||||
),
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
content: Column(
|
children: [
|
||||||
mainAxisSize: MainAxisSize.min,
|
Row(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Row(
|
const SizedBox(),
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
Text(
|
||||||
children: [
|
isEdit ? 'Edit Schedule' : 'Add Schedule',
|
||||||
const SizedBox(),
|
style: Theme.of(context).textTheme.titleLarge!.copyWith(
|
||||||
Text(
|
color: Colors.blue,
|
||||||
'Scheduling',
|
|
||||||
style: context.textTheme.titleLarge!.copyWith(
|
|
||||||
color: ColorsManager.dialogBlueTitle,
|
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
),
|
),
|
||||||
),
|
|
||||||
const SizedBox(),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
const SizedBox(height: 24),
|
const SizedBox(),
|
||||||
SizedBox(
|
|
||||||
width: 150,
|
|
||||||
height: 40,
|
|
||||||
child: DefaultButton(
|
|
||||||
padding: 8,
|
|
||||||
backgroundColor: ColorsManager.boxColor,
|
|
||||||
borderRadius: 15,
|
|
||||||
onPressed: () async {
|
|
||||||
TimeOfDay? time = await showTimePicker(
|
|
||||||
context: context,
|
|
||||||
initialTime: state.selectedTime ?? TimeOfDay.now(),
|
|
||||||
builder: (context, child) {
|
|
||||||
return Theme(
|
|
||||||
data: Theme.of(context).copyWith(
|
|
||||||
colorScheme: const ColorScheme.light(
|
|
||||||
primary: ColorsManager.primaryColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: child!,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
if (time != null) {
|
|
||||||
bloc.add(UpdateSelectedTimeEvent(time));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
state.selectedTime == null ? 'Time' : state.selectedTime!.format(context),
|
|
||||||
style: context.textTheme.bodySmall!.copyWith(
|
|
||||||
color: ColorsManager.grayColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const Icon(
|
|
||||||
Icons.access_time,
|
|
||||||
color: ColorsManager.grayColor,
|
|
||||||
size: 18,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 16),
|
|
||||||
_buildDayCheckboxes(context, state.selectedDays, isEdit: isEdit),
|
|
||||||
const SizedBox(height: 16),
|
|
||||||
_buildFunctionSwitch(context, state.functionOn, isEdit),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
actions: [
|
const SizedBox(height: 24),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
width: 200,
|
width: 150,
|
||||||
child: DefaultButton(
|
height: 40,
|
||||||
height: 40,
|
child: ElevatedButton(
|
||||||
onPressed: () {
|
style: ElevatedButton.styleFrom(
|
||||||
Navigator.pop(context);
|
backgroundColor: Colors.grey[200],
|
||||||
},
|
shape: RoundedRectangleBorder(
|
||||||
backgroundColor: ColorsManager.boxColor,
|
borderRadius: BorderRadius.circular(15),
|
||||||
child: Text(
|
|
||||||
'Cancel',
|
|
||||||
style: context.textTheme.bodyMedium,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
onPressed: () async {
|
||||||
SizedBox(
|
TimeOfDay? time = await showTimePicker(
|
||||||
width: 200,
|
context: ctx,
|
||||||
child: DefaultButton(
|
initialTime: selectedTime,
|
||||||
height: 40,
|
);
|
||||||
onPressed: () {
|
if (time != null) {
|
||||||
if (state.selectedTime != null) {
|
setState(() => selectedTime = time);
|
||||||
if (state.isEditing && index != null) {
|
}
|
||||||
bloc.add(EditWaterHeaterScheduleEvent(
|
},
|
||||||
scheduleId: schedule?.scheduleId ?? '',
|
child: Row(
|
||||||
category: 'switch_1',
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
time: state.selectedTime!,
|
children: [
|
||||||
selectedDays: state.selectedDays,
|
Text(
|
||||||
functionOn: state.functionOn,
|
selectedTime.format(context),
|
||||||
));
|
style: Theme.of(context)
|
||||||
} else {
|
.textTheme
|
||||||
bloc.add(AddScheduleEvent(
|
.bodySmall!
|
||||||
category: 'switch_1',
|
.copyWith(color: Colors.grey),
|
||||||
time: state.selectedTime!,
|
),
|
||||||
selectedDays: state.selectedDays,
|
const Icon(Icons.access_time,
|
||||||
functionOn: state.functionOn,
|
color: Colors.grey, size: 18),
|
||||||
));
|
],
|
||||||
}
|
|
||||||
Navigator.pop(context);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
backgroundColor: ColorsManager.primaryColor,
|
|
||||||
child: const Text('Save'),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
),
|
||||||
);
|
const SizedBox(height: 16),
|
||||||
}
|
_buildDayCheckboxes(ctx, selectedDays, (i, v) {
|
||||||
return const SizedBox();
|
setState(() => selectedDays[i] = v);
|
||||||
},
|
}),
|
||||||
),
|
const SizedBox(height: 16),
|
||||||
|
_buildFunctionSwitch(ctx, functionOn, (v) {
|
||||||
|
setState(() => functionOn = v);
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
SizedBox(
|
||||||
|
width: 100,
|
||||||
|
child: OutlinedButton(
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.pop(ctx, null);
|
||||||
|
},
|
||||||
|
child: const Text('Cancel'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
SizedBox(
|
||||||
|
width: 100,
|
||||||
|
child: ElevatedButton(
|
||||||
|
onPressed: () {
|
||||||
|
final entry = ScheduleEntry(
|
||||||
|
category: schedule?.category ?? 'switch_1',
|
||||||
|
time: _formatTimeOfDayToISO(selectedTime),
|
||||||
|
function: Status(code: 'switch_1', value: functionOn),
|
||||||
|
days: _convertSelectedDaysToStrings(selectedDays),
|
||||||
|
scheduleId: schedule?.scheduleId,
|
||||||
|
);
|
||||||
|
Navigator.pop(ctx, entry);
|
||||||
|
},
|
||||||
|
child: const Text('Save'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
static TimeOfDay _convertStringToTimeOfDay(String timeString) {
|
static TimeOfDay _convertStringToTimeOfDay(String iso) {
|
||||||
final regex = RegExp(r'^(\d{2}):(\d{2})$');
|
final dt = DateTime.tryParse(iso);
|
||||||
final match = regex.firstMatch(timeString);
|
if (dt != null) return TimeOfDay(hour: dt.hour, minute: dt.minute);
|
||||||
if (match != null) {
|
return const TimeOfDay(hour: 9, minute: 0);
|
||||||
final hour = int.parse(match.group(1)!);
|
|
||||||
final minute = int.parse(match.group(2)!);
|
|
||||||
return TimeOfDay(hour: hour, minute: minute);
|
|
||||||
} else {
|
|
||||||
throw const FormatException('Invalid time format');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static List<bool> _convertDaysStringToBooleans(List<String> selectedDays) {
|
static List<bool> _convertDaysStringToBooleans(List<String> selectedDays) {
|
||||||
final daysOfWeek = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
final daysOfWeek = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
||||||
List<bool> daysBoolean = List.filled(7, false);
|
return daysOfWeek
|
||||||
|
.map((d) =>
|
||||||
for (int i = 0; i < daysOfWeek.length; i++) {
|
selectedDays.map((e) => e.toLowerCase()).contains(d.toLowerCase()))
|
||||||
if (selectedDays.contains(daysOfWeek[i])) {
|
.toList();
|
||||||
daysBoolean[i] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return daysBoolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static Widget _buildDayCheckboxes(BuildContext context, List<bool> selectedDays, {bool? isEdit}) {
|
static String _formatTimeOfDayToISO(TimeOfDay t) {
|
||||||
final dayLabels = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
final now = DateTime.now();
|
||||||
|
final dt = DateTime(now.year, now.month, now.day, t.hour, t.minute);
|
||||||
|
return dt.toIso8601String();
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<String> _convertSelectedDaysToStrings(List<bool> selectedDays) {
|
||||||
|
const allDays = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
||||||
|
List<String> result = [];
|
||||||
|
for (int i = 0; i < selectedDays.length; i++) {
|
||||||
|
if (selectedDays[i]) result.add(allDays[i]);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static Widget _buildDayCheckboxes(BuildContext ctx, List<bool> selectedDays,
|
||||||
|
Function(int, bool) onChanged) {
|
||||||
|
final dayLabels = ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'];
|
||||||
return Row(
|
return Row(
|
||||||
children: List.generate(7, (index) {
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
return Row(
|
children: List.generate(
|
||||||
|
7,
|
||||||
|
(index) => Row(
|
||||||
children: [
|
children: [
|
||||||
Checkbox(
|
Checkbox(
|
||||||
value: selectedDays[index],
|
value: selectedDays[index],
|
||||||
onChanged: (bool? value) {
|
onChanged: (val) => onChanged(index, val!),
|
||||||
context.read<WaterHeaterBloc>().add(UpdateSelectedDayEvent(index, value!));
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
Text(dayLabels[index]),
|
Text(dayLabels[index]),
|
||||||
],
|
],
|
||||||
);
|
),
|
||||||
}),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Widget _buildFunctionSwitch(BuildContext context, bool isOn, bool? isEdit) {
|
static Widget _buildFunctionSwitch(
|
||||||
|
BuildContext ctx, bool isOn, Function(bool) onChanged) {
|
||||||
return Row(
|
return Row(
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
'Function:',
|
'Function:',
|
||||||
style: context.textTheme.bodySmall!.copyWith(color: ColorsManager.grayColor),
|
style:
|
||||||
|
Theme.of(ctx).textTheme.bodySmall!.copyWith(color: Colors.grey),
|
||||||
),
|
),
|
||||||
const SizedBox(width: 10),
|
const SizedBox(width: 10),
|
||||||
Radio<bool>(
|
Radio<bool>(
|
||||||
value: true,
|
value: true,
|
||||||
groupValue: isOn,
|
groupValue: isOn,
|
||||||
onChanged: (bool? value) {
|
onChanged: (val) => onChanged(true),
|
||||||
context.read<WaterHeaterBloc>().add(const UpdateFunctionOnEvent(true));
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
const Text('On'),
|
const Text('On'),
|
||||||
const SizedBox(width: 10),
|
const SizedBox(width: 10),
|
||||||
Radio<bool>(
|
Radio<bool>(
|
||||||
value: false,
|
value: false,
|
||||||
groupValue: isOn,
|
groupValue: isOn,
|
||||||
onChanged: (bool? value) {
|
onChanged: (val) => onChanged(false),
|
||||||
context.read<WaterHeaterBloc>().add(const UpdateFunctionOnEvent(false));
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
const Text('Off'),
|
const Text('Off'),
|
||||||
],
|
],
|
||||||
|
@ -2,6 +2,7 @@ import 'dart:convert';
|
|||||||
|
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/all_devices/models/device_status.dart';
|
import 'package:syncrow_web/pages/device_managment/all_devices/models/device_status.dart';
|
||||||
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/schedule_model.dart';
|
||||||
|
|
||||||
class ScheduleEntry {
|
class ScheduleEntry {
|
||||||
final String category;
|
final String category;
|
||||||
@ -58,7 +59,8 @@ class ScheduleEntry {
|
|||||||
|
|
||||||
String toJson() => json.encode(toMap());
|
String toJson() => json.encode(toMap());
|
||||||
|
|
||||||
factory ScheduleEntry.fromJson(String source) => ScheduleEntry.fromMap(json.decode(source));
|
factory ScheduleEntry.fromJson(String source) =>
|
||||||
|
ScheduleEntry.fromMap(json.decode(source));
|
||||||
|
|
||||||
@override
|
@override
|
||||||
bool operator ==(Object other) {
|
bool operator ==(Object other) {
|
||||||
@ -73,6 +75,23 @@ class ScheduleEntry {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
int get hashCode {
|
int get hashCode {
|
||||||
return category.hashCode ^ time.hashCode ^ function.hashCode ^ days.hashCode;
|
return category.hashCode ^
|
||||||
|
time.hashCode ^
|
||||||
|
function.hashCode ^
|
||||||
|
days.hashCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Existing properties and methods
|
||||||
|
|
||||||
|
// Add the fromScheduleModel method
|
||||||
|
|
||||||
|
static ScheduleEntry fromScheduleModel(ScheduleModel scheduleModel) {
|
||||||
|
return ScheduleEntry(
|
||||||
|
days: scheduleModel.days,
|
||||||
|
time: scheduleModel.time,
|
||||||
|
function: scheduleModel.function,
|
||||||
|
category: scheduleModel.category,
|
||||||
|
scheduleId: scheduleModel.scheduleId,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@ class WaterHeaterStatusModel extends Equatable {
|
|||||||
final String cycleTiming;
|
final String cycleTiming;
|
||||||
final List<ScheduleModel> schedules;
|
final List<ScheduleModel> schedules;
|
||||||
|
|
||||||
const WaterHeaterStatusModel({
|
const WaterHeaterStatusModel({
|
||||||
required this.uuid,
|
required this.uuid,
|
||||||
required this.heaterSwitch,
|
required this.heaterSwitch,
|
||||||
required this.countdownHours,
|
required this.countdownHours,
|
||||||
|
@ -7,7 +7,7 @@ import 'package:syncrow_web/pages/device_managment/shared/toggle_widget.dart';
|
|||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/factories/water_heater_bloc_factory.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/factories/water_heater_bloc_factory.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedual_view.dart';
|
import 'package:syncrow_web/pages/device_managment/schedule_device/schedule_widgets/schedual_view.dart';
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
import 'package:syncrow_web/utils/color_manager.dart';
|
||||||
import 'package:syncrow_web/utils/constants/assets.dart';
|
import 'package:syncrow_web/utils/constants/assets.dart';
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
||||||
@ -35,7 +35,8 @@ class WaterHeaterDeviceControlView extends StatelessWidget
|
|||||||
state is WaterHeaterBatchFailedState) {
|
state is WaterHeaterBatchFailedState) {
|
||||||
return const Center(child: Text('Error fetching status'));
|
return const Center(child: Text('Error fetching status'));
|
||||||
} else {
|
} else {
|
||||||
return const SizedBox(height: 200, child: Center(child: SizedBox()));
|
return const SizedBox(
|
||||||
|
height: 200, child: Center(child: SizedBox()));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
));
|
));
|
||||||
@ -79,7 +80,9 @@ class WaterHeaterDeviceControlView extends StatelessWidget
|
|||||||
context: context,
|
context: context,
|
||||||
builder: (ctx) => BlocProvider.value(
|
builder: (ctx) => BlocProvider.value(
|
||||||
value: BlocProvider.of<WaterHeaterBloc>(context),
|
value: BlocProvider.of<WaterHeaterBloc>(context),
|
||||||
child: BuildScheduleView(status: status),
|
child: BuildScheduleView(
|
||||||
|
deviceUuid: device.uuid ?? '',
|
||||||
|
),
|
||||||
));
|
));
|
||||||
},
|
},
|
||||||
child: DeviceControlsContainer(
|
child: DeviceControlsContainer(
|
||||||
|
@ -1,223 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
|
||||||
|
|
||||||
class CountdownInchingView extends StatelessWidget {
|
|
||||||
final WaterHeaterDeviceStatusLoaded state;
|
|
||||||
|
|
||||||
const CountdownInchingView({
|
|
||||||
super.key,
|
|
||||||
required this.state,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final isCountDown =
|
|
||||||
state.scheduleMode?.name == ScheduleModes.countdown.name;
|
|
||||||
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
isCountDown ? 'Countdown:' : 'Inching:',
|
|
||||||
style: context.textTheme.bodySmall!.copyWith(
|
|
||||||
fontSize: 13,
|
|
||||||
color: ColorsManager.grayColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
Visibility(
|
|
||||||
visible: !isCountDown,
|
|
||||||
child: const Text(
|
|
||||||
'Once enabled this feature, each time the device is turned on, it will automatically turn off after a preset time.'),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
_hourMinutesWheel(context, state),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Row _hourMinutesWheel(
|
|
||||||
BuildContext context, WaterHeaterDeviceStatusLoaded state) {
|
|
||||||
final isCountDown =
|
|
||||||
state.scheduleMode?.name == ScheduleModes.countdown.name;
|
|
||||||
late bool isActive;
|
|
||||||
if (isCountDown &&
|
|
||||||
state.countdownRemaining != null &&
|
|
||||||
state.isCountdownActive == true) {
|
|
||||||
isActive = true;
|
|
||||||
} else if (!isCountDown &&
|
|
||||||
state.countdownRemaining != null &&
|
|
||||||
state.isInchingActive == true) {
|
|
||||||
isActive = true;
|
|
||||||
} else {
|
|
||||||
isActive = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
_buildPickerColumn(
|
|
||||||
context,
|
|
||||||
'h',
|
|
||||||
isCountDown
|
|
||||||
? (state.countdownHours ?? 0)
|
|
||||||
: (state.inchingHours ?? 0),
|
|
||||||
24, (value) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: state.scheduleMode ?? ScheduleModes.countdown,
|
|
||||||
hours: value,
|
|
||||||
minutes: isCountDown
|
|
||||||
? (state.countdownMinutes ?? 0)
|
|
||||||
: (state.inchingMinutes ?? 0),
|
|
||||||
));
|
|
||||||
}, isActive: isActive),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
_buildPickerColumn(
|
|
||||||
context,
|
|
||||||
'm',
|
|
||||||
isCountDown
|
|
||||||
? (state.countdownMinutes ?? 0)
|
|
||||||
: (state.inchingMinutes ?? 0),
|
|
||||||
60, (value) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: state.scheduleMode ?? ScheduleModes.countdown,
|
|
||||||
hours: isCountDown
|
|
||||||
? (state.countdownHours ?? 0)
|
|
||||||
: (state.inchingHours ?? 0),
|
|
||||||
minutes: value,
|
|
||||||
));
|
|
||||||
}, isActive: isActive),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Row _hourMinutesSecondWheel(
|
|
||||||
BuildContext context, WaterHeaterDeviceStatusLoaded state) {
|
|
||||||
final isCountDown =
|
|
||||||
state.scheduleMode?.name == ScheduleModes.countdown.name;
|
|
||||||
late bool isActive;
|
|
||||||
if (isCountDown &&
|
|
||||||
state.countdownRemaining != null &&
|
|
||||||
state.isCountdownActive == true) {
|
|
||||||
isActive = true;
|
|
||||||
} else if (!isCountDown &&
|
|
||||||
state.countdownRemaining != null &&
|
|
||||||
state.isInchingActive == true) {
|
|
||||||
isActive = true;
|
|
||||||
} else {
|
|
||||||
isActive = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
_buildPickerColumn(
|
|
||||||
context,
|
|
||||||
'h',
|
|
||||||
isCountDown
|
|
||||||
? (state.countdownHours ?? 0)
|
|
||||||
: (state.inchingHours ?? 0),
|
|
||||||
24, (value) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: state.scheduleMode ?? ScheduleModes.countdown,
|
|
||||||
hours: value,
|
|
||||||
minutes: isCountDown
|
|
||||||
? (state.countdownMinutes ?? 0)
|
|
||||||
: (state.inchingMinutes ?? 0),
|
|
||||||
));
|
|
||||||
}, isActive: isActive),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
_buildPickerColumn(
|
|
||||||
context,
|
|
||||||
'm',
|
|
||||||
isCountDown
|
|
||||||
? (state.countdownMinutes ?? 0)
|
|
||||||
: (state.inchingMinutes ?? 0),
|
|
||||||
60, (value) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: state.scheduleMode ?? ScheduleModes.countdown,
|
|
||||||
hours: isCountDown
|
|
||||||
? (state.countdownHours ?? 0)
|
|
||||||
: (state.inchingHours ?? 0),
|
|
||||||
minutes: value,
|
|
||||||
));
|
|
||||||
}, isActive: isActive),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
_buildPickerColumn(
|
|
||||||
context,
|
|
||||||
'S',
|
|
||||||
isCountDown
|
|
||||||
? (state.countdownMinutes ?? 0)
|
|
||||||
: (state.inchingMinutes ?? 0),
|
|
||||||
60, (value) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: state.scheduleMode ?? ScheduleModes.countdown,
|
|
||||||
hours: isCountDown
|
|
||||||
? (state.countdownHours ?? 0)
|
|
||||||
: (state.inchingHours ?? 0),
|
|
||||||
minutes: value,
|
|
||||||
));
|
|
||||||
}, isActive: isActive),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildPickerColumn(
|
|
||||||
BuildContext context,
|
|
||||||
String label,
|
|
||||||
int initialValue,
|
|
||||||
int itemCount,
|
|
||||||
ValueChanged<int> onSelected, {
|
|
||||||
required bool isActive,
|
|
||||||
}) {
|
|
||||||
return Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
height: 40,
|
|
||||||
width: 80,
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 16),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: ColorsManager.boxColor,
|
|
||||||
borderRadius: BorderRadius.circular(8),
|
|
||||||
),
|
|
||||||
child: ListWheelScrollView.useDelegate(
|
|
||||||
key: ValueKey('$label-$initialValue'),
|
|
||||||
controller: FixedExtentScrollController(
|
|
||||||
initialItem: initialValue,
|
|
||||||
),
|
|
||||||
itemExtent: 40.0,
|
|
||||||
physics: const FixedExtentScrollPhysics(),
|
|
||||||
onSelectedItemChanged: onSelected,
|
|
||||||
childDelegate: ListWheelChildBuilderDelegate(
|
|
||||||
builder: (context, index) {
|
|
||||||
return Center(
|
|
||||||
child: Text(
|
|
||||||
index.toString().padLeft(2, '0'),
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 24,
|
|
||||||
color: isActive ? ColorsManager.grayColor : Colors.black,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
childCount: itemCount,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: 8),
|
|
||||||
Text(
|
|
||||||
label,
|
|
||||||
style: const TextStyle(
|
|
||||||
color: ColorsManager.grayColor,
|
|
||||||
fontSize: 18,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,117 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/helper/add_schedule_dialog_helper.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/count_down_button.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/count_down_inching_view.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/inching_mode_buttons.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedule_header.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedule_managment_ui.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedule_mode_buttons.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/widgets/schedule_mode_selector.dart';
|
|
||||||
|
|
||||||
class BuildScheduleView extends StatefulWidget {
|
|
||||||
const BuildScheduleView({super.key, required this.status});
|
|
||||||
|
|
||||||
final WaterHeaterStatusModel status;
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<BuildScheduleView> createState() => _BuildScheduleViewState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _BuildScheduleViewState extends State<BuildScheduleView> {
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final bloc = BlocProvider.of<WaterHeaterBloc>(context);
|
|
||||||
|
|
||||||
return BlocProvider.value(
|
|
||||||
value: bloc,
|
|
||||||
child: Dialog(
|
|
||||||
backgroundColor: Colors.white,
|
|
||||||
insetPadding: const EdgeInsets.all(20),
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
),
|
|
||||||
child: SizedBox(
|
|
||||||
width: 700,
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
child: Padding(
|
|
||||||
padding:
|
|
||||||
const EdgeInsets.symmetric(horizontal: 40.0, vertical: 20),
|
|
||||||
child: BlocBuilder<WaterHeaterBloc, WaterHeaterState>(
|
|
||||||
builder: (context, state) {
|
|
||||||
if (state is WaterHeaterDeviceStatusLoaded) {
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
const ScheduleHeader(),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
ScheduleModeSelector(state: state),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
if (state.scheduleMode == ScheduleModes.schedule)
|
|
||||||
ScheduleManagementUI(
|
|
||||||
state: state,
|
|
||||||
onAddSchedule: () {
|
|
||||||
ScheduleDialogHelper.showAddScheduleDialog(
|
|
||||||
context,
|
|
||||||
schedule: null,
|
|
||||||
index: null,
|
|
||||||
isEdit: false);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
if (state.scheduleMode == ScheduleModes.countdown ||
|
|
||||||
state.scheduleMode == ScheduleModes.inching)
|
|
||||||
CountdownInchingView(state: state),
|
|
||||||
const SizedBox(height: 20),
|
|
||||||
if (state.scheduleMode == ScheduleModes.countdown)
|
|
||||||
CountdownModeButtons(
|
|
||||||
isActive: state.isCountdownActive ?? false,
|
|
||||||
deviceId: widget.status.uuid,
|
|
||||||
hours: state.countdownHours ?? 0,
|
|
||||||
minutes: state.countdownMinutes ?? 0,
|
|
||||||
),
|
|
||||||
if (state.scheduleMode == ScheduleModes.inching)
|
|
||||||
InchingModeButtons(
|
|
||||||
isActive: state.isInchingActive ?? false,
|
|
||||||
deviceId: widget.status.uuid,
|
|
||||||
hours: state.inchingHours ?? 0,
|
|
||||||
minutes: state.inchingMinutes ?? 0,
|
|
||||||
),
|
|
||||||
if (state.scheduleMode != ScheduleModes.countdown &&
|
|
||||||
state.scheduleMode != ScheduleModes.inching)
|
|
||||||
ScheduleModeButtons(
|
|
||||||
onSave: () {
|
|
||||||
Navigator.pop(context);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (state is WaterHeaterLoadingState) {
|
|
||||||
return const SizedBox(
|
|
||||||
height: 200,
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
ScheduleHeader(),
|
|
||||||
SizedBox(
|
|
||||||
height: 20,
|
|
||||||
),
|
|
||||||
Center(child: CircularProgressIndicator()),
|
|
||||||
],
|
|
||||||
));
|
|
||||||
}
|
|
||||||
return const SizedBox(
|
|
||||||
height: 200,
|
|
||||||
child: ScheduleHeader(),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,86 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/bloc/water_heater_bloc.dart';
|
|
||||||
import 'package:syncrow_web/utils/color_manager.dart';
|
|
||||||
import 'package:syncrow_web/utils/extension/build_context_x.dart';
|
|
||||||
import 'package:syncrow_web/pages/device_managment/water_heater/models/water_heater_status_model.dart';
|
|
||||||
|
|
||||||
class ScheduleModeSelector extends StatelessWidget {
|
|
||||||
final WaterHeaterDeviceStatusLoaded state;
|
|
||||||
|
|
||||||
const ScheduleModeSelector({super.key, required this.state});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
'Type:',
|
|
||||||
style: context.textTheme.bodySmall!.copyWith(
|
|
||||||
fontSize: 13,
|
|
||||||
color: ColorsManager.grayColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
||||||
children: [
|
|
||||||
_buildRadioTile(
|
|
||||||
context, 'Countdown', ScheduleModes.countdown, state),
|
|
||||||
_buildRadioTile(context, 'Schedule', ScheduleModes.schedule, state),
|
|
||||||
_buildRadioTile(
|
|
||||||
context, 'Circulate', ScheduleModes.circulate, state),
|
|
||||||
_buildRadioTile(context, 'Inching', ScheduleModes.inching, state),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildRadioTile(BuildContext context, String label, ScheduleModes mode,
|
|
||||||
WaterHeaterDeviceStatusLoaded state) {
|
|
||||||
return Flexible(
|
|
||||||
child: ListTile(
|
|
||||||
contentPadding: EdgeInsets.zero,
|
|
||||||
title: Text(
|
|
||||||
label,
|
|
||||||
style: context.textTheme.bodySmall!.copyWith(
|
|
||||||
fontSize: 13,
|
|
||||||
color: ColorsManager.blackColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
leading: Radio<ScheduleModes>(
|
|
||||||
value: mode,
|
|
||||||
groupValue: state.scheduleMode,
|
|
||||||
onChanged: (ScheduleModes? value) {
|
|
||||||
if (value != null) {
|
|
||||||
if (value == ScheduleModes.countdown) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: value,
|
|
||||||
hours: state.countdownHours ?? 0,
|
|
||||||
minutes: state.countdownMinutes ?? 0,
|
|
||||||
));
|
|
||||||
} else if (value == ScheduleModes.inching) {
|
|
||||||
context.read<WaterHeaterBloc>().add(UpdateScheduleEvent(
|
|
||||||
scheduleMode: value,
|
|
||||||
hours: state.inchingHours ?? 0,
|
|
||||||
minutes: state.inchingMinutes ?? 0,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value == ScheduleModes.schedule) {
|
|
||||||
context.read<WaterHeaterBloc>().add(
|
|
||||||
GetSchedulesEvent(
|
|
||||||
category: 'switch_1',
|
|
||||||
uuid: state.status.uuid,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
Reference in New Issue
Block a user