2022-10-17 22:26:18 +09:00
import ' dart:async ' ;
2022-06-17 22:57:41 +08:00
import ' dart:io ' ;
2022-07-09 19:14:40 +08:00
import ' dart:math ' ;
2022-06-17 22:57:41 +08:00
2023-02-24 14:15:54 +08:00
import ' package:flutter_hbb/desktop/widgets/dragable_divider.dart ' ;
2023-02-22 22:13:21 +01:00
import ' package:percent_indicator/percent_indicator.dart ' ;
2022-08-16 13:28:48 +08:00
import ' package:desktop_drop/desktop_drop.dart ' ;
2022-10-20 11:20:41 +09:00
import ' package:flutter/gestures.dart ' ;
2022-06-17 22:57:41 +08:00
import ' package:flutter/material.dart ' ;
2022-10-17 22:26:18 +09:00
import ' package:flutter/services.dart ' ;
2022-08-16 11:46:51 +08:00
import ' package:flutter_breadcrumb/flutter_breadcrumb.dart ' ;
2022-12-13 12:55:41 +08:00
import ' package:flutter_hbb/desktop/widgets/list_search_action_listener.dart ' ;
2023-02-22 22:13:21 +01:00
import ' package:flutter_hbb/desktop/widgets/menu_button.dart ' ;
2022-10-18 23:56:36 +09:00
import ' package:flutter_hbb/desktop/widgets/tabbar_widget.dart ' ;
2022-06-17 22:57:41 +08:00
import ' package:flutter_hbb/models/file_model.dart ' ;
2023-02-22 22:13:21 +01:00
import ' package:flutter_svg/flutter_svg.dart ' ;
2022-06-17 22:57:41 +08:00
import ' package:get/get.dart ' ;
import ' package:provider/provider.dart ' ;
import ' package:wakelock/wakelock.dart ' ;
2023-02-22 22:13:21 +01:00
2022-10-13 20:22:11 +09:00
import ' ../../consts.dart ' ;
import ' ../../desktop/widgets/material_mod_popup_menu.dart ' as mod_menu ;
2022-06-17 22:57:41 +08:00
import ' ../../common.dart ' ;
import ' ../../models/model.dart ' ;
2022-08-03 22:03:31 +08:00
import ' ../../models/platform_model.dart ' ;
2022-10-13 20:22:11 +09:00
import ' ../widgets/popup_menu.dart ' ;
2022-06-17 22:57:41 +08:00
2022-10-13 10:17:20 +09:00
/// status of location bar
enum LocationStatus {
/// normal bread crumb bar
bread ,
/// show path text field
pathLocation ,
/// show file search bar text field
fileSearchBar
}
2022-08-16 11:46:51 +08:00
2022-12-13 12:55:41 +08:00
/// The status of currently focused scope of the mouse
enum MouseFocusScope {
/// Mouse is in local field.
local ,
/// Mouse is in remote field.
remote ,
/// Mouse is not in local field, remote neither.
none
}
2022-06-17 22:57:41 +08:00
class FileManagerPage extends StatefulWidget {
2023-02-13 16:40:24 +08:00
const FileManagerPage ( { Key ? key , required this . id , this . forceRelay } )
: super ( key: key ) ;
2022-06-17 22:57:41 +08:00
final String id ;
2023-02-13 16:40:24 +08:00
final bool ? forceRelay ;
2022-06-17 22:57:41 +08:00
@ override
State < StatefulWidget > createState ( ) = > _FileManagerPageState ( ) ;
}
class _FileManagerPageState extends State < FileManagerPage >
with AutomaticKeepAliveClientMixin {
2022-07-09 19:14:40 +08:00
final _localSelectedItems = SelectedItems ( ) ;
final _remoteSelectedItems = SelectedItems ( ) ;
2022-06-17 22:57:41 +08:00
2022-08-16 11:46:51 +08:00
final _locationStatusLocal = LocationStatus . bread . obs ;
final _locationStatusRemote = LocationStatus . bread . obs ;
2022-10-17 22:26:18 +09:00
final _locationNodeLocal = FocusNode ( debugLabel: " locationNodeLocal " ) ;
final _locationNodeRemote = FocusNode ( debugLabel: " locationNodeRemote " ) ;
2022-10-19 10:52:29 +09:00
final _locationBarKeyLocal = GlobalKey ( debugLabel: " locationBarKeyLocal " ) ;
final _locationBarKeyRemote = GlobalKey ( debugLabel: " locationBarKeyRemote " ) ;
2022-08-16 12:28:12 +08:00
final _searchTextLocal = " " . obs ;
final _searchTextRemote = " " . obs ;
final _breadCrumbScrollerLocal = ScrollController ( ) ;
final _breadCrumbScrollerRemote = ScrollController ( ) ;
2022-12-13 12:55:41 +08:00
final _mouseFocusScope = Rx < MouseFocusScope > ( MouseFocusScope . none ) ;
final _keyboardNodeLocal = FocusNode ( debugLabel: " keyboardNodeLocal " ) ;
final _keyboardNodeRemote = FocusNode ( debugLabel: " keyboardNodeRemote " ) ;
final _listSearchBufferLocal = TimeoutStringBuffer ( ) ;
final _listSearchBufferRemote = TimeoutStringBuffer ( ) ;
2023-02-24 14:15:54 +08:00
final _nameColWidthLocal = kDesktopFileTransferNameColWidth . obs ;
final _modifiedColWidthLocal = kDesktopFileTransferModifiedColWidth . obs ;
final _nameColWidthRemote = kDesktopFileTransferNameColWidth . obs ;
final _modifiedColWidthRemote = kDesktopFileTransferModifiedColWidth . obs ;
2022-08-16 12:28:12 +08:00
2022-10-17 22:26:18 +09:00
/// [_lastClickTime], [_lastClickEntry] help to handle double click
2022-11-02 11:32:30 +08:00
int _lastClickTime =
DateTime . now ( ) . millisecondsSinceEpoch - bind . getDoubleClickTime ( ) - 1000 ;
2022-10-17 22:26:18 +09:00
Entry ? _lastClickEntry ;
2022-10-13 10:17:20 +09:00
final _dropMaskVisible = false . obs ; // TODO impl drop mask
2023-02-08 22:01:15 +09:00
final _overlayKeyState = OverlayKeyState ( ) ;
2022-08-16 13:28:48 +08:00
2022-08-16 12:28:12 +08:00
ScrollController getBreadCrumbScrollController ( bool isLocal ) {
return isLocal ? _breadCrumbScrollerLocal : _breadCrumbScrollerRemote ;
}
2022-08-16 11:46:51 +08:00
2022-10-19 10:52:29 +09:00
GlobalKey getLocationBarKey ( bool isLocal ) {
return isLocal ? _locationBarKeyLocal : _locationBarKeyRemote ;
}
2022-08-12 18:42:02 +08:00
late FFI _ffi ;
2022-06-17 22:57:41 +08:00
FileModel get model = > _ffi . fileModel ;
2022-10-19 10:52:29 +09:00
SelectedItems getSelectedItems ( bool isLocal ) {
2022-07-09 19:14:40 +08:00
return isLocal ? _localSelectedItems : _remoteSelectedItems ;
}
2022-06-17 22:57:41 +08:00
@ override
void initState ( ) {
super . initState ( ) ;
2022-08-12 18:42:02 +08:00
_ffi = FFI ( ) ;
2023-02-13 16:40:24 +08:00
_ffi . start ( widget . id , isFileTransfer: true , forceRelay: widget . forceRelay ) ;
2022-09-06 19:08:45 +08:00
WidgetsBinding . instance . addPostFrameCallback ( ( _ ) {
_ffi . dialogManager
. showLoading ( translate ( ' Connecting... ' ) , onCancel: closeConnection ) ;
} ) ;
2022-08-12 18:42:02 +08:00
Get . put ( _ffi , tag: ' ft_ ${ widget . id } ' ) ;
2022-06-17 22:57:41 +08:00
if ( ! Platform . isLinux ) {
Wakelock . enable ( ) ;
}
2022-09-13 21:36:38 +08:00
debugPrint ( " File manager page init success with id ${ widget . id } " ) ;
2022-12-04 22:41:44 +09:00
model . onDirChanged = breadCrumbScrollToEnd ;
2022-08-16 11:46:51 +08:00
// register location listener
_locationNodeLocal . addListener ( onLocalLocationFocusChanged ) ;
_locationNodeRemote . addListener ( onRemoteLocationFocusChanged ) ;
2023-02-08 22:01:15 +09:00
_ffi . dialogManager . setOverlayState ( _overlayKeyState ) ;
2022-06-17 22:57:41 +08:00
}
@ override
void dispose ( ) {
2022-12-04 22:41:44 +09:00
model . onClose ( ) . whenComplete ( ( ) {
_ffi . close ( ) ;
_ffi . dialogManager . dismissAll ( ) ;
if ( ! Platform . isLinux ) {
Wakelock . disable ( ) ;
}
Get . delete < FFI > ( tag: ' ft_ ${ widget . id } ' ) ;
_locationNodeLocal . removeListener ( onLocalLocationFocusChanged ) ;
_locationNodeRemote . removeListener ( onRemoteLocationFocusChanged ) ;
_locationNodeLocal . dispose ( ) ;
_locationNodeRemote . dispose ( ) ;
} ) ;
2022-06-17 22:57:41 +08:00
super . dispose ( ) ;
}
@ override
Widget build ( BuildContext context ) {
super . build ( context ) ;
2023-02-08 22:01:15 +09:00
return Overlay ( key: _overlayKeyState . key , initialEntries: [
OverlayEntry ( builder: ( _ ) {
2022-08-19 12:44:35 +08:00
return ChangeNotifierProvider . value (
value: _ffi . fileModel ,
2022-10-11 09:59:27 +09:00
child: Consumer < FileModel > ( builder: ( context , model , child ) {
2022-10-13 10:17:20 +09:00
return Scaffold (
2023-02-22 22:13:21 +01:00
backgroundColor: Theme . of ( context ) . scaffoldBackgroundColor ,
2022-10-13 10:17:20 +09:00
body: Row (
children: [
2023-02-26 09:13:42 +01:00
Flexible ( flex: 3 , child: body ( isLocal: true ) ) ,
Flexible ( flex: 3 , child: body ( isLocal: false ) ) ,
Flexible ( flex: 2 , child: statusList ( ) )
2022-10-13 10:17:20 +09:00
] ,
) ,
) ;
2022-08-19 12:44:35 +08:00
} ) ) ;
} )
] ) ;
2022-06-17 22:57:41 +08:00
}
2022-06-27 16:44:34 +08:00
Widget menu ( { bool isLocal = false } ) {
2022-10-13 20:22:11 +09:00
var menuPos = RelativeRect . fill ;
2022-10-19 10:52:29 +09:00
final List < MenuEntryBase < String > > items = [
2022-10-13 20:22:11 +09:00
MenuEntrySwitch < String > (
switchType: SwitchType . scheckbox ,
text: translate ( " Show Hidden Files " ) ,
getter: ( ) async {
return model . getCurrentShowHidden ( isLocal ) ;
} ,
setter: ( bool v ) async {
model . toggleShowHidden ( local: isLocal ) ;
2022-06-27 16:44:34 +08:00
} ,
2022-10-13 20:22:11 +09:00
padding: kDesktopMenuPadding ,
dismissOnClicked: true ,
) ,
2022-10-19 10:52:29 +09:00
MenuEntryButton (
childBuilder: ( style ) = > Text ( translate ( " Select All " ) , style: style ) ,
proc: ( ) = > setState ( ( ) = > getSelectedItems ( isLocal )
. selectAll ( model . getCurrentDir ( isLocal ) . entries ) ) ,
padding: kDesktopMenuPadding ,
dismissOnClicked: true ) ,
MenuEntryButton (
childBuilder: ( style ) = >
Text ( translate ( " Unselect All " ) , style: style ) ,
proc: ( ) = > setState ( ( ) = > getSelectedItems ( isLocal ) . clear ( ) ) ,
padding: kDesktopMenuPadding ,
dismissOnClicked: true )
2022-10-13 20:22:11 +09:00
] ;
return Listener (
2023-02-22 22:13:21 +01:00
onPointerDown: ( e ) {
final x = e . position . dx ;
final y = e . position . dy ;
menuPos = RelativeRect . fromLTRB ( x , y , x , y ) ;
} ,
child: MenuButton (
onPressed: ( ) = > mod_menu . showMenu (
context: context ,
position: menuPos ,
items: items
. map (
( e ) = > e . build (
context ,
MenuConfig (
commonColor: CustomPopupMenuTheme . commonColor ,
height: CustomPopupMenuTheme . height ,
dividerHeight: CustomPopupMenuTheme . dividerHeight ) ,
) ,
)
. expand ( ( i ) = > i )
. toList ( ) ,
elevation: 8 ,
) ,
child: SvgPicture . asset (
" assets/dots.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ,
) ;
2022-06-27 16:44:34 +08:00
}
2022-06-21 18:14:44 +08:00
Widget body ( { bool isLocal = false } ) {
2022-12-13 12:55:41 +08:00
final scrollController = ScrollController ( ) ;
2022-07-01 17:17:25 +08:00
return Container (
margin: const EdgeInsets . all ( 16.0 ) ,
padding: const EdgeInsets . all ( 8.0 ) ,
2022-08-16 13:28:48 +08:00
child: DropTarget (
onDragDone: ( detail ) = > handleDragDone ( detail , isLocal ) ,
onDragEntered: ( enter ) {
_dropMaskVisible . value = true ;
} ,
onDragExited: ( exit ) {
_dropMaskVisible . value = false ;
} ,
2023-02-22 22:13:21 +01:00
child: Column (
crossAxisAlignment: CrossAxisAlignment . start ,
children: [
headTools ( isLocal ) ,
Expanded (
2022-08-16 13:28:48 +08:00
child: Row (
2023-02-22 22:13:21 +01:00
crossAxisAlignment: CrossAxisAlignment . start ,
children: [
Expanded (
child: _buildFileList ( context , isLocal , scrollController ) ,
)
] ,
) ,
) ,
] ,
) ,
2022-08-16 13:28:48 +08:00
) ,
2022-07-01 17:17:25 +08:00
) ;
2022-06-17 22:57:41 +08:00
}
2023-02-15 15:03:19 +08:00
Widget _buildFileList (
2022-12-13 12:55:41 +08:00
BuildContext context , bool isLocal , ScrollController scrollController ) {
2022-10-17 22:26:18 +09:00
final fd = model . getCurrentDir ( isLocal ) ;
final entries = fd . entries ;
2023-02-15 15:03:19 +08:00
final selectedEntries = getSelectedItems ( isLocal ) ;
2022-10-17 22:26:18 +09:00
2022-12-13 12:55:41 +08:00
return MouseRegion (
onEnter: ( evt ) {
_mouseFocusScope . value =
isLocal ? MouseFocusScope . local : MouseFocusScope . remote ;
if ( isLocal ) {
_keyboardNodeLocal . requestFocus ( ) ;
} else {
_keyboardNodeRemote . requestFocus ( ) ;
}
} ,
onExit: ( evt ) {
_mouseFocusScope . value = MouseFocusScope . none ;
} ,
child: ListSearchActionListener (
node: isLocal ? _keyboardNodeLocal : _keyboardNodeRemote ,
buffer: isLocal ? _listSearchBufferLocal : _listSearchBufferRemote ,
onNext: ( buffer ) {
debugPrint ( " searching next for $ buffer " ) ;
assert ( buffer . length = = 1 ) ;
assert ( selectedEntries . length < = 1 ) ;
var skipCount = 0 ;
if ( selectedEntries . items . isNotEmpty ) {
final index = entries . indexOf ( selectedEntries . items . first ) ;
if ( index < 0 ) {
return ;
}
skipCount = index + 1 ;
}
2023-02-27 09:44:52 +01:00
var searchResult = entries . skip ( skipCount ) . where (
( element ) = > element . name . toLowerCase ( ) . startsWith ( buffer ) ) ;
2022-12-13 12:55:41 +08:00
if ( searchResult . isEmpty ) {
2022-12-14 11:56:36 +08:00
// cannot find next, lets restart search from head
2023-02-24 14:15:54 +08:00
debugPrint ( " restart search from head " ) ;
2023-02-27 09:44:52 +01:00
searchResult = entries . where (
( element ) = > element . name . toLowerCase ( ) . startsWith ( buffer ) ) ;
2022-12-13 12:55:41 +08:00
}
if ( searchResult . isEmpty ) {
2022-12-14 11:56:36 +08:00
setState ( ( ) {
getSelectedItems ( isLocal ) . clear ( ) ;
} ) ;
2022-12-13 12:55:41 +08:00
return ;
}
2023-02-22 22:13:21 +01:00
_jumpToEntry ( isLocal , searchResult . first , scrollController ,
2023-02-24 15:56:37 +08:00
kDesktopFileTransferRowHeight ) ;
2022-12-13 12:55:41 +08:00
} ,
onSearch: ( buffer ) {
debugPrint ( " searching for $ buffer " ) ;
final selectedEntries = getSelectedItems ( isLocal ) ;
2023-02-27 09:44:52 +01:00
final searchResult = entries . where (
( element ) = > element . name . toLowerCase ( ) . startsWith ( buffer ) ) ;
2022-12-13 12:55:41 +08:00
selectedEntries . clear ( ) ;
if ( searchResult . isEmpty ) {
2022-12-14 11:56:36 +08:00
setState ( ( ) {
getSelectedItems ( isLocal ) . clear ( ) ;
} ) ;
2022-12-13 12:55:41 +08:00
return ;
}
2023-02-22 22:13:21 +01:00
_jumpToEntry ( isLocal , searchResult . first , scrollController ,
2023-02-24 15:56:37 +08:00
kDesktopFileTransferRowHeight ) ;
2022-12-13 12:55:41 +08:00
} ,
child: ObxValue < RxString > (
( searchText ) {
final filteredEntries = searchText . isNotEmpty
? entries . where ( ( element ) {
return element . name . contains ( searchText . value ) ;
} ) . toList ( growable: false )
: entries ;
2023-02-15 15:03:19 +08:00
final rows = filteredEntries . map ( ( entry ) {
2023-02-22 22:13:21 +01:00
final sizeStr =
entry . isFile ? readableFileSize ( entry . size . toDouble ( ) ) : " " ;
final lastModifiedStr = entry . isDrive
? " "
: " ${ entry . lastModified ( ) . toString ( ) . replaceAll ( " .000 " , " " ) } " ;
2023-02-15 15:03:19 +08:00
final isSelected = selectedEntries . contains ( entry ) ;
2023-02-22 22:13:21 +01:00
return Padding (
padding: EdgeInsets . symmetric ( vertical: 1 ) ,
child: Container (
decoration: BoxDecoration (
color: isSelected
? Theme . of ( context ) . hoverColor
: Theme . of ( context ) . cardColor ,
borderRadius: BorderRadius . all (
Radius . circular ( 5.0 ) ,
) ,
2023-02-15 15:03:19 +08:00
) ,
2023-02-22 22:13:21 +01:00
key: ValueKey ( entry . name ) ,
height: kDesktopFileTransferRowHeight ,
child: Column (
mainAxisAlignment: MainAxisAlignment . spaceAround ,
children: [
Expanded (
child: InkWell (
child: Row (
children: [
GestureDetector (
2023-02-24 14:15:54 +08:00
child: Obx (
( ) = > Container (
width: isLocal
? _nameColWidthLocal . value
: _nameColWidthRemote . value ,
child: Tooltip (
waitDuration:
Duration ( milliseconds: 500 ) ,
message: entry . name ,
child: Row ( children: [
entry . isDrive
? Image (
image: iconHardDrive ,
fit: BoxFit . scaleDown ,
color: Theme . of ( context )
. iconTheme
. color
? . withOpacity ( 0.7 ) )
. paddingAll ( 4 )
: SvgPicture . asset (
entry . isFile
? " assets/file.svg "
: " assets/folder.svg " ,
color: Theme . of ( context )
. tabBarTheme
. labelColor ,
) ,
Expanded (
child: Text (
entry . name . nonBreaking ,
overflow:
TextOverflow . ellipsis ) )
] ) ,
) ) ,
) ,
2023-02-22 22:13:21 +01:00
onTap: ( ) {
final items = getSelectedItems ( isLocal ) ;
// handle double click
if ( _checkDoubleClick ( entry ) ) {
openDirectory ( entry . path ,
isLocal: isLocal ) ;
items . clear ( ) ;
return ;
}
_onSelectedChanged (
items , filteredEntries , entry , isLocal ) ;
} ,
) ,
2023-02-24 14:15:54 +08:00
SizedBox (
width: 2.0 ,
) ,
2023-02-22 22:52:29 +01:00
GestureDetector (
2023-02-24 14:15:54 +08:00
child: Obx (
( ) = > SizedBox (
width: isLocal
? _modifiedColWidthLocal . value
: _modifiedColWidthRemote . value ,
child: Tooltip (
waitDuration:
Duration ( milliseconds: 500 ) ,
message: lastModifiedStr ,
child: Text (
lastModifiedStr ,
2023-03-03 19:33:55 +01:00
overflow: TextOverflow . ellipsis ,
2023-02-24 14:15:54 +08:00
style: TextStyle (
fontSize: 12 ,
color: MyTheme . darkGray ,
) ,
) ) ,
) ,
2023-02-22 22:13:21 +01:00
) ,
) ,
2023-02-24 14:15:54 +08:00
// Divider from header.
2023-02-22 22:13:21 +01:00
SizedBox (
2023-02-24 14:15:54 +08:00
width: 2.0 ,
) ,
Expanded (
// width: 100,
2023-02-22 22:13:21 +01:00
child: GestureDetector (
child: Tooltip (
waitDuration: Duration ( milliseconds: 500 ) ,
message: sizeStr ,
child: Text (
sizeStr ,
overflow: TextOverflow . ellipsis ,
style: TextStyle (
fontSize: 10 ,
color: MyTheme . darkGray ) ,
) ,
) ,
) ,
) ,
] ,
2023-02-15 15:03:19 +08:00
) ,
2023-02-22 22:13:21 +01:00
) ,
2023-02-15 15:03:19 +08:00
) ,
2023-02-22 22:13:21 +01:00
] ,
) ) ,
2023-02-15 15:03:19 +08:00
) ;
} ) . toList ( growable: false ) ;
return Column (
children: [
// Header
2023-02-24 15:56:37 +08:00
Row (
children: [
Expanded ( child: _buildFileBrowserHeader ( context , isLocal ) ) ,
] ,
) ,
2023-02-15 15:03:19 +08:00
// Body
Expanded (
child: ListView . builder (
controller: scrollController ,
itemExtent: kDesktopFileTransferRowHeight ,
itemBuilder: ( context , index ) {
return rows [ index ] ;
} ,
itemCount: rows . length ,
) ,
) ,
] ,
2022-12-13 12:55:41 +08:00
) ;
} ,
isLocal ? _searchTextLocal : _searchTextRemote ,
) ,
) ,
2022-10-17 22:26:18 +09:00
) ;
}
2022-12-14 11:56:36 +08:00
void _jumpToEntry ( bool isLocal , Entry entry ,
2023-02-24 15:56:37 +08:00
ScrollController scrollController , double rowHeight ) {
2022-12-14 11:56:36 +08:00
final entries = model . getCurrentDir ( isLocal ) . entries ;
final index = entries . indexOf ( entry ) ;
if ( index = = - 1 ) {
debugPrint ( " entry is not valid: ${ entry . path } " ) ;
}
final selectedEntries = getSelectedItems ( isLocal ) ;
2023-02-27 09:44:52 +01:00
final searchResult = entries . where ( ( element ) = > element = = entry ) ;
2022-12-14 11:56:36 +08:00
selectedEntries . clear ( ) ;
if ( searchResult . isEmpty ) {
return ;
}
final offset = min (
max ( scrollController . position . minScrollExtent ,
entries . indexOf ( searchResult . first ) * rowHeight ) ,
scrollController . position . maxScrollExtent ) ;
scrollController . jumpTo ( offset ) ;
setState ( ( ) {
selectedEntries . add ( isLocal , searchResult . first ) ;
debugPrint ( " focused on ${ searchResult . first . name } " ) ;
} ) ;
}
2022-10-17 23:07:40 +09:00
void _onSelectedChanged ( SelectedItems selectedItems , List < Entry > entries ,
Entry entry , bool isLocal ) {
final isCtrlDown = RawKeyboard . instance . keysPressed
. contains ( LogicalKeyboardKey . controlLeft ) ;
final isShiftDown =
RawKeyboard . instance . keysPressed . contains ( LogicalKeyboardKey . shiftLeft ) ;
if ( isCtrlDown ) {
if ( selectedItems . contains ( entry ) ) {
selectedItems . remove ( entry ) ;
} else {
selectedItems . add ( isLocal , entry ) ;
}
} else if ( isShiftDown ) {
final List < int > indexGroup = [ ] ;
for ( var selected in selectedItems . items ) {
indexGroup . add ( entries . indexOf ( selected ) ) ;
}
indexGroup . add ( entries . indexOf ( entry ) ) ;
indexGroup . removeWhere ( ( e ) = > e = = - 1 ) ;
final maxIndex = indexGroup . reduce ( max ) ;
final minIndex = indexGroup . reduce ( min ) ;
selectedItems . clear ( ) ;
entries
. getRange ( minIndex , maxIndex + 1 )
. forEach ( ( e ) = > selectedItems . add ( isLocal , e ) ) ;
} else {
selectedItems . clear ( ) ;
selectedItems . add ( isLocal , entry ) ;
}
setState ( ( ) { } ) ;
}
2022-10-17 22:26:18 +09:00
bool _checkDoubleClick ( Entry entry ) {
final current = DateTime . now ( ) . millisecondsSinceEpoch ;
final elapsed = current - _lastClickTime ;
_lastClickTime = current ;
if ( _lastClickEntry = = entry ) {
2022-11-02 11:32:30 +08:00
if ( elapsed < bind . getDoubleClickTime ( ) ) {
2022-10-17 22:26:18 +09:00
return true ;
}
} else {
_lastClickEntry = entry ;
}
return false ;
}
2023-02-26 09:13:42 +01:00
Widget generateCard ( Widget child ) {
return Container (
decoration: BoxDecoration (
color: Theme . of ( context ) . cardColor ,
borderRadius: BorderRadius . all (
Radius . circular ( 15.0 ) ,
) ,
) ,
child: child ,
) ;
}
2022-07-01 12:08:52 +08:00
/// transfer status list
/// watch transfer status
Widget statusList ( ) {
2022-07-01 17:17:25 +08:00
return PreferredSize (
2023-02-25 09:44:23 +01:00
preferredSize: const Size ( 200 , double . infinity ) ,
child: Container (
margin: const EdgeInsets . only ( top: 16.0 , bottom: 16.0 , right: 16.0 ) ,
padding: const EdgeInsets . all ( 8.0 ) ,
2023-02-26 09:13:42 +01:00
child: model . jobTable . isEmpty
? generateCard (
Center (
2023-02-25 09:44:23 +01:00
child: Column (
2023-02-26 09:13:42 +01:00
mainAxisAlignment: MainAxisAlignment . center ,
2023-02-25 09:44:23 +01:00
children: [
2023-02-26 09:13:42 +01:00
SvgPicture . asset (
" assets/transfer.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
height: 40 ,
) . paddingOnly ( bottom: 10 ) ,
Text (
translate ( " No transfers in progress " ) ,
textAlign: TextAlign . center ,
textScaleFactor: 1.20 ,
style: TextStyle (
color: Theme . of ( context ) . tabBarTheme . labelColor ) ,
) ,
] ,
) ,
) ,
)
: Obx (
( ) = > ListView . builder (
controller: ScrollController ( ) ,
itemBuilder: ( BuildContext context , int index ) {
final item = model . jobTable [ index ] ;
return Padding (
padding: const EdgeInsets . only ( bottom: 5 ) ,
child: generateCard (
Column (
mainAxisSize: MainAxisSize . min ,
children: [
Row (
crossAxisAlignment: CrossAxisAlignment . center ,
2023-02-25 09:44:23 +01:00
children: [
2023-02-26 09:13:42 +01:00
Transform . rotate (
angle: item . isRemote ? pi : 0 ,
child: SvgPicture . asset (
" assets/arrow.svg " ,
color: Theme . of ( context )
. tabBarTheme
. labelColor ,
2023-02-25 09:44:23 +01:00
) ,
2023-02-26 09:13:42 +01:00
) . paddingOnly ( left: 15 ) ,
const SizedBox (
width: 16.0 ,
2023-02-25 09:44:23 +01:00
) ,
2023-02-26 09:13:42 +01:00
Expanded (
child: Column (
mainAxisSize: MainAxisSize . min ,
crossAxisAlignment:
CrossAxisAlignment . start ,
children: [
Tooltip (
waitDuration:
Duration ( milliseconds: 500 ) ,
message: item . jobName ,
child: Text (
2023-03-03 20:26:13 +01:00
item . fileName ,
2023-02-26 09:13:42 +01:00
maxLines: 1 ,
overflow: TextOverflow . ellipsis ,
) . paddingSymmetric ( vertical: 10 ) ,
) ,
Text (
' ${ translate ( " Total " ) } ${ readableFileSize ( item . totalSize . toDouble ( ) ) } ' ,
style: TextStyle (
fontSize: 12 ,
color: MyTheme . darkGray ,
) ,
) ,
Offstage (
offstage:
item . state ! = JobState . inProgress ,
child: Text (
' ${ translate ( " Speed " ) } ${ readableFileSize ( item . speed ) } /s ' ,
style: TextStyle (
fontSize: 12 ,
color: MyTheme . darkGray ,
) ,
) ,
) ,
Offstage (
offstage:
item . state = = JobState . inProgress ,
child: Text (
translate (
item . display ( ) ,
) ,
style: TextStyle (
fontSize: 12 ,
color: MyTheme . darkGray ,
) ,
) ,
) ,
Offstage (
offstage:
item . state ! = JobState . inProgress ,
child: LinearPercentIndicator (
padding: EdgeInsets . only ( right: 15 ) ,
animateFromLastPercent: true ,
center: Text (
' ${ ( item . finishedSize / item . totalSize * 100 ) . toStringAsFixed ( 0 ) } % ' ,
) ,
barRadius: Radius . circular ( 15 ) ,
percent: item . finishedSize /
item . totalSize ,
progressColor: MyTheme . accent ,
backgroundColor:
Theme . of ( context ) . hoverColor ,
lineHeight:
kDesktopFileTransferRowHeight ,
) . paddingSymmetric ( vertical: 15 ) ,
) ,
] ,
2023-02-22 22:13:21 +01:00
) ,
2023-02-25 09:44:23 +01:00
) ,
2023-02-26 09:13:42 +01:00
Row (
mainAxisAlignment: MainAxisAlignment . end ,
children: [
Offstage (
offstage: item . state ! = JobState . paused ,
child: MenuButton (
onPressed: ( ) {
model . resumeJob ( item . id ) ;
} ,
child: SvgPicture . asset (
" assets/refresh.svg " ,
color: Colors . white ,
) ,
color: MyTheme . accent ,
hoverColor: MyTheme . accent80 ,
) ,
2023-02-22 22:13:21 +01:00
) ,
2023-02-26 09:13:42 +01:00
MenuButton (
padding: EdgeInsets . only ( right: 15 ) ,
child: SvgPicture . asset (
" assets/close.svg " ,
color: Colors . white ,
) ,
onPressed: ( ) {
model . jobTable . removeAt ( index ) ;
model . cancelJob ( item . id ) ;
} ,
color: MyTheme . accent ,
hoverColor: MyTheme . accent80 ,
2023-02-25 09:44:23 +01:00
) ,
2023-02-26 09:13:42 +01:00
] ,
2023-02-25 09:44:23 +01:00
) ,
] ,
) ,
2023-02-26 09:13:42 +01:00
] ,
) . paddingSymmetric ( vertical: 10 ) ,
2023-02-25 09:44:23 +01:00
) ,
2023-02-26 09:13:42 +01:00
) ;
} ,
itemCount: model . jobTable . length ,
2023-02-22 22:13:21 +01:00
) ,
2023-02-26 09:13:42 +01:00
) ,
2023-02-25 09:44:23 +01:00
) ,
) ;
2022-07-01 12:08:52 +08:00
}
2022-08-16 11:46:51 +08:00
Widget headTools ( bool isLocal ) {
2022-10-11 09:59:27 +09:00
final locationStatus =
2022-08-16 11:46:51 +08:00
isLocal ? _locationStatusLocal : _locationStatusRemote ;
2022-10-11 09:59:27 +09:00
final locationFocus = isLocal ? _locationNodeLocal : _locationNodeRemote ;
2022-10-19 10:52:29 +09:00
final selectedItems = getSelectedItems ( isLocal ) ;
2022-08-16 11:46:51 +08:00
return Container (
2023-02-22 22:13:21 +01:00
child: Column (
children: [
// symbols
PreferredSize (
2022-08-16 11:46:51 +08:00
child: Row (
2023-02-22 22:13:21 +01:00
crossAxisAlignment: CrossAxisAlignment . center ,
2022-08-16 11:46:51 +08:00
children: [
2023-02-22 22:13:21 +01:00
Container (
width: 50 ,
height: 50 ,
decoration: BoxDecoration (
borderRadius: BorderRadius . all ( Radius . circular ( 8 ) ) ,
color: MyTheme . accent ,
) ,
padding: EdgeInsets . all ( 8.0 ) ,
child: FutureBuilder < String > (
future: bind . sessionGetPlatform (
id: _ffi . id , isRemote: ! isLocal ) ,
builder: ( context , snapshot ) {
if ( snapshot . hasData & &
snapshot . data ! . isNotEmpty ) {
return getPlatformImage ( ' ${ snapshot . data } ' ) ;
} else {
return CircularProgressIndicator (
color: Theme . of ( context )
. tabBarTheme
. labelColor ,
) ;
}
} ) ) ,
Text ( isLocal
? translate ( " Local Computer " )
: translate ( " Remote Computer " ) )
. marginOnly ( left: 8.0 )
2022-08-16 11:46:51 +08:00
] ,
2023-02-22 22:13:21 +01:00
) ,
preferredSize: Size ( double . infinity , 70 ) )
. paddingOnly ( bottom: 15 ) ,
// buttons
Row (
children: [
Row (
children: [
MenuButton (
padding: EdgeInsets . only (
right: 3 ,
) ,
2023-02-23 14:53:24 +01:00
child: RotatedBox (
quarterTurns: 2 ,
child: SvgPicture . asset (
" assets/arrow.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
2023-02-22 22:13:21 +01:00
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
onPressed: ( ) {
selectedItems . clear ( ) ;
model . goBack ( isLocal: isLocal ) ;
} ,
) ,
MenuButton (
child: RotatedBox (
quarterTurns: 3 ,
child: SvgPicture . asset (
" assets/arrow.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
onPressed: ( ) {
selectedItems . clear ( ) ;
model . goToParentDirectory ( isLocal: isLocal ) ;
} ,
) ,
] ,
) ,
Expanded (
child: Padding (
padding: const EdgeInsets . symmetric ( horizontal: 3.0 ) ,
child: Container (
decoration: BoxDecoration (
color: Theme . of ( context ) . cardColor ,
borderRadius: BorderRadius . all (
Radius . circular ( 8.0 ) ,
) ,
) ,
child: Padding (
padding: EdgeInsets . symmetric ( vertical: 2.5 ) ,
child: GestureDetector (
onTap: ( ) {
locationStatus . value =
locationStatus . value = = LocationStatus . bread
? LocationStatus . pathLocation
: LocationStatus . bread ;
Future . delayed ( Duration . zero , ( ) {
if ( locationStatus . value = =
LocationStatus . pathLocation ) {
locationFocus . requestFocus ( ) ;
}
} ) ;
} ,
child: Obx (
( ) = > Container (
child: Row (
children: [
Expanded (
child: locationStatus . value = =
LocationStatus . bread
? buildBread ( isLocal )
: buildPathLocation ( isLocal ) ) ,
] ,
) ,
) ,
) ,
) ,
) ,
) ,
) ,
) ,
Obx ( ( ) {
switch ( locationStatus . value ) {
case LocationStatus . bread:
return MenuButton (
2022-10-13 10:17:20 +09:00
onPressed: ( ) {
locationStatus . value = LocationStatus . fileSearchBar ;
final focusNode =
isLocal ? _locationNodeLocal : _locationNodeRemote ;
Future . delayed (
Duration . zero , ( ) = > focusNode . requestFocus ( ) ) ;
} ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/search.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ;
case LocationStatus . pathLocation:
return MenuButton (
2022-10-13 10:17:20 +09:00
onPressed: null ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/close.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
2022-10-13 10:17:20 +09:00
color: Theme . of ( context ) . disabledColor ,
2023-02-22 22:13:21 +01:00
hoverColor: Theme . of ( context ) . hoverColor ,
) ;
case LocationStatus . fileSearchBar:
return MenuButton (
2022-10-13 10:17:20 +09:00
onPressed: ( ) {
onSearchText ( " " , isLocal ) ;
locationStatus . value = LocationStatus . bread ;
} ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/close.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ;
}
} ) ,
MenuButton (
padding: EdgeInsets . only (
left: 3 ,
) ,
2022-08-16 11:46:51 +08:00
onPressed: ( ) {
model . refresh ( isLocal: isLocal ) ;
} ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/refresh.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ,
] ,
) ,
Row (
textDirection: isLocal ? TextDirection . ltr : TextDirection . rtl ,
children: [
Expanded (
child: Row (
mainAxisAlignment:
isLocal ? MainAxisAlignment . start : MainAxisAlignment . end ,
children: [
MenuButton (
padding: EdgeInsets . only (
right: 3 ,
) ,
onPressed: ( ) {
model . goHome ( isLocal: isLocal ) ;
} ,
child: SvgPicture . asset (
" assets/home.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ,
MenuButton (
2022-07-11 16:07:49 +08:00
onPressed: ( ) {
2022-08-16 11:46:51 +08:00
final name = TextEditingController ( ) ;
2022-09-03 18:19:50 +08:00
_ffi . dialogManager . show ( ( setState , close ) {
submit ( ) {
if ( name . value . text . isNotEmpty ) {
model . createDir (
PathUtil . join (
model . getCurrentDir ( isLocal ) . path ,
name . value . text ,
model . getCurrentIsWindows ( isLocal ) ) ,
isLocal: isLocal ) ;
close ( ) ;
}
}
cancel ( ) = > close ( false ) ;
return CustomAlertDialog (
2023-02-27 09:44:52 +01:00
title: Row (
mainAxisAlignment: MainAxisAlignment . center ,
children: [
SvgPicture . asset ( " assets/folder_new.svg " ,
color: MyTheme . accent ) ,
Text (
translate ( " Create Folder " ) ,
) . paddingOnly (
left: 10 ,
) ,
] ,
) ,
2022-09-03 18:19:50 +08:00
content: Column (
mainAxisSize: MainAxisSize . min ,
children: [
2023-03-01 14:50:50 +01:00
TextFormField (
decoration: InputDecoration (
labelText: translate (
" Please enter the folder name " ,
2023-02-27 09:44:52 +01:00
) ,
2023-03-01 14:50:50 +01:00
) ,
controller: name ,
autofocus: true ,
2022-09-03 18:19:50 +08:00
) ,
] ,
) ,
2023-03-01 14:50:50 +01:00
actions: [
dialogButton (
" Cancel " ,
icon: Icon ( Icons . close_rounded ) ,
onPressed: cancel ,
isOutline: true ,
) ,
dialogButton (
" Ok " ,
icon: Icon ( Icons . done_rounded ) ,
onPressed: submit ,
) ,
] ,
2022-09-03 18:19:50 +08:00
onSubmit: submit ,
onCancel: cancel ,
) ;
} ) ;
2022-07-11 16:07:49 +08:00
} ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/folder_new.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ,
MenuButton (
2022-10-19 10:52:29 +09:00
onPressed: validItems ( selectedItems )
? ( ) async {
await ( model . removeAction ( selectedItems ,
isLocal: isLocal ) ) ;
selectedItems . clear ( ) ;
}
: null ,
2023-02-22 22:13:21 +01:00
child: SvgPicture . asset (
" assets/trash.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
color: Theme . of ( context ) . cardColor ,
hoverColor: Theme . of ( context ) . hoverColor ,
) ,
menu ( isLocal: isLocal ) ,
] ,
) ,
2022-06-17 22:57:41 +08:00
) ,
2023-02-22 22:13:21 +01:00
ElevatedButton . icon (
style: ButtonStyle (
padding: MaterialStateProperty . all < EdgeInsetsGeometry > ( isLocal
? EdgeInsets . only ( left: 10 )
: EdgeInsets . only ( right: 10 ) ) ,
backgroundColor: MaterialStateProperty . all (
selectedItems . length = = 0
? MyTheme . accent80
: MyTheme . accent ,
) ,
) ,
2022-10-19 10:52:29 +09:00
onPressed: validItems ( selectedItems )
? ( ) {
model . sendFiles ( selectedItems , isRemote: ! isLocal ) ;
selectedItems . clear ( ) ;
}
: null ,
2023-02-22 22:13:21 +01:00
icon: isLocal
? Text (
translate ( ' Send ' ) ,
textAlign: TextAlign . right ,
style: TextStyle (
color: selectedItems . length = = 0
2023-02-25 09:50:40 +01:00
? Theme . of ( context ) . brightness = = Brightness . light
? MyTheme . grayBg
: MyTheme . darkGray
2023-02-22 22:13:21 +01:00
: Colors . white ,
) ,
)
: RotatedBox (
quarterTurns: 2 ,
child: SvgPicture . asset (
" assets/arrow.svg " ,
color: selectedItems . length = = 0
2023-02-25 09:50:40 +01:00
? Theme . of ( context ) . brightness = = Brightness . light
? MyTheme . grayBg
: MyTheme . darkGray
2023-02-22 22:13:21 +01:00
: Colors . white ,
alignment: Alignment . bottomRight ,
) ,
) ,
label: isLocal
? SvgPicture . asset (
" assets/arrow.svg " ,
color: selectedItems . length = = 0
2023-02-25 09:50:40 +01:00
? Theme . of ( context ) . brightness = = Brightness . light
? MyTheme . grayBg
: MyTheme . darkGray
2023-02-22 22:13:21 +01:00
: Colors . white ,
)
: Text (
translate ( ' Receive ' ) ,
style: TextStyle (
color: selectedItems . length = = 0
2023-02-25 09:50:40 +01:00
? Theme . of ( context ) . brightness = = Brightness . light
? MyTheme . grayBg
: MyTheme . darkGray
2023-02-22 22:13:21 +01:00
: Colors . white ,
) ,
) ,
) ,
] ,
) . marginOnly ( top: 8.0 )
] ,
) ,
) ;
2022-08-16 11:46:51 +08:00
}
2022-06-17 22:57:41 +08:00
2022-10-19 10:52:29 +09:00
bool validItems ( SelectedItems items ) {
if ( items . length > 0 ) {
// exclude DirDrive type
return items . items . any ( ( item ) = > ! item . isDrive ) ;
}
return false ;
}
2022-06-17 22:57:41 +08:00
@ override
bool get wantKeepAlive = > true ;
2022-07-11 16:07:49 +08:00
2022-08-16 11:46:51 +08:00
void onLocalLocationFocusChanged ( ) {
debugPrint ( " focus changed on local " ) ;
if ( _locationNodeLocal . hasFocus ) {
// ignore
} else {
// lost focus, change to bread
2022-10-13 10:17:20 +09:00
if ( _locationStatusLocal . value ! = LocationStatus . fileSearchBar ) {
_locationStatusLocal . value = LocationStatus . bread ;
}
2022-08-16 11:46:51 +08:00
}
}
void onRemoteLocationFocusChanged ( ) {
debugPrint ( " focus changed on remote " ) ;
if ( _locationNodeRemote . hasFocus ) {
// ignore
} else {
// lost focus, change to bread
2022-10-13 10:17:20 +09:00
if ( _locationStatusRemote . value ! = LocationStatus . fileSearchBar ) {
_locationStatusRemote . value = LocationStatus . bread ;
}
2022-08-16 11:46:51 +08:00
}
}
Widget buildBread ( bool isLocal ) {
2022-08-16 12:50:08 +08:00
final items = getPathBreadCrumbItems ( isLocal , ( list ) {
var path = " " ;
for ( var item in list ) {
path = PathUtil . join ( path , item , model . getCurrentIsWindows ( isLocal ) ) ;
}
openDirectory ( path , isLocal: isLocal ) ;
} ) ;
2022-10-19 10:52:29 +09:00
final locationBarKey = getLocationBarKey ( isLocal ) ;
2022-10-18 23:56:36 +09:00
2022-08-16 12:50:08 +08:00
return items . isEmpty
? Offstage ( )
2022-10-18 23:56:36 +09:00
: Row (
key: locationBarKey ,
mainAxisAlignment: MainAxisAlignment . spaceBetween ,
children: [
Expanded (
2023-02-22 22:13:21 +01:00
child: Listener (
// handle mouse wheel
onPointerSignal: ( e ) {
if ( e is PointerScrollEvent ) {
final sc = getBreadCrumbScrollController ( isLocal ) ;
final scale = Platform . isWindows ? 2 : 4 ;
sc . jumpTo ( sc . offset + e . scrollDelta . dy / scale ) ;
}
} ,
child: BreadCrumb (
items: items ,
divider: const Icon ( Icons . keyboard_arrow_right_rounded ) ,
overflow: ScrollableOverflow (
controller: getBreadCrumbScrollController ( isLocal ) ,
) ,
) ,
) ,
) ,
2022-10-18 23:56:36 +09:00
ActionIcon (
message: " " ,
2023-02-22 22:13:21 +01:00
icon: Icons . keyboard_arrow_down_rounded ,
2022-10-18 23:56:36 +09:00
onTap: ( ) async {
final renderBox = locationBarKey . currentContext
? . findRenderObject ( ) as RenderBox ;
locationBarKey . currentContext ? . size ;
final size = renderBox . size ;
final offset = renderBox . localToGlobal ( Offset . zero ) ;
final x = offset . dx ;
final y = offset . dy + size . height + 1 ;
2022-12-04 23:44:03 +09:00
final isPeerWindows = model . getCurrentIsWindows ( isLocal ) ;
2022-10-19 11:24:44 +09:00
final List < MenuEntryBase > menuItems = [
MenuEntryButton (
2022-11-02 23:48:02 +09:00
childBuilder: ( TextStyle ? style ) = > isPeerWindows
? buildWindowsThisPC ( style )
: Text (
' / ' ,
style: style ,
) ,
2022-10-19 11:24:44 +09:00
proc: ( ) {
openDirectory ( ' / ' , isLocal: isLocal ) ;
} ,
dismissOnClicked: true ) ,
MenuEntryDivider ( )
] ;
2022-11-02 23:48:02 +09:00
if ( isPeerWindows ) {
2022-10-19 10:52:29 +09:00
var loadingTag = " " ;
if ( ! isLocal ) {
loadingTag = _ffi . dialogManager . showLoading ( " Waiting " ) ;
}
2022-10-18 23:56:36 +09:00
try {
final fd =
2022-10-19 10:52:29 +09:00
await model . fetchDirectory ( " / " , isLocal , isLocal ) ;
2022-10-18 23:56:36 +09:00
for ( var entry in fd . entries ) {
menuItems . add ( MenuEntryButton (
2022-10-19 23:59:02 +09:00
childBuilder: ( TextStyle ? style ) = >
Row ( children: [
2022-10-20 10:31:31 +09:00
Image (
image: iconHardDrive ,
fit: BoxFit . scaleDown ,
color: Theme . of ( context )
. iconTheme
. color
? . withOpacity ( 0.7 ) ) ,
2022-10-19 23:59:02 +09:00
SizedBox ( width: 10 ) ,
Text (
entry . name ,
style: style ,
)
] ) ,
2022-10-18 23:56:36 +09:00
proc: ( ) {
2022-11-02 23:48:02 +09:00
openDirectory ( ' ${ entry . name } \\ ' ,
isLocal: isLocal ) ;
2022-10-19 10:52:29 +09:00
} ,
dismissOnClicked: true ) ) ;
2022-10-18 23:56:36 +09:00
}
2022-12-20 23:55:54 +09:00
} catch ( e ) {
debugPrint ( " buildBread fetchDirectory err= $ e " ) ;
2022-10-18 23:56:36 +09:00
} finally {
2022-10-19 10:52:29 +09:00
if ( ! isLocal ) {
_ffi . dialogManager . dismissByTag ( loadingTag ) ;
}
2022-10-18 23:56:36 +09:00
}
}
2022-10-19 23:59:02 +09:00
menuItems . add ( MenuEntryDivider ( ) ) ;
2022-10-18 23:56:36 +09:00
mod_menu . showMenu (
context: context ,
position: RelativeRect . fromLTRB ( x , y , x , y ) ,
elevation: 4 ,
items: menuItems
. map ( ( e ) = > e . build (
context ,
MenuConfig (
commonColor:
CustomPopupMenuTheme . commonColor ,
height: CustomPopupMenuTheme . height ,
dividerHeight:
CustomPopupMenuTheme . dividerHeight ,
boxWidth: size . width ) ) )
. expand ( ( i ) = > i )
. toList ( ) ) ;
} ,
iconSize: 20 ,
)
] ) ;
2022-08-16 11:46:51 +08:00
}
2022-11-02 23:48:02 +09:00
Widget buildWindowsThisPC ( [ TextStyle ? textStyle ] ) {
final color = Theme . of ( context ) . iconTheme . color ? . withOpacity ( 0.7 ) ;
return Row ( children: [
Icon ( Icons . computer , size: 20 , color: color ) ,
SizedBox ( width: 10 ) ,
Text ( translate ( ' This PC ' ) , style: textStyle )
] ) ;
}
2022-08-16 11:46:51 +08:00
List < BreadCrumbItem > getPathBreadCrumbItems (
bool isLocal , void Function ( List < String > ) onPressed ) {
final path = model . getCurrentDir ( isLocal ) . path ;
final breadCrumbList = List < BreadCrumbItem > . empty ( growable: true ) ;
2022-12-04 23:44:03 +09:00
final isWindows = model . getCurrentIsWindows ( isLocal ) ;
if ( isWindows & & path = = ' / ' ) {
2022-11-02 23:48:02 +09:00
breadCrumbList . add ( BreadCrumbItem (
content: TextButton (
child: buildWindowsThisPC ( ) ,
style: ButtonStyle (
minimumSize: MaterialStateProperty . all ( Size ( 0 , 0 ) ) ) ,
onPressed: ( ) = > onPressed ( [ ' / ' ] ) )
. marginSymmetric ( horizontal: 4 ) ) ) ;
} else {
2022-12-04 23:44:03 +09:00
final list = PathUtil . split ( path , isWindows ) ;
2023-02-22 22:13:21 +01:00
breadCrumbList . addAll (
list . asMap ( ) . entries . map (
( e ) = > BreadCrumbItem (
content: TextButton (
2022-11-02 23:48:02 +09:00
child: Text ( e . value ) ,
style: ButtonStyle (
2023-02-22 22:13:21 +01:00
minimumSize: MaterialStateProperty . all (
Size ( 0 , 0 ) ,
) ,
) ,
onPressed: ( ) = > onPressed (
list . sublist ( 0 , e . key + 1 ) ,
) ,
) . marginSymmetric ( horizontal: 4 ) ,
) ,
) ,
) ;
2022-11-02 23:48:02 +09:00
}
2022-08-16 11:46:51 +08:00
return breadCrumbList ;
}
2022-08-16 12:28:12 +08:00
breadCrumbScrollToEnd ( bool isLocal ) {
Future . delayed ( Duration ( milliseconds: 200 ) , ( ) {
2022-10-11 09:59:27 +09:00
final breadCrumbScroller = getBreadCrumbScrollController ( isLocal ) ;
2022-10-13 20:22:11 +09:00
if ( breadCrumbScroller . hasClients ) {
breadCrumbScroller . animateTo (
breadCrumbScroller . position . maxScrollExtent ,
duration: Duration ( milliseconds: 200 ) ,
curve: Curves . fastLinearToSlowEaseIn ) ;
}
2022-08-16 12:28:12 +08:00
} ) ;
}
2022-08-16 11:46:51 +08:00
Widget buildPathLocation ( bool isLocal ) {
2022-10-13 10:17:20 +09:00
final searchTextObs = isLocal ? _searchTextLocal : _searchTextRemote ;
final locationStatus =
isLocal ? _locationStatusLocal : _locationStatusRemote ;
final focusNode = isLocal ? _locationNodeLocal : _locationNodeRemote ;
2022-10-13 19:57:59 +09:00
final text = locationStatus . value = = LocationStatus . pathLocation
? model . getCurrentDir ( isLocal ) . path
: searchTextObs . value ;
final textController = TextEditingController ( text: text )
. . selection = TextSelection . collapsed ( offset: text . length ) ;
2023-02-22 22:13:21 +01:00
return Row (
children: [
SvgPicture . asset (
locationStatus . value = = LocationStatus . pathLocation
? " assets/folder.svg "
: " assets/search.svg " ,
color: Theme . of ( context ) . tabBarTheme . labelColor ,
) ,
Expanded (
2022-10-13 10:17:20 +09:00
child: TextField (
2023-02-22 22:13:21 +01:00
focusNode: focusNode ,
decoration: InputDecoration (
border: InputBorder . none ,
isDense: true ,
prefix: Padding (
padding: EdgeInsets . only ( left: 4.0 ) ,
) ,
) ,
controller: textController ,
onSubmitted: ( path ) {
openDirectory ( path , isLocal: isLocal ) ;
} ,
onChanged: locationStatus . value = = LocationStatus . fileSearchBar
? ( searchText ) = > onSearchText ( searchText , isLocal )
: null ,
) ,
)
] ,
) ;
2022-08-16 11:46:51 +08:00
}
2022-08-16 12:06:54 +08:00
onSearchText ( String searchText , bool isLocal ) {
if ( isLocal ) {
2022-10-17 23:09:38 +09:00
_localSelectedItems . clear ( ) ;
2022-08-16 12:28:12 +08:00
_searchTextLocal . value = searchText ;
2022-08-16 12:06:54 +08:00
} else {
2022-10-17 23:09:38 +09:00
_remoteSelectedItems . clear ( ) ;
2022-08-16 12:28:12 +08:00
_searchTextRemote . value = searchText ;
2022-08-16 12:06:54 +08:00
}
}
2022-08-16 12:28:12 +08:00
openDirectory ( String path , { bool isLocal = false } ) {
2022-12-04 22:41:44 +09:00
model . openDirectory ( path , isLocal: isLocal ) ;
2022-08-16 12:28:12 +08:00
}
2022-08-16 13:28:48 +08:00
void handleDragDone ( DropDoneDetails details , bool isLocal ) {
if ( isLocal ) {
// ignore local
return ;
}
var items = SelectedItems ( ) ;
2022-10-11 09:59:27 +09:00
for ( var file in details . files ) {
2022-08-16 13:28:48 +08:00
final f = File ( file . path ) ;
items . add (
true ,
Entry ( )
. . path = file . path
. . name = file . name
. . size =
FileSystemEntity . isDirectorySync ( f . path ) ? 0 : f . lengthSync ( ) ) ;
2022-10-11 09:59:27 +09:00
}
2022-08-16 13:28:48 +08:00
model . sendFiles ( items , isRemote: false ) ;
}
2022-12-13 12:55:41 +08:00
void refocusKeyboardListener ( bool isLocal ) {
Future . delayed ( Duration . zero , ( ) {
if ( isLocal ) {
_keyboardNodeLocal . requestFocus ( ) ;
} else {
_keyboardNodeRemote . requestFocus ( ) ;
}
} ) ;
}
2023-02-15 15:03:19 +08:00
Widget headerItemFunc (
double ? width , SortBy sortBy , String name , bool isLocal ) {
final headerTextStyle =
Theme . of ( context ) . dataTableTheme . headingTextStyle ? ? TextStyle ( ) ;
return ObxValue < Rx < bool ? > > (
( ascending ) = > InkWell (
onTap: ( ) {
if ( ascending . value = = null ) {
ascending . value = true ;
} else {
ascending . value = ! ascending . value ! ;
}
model . changeSortStyle ( sortBy ,
isLocal: isLocal , ascending: ascending . value ! ) ;
} ,
child: SizedBox (
width: width ,
height: kDesktopFileTransferHeaderHeight ,
child: Row (
children: [
2023-02-24 15:56:37 +08:00
Flexible (
flex: 2 ,
child: Text (
name ,
style: headerTextStyle ,
overflow: TextOverflow . ellipsis ,
) . marginSymmetric ( horizontal: 4 ) ,
) ,
Flexible (
2023-02-27 09:44:52 +01:00
flex: 1 ,
child: ascending . value ! = null
? Icon (
ascending . value !
? Icons . keyboard_arrow_up_rounded
: Icons . keyboard_arrow_down_rounded ,
)
: const Offstage ( ) )
2023-02-15 15:03:19 +08:00
] ,
) ,
) ,
) , ( ) {
if ( model . getSortStyle ( isLocal ) = = sortBy ) {
return model . getSortAscending ( isLocal ) . obs ;
} else {
return Rx < bool ? > ( null ) ;
}
} ( ) ) ;
}
Widget _buildFileBrowserHeader ( BuildContext context , bool isLocal ) {
2023-02-24 14:15:54 +08:00
final nameColWidth = isLocal ? _nameColWidthLocal : _nameColWidthRemote ;
final modifiedColWidth =
isLocal ? _modifiedColWidthLocal : _modifiedColWidthRemote ;
final padding = EdgeInsets . all ( 1.0 ) ;
return SizedBox (
height: kDesktopFileTransferHeaderHeight ,
child: Row (
children: [
Obx (
( ) = > headerItemFunc (
nameColWidth . value , SortBy . name , translate ( " Name " ) , isLocal ) ,
) ,
DraggableDivider (
axis: Axis . vertical ,
onPointerMove: ( dx ) {
nameColWidth . value + = dx ;
2023-02-27 09:44:52 +01:00
nameColWidth . value = min ( kDesktopFileTransferMaximumWidth ,
max ( kDesktopFileTransferMinimumWidth , nameColWidth . value ) ) ;
2023-02-24 14:15:54 +08:00
} ,
padding: padding ,
) ,
Obx (
( ) = > headerItemFunc ( modifiedColWidth . value , SortBy . modified ,
translate ( " Modified " ) , isLocal ) ,
) ,
DraggableDivider (
axis: Axis . vertical ,
onPointerMove: ( dx ) {
modifiedColWidth . value + = dx ;
modifiedColWidth . value = min (
kDesktopFileTransferMaximumWidth ,
max ( kDesktopFileTransferMinimumWidth ,
modifiedColWidth . value ) ) ;
} ,
padding: padding ) ,
Expanded (
child:
headerItemFunc ( null , SortBy . size , translate ( " Size " ) , isLocal ) )
] ,
) ,
2023-02-15 15:03:19 +08:00
) ;
}
2022-06-17 22:57:41 +08:00
}