import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter_hbb/common/formatter/id_formatter.dart'; import 'package:flutter_hbb/mobile/pages/file_manager_page.dart'; import 'package:get/get.dart'; import 'package:provider/provider.dart'; import 'package:url_launcher/url_launcher.dart'; import '../../common.dart'; import '../../common/widgets/address_book.dart'; import '../../common/widgets/peer_tab_page.dart'; import '../../common/widgets/peer_widget.dart'; import '../../consts.dart'; import '../../models/model.dart'; import '../../models/platform_model.dart'; import 'home_page.dart'; import 'remote_page.dart'; import 'scan_page.dart'; import 'settings_page.dart'; /// Connection page for connecting to a remote peer. class ConnectionPage extends StatefulWidget implements PageShape { ConnectionPage({Key? key}) : super(key: key); @override final icon = const Icon(Icons.connected_tv); @override final title = translate("Connection"); @override final appBarActions = !isAndroid ? [const WebMenu()] : []; @override State createState() => _ConnectionPageState(); } /// State for the connection page. class _ConnectionPageState extends State { /// Controller for the id input bar. final _idController = IDTextEditingController(); /// Update url. If it's not null, means an update is available. var _updateUrl = ''; @override void initState() { super.initState(); if (_idController.text.isEmpty) { () async { final lastRemoteId = await bind.mainGetLastRemoteId(); if (lastRemoteId != _idController.id) { setState(() { _idController.id = lastRemoteId; }); } }(); } if (isAndroid) { Timer(const Duration(seconds: 5), () async { _updateUrl = await bind.mainGetSoftwareUpdateUrl(); if (_updateUrl.isNotEmpty) setState(() {}); }); } } @override Widget build(BuildContext context) { Provider.of(context); return Column( mainAxisAlignment: MainAxisAlignment.start, mainAxisSize: MainAxisSize.max, crossAxisAlignment: CrossAxisAlignment.center, children: [ _buildUpdateUI(), _buildRemoteIDTextField(), Expanded( child: PeerTabPage( tabs: [ translate('Recent Sessions'), translate('Favorites'), translate('Discovered'), translate('Address Book') ], children: [ RecentPeerWidget(), FavoritePeerWidget(), DiscoveredPeerWidget(), const AddressBook(), ], )), ]).marginOnly(top: 2, left: 10, right: 10); } /// Callback for the connect button. /// Connects to the selected peer. void onConnect() { var id = _idController.id; connect(id); } /// Connect to a peer with [id]. /// If [isFileTransfer], starts a session only for file transfer. void connect(String id, {bool isFileTransfer = false}) async { if (id == '') return; id = id.replaceAll(' ', ''); if (isFileTransfer) { if (!await PermissionManager.check("file")) { if (!await PermissionManager.request("file")) { return; } } Navigator.push( context, MaterialPageRoute( builder: (BuildContext context) => FileManagerPage(id: id), ), ); } else { Navigator.push( context, MaterialPageRoute( builder: (BuildContext context) => RemotePage(id: id), ), ); } FocusScopeNode currentFocus = FocusScope.of(context); if (!currentFocus.hasPrimaryFocus) { currentFocus.unfocus(); } } /// UI for software update. /// If [_updateUrl] is not empty, shows a button to update the software. Widget _buildUpdateUI() { return _updateUrl.isEmpty ? const SizedBox(height: 0) : InkWell( onTap: () async { final url = '$_updateUrl.apk'; if (await canLaunchUrl(Uri.parse(url))) { await launchUrl(Uri.parse(url)); } }, child: Container( alignment: AlignmentDirectional.center, width: double.infinity, color: Colors.pinkAccent, padding: const EdgeInsets.symmetric(vertical: 12), child: Text(translate('Download new version'), style: const TextStyle( color: Colors.white, fontWeight: FontWeight.bold)))); } /// UI for the remote ID TextField. /// Search for a peer and connect to it if the id exists. Widget _buildRemoteIDTextField() { final w = SizedBox( height: 84, child: Padding( padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 2), child: Ink( decoration: const BoxDecoration( color: MyTheme.white, borderRadius: BorderRadius.all(Radius.circular(13)), ), child: Row( children: [ Expanded( child: Container( padding: const EdgeInsets.only(left: 16, right: 16), child: TextField( autocorrect: false, enableSuggestions: false, keyboardType: TextInputType.visiblePassword, // keyboardType: TextInputType.number, style: const TextStyle( fontFamily: 'WorkSans', fontWeight: FontWeight.bold, fontSize: 30, color: MyTheme.idColor, ), decoration: InputDecoration( labelText: translate('Remote ID'), // hintText: 'Enter your remote ID', border: InputBorder.none, helperStyle: const TextStyle( fontWeight: FontWeight.bold, fontSize: 16, color: MyTheme.darkGray, ), labelStyle: const TextStyle( fontWeight: FontWeight.w600, fontSize: 16, letterSpacing: 0.2, color: MyTheme.darkGray, ), ), controller: _idController, inputFormatters: [IDTextInputFormatter()], ), ), ), SizedBox( width: 60, height: 60, child: IconButton( icon: const Icon(Icons.arrow_forward, color: MyTheme.darkGray, size: 45), onPressed: onConnect, ), ), ], ), ), ), ); return Align( alignment: Alignment.topLeft, child: Container(constraints: kMobilePageConstraints, child: w)); } @override void dispose() { _idController.dispose(); super.dispose(); } } class WebMenu extends StatefulWidget { const WebMenu({Key? key}) : super(key: key); @override State createState() => _WebMenuState(); } class _WebMenuState extends State { String? username; String url = ""; @override void initState() { super.initState(); () async { final usernameRes = await getUsername(); final urlRes = await getUrl(); var update = false; if (usernameRes != username) { username = usernameRes; update = true; } if (urlRes != url) { url = urlRes; update = true; } if (update) { setState(() {}); } }(); } @override Widget build(BuildContext context) { Provider.of(context); return PopupMenuButton( icon: const Icon(Icons.more_vert), itemBuilder: (context) { return (isIOS ? [ const PopupMenuItem( value: "scan", child: Icon(Icons.qr_code_scanner, color: Colors.black), ) ] : >[]) + [ PopupMenuItem( value: "server", child: Text(translate('ID/Relay Server')), ) ] + (url.contains('admin.rustdesk.com') ? >[] : [ PopupMenuItem( value: "login", child: Text(username == null ? translate("Login") : '${translate("Logout")} ($username)'), ) ]) + [ PopupMenuItem( value: "about", child: Text('${translate('About')} RustDesk'), ) ]; }, onSelected: (value) { if (value == 'server') { showServerSettings(gFFI.dialogManager); } if (value == 'about') { showAbout(gFFI.dialogManager); } if (value == 'login') { if (username == null) { showLogin(gFFI.dialogManager); } else { logout(gFFI.dialogManager); } } if (value == 'scan') { Navigator.push( context, MaterialPageRoute( builder: (BuildContext context) => ScanPage(), ), ); } }); } }