mirror of
https://github.com/SyncrowIOT/web.git
synced 2025-07-17 02:25:31 +00:00
connect curtain controller
This commit is contained in:
@ -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,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
@ -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':
|
||||||
|
@ -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';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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 {
|
||||||
|
@ -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();
|
||||||
|
@ -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(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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(),
|
||||||
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
Reference in New Issue
Block a user