This repository has been archived on 2023-11-18. You can view files and clone it, but cannot push or open issues or pull requests.
ha_client/lib/main.dart

556 lines
17 KiB
Dart
Raw Normal View History

2018-09-10 00:34:52 +03:00
import 'dart:convert';
import 'dart:async';
import 'package:flutter/rendering.dart';
2018-09-10 00:34:52 +03:00
import 'package:flutter/material.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:web_socket_channel/io.dart';
import 'package:progress_indicators/progress_indicators.dart';
import 'package:event_bus/event_bus.dart';
2018-09-16 19:24:26 +03:00
import 'package:flutter/widgets.dart';
import 'package:cached_network_image/cached_network_image.dart';
2018-09-24 22:12:56 +03:00
import 'package:url_launcher/url_launcher.dart';
import 'package:flutter/services.dart';
2018-09-29 16:19:01 +03:00
import 'package:date_format/date_format.dart';
2018-10-07 23:06:06 +03:00
import 'package:http/http.dart' as http;
2018-10-17 02:19:46 +03:00
import 'package:flutter_colorpicker/material_picker.dart';
import 'package:charts_flutter/flutter.dart' as charts;
2018-10-27 14:27:41 +03:00
part 'entity_class/entity.class.dart';
part 'entity_class/switch_entity.class.dart';
part 'entity_class/button_entity.class.dart';
part 'entity_class/text_entity.class.dart';
part 'entity_class/climate_entity.class.dart';
part 'entity_class/cover_entity.class.dart';
part 'entity_class/date_time_entity.class.dart';
part 'entity_class/light_entity.class.dart';
part 'entity_class/select_entity.class.dart';
2018-10-28 20:01:01 +02:00
part 'entity_class/other_entity.class.dart';
part 'entity_class/slider_entity.dart';
2018-11-04 18:20:06 +02:00
part 'entity_class/media_player_entity.class.dart';
2018-10-27 14:27:41 +03:00
part 'entity_widgets/badge.dart';
part 'entity_widgets/model_widgets.dart';
2018-10-27 14:27:41 +03:00
part 'entity_widgets/default_entity_container.dart';
part 'entity_widgets/entity_attributes_list.dart';
part 'entity_widgets/entity_icon.dart';
part 'entity_widgets/entity_name.dart';
part 'entity_widgets/last_updated.dart';
part 'entity_widgets/mode_swicth.dart';
part 'entity_widgets/mode_selector.dart';
2018-10-28 21:02:38 +02:00
part 'entity_widgets/entity_colors.class.dart';
2018-10-27 14:27:41 +03:00
part 'entity_widgets/entity_page_container.dart';
2018-10-28 18:07:52 +02:00
part 'entity_widgets/history_chart/entity_history.dart';
part 'entity_widgets/history_chart/simple_state_history_chart.dart';
2018-10-28 20:01:01 +02:00
part 'entity_widgets/history_chart/numeric_state_history_chart.dart';
part 'entity_widgets/history_chart/combined_history_chart.dart';
2018-11-03 22:50:21 +02:00
part 'entity_widgets/history_chart/history_control_widget.dart';
part 'entity_widgets/history_chart/entity_history_moment.dart';
2018-10-27 14:27:41 +03:00
part 'entity_widgets/state/switch_state.dart';
part 'entity_widgets/state/slider_state.dart';
part 'entity_widgets/state/text_input_state.dart';
part 'entity_widgets/state/select_state.dart';
part 'entity_widgets/state/simple_state.dart';
part 'entity_widgets/state/climate_state.dart';
part 'entity_widgets/state/cover_state.dart';
part 'entity_widgets/state/date_time_state.dart';
part 'entity_widgets/state/button_state.dart';
part 'entity_widgets/controls/climate_controls.dart';
part 'entity_widgets/controls/cover_controls.dart';
part 'entity_widgets/controls/light_controls.dart';
2018-09-25 22:47:06 +03:00
part 'settings.page.dart';
part 'home_assistant.class.dart';
2018-09-25 22:47:06 +03:00
part 'log.page.dart';
2018-09-29 16:19:01 +03:00
part 'entity.page.dart';
2018-09-25 22:47:06 +03:00
part 'utils.class.dart';
part 'mdi.class.dart';
2018-09-26 22:16:50 +03:00
part 'entity_collection.class.dart';
2018-10-27 17:28:47 +03:00
part 'ui_class/ui.dart';
part 'ui_class/view.class.dart';
part 'ui_class/card.class.dart';
part 'ui_widgets/view.dart';
part 'ui_widgets/entities_card.dart';
part 'ui_widgets/unsupported_card.dart';
part 'ui_widgets/media_control_card.dart';
part 'ui_widgets/card_header_widget.dart';
2018-09-10 00:34:52 +03:00
EventBus eventBus = new EventBus();
const String appName = "HA Client";
2018-11-04 23:20:58 +02:00
const appVersion = "0.3.7";
String homeAssistantWebHost;
void main() {
FlutterError.onError = (errorDetails) {
2018-10-27 01:24:23 +03:00
TheLogger.error( "${errorDetails.exception}");
if (TheLogger.isInDebugMode) {
FlutterError.dumpErrorToConsole(errorDetails);
}
};
runZoned(() {
2018-09-25 22:47:06 +03:00
runApp(new HAClientApp());
}, onError: (error, stack) {
2018-10-27 01:24:23 +03:00
TheLogger.error("$error");
TheLogger.error("$stack");
if (TheLogger.isInDebugMode) {
debugPrint("$stack");
}
});
}
2018-09-10 00:34:52 +03:00
2018-09-25 22:47:06 +03:00
class HAClientApp extends StatelessWidget {
2018-09-10 00:34:52 +03:00
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return new MaterialApp(
title: appName,
2018-09-10 00:34:52 +03:00
theme: new ThemeData(
primarySwatch: Colors.blue,
),
initialRoute: "/",
routes: {
2018-10-02 00:41:40 +03:00
"/": (context) => MainPage(title: 'HA Client'),
2018-10-27 00:54:05 +03:00
"/connection-settings": (context) => ConnectionSettingsPage(title: "Settings"),
"/log-view": (context) => LogViewPage(title: "Log")
},
2018-09-10 00:34:52 +03:00
);
}
}
class MainPage extends StatefulWidget {
MainPage({Key key, this.title}) : super(key: key);
2018-09-10 00:34:52 +03:00
final String title;
@override
_MainPageState createState() => new _MainPageState();
2018-09-10 00:34:52 +03:00
}
2018-09-16 19:24:26 +03:00
class _MainPageState extends State<MainPage> with WidgetsBindingObserver {
HomeAssistant _homeAssistant;
EntityCollection _entities;
//Map _instanceConfig;
2018-10-07 23:06:06 +03:00
String _webSocketApiEndpoint;
String _password;
String _authType;
2018-10-25 00:54:20 +03:00
//int _uiViewsCount = 0;
String _instanceHost;
StreamSubscription _stateSubscription;
2018-09-21 00:39:49 +03:00
StreamSubscription _settingsSubscription;
StreamSubscription _serviceCallSubscription;
2018-09-29 16:19:01 +03:00
StreamSubscription _showEntityPageSubscription;
2018-10-07 02:17:14 +03:00
StreamSubscription _refreshDataSubscription;
StreamSubscription _showErrorSubscription;
int _isLoading = 1;
2018-10-03 15:25:01 +03:00
bool _settingsLoaded = false;
bool _accountMenuExpanded = false;
2018-10-27 00:54:05 +03:00
bool _useLovelaceUI;
@override
void initState() {
super.initState();
2018-10-03 15:25:01 +03:00
_settingsLoaded = false;
2018-09-16 19:24:26 +03:00
WidgetsBinding.instance.addObserver(this);
_homeAssistant = HomeAssistant();
2018-09-21 00:39:49 +03:00
_settingsSubscription = eventBus.on<SettingsChangedEvent>().listen((event) {
2018-10-27 01:24:23 +03:00
TheLogger.debug("Settings change event: reconnect=${event.reconnect}");
if (event.reconnect) {
_homeAssistant.disconnect().then((_){
_initialLoad();
});
}
});
_initialLoad();
}
void _initialLoad() {
_loadConnectionSettings().then((_){
_subscribe();
_refreshData();
}, onError: (_) {
setState(() {
_isLoading = 2;
});
_showErrorSnackBar(message: _, errorCode: 5);
});
}
2018-09-10 00:34:52 +03:00
2018-09-16 19:24:26 +03:00
@override
void didChangeAppLifecycleState(AppLifecycleState state) {
2018-10-27 01:24:23 +03:00
TheLogger.debug("$state");
2018-10-03 15:25:01 +03:00
if (state == AppLifecycleState.resumed && _settingsLoaded) {
2018-09-16 19:24:26 +03:00
_refreshData();
}
}
_loadConnectionSettings() async {
SharedPreferences prefs = await SharedPreferences.getInstance();
String domain = prefs.getString('hassio-domain');
String port = prefs.getString('hassio-port');
2018-09-17 22:20:36 +03:00
_instanceHost = "$domain:$port";
2018-10-07 23:06:06 +03:00
_webSocketApiEndpoint = "${prefs.getString('hassio-protocol')}://$domain:$port/api/websocket";
homeAssistantWebHost = "${prefs.getString('hassio-res-protocol')}://$domain:$port";
2018-10-07 23:06:06 +03:00
_password = prefs.getString('hassio-password');
_authType = prefs.getString('hassio-auth-type');
2018-10-27 00:54:05 +03:00
_useLovelaceUI = prefs.getBool('use-lovelace') ?? false;
2018-10-07 23:06:06 +03:00
if ((domain == null) || (port == null) || (_password == null) ||
(domain.length == 0) || (port.length == 0) || (_password.length == 0)) {
throw("Check connection settings");
2018-10-03 15:25:01 +03:00
} else {
_settingsLoaded = true;
}
}
_subscribe() {
if (_stateSubscription == null) {
2018-10-25 00:54:20 +03:00
//TODO Move to homeAssistant or remove
_stateSubscription = eventBus.on<StateChangedEvent>().listen((event) {
setState(() {
if (event.localChange) {
_entities
.get(event.entityId)
.state = event.newState;
}
});
});
}
if (_serviceCallSubscription == null) {
_serviceCallSubscription =
eventBus.on<ServiceCallEvent>().listen((event) {
_callService(event.domain, event.service, event.entityId,
event.additionalParams);
});
}
2018-09-29 16:19:01 +03:00
if (_showEntityPageSubscription == null) {
_showEntityPageSubscription =
eventBus.on<ShowEntityPageEvent>().listen((event) {
_showEntityPage(event.entity);
});
}
2018-10-07 02:17:14 +03:00
if (_refreshDataSubscription == null) {
_refreshDataSubscription = eventBus.on<RefreshDataEvent>().listen((event){
_refreshData();
});
}
if (_showErrorSubscription == null) {
_showErrorSubscription = eventBus.on<ShowErrorEvent>().listen((event){
_showErrorSnackBar(message: event.text, errorCode: event.errorCode);
});
}
}
_refreshData() async {
2018-10-27 00:54:05 +03:00
_homeAssistant.updateSettings(_webSocketApiEndpoint, _password, _authType, _useLovelaceUI);
setState(() {
_hideErrorSnackBar();
_isLoading = 1;
});
await _homeAssistant.fetch().then((result) {
setState(() {
//_instanceConfig = _homeAssistant.instanceConfig;
_entities = _homeAssistant.entities;
2018-10-25 00:54:20 +03:00
//_uiViewsCount = _homeAssistant.viewsCount;
2018-10-27 01:24:23 +03:00
//TheLogger.debug("_uiViewsCount=$_uiViewsCount");
_isLoading = 0;
});
}).catchError((e) {
_setErrorState(e);
});
2018-10-07 02:17:14 +03:00
eventBus.fire(RefreshDataFinishedEvent());
2018-09-11 01:09:21 +03:00
}
_setErrorState(e) {
setState(() {
_isLoading = 2;
});
2018-11-04 18:20:06 +02:00
if (e is Error) {
TheLogger.error(e.toString());
TheLogger.error("${e.stackTrace}");
_showErrorSnackBar(
message: "There was some error",
errorCode: 13
);
} else {
_showErrorSnackBar(
message: e != null ? e["errorMessage"] ?? "$e" : "Unknown error",
errorCode: e["errorCode"] != null ? e["errorCode"] : 99
);
}
}
2018-10-16 17:35:13 +03:00
void _callService(String domain, String service, String entityId, Map<String, dynamic> additionalParams) {
_homeAssistant.callService(domain, service, entityId, additionalParams).catchError((e) => _setErrorState(e));
}
2018-09-29 16:19:01 +03:00
void _showEntityPage(Entity entity) {
Navigator.push(
context,
MaterialPageRoute(
2018-10-07 23:06:06 +03:00
builder: (context) => EntityViewPage(entity: entity, homeAssistant: _homeAssistant),
2018-09-29 16:19:01 +03:00
)
);
}
2018-10-07 02:17:14 +03:00
List<Tab> buildUIViewTabs() {
List<Tab> result = [];
2018-10-27 00:54:05 +03:00
if (_homeAssistant.ui.views.isNotEmpty) {
_homeAssistant.ui.views.forEach((HAView view) {
2018-11-04 22:19:45 +02:00
result.add(view.buildTab());
2018-10-27 00:54:05 +03:00
});
}
return result;
}
Widget _buildAppTitle() {
Row titleRow = Row(
children: [Text(_homeAssistant != null ? _homeAssistant.locationName : "")],
);
if (_isLoading == 1) {
titleRow.children.add(Padding(
child: JumpingDotsProgressIndicator(
fontSize: 26.0,
color: Colors.white,
),
padding: const EdgeInsets.fromLTRB(5.0, 0.0, 0.0, 30.0),
));
} else if (_isLoading == 2) {
titleRow.children.add(Padding(
child: Icon(
Icons.error_outline,
size: 20.0,
color: Colors.red,
),
padding: const EdgeInsets.fromLTRB(5.0, 0.0, 0.0, 0.0),
));
}
return titleRow;
2018-09-10 00:34:52 +03:00
}
Drawer _buildAppDrawer() {
List<Widget> menuItems = [];
menuItems.add(
UserAccountsDrawerHeader(
accountName: Text(_homeAssistant.userName),
accountEmail: Text(_instanceHost ?? "Not configured"),
onDetailsPressed: () {
setState(() {
_accountMenuExpanded = !_accountMenuExpanded;
});
},
currentAccountPicture: CircleAvatar(
child: Text(
2018-10-08 23:11:56 +03:00
_homeAssistant.userAvatarText,
style: TextStyle(
fontSize: 32.0
),
),
),
)
);
if (_accountMenuExpanded) {
menuItems.addAll([
ListTile(
leading: Icon(Icons.settings),
2018-10-27 00:54:05 +03:00
title: Text("Settings"),
onTap: () {
Navigator.of(context).pop();
Navigator.of(context).pushNamed('/connection-settings');
},
),
Divider(),
]);
} else {
menuItems.addAll([
new ListTile(
leading: Icon(Icons.insert_drive_file),
title: Text("Log"),
onTap: () {
Navigator.of(context).pop();
Navigator.of(context).pushNamed('/log-view');
},
),
new ListTile(
leading: Icon(MaterialDesignIcons.createIconDataFromIconName("mdi:github-circle")),
title: Text("Report an issue"),
onTap: () {
Navigator.of(context).pop();
HAUtils.launchURL("https://github.com/estevez-dev/ha_client_pub/issues/new");
},
),
Divider(),
new AboutListTile(
applicationName: appName,
applicationVersion: appVersion,
applicationLegalese: "Keyboard Crumbs | www.keyboardcrumbs.io",
)
]);
}
return new Drawer(
child: ListView(
children: menuItems,
),
);
}
2018-10-07 12:14:48 +03:00
void _hideErrorSnackBar() {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
}
void _showErrorSnackBar({Key key, @required String message, @required int errorCode}) {
SnackBarAction action;
switch (errorCode) {
case 9:
case 11:
case 7:
case 1: {
action = SnackBarAction(
label: "Retry",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
_refreshData();
},
);
break;
}
case 5: {
message = "Check connection settings";
action = SnackBarAction(
label: "Open",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
Navigator.pushNamed(context, '/connection-settings');
},
);
break;
}
case 6: {
action = SnackBarAction(
label: "Settings",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
Navigator.pushNamed(context, '/connection-settings');
},
);
break;
}
case 10: {
action = SnackBarAction(
label: "Refresh",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
_refreshData();
},
);
break;
}
case 8: {
action = SnackBarAction(
label: "Reconnect",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
_refreshData();
},
);
break;
}
2018-11-04 18:20:06 +02:00
default: {
action = SnackBarAction(
label: "Reload",
onPressed: () {
_scaffoldKey?.currentState?.hideCurrentSnackBar();
_refreshData();
},
);
break;
}
}
_scaffoldKey.currentState.hideCurrentSnackBar();
_scaffoldKey.currentState.showSnackBar(
SnackBar(
content: Text("$message (code: $errorCode)"),
action: action,
duration: Duration(hours: 1),
)
);
}
final GlobalKey<ScaffoldState> _scaffoldKey = new GlobalKey<ScaffoldState>();
2018-09-24 22:54:51 +03:00
Scaffold _buildScaffold(bool empty) {
return Scaffold(
key: _scaffoldKey,
appBar: AppBar(
title: _buildAppTitle(),
leading: IconButton(
icon: Icon(Icons.menu),
onPressed: () {
_scaffoldKey.currentState.openDrawer();
setState(() {
_accountMenuExpanded = false;
});
},
),
2018-10-27 00:54:05 +03:00
primary: true,
2018-10-07 17:16:24 +03:00
bottom: empty ? null : TabBar(
tabs: buildUIViewTabs(),
isScrollable: true,
),
2018-09-24 22:54:51 +03:00
),
drawer: _buildAppDrawer(),
body: empty ?
Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
MaterialDesignIcons.createIconDataFromIconName("mdi:home-assistant"),
size: 100.0,
color: _isLoading == 2 ? Colors.redAccent : Colors.blue,
2018-09-24 22:54:51 +03:00
),
]
),
)
:
2018-10-27 00:54:05 +03:00
_homeAssistant.buildViews(context, _useLovelaceUI)
2018-09-24 22:54:51 +03:00
);
}
2018-09-10 00:34:52 +03:00
@override
Widget build(BuildContext context) {
// This method is rerun every time setState is called.
2018-11-04 11:23:21 +02:00
if (_homeAssistant.ui == null || _homeAssistant.ui.views == null) {
2018-09-24 22:54:51 +03:00
return _buildScaffold(true);
} else {
return DefaultTabController(
2018-10-25 00:54:20 +03:00
length: _homeAssistant.ui.views.length,
2018-09-24 22:54:51 +03:00
child: _buildScaffold(false)
);
}
2018-09-10 00:34:52 +03:00
}
@override
void dispose() {
2018-09-16 19:24:26 +03:00
WidgetsBinding.instance.removeObserver(this);
2018-09-21 00:39:49 +03:00
if (_stateSubscription != null) _stateSubscription.cancel();
if (_settingsSubscription != null) _settingsSubscription.cancel();
if (_serviceCallSubscription != null) _serviceCallSubscription.cancel();
2018-09-29 16:19:01 +03:00
if (_showEntityPageSubscription != null) _showEntityPageSubscription.cancel();
2018-10-07 02:17:14 +03:00
if (_refreshDataSubscription != null) _refreshDataSubscription.cancel();
if (_showErrorSubscription != null) _showErrorSubscription.cancel();
_homeAssistant.disconnect();
super.dispose();
}
2018-09-10 00:34:52 +03:00
}