2022-08-26 23:28:08 +08:00
import ' package:flutter/material.dart ' ;
import ' package:flutter/services.dart ' ;
import ' package:flutter_hbb/models/chat_model.dart ' ;
import ' package:get/get.dart ' ;
2022-08-31 23:02:02 -07:00
import ' package:rxdart/rxdart.dart ' as rxdart ;
2022-08-26 23:28:08 +08:00
import ' ../../common.dart ' ;
import ' ../../mobile/widgets/dialog.dart ' ;
import ' ../../mobile/widgets/overlay.dart ' ;
import ' ../../models/model.dart ' ;
import ' ../../models/platform_model.dart ' ;
2022-08-29 18:48:12 +08:00
import ' ../../common/shared_state.dart ' ;
2022-08-26 23:28:08 +08:00
import ' ./popup_menu.dart ' ;
2022-08-28 21:55:16 +08:00
import ' ./material_mod_popup_menu.dart ' as mod_menu ;
2022-08-26 23:28:08 +08:00
class _MenubarTheme {
static const Color commonColor = MyTheme . accent ;
2022-08-29 18:48:12 +08:00
// kMinInteractiveDimension
2022-09-01 03:56:12 -07:00
static const double height = 25.0 ;
2022-08-29 18:48:12 +08:00
static const double dividerHeight = 12.0 ;
2022-08-26 23:28:08 +08:00
}
class RemoteMenubar extends StatefulWidget {
final String id ;
final FFI ffi ;
const RemoteMenubar ( {
Key ? key ,
required this . id ,
required this . ffi ,
} ) : super ( key: key ) ;
@ override
State < RemoteMenubar > createState ( ) = > _RemoteMenubarState ( ) ;
}
class _RemoteMenubarState extends State < RemoteMenubar > {
final RxBool _show = false . obs ;
final Rx < Color > _hideColor = Colors . white12 . obs ;
bool get isFullscreen = > Get . find < RxBool > ( tag: ' fullscreen ' ) . isTrue ;
void setFullscreen ( bool v ) {
Get . find < RxBool > ( tag: ' fullscreen ' ) . value = v ;
}
@ override
Widget build ( BuildContext context ) {
return Align (
alignment: Alignment . topCenter ,
child: Obx (
( ) = > _show . value ? _buildMenubar ( context ) : _buildShowHide ( context ) ) ,
) ;
}
Widget _buildShowHide ( BuildContext context ) {
2022-08-28 21:55:16 +08:00
return Obx ( ( ) = > Tooltip (
message: translate ( _show . value ? " Hide Menubar " : " Show Menubar " ) ,
child: SizedBox (
width: 100 ,
height: 5 ,
child: TextButton (
onHover: ( bool v ) {
_hideColor . value = v ? Colors . white60 : Colors . white24 ;
} ,
onPressed: ( ) {
_show . value = ! _show . value ;
} ,
child: Obx ( ( ) = > Container (
color: _hideColor . value ,
) ) ) ) ,
) ) ;
2022-08-26 23:28:08 +08:00
}
Widget _buildMenubar ( BuildContext context ) {
final List < Widget > menubarItems = [ ] ;
if ( ! isWebDesktop ) {
menubarItems . add ( _buildFullscreen ( context ) ) ;
if ( widget . ffi . ffiModel . isPeerAndroid ) {
menubarItems . add ( IconButton (
tooltip: translate ( ' Mobile Actions ' ) ,
color: _MenubarTheme . commonColor ,
2022-08-28 21:55:16 +08:00
icon: const Icon ( Icons . build ) ,
2022-08-26 23:28:08 +08:00
onPressed: ( ) {
if ( mobileActionsOverlayEntry = = null ) {
showMobileActionsOverlay ( ) ;
} else {
hideMobileActionsOverlay ( ) ;
}
} ,
) ) ;
}
}
menubarItems . add ( _buildMonitor ( context ) ) ;
menubarItems . add ( _buildControl ( context ) ) ;
menubarItems . add ( _buildDisplay ( context ) ) ;
if ( ! isWeb ) {
menubarItems . add ( _buildChat ( context ) ) ;
}
menubarItems . add ( _buildClose ( context ) ) ;
return PopupMenuTheme (
2022-08-28 21:55:16 +08:00
data: const PopupMenuThemeData (
2022-08-26 23:28:08 +08:00
textStyle: TextStyle ( color: _MenubarTheme . commonColor ) ) ,
child: Column ( mainAxisSize: MainAxisSize . min , children: [
Container (
color: Colors . white ,
child: Row (
mainAxisSize: MainAxisSize . min ,
children: menubarItems ,
) ) ,
_buildShowHide ( context ) ,
] ) ) ;
}
Widget _buildFullscreen ( BuildContext context ) {
return IconButton (
tooltip: translate ( isFullscreen ? ' Exit Fullscreen ' : ' Fullscreen ' ) ,
onPressed: ( ) {
setFullscreen ( ! isFullscreen ) ;
} ,
icon: Obx ( ( ) = > isFullscreen
2022-08-28 21:55:16 +08:00
? const Icon (
2022-08-26 23:28:08 +08:00
Icons . fullscreen_exit ,
color: _MenubarTheme . commonColor ,
)
2022-08-28 21:55:16 +08:00
: const Icon (
2022-08-26 23:28:08 +08:00
Icons . fullscreen ,
color: _MenubarTheme . commonColor ,
) ) ,
) ;
}
Widget _buildChat ( BuildContext context ) {
return IconButton (
tooltip: translate ( ' Chat ' ) ,
onPressed: ( ) {
widget . ffi . chatModel . changeCurrentID ( ChatModel . clientModeID ) ;
widget . ffi . chatModel . toggleChatOverlay ( ) ;
} ,
2022-08-28 21:55:16 +08:00
icon: const Icon (
2022-08-26 23:28:08 +08:00
Icons . message ,
color: _MenubarTheme . commonColor ,
) ,
) ;
}
Widget _buildMonitor ( BuildContext context ) {
final pi = widget . ffi . ffiModel . pi ;
2022-08-28 21:55:16 +08:00
return mod_menu . PopupMenuButton (
2022-08-26 23:28:08 +08:00
tooltip: translate ( ' Select Monitor ' ) ,
padding: EdgeInsets . zero ,
2022-08-28 21:55:16 +08:00
position: mod_menu . PopupMenuPosition . under ,
2022-08-26 23:28:08 +08:00
icon: Stack (
alignment: Alignment . center ,
children: [
2022-08-28 21:55:16 +08:00
const Icon (
2022-08-26 23:28:08 +08:00
Icons . personal_video ,
color: _MenubarTheme . commonColor ,
) ,
Padding (
2022-08-28 21:55:16 +08:00
padding: const EdgeInsets . only ( bottom: 3.9 ) ,
2022-08-26 23:28:08 +08:00
child: Obx ( ( ) {
RxInt display = CurrentDisplayState . find ( widget . id ) ;
return Text (
" ${ display . value + 1 } / ${ pi . displays . length } " ,
2022-08-28 21:55:16 +08:00
style: const TextStyle (
color: _MenubarTheme . commonColor , fontSize: 8 ) ,
2022-08-26 23:28:08 +08:00
) ;
} ) ,
)
] ,
) ,
itemBuilder: ( BuildContext context ) {
final List < Widget > rowChildren = [ ] ;
for ( int i = 0 ; i < pi . displays . length ; i + + ) {
2022-08-29 18:48:12 +08:00
rowChildren . add (
Stack (
2022-08-26 23:28:08 +08:00
alignment: Alignment . center ,
children: [
2022-08-28 21:55:16 +08:00
const Icon (
2022-08-26 23:28:08 +08:00
Icons . personal_video ,
color: _MenubarTheme . commonColor ,
) ,
TextButton (
child: Container (
alignment: AlignmentDirectional . center ,
constraints:
2022-08-28 21:55:16 +08:00
const BoxConstraints ( minHeight: _MenubarTheme . height ) ,
2022-08-26 23:28:08 +08:00
child: Padding (
2022-08-28 21:55:16 +08:00
padding: const EdgeInsets . only ( bottom: 2.5 ) ,
2022-08-26 23:28:08 +08:00
child: Text (
( i + 1 ) . toString ( ) ,
2022-08-28 21:55:16 +08:00
style:
const TextStyle ( color: _MenubarTheme . commonColor ) ,
2022-08-26 23:28:08 +08:00
) ,
) ) ,
onPressed: ( ) {
RxInt display = CurrentDisplayState . find ( widget . id ) ;
if ( display . value ! = i ) {
bind . sessionSwitchDisplay ( id: widget . id , value: i ) ;
pi . currentDisplay = i ;
display . value = i ;
}
} ,
)
] ,
) ,
2022-08-29 18:48:12 +08:00
) ;
2022-08-26 23:28:08 +08:00
}
2022-08-28 21:55:16 +08:00
return < mod_menu . PopupMenuEntry < String > > [
mod_menu . PopupMenuItem < String > (
2022-08-26 23:28:08 +08:00
height: _MenubarTheme . height ,
padding: EdgeInsets . zero ,
child: Row (
mainAxisAlignment: MainAxisAlignment . center ,
children: rowChildren ) ,
)
] ;
} ,
) ;
}
Widget _buildControl ( BuildContext context ) {
2022-08-28 21:55:16 +08:00
return mod_menu . PopupMenuButton (
2022-08-26 23:28:08 +08:00
padding: EdgeInsets . zero ,
2022-08-28 21:55:16 +08:00
icon: const Icon (
2022-08-26 23:28:08 +08:00
Icons . bolt ,
color: _MenubarTheme . commonColor ,
) ,
tooltip: translate ( ' Control Actions ' ) ,
2022-08-28 21:55:16 +08:00
position: mod_menu . PopupMenuPosition . under ,
2022-08-26 23:28:08 +08:00
itemBuilder: ( BuildContext context ) = > _getControlMenu ( )
. map ( ( entry ) = > entry . build (
context ,
2022-08-28 21:55:16 +08:00
const MenuConfig (
2022-08-26 23:28:08 +08:00
commonColor: _MenubarTheme . commonColor ,
2022-08-29 18:48:12 +08:00
height: _MenubarTheme . height ,
dividerHeight: _MenubarTheme . dividerHeight ,
2022-08-26 23:28:08 +08:00
) ) )
2022-08-30 17:20:25 +08:00
. expand ( ( i ) = > i )
2022-08-26 23:28:08 +08:00
. toList ( ) ,
) ;
}
Widget _buildDisplay ( BuildContext context ) {
2022-08-28 21:55:16 +08:00
return mod_menu . PopupMenuButton (
2022-08-26 23:28:08 +08:00
padding: EdgeInsets . zero ,
2022-08-28 21:55:16 +08:00
icon: const Icon (
2022-08-26 23:28:08 +08:00
Icons . tv ,
color: _MenubarTheme . commonColor ,
) ,
tooltip: translate ( ' Display Settings ' ) ,
2022-08-28 21:55:16 +08:00
position: mod_menu . PopupMenuPosition . under ,
2022-08-26 23:28:08 +08:00
onSelected: ( String item ) { } ,
itemBuilder: ( BuildContext context ) = > _getDisplayMenu ( )
. map ( ( entry ) = > entry . build (
context ,
2022-08-28 21:55:16 +08:00
const MenuConfig (
2022-08-26 23:28:08 +08:00
commonColor: _MenubarTheme . commonColor ,
2022-08-29 18:48:12 +08:00
height: _MenubarTheme . height ,
dividerHeight: _MenubarTheme . dividerHeight ,
2022-08-26 23:28:08 +08:00
) ) )
2022-08-30 17:20:25 +08:00
. expand ( ( i ) = > i )
2022-08-26 23:28:08 +08:00
. toList ( ) ,
) ;
}
Widget _buildClose ( BuildContext context ) {
return IconButton (
tooltip: translate ( ' Close ' ) ,
onPressed: ( ) {
clientClose ( widget . ffi . dialogManager ) ;
} ,
2022-08-28 21:55:16 +08:00
icon: const Icon (
2022-08-26 23:28:08 +08:00
Icons . close ,
color: _MenubarTheme . commonColor ,
) ,
) ;
}
List < MenuEntryBase < String > > _getControlMenu ( ) {
final pi = widget . ffi . ffiModel . pi ;
final perms = widget . ffi . ffiModel . permissions ;
final List < MenuEntryBase < String > > displayMenu = [ ] ;
if ( pi . version . isNotEmpty ) {
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' Refresh ' ) ,
style: style ,
) ,
proc: ( ) {
bind . sessionRefresh ( id: widget . id ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' OS Password ' ) ,
style: style ,
) ,
proc: ( ) {
showSetOSPassword ( widget . id , false , widget . ffi . dialogManager ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
if ( ! isWebDesktop ) {
if ( perms [ ' keyboard ' ] ! = false & & perms [ ' clipboard ' ] ! = false ) {
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' Paste ' ) ,
style: style ,
) ,
proc: ( ) {
( ) async {
ClipboardData ? data =
await Clipboard . getData ( Clipboard . kTextPlain ) ;
if ( data ! = null & & data . text ! = null ) {
bind . sessionInputString ( id: widget . id , value: data . text ? ? " " ) ;
}
} ( ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' Reset canvas ' ) ,
style: style ,
) ,
proc: ( ) {
widget . ffi . cursorModel . reset ( ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
if ( perms [ ' keyboard ' ] ! = false ) {
if ( pi . platform = = ' Linux ' | | pi . sasEnabled ) {
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
2022-08-28 21:55:16 +08:00
' ${ translate ( " Insert " ) } Ctrl + Alt + Del ' ,
2022-08-26 23:28:08 +08:00
style: style ,
) ,
proc: ( ) {
bind . sessionCtrlAltDel ( id: widget . id ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' Insert Lock ' ) ,
style: style ,
) ,
proc: ( ) {
bind . sessionLockScreen ( id: widget . id ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
if ( pi . platform = = ' Windows ' ) {
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Obx ( ( ) = > Text (
translate (
2022-08-28 21:55:16 +08:00
' ${ BlockInputState . find ( widget . id ) . value ? " Unb " : " B " } lock user input ' ) ,
2022-08-26 23:28:08 +08:00
style: style ,
) ) ,
proc: ( ) {
RxBool blockInput = BlockInputState . find ( widget . id ) ;
bind . sessionToggleOption (
id: widget . id ,
2022-08-28 21:55:16 +08:00
value: ' ${ blockInput . value ? " un " : " " } block-input ' ) ;
2022-08-26 23:28:08 +08:00
blockInput . value = ! blockInput . value ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
}
if ( gFFI . ffiModel . permissions [ " restart " ] ! = false & &
( pi . platform = = " Linux " | |
pi . platform = = " Windows " | |
pi . platform = = " Mac OS " ) ) {
displayMenu . add ( MenuEntryButton < String > (
childBuilder: ( TextStyle ? style ) = > Text (
translate ( ' Restart Remote Device ' ) ,
style: style ,
) ,
proc: ( ) {
showRestartRemoteDevice ( pi , widget . id , gFFI . dialogManager ) ;
} ,
2022-08-31 18:41:55 +08:00
dismissOnClicked: true ,
2022-08-26 23:28:08 +08:00
) ) ;
}
return displayMenu ;
}
List < MenuEntryBase < String > > _getDisplayMenu ( ) {
final displayMenu = [
2022-08-30 17:20:25 +08:00
MenuEntryRadios < String > (
2022-08-26 23:28:08 +08:00
text: translate ( ' Ratio ' ) ,
optionsGetter: ( ) = > [
2022-08-31 18:41:55 +08:00
MenuEntryRadioOption (
text: translate ( ' Scale original ' ) , value: ' original ' ) ,
MenuEntryRadioOption (
text: translate ( ' Scale adaptive ' ) , value: ' adaptive ' ) ,
2022-08-26 23:28:08 +08:00
] ,
curOptionGetter: ( ) async {
return await bind . sessionGetOption (
id: widget . id , arg: ' view-style ' ) ? ?
2022-08-31 16:41:05 +08:00
' adaptive ' ;
2022-08-26 23:28:08 +08:00
} ,
2022-08-31 18:41:55 +08:00
optionSetter: ( String oldValue , String newValue ) async {
2022-08-26 23:28:08 +08:00
await bind . sessionPeerOption (
2022-08-31 18:41:55 +08:00
id: widget . id , name: " view-style " , value: newValue ) ;
2022-08-26 23:28:08 +08:00
widget . ffi . canvasModel . updateViewStyle ( ) ;
} ) ,
2022-08-30 17:20:25 +08:00
MenuEntryDivider < String > ( ) ,
MenuEntryRadios < String > (
2022-08-26 23:28:08 +08:00
text: translate ( ' Scroll Style ' ) ,
optionsGetter: ( ) = > [
2022-08-31 18:41:55 +08:00
MenuEntryRadioOption (
text: translate ( ' ScrollAuto ' ) , value: ' scrollauto ' ) ,
MenuEntryRadioOption (
text: translate ( ' Scrollbar ' ) , value: ' scrollbar ' ) ,
2022-08-26 23:28:08 +08:00
] ,
curOptionGetter: ( ) async {
return await bind . sessionGetOption (
id: widget . id , arg: ' scroll-style ' ) ? ?
' ' ;
} ,
2022-08-31 18:41:55 +08:00
optionSetter: ( String oldValue , String newValue ) async {
2022-08-26 23:28:08 +08:00
await bind . sessionPeerOption (
2022-08-31 18:41:55 +08:00
id: widget . id , name: " scroll-style " , value: newValue ) ;
2022-08-26 23:28:08 +08:00
widget . ffi . canvasModel . updateScrollStyle ( ) ;
} ) ,
2022-08-30 17:20:25 +08:00
MenuEntryDivider < String > ( ) ,
MenuEntryRadios < String > (
2022-08-26 23:28:08 +08:00
text: translate ( ' Image Quality ' ) ,
optionsGetter: ( ) = > [
2022-08-31 18:41:55 +08:00
MenuEntryRadioOption (
text: translate ( ' Good image quality ' ) , value: ' best ' ) ,
MenuEntryRadioOption (
text: translate ( ' Balanced ' ) , value: ' balanced ' ) ,
MenuEntryRadioOption (
text: translate ( ' Optimize reaction time ' ) , value: ' low ' ) ,
MenuEntryRadioOption (
text: translate ( ' Custom ' ) ,
value: ' custom ' ,
dismissOnClicked: true ) ,
2022-08-26 23:28:08 +08:00
] ,
curOptionGetter: ( ) async {
String quality =
await bind . sessionGetImageQuality ( id: widget . id ) ? ? ' balanced ' ;
if ( quality = = ' ' ) quality = ' balanced ' ;
return quality ;
} ,
2022-08-31 18:41:55 +08:00
optionSetter: ( String oldValue , String newValue ) async {
if ( oldValue ! = newValue ) {
await bind . sessionSetImageQuality ( id: widget . id , value: newValue ) ;
}
if ( newValue = = ' custom ' ) {
2022-08-31 23:02:02 -07:00
final btnCancel = msgBoxButton ( translate ( ' Close ' ) , ( ) {
2022-08-31 18:41:55 +08:00
widget . ffi . dialogManager . dismissAll ( ) ;
} ) ;
final quality =
await bind . sessionGetCustomImageQuality ( id: widget . id ) ;
final double initValue = quality ! = null & & quality . isNotEmpty
? quality [ 0 ] . toDouble ( )
: 50.0 ;
final RxDouble sliderValue = RxDouble ( initValue ) ;
2022-08-31 23:02:02 -07:00
final rxReplay = rxdart . ReplaySubject < double > ( ) ;
rxReplay
. throttleTime ( const Duration ( milliseconds: 1000 ) ,
trailing: true , leading: false )
. listen ( ( double v ) {
( ) async {
await bind . sessionSetCustomImageQuality (
id: widget . id , value: v . toInt ( ) ) ;
} ( ) ;
} ) ;
final slider = Obx ( ( ) {
return Slider (
value: sliderValue . value ,
max: 100 ,
divisions: 100 ,
label: sliderValue . value . round ( ) . toString ( ) ,
onChanged: ( double value ) {
sliderValue . value = value ;
rxReplay . add ( value ) ;
} ,
) ;
} ) ;
2022-08-31 18:41:55 +08:00
msgBoxCommon ( widget . ffi . dialogManager , ' Custom Image Quality ' ,
slider , [ btnCancel ] ) ;
}
2022-08-26 23:28:08 +08:00
} ) ,
2022-08-30 17:20:25 +08:00
MenuEntryDivider < String > ( ) ,
2022-08-26 23:28:08 +08:00
MenuEntrySwitch < String > (
text: translate ( ' Show remote cursor ' ) ,
getter: ( ) async {
2022-08-28 21:55:16 +08:00
return bind . sessionGetToggleOptionSync (
2022-08-26 23:28:08 +08:00
id: widget . id , arg: ' show-remote-cursor ' ) ;
} ,
setter: ( bool v ) async {
await bind . sessionToggleOption (
id: widget . id , value: ' show-remote-cursor ' ) ;
} ) ,
MenuEntrySwitch < String > (
text: translate ( ' Show quality monitor ' ) ,
getter: ( ) async {
2022-08-28 21:55:16 +08:00
return bind . sessionGetToggleOptionSync (
2022-08-26 23:28:08 +08:00
id: widget . id , arg: ' show-quality-monitor ' ) ;
} ,
setter: ( bool v ) async {
await bind . sessionToggleOption (
id: widget . id , value: ' show-quality-monitor ' ) ;
widget . ffi . qualityMonitorModel . checkShowQualityMonitor ( widget . id ) ;
} ) ,
] ;
final perms = widget . ffi . ffiModel . permissions ;
final pi = widget . ffi . ffiModel . pi ;
if ( perms [ ' audio ' ] ! = false ) {
displayMenu . add ( _createSwitchMenuEntry ( ' Mute ' , ' disable-audio ' ) ) ;
}
if ( perms [ ' keyboard ' ] ! = false ) {
if ( perms [ ' clipboard ' ] ! = false ) {
displayMenu . add (
_createSwitchMenuEntry ( ' Disable clipboard ' , ' disable-clipboard ' ) ) ;
}
displayMenu . add ( _createSwitchMenuEntry (
' Lock after session end ' , ' lock-after-session-end ' ) ) ;
if ( pi . platform = = ' Windows ' ) {
displayMenu . add ( MenuEntrySwitch2 < String > (
text: translate ( ' Privacy mode ' ) ,
getter: ( ) {
return PrivacyModeState . find ( widget . id ) ;
} ,
setter: ( bool v ) async {
Navigator . pop ( context ) ;
await bind . sessionToggleOption (
id: widget . id , value: ' privacy-mode ' ) ;
} ) ) ;
}
}
return displayMenu ;
}
MenuEntrySwitch < String > _createSwitchMenuEntry ( String text , String option ) {
return MenuEntrySwitch < String > (
text: translate ( text ) ,
getter: ( ) async {
return bind . sessionGetToggleOptionSync ( id: widget . id , arg: option ) ;
} ,
setter: ( bool v ) async {
await bind . sessionToggleOption ( id: widget . id , value: option ) ;
} ) ;
}
}
void showSetOSPassword (
String id , bool login , OverlayDialogManager dialogManager ) async {
final controller = TextEditingController ( ) ;
var password = await bind . sessionGetOption ( id: id , arg: " os-password " ) ? ? " " ;
var autoLogin = await bind . sessionGetOption ( id: id , arg: " auto-login " ) ! = " " ;
controller . text = password ;
dialogManager . show ( ( setState , close ) {
return CustomAlertDialog (
title: Text ( translate ( ' OS Password ' ) ) ,
content: Column ( mainAxisSize: MainAxisSize . min , children: [
PasswordWidget ( controller: controller ) ,
CheckboxListTile (
contentPadding: const EdgeInsets . all ( 0 ) ,
dense: true ,
controlAffinity: ListTileControlAffinity . leading ,
title: Text (
translate ( ' Auto Login ' ) ,
) ,
value: autoLogin ,
onChanged: ( v ) {
if ( v = = null ) return ;
setState ( ( ) = > autoLogin = v ) ;
} ,
) ,
] ) ,
actions: [
TextButton (
style: flatButtonStyle ,
onPressed: ( ) {
close ( ) ;
} ,
child: Text ( translate ( ' Cancel ' ) ) ,
) ,
TextButton (
style: flatButtonStyle ,
onPressed: ( ) {
var text = controller . text . trim ( ) ;
bind . sessionPeerOption ( id: id , name: " os-password " , value: text ) ;
bind . sessionPeerOption (
id: id , name: " auto-login " , value: autoLogin ? ' Y ' : ' ' ) ;
if ( text ! = " " & & login ) {
bind . sessionInputOsPassword ( id: id , value: text ) ;
}
close ( ) ;
} ,
child: Text ( translate ( ' OK ' ) ) ,
) ,
] ) ;
} ) ;
}