connect curtain controller

This commit is contained in:
ashrafzarkanisala
2024-09-08 20:12:51 +03:00
parent 85a04e504f
commit ce861efa3f
7 changed files with 110 additions and 69 deletions

View File

@ -1,11 +1,9 @@
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
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/flutter_svg.dart';
import 'package:syncrow_web/pages/device_managment/three_gang_switch/bloc/living_room_bloc.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_bloc.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_event.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';
@ -40,15 +38,16 @@ class CurtainToggle extends StatelessWidget {
crossAxisAlignment: CrossAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.center,
children: [ children: [
ClipOval( ClipOval(
child: Container( child: Container(
color: ColorsManager.whiteColors, color: ColorsManager.whiteColors,
child: SvgPicture.asset( child: SvgPicture.asset(
Assets.curtainIcon, Assets.curtainIcon,
width: 60, width: 60,
height: 60, height: 60,
fit: BoxFit.cover, fit: BoxFit.cover,
), ),
)), ),
),
SizedBox( SizedBox(
height: 20, height: 20,
width: 35, width: 35,
@ -56,13 +55,13 @@ class CurtainToggle extends StatelessWidget {
value: value, value: value,
activeColor: ColorsManager.dialogBlueTitle, activeColor: ColorsManager.dialogBlueTitle,
onChanged: (newValue) { onChanged: (newValue) {
context.read<LivingRoomBloc>().add( context.read<CurtainBloc>().add(
LivingRoomControl( CurtainControl(
deviceId: deviceId, deviceId: deviceId,
code: code, code: code,
value: newValue, value: newValue,
), ),
); );
}, },
), ),
), ),

View File

@ -3,7 +3,7 @@ import 'package:flutter/material.dart';
import 'package:syncrow_web/pages/device_managment/ac/view/ac_device_control.dart'; import 'package:syncrow_web/pages/device_managment/ac/view/ac_device_control.dart';
import 'package:syncrow_web/pages/device_managment/all_devices/models/devices_model.dart'; import 'package:syncrow_web/pages/device_managment/all_devices/models/devices_model.dart';
import 'package:syncrow_web/pages/device_managment/ceiling_sensor/view/ceiling_sensor_controls.dart'; import 'package:syncrow_web/pages/device_managment/ceiling_sensor/view/ceiling_sensor_controls.dart';
import 'package:syncrow_web/pages/device_managment/curtain/view/curtain_control.dart'; import 'package:syncrow_web/pages/device_managment/curtain/view/curtain_status_view.dart';
import 'package:syncrow_web/pages/device_managment/door_lock/view/door_lock_status_view.dart'; import 'package:syncrow_web/pages/device_managment/door_lock/view/door_lock_status_view.dart';
import 'package:syncrow_web/pages/device_managment/gateway/view/gateway_view.dart'; import 'package:syncrow_web/pages/device_managment/gateway/view/gateway_view.dart';
import 'package:syncrow_web/pages/device_managment/three_gang_switch/view/living_room_device_control.dart'; import 'package:syncrow_web/pages/device_managment/three_gang_switch/view/living_room_device_control.dart';
@ -29,7 +29,7 @@ mixin RouteControlsBasedCode {
device: device, device: device,
); );
case 'CUR': case 'CUR':
return CurtainControl( return CurtainStatusView(
deviceId: device.uuid!, deviceId: device.uuid!,
); );
case 'AC': case 'AC':

View File

@ -1,7 +1,6 @@
import 'dart:async'; import 'dart:async';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:syncrow_web/pages/device_managment/all_devices/models/device_status.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_event.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_event.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_state.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_state.dart';
import 'package:syncrow_web/services/devices_mang_api.dart'; import 'package:syncrow_web/services/devices_mang_api.dart';
@ -9,32 +8,84 @@ import 'package:syncrow_web/services/devices_mang_api.dart';
class CurtainBloc extends Bloc<CurtainEvent, CurtainState> { class CurtainBloc extends Bloc<CurtainEvent, CurtainState> {
late bool deviceStatus; late bool deviceStatus;
final String deviceId; final String deviceId;
Timer? _timer;
CurtainBloc({required this.deviceId}) : super(CurtainInitial()) { CurtainBloc({required this.deviceId}) : super(CurtainInitial()) {
on<CurtainFetchDeviceStatus>(_onFetchDeviceStatus); on<CurtainFetchDeviceStatus>(_onFetchDeviceStatus);
on<CurtainControl>(_onCurtainControl);
} }
FutureOr<void> _onFetchDeviceStatus( FutureOr<void> _onFetchDeviceStatus(
CurtainFetchDeviceStatus event, Emitter<CurtainState> emit) async { CurtainFetchDeviceStatus event, Emitter<CurtainState> emit) async {
emit(CurtainStatusLoading()); emit(CurtainStatusLoading());
try { try {
final status = await DevicesManagementApi().getDeviceStatus(event.deviceId); final status =
deviceStatus =checkStatus(status.status[0].value) ; // Assuming this is a Map<String, dynamic> await DevicesManagementApi().getDeviceStatus(event.deviceId);
deviceStatus = _checkStatus(status.status[0].value);
emit(CurtainStatusLoaded(deviceStatus)); emit(CurtainStatusLoaded(deviceStatus));
} catch (e) { } catch (e) {
emit(CurtainError(e.toString())); emit(CurtainError(e.toString()));
} }
} }
bool checkStatus(String command) { FutureOr<void> _onCurtainControl(
if (command.toLowerCase() == 'open') { CurtainControl event, Emitter<CurtainState> emit) async {
return true; final oldValue = deviceStatus;
} else {
return false; _updateLocalValue(event.value, emit);
}
emit(CurtainStatusLoaded(deviceStatus));
await _runDebounce(
deviceId: event.deviceId,
code: event.code,
value: event.value,
oldValue: oldValue,
emit: emit,
);
} }
Future<void> _runDebounce({
required String deviceId,
required String code,
required bool value,
required bool oldValue,
required Emitter<CurtainState> emit,
}) async {
if (_timer != null) {
_timer!.cancel();
}
_timer = Timer(const Duration(seconds: 1), () async {
try {
final controlValue = value ? 'open' : 'stop';
final response = await DevicesManagementApi()
.deviceControl(deviceId, Status(code: code, value: controlValue));
if (!response) {
_revertValueAndEmit(deviceId, oldValue, emit);
}
} catch (e) {
_revertValueAndEmit(deviceId, oldValue, emit);
}
});
}
void _revertValueAndEmit(
String deviceId, bool oldValue, Emitter<CurtainState> emit) {
_updateLocalValue(oldValue, emit);
emit(CurtainStatusLoaded(deviceStatus));
emit(const CurtainControlError('Failed to control the device.'));
}
void _updateLocalValue(bool value, Emitter<CurtainState> emit) {
deviceStatus = value;
emit(CurtainStatusLoaded(deviceStatus));
}
bool _checkStatus(String command) {
return command.toLowerCase() == 'open';
}
} }

View File

@ -1,6 +1,5 @@
import 'package:equatable/equatable.dart'; import 'package:equatable/equatable.dart';
import 'package:flutter/material.dart';
sealed class CurtainEvent extends Equatable { sealed class CurtainEvent extends Equatable {

View File

@ -1,6 +1,5 @@
import 'package:equatable/equatable.dart'; import 'package:equatable/equatable.dart';
import 'package:syncrow_web/pages/device_managment/curtain/model/curtain_model.dart';
sealed class CurtainState extends Equatable { sealed class CurtainState extends Equatable {
const CurtainState(); const CurtainState();

View File

@ -26,7 +26,7 @@ class CurtainModel {
return { return {
'productUuid': productUuid, 'productUuid': productUuid,
'productType': productType, 'productType': productType,
'status': status.map((s) => s!.toJson()).toList(), 'status': status.map((s) => s.toJson()).toList(),
}; };
} }
} }

View File

@ -1,27 +1,21 @@
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/pages/common/curtain_toggle.dart'; import 'package:syncrow_web/pages/common/curtain_toggle.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_bloc.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_bloc.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_event.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_event.dart';
import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_state.dart'; import 'package:syncrow_web/pages/device_managment/curtain/bloc/curtain_state.dart';
import 'package:syncrow_web/pages/device_managment/curtain/model/curtain_model.dart';
import 'package:syncrow_web/pages/device_managment/three_gang_switch/bloc/living_room_bloc.dart';
import 'package:syncrow_web/pages/device_managment/three_gang_switch/models/living_room_model.dart';
import 'package:syncrow_web/pages/device_managment/three_gang_switch/widgets/living_toggle_widget.dart';
import 'package:syncrow_web/utils/helpers/responsice_layout_helper/responsive_layout_helper.dart'; import 'package:syncrow_web/utils/helpers/responsice_layout_helper/responsive_layout_helper.dart';
class CurtainControl extends StatelessWidget with HelperResponsiveLayout { class CurtainStatusView extends StatelessWidget with HelperResponsiveLayout {
final String deviceId; final String deviceId;
const CurtainControl({super.key, required this.deviceId}); const CurtainStatusView({super.key, required this.deviceId});
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocProvider( return BlocProvider(
create: (context) => create: (context) => CurtainBloc(deviceId: deviceId)
CurtainBloc(deviceId: deviceId)..add(CurtainFetchDeviceStatus(deviceId)), ..add(CurtainFetchDeviceStatus(deviceId)),
child: BlocBuilder<CurtainBloc, CurtainState>( child: BlocBuilder<CurtainBloc, CurtainState>(
builder: (context, state) { builder: (context, state) {
if (state is CurtainStatusLoading) { if (state is CurtainStatusLoading) {
@ -42,31 +36,30 @@ class CurtainControl extends StatelessWidget with HelperResponsiveLayout {
final isExtraLarge = isExtraLargeScreenSize(context); final isExtraLarge = isExtraLargeScreenSize(context);
final isLarge = isLargeScreenSize(context); final isLarge = isLargeScreenSize(context);
final isMedium = isMediumScreenSize(context); final isMedium = isMediumScreenSize(context);
return Container( return GridView(
child: GridView( padding: const EdgeInsets.symmetric(horizontal: 50),
padding: const EdgeInsets.symmetric(horizontal: 50), shrinkWrap: true,
shrinkWrap: true, physics: const NeverScrollableScrollPhysics(),
physics: const NeverScrollableScrollPhysics(), gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
gridDelegate: SliverGridDelegateWithFixedCrossAxisCount( crossAxisCount: isLarge || isExtraLarge
crossAxisCount: isLarge || isExtraLarge ? 3
? 3 : isMedium
: isMedium ? 2
? 2 : 1,
: 1, mainAxisExtent: 140,
mainAxisExtent: 140, crossAxisSpacing: 12,
crossAxisSpacing: 12, mainAxisSpacing: 12,
mainAxisSpacing: 12,
),
children: [
CurtainToggle(
value: status,
code: 'Curtains',
deviceId: deviceId,
label: 'Curtains',
),
],
), ),
children: [
const SizedBox.shrink(),
CurtainToggle(
value: status,
code: 'control',
deviceId: deviceId,
label: 'Curtains',
),
const SizedBox.shrink(),
],
); );
} }
} }