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

499 lines
15 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';
part 'entity_class/entity.class.dart';
part 'entity_class/button_entity.class.dart';
part 'entity_class/datetime_entity.class.dart';
part 'entity_class/select_entity.class.dart';
part 'entity_class/slider_entity.class.dart';
part 'entity_class/switch_entity.class.dart';
part 'entity_class/text_entity.class.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-07 02:17:14 +03:00
part 'view_builder.class.dart';
part 'view_class.dart';
part 'card_class.dart';
2018-09-10 00:34:52 +03:00
EventBus eventBus = new EventBus();
const String appName = "HA Client";
2018-10-03 15:26:46 +03:00
const appVersion = "0.2.4";
String homeAssistantWebHost;
void main() {
FlutterError.onError = (errorDetails) {
TheLogger.log("Error", "${errorDetails.exception}");
if (TheLogger.isInDebugMode) {
FlutterError.dumpErrorToConsole(errorDetails);
}
};
runZoned(() {
2018-09-25 22:47:06 +03:00
runApp(new HAClientApp());
}, onError: (error, stack) {
TheLogger.log("Global error", "$error");
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'),
"/connection-settings": (context) => ConnectionSettingsPage(title: "Connection 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;
String _apiEndpoint;
String _apiPassword;
String _authType;
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;
@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) {
TheLogger.log("Debug","Settings change event: reconnect=${event.reconnect}");
if (event.reconnect) {
_homeAssistant.disconnect().then((_){
_loadConnectionSettings().then((b){
_refreshData();
}, onError: (_) {
setState(() {
_isLoading = 2;
});
_showErrorSnackBar(message: _, errorCode: 5);
}
);
});
}
});
_loadConnectionSettings().then((_){
_createConnection();
}, 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) {
TheLogger.log("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";
_apiEndpoint = "${prefs.getString('hassio-protocol')}://$domain:$port/api/websocket";
homeAssistantWebHost = "${prefs.getString('hassio-res-protocol')}://$domain:$port";
_apiPassword = prefs.getString('hassio-password');
_authType = prefs.getString('hassio-auth-type');
if ((domain == null) || (port == null) || (_apiPassword == null) ||
(domain.length == 0) || (port.length == 0) || (_apiPassword.length == 0)) {
throw("Check connection settings");
2018-10-03 15:25:01 +03:00
} else {
_settingsLoaded = true;
}
}
_createConnection() {
_refreshData();
if (_stateSubscription == null) {
_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 {
_homeAssistant.updateConnectionSettings(_apiEndpoint, _apiPassword, _authType);
setState(() {
_isLoading = 1;
});
await _homeAssistant.fetch().then((result) {
setState(() {
//_instanceConfig = _homeAssistant.instanceConfig;
_entities = _homeAssistant.entities;
_uiViewsCount = _homeAssistant.viewsCount;
_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;
});
_showErrorSnackBar(
message: e != null ? e["errorMessage"] ?? "$e" : "Unknown error",
errorCode: e["errorCode"] != null ? e["errorCode"] : 99
);
}
void _callService(String domain, String service, String entityId, Map<String, String> 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(
builder: (context) => EntityViewPage(entity: entity),
)
);
}
2018-10-07 02:17:14 +03:00
List<Tab> buildUIViewTabs() {
//TODO move somewhere to ViewBuilder
List<Tab> result = [];
if (!_entities.isEmpty) {
if (!_entities.hasDefaultView) {
result.add(
2018-10-07 02:17:14 +03:00
Tab(
icon:
Icon(
MaterialDesignIcons.createIconDataFromIconName("mdi:home-assistant"),
size: 24.0,
)
)
);
2018-09-17 22:20:36 +03:00
}
2018-10-07 02:17:14 +03:00
_entities.viewList.forEach((viewId) {
result.add(
2018-09-17 22:20:36 +03:00
Tab(
icon: MaterialDesignIcons.createIconWidgetFromEntityData(_entities.get(viewId), 24.0, null) ??
Icon(
MaterialDesignIcons.createIconDataFromIconName("mdi:home-assistant"),
size: 24.0,
)
2018-09-17 22:20:36 +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() {
return new Drawer(
child: ListView(
children: <Widget>[
new UserAccountsDrawerHeader(
accountName: Text(_homeAssistant != null ? _homeAssistant.locationName : "Unknown"),
accountEmail: Text(_instanceHost ?? "Not configured"),
currentAccountPicture: new Image.asset('images/hassio-192x192.png'),
),
new ListTile(
leading: Icon(Icons.settings),
title: Text("Connection settings"),
onTap: () {
2018-09-24 22:23:01 +03:00
Navigator.of(context).pop();
Navigator.of(context).pushNamed('/connection-settings');
},
),
2018-10-03 15:55:11 +03:00
Container(
height: 16.0,
decoration: new BoxDecoration(
border: new Border(
bottom: BorderSide(
width: 1.0,
color: Colors.black26,
)
),
)
),
new ListTile(
leading: Icon(Icons.insert_drive_file),
title: Text("Log"),
onTap: () {
2018-09-24 22:23:01 +03:00
Navigator.of(context).pop();
Navigator.of(context).pushNamed('/log-view');
},
),
2018-09-24 22:12:56 +03:00
new ListTile(
leading: Icon(MaterialDesignIcons.createIconDataFromIconName("mdi:github-circle")),
2018-09-29 12:09:01 +03:00
title: Text("Report an issue"),
2018-09-24 22:12:56 +03:00
onTap: () {
2018-09-24 22:23:01 +03:00
Navigator.of(context).pop();
2018-10-02 00:41:40 +03:00
HAUtils.launchURL("https://github.com/estevez-dev/ha_client_pub/issues/new");
2018-09-24 22:12:56 +03:00
},
),
Container(
2018-10-03 15:55:11 +03:00
height: 16.0,
decoration: new BoxDecoration(
border: new Border(
2018-10-03 15:55:11 +03:00
bottom: BorderSide(
width: 1.0,
color: Colors.black26,
)
),
)
),
2018-10-03 15:55:11 +03:00
new AboutListTile(
applicationName: appName,
applicationVersion: appVersion,
applicationLegalese: "Keyboard Crumbs | www.keyboardcrumbs.io",
),
new ListTile(
leading: Icon(MaterialDesignIcons.createIconDataFromIconName("mdi:coffee")),
2018-10-03 15:55:11 +03:00
title: Text("Buy me a coffee"),
onTap: () {
Navigator.of(context).pop();
HAUtils.launchURL("https://www.buymeacoffee.com/estevez");
},
)
],
),
);
}
_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;
}
}
_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(),
bottom: empty ? null : TabBar(tabs: buildUIViewTabs()),
),
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-07 02:17:14 +03:00
_homeAssistant.buildViews(context)
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.
if (_entities == null) {
2018-09-24 22:54:51 +03:00
return _buildScaffold(true);
} else {
return DefaultTabController(
length: _uiViewsCount,
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
}