2020-02-19 10:13:33 +00:00
import * as React from 'react' ;
2020-09-15 14:01:07 +01:00
import ButtonBar from './ConfigScreen/ButtonBar' ;
2020-11-07 15:59:37 +00:00
import { _ } from '@joplin/lib/locale' ;
2020-02-19 10:13:33 +00:00
const { connect } = require ( 'react-redux' ) ;
2020-11-07 15:59:37 +00:00
const { themeStyle } = require ( '@joplin/lib/theme' ) ;
2024-02-22 13:29:16 -08:00
import bridge from '../services/bridge' ;
2020-02-19 10:13:33 +00:00
const prettyBytes = require ( 'pretty-bytes' ) ;
2021-01-22 17:41:11 +00:00
import Resource from '@joplin/lib/models/Resource' ;
2024-05-21 02:14:39 -07:00
import { LoadOptions } from '@joplin/lib/models/utils/types' ;
2020-02-19 10:13:33 +00:00
2020-04-08 11:12:25 +01:00
interface Style {
2020-11-12 19:29:22 +00:00
width : number ;
height : number ;
2020-04-08 11:12:25 +01:00
}
2020-02-19 10:13:33 +00:00
interface Props {
2020-09-15 14:01:07 +01:00
themeId : number ;
2020-11-12 19:29:22 +00:00
style : Style ;
2023-06-30 10:30:29 +01:00
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
2020-11-12 19:29:22 +00:00
dispatch : Function ;
2020-02-19 10:13:33 +00:00
}
2020-11-05 16:58:23 +00:00
interface InnerResource {
2020-11-12 19:29:22 +00:00
title : string ;
id : string ;
size : number ;
file_extension : string ;
2020-02-19 10:13:33 +00:00
}
interface State {
2020-11-12 19:29:22 +00:00
resources : InnerResource [ ] | undefined ;
sorting : ActiveSorting ;
isLoading : boolean ;
2024-05-21 02:14:39 -07:00
filter : string ;
2020-02-19 10:13:33 +00:00
}
interface ResourceTable {
2020-11-12 19:29:22 +00:00
resources : InnerResource [ ] ;
sorting : ActiveSorting ;
2024-04-05 12:16:49 +01:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2020-11-12 19:29:22 +00:00
onResourceClick : ( resource : InnerResource ) = > any ;
2024-04-05 12:16:49 +01:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2020-11-12 19:29:22 +00:00
onResourceDelete : ( resource : InnerResource ) = > any ;
2024-04-05 12:16:49 +01:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2020-11-12 19:29:22 +00:00
onToggleSorting : ( order : SortingOrder ) = > any ;
2024-05-21 02:14:39 -07:00
filter : string ;
2020-11-12 19:29:22 +00:00
themeId : number ;
style : Style ;
2020-02-19 10:13:33 +00:00
}
2020-11-12 19:29:22 +00:00
type SortingOrder = 'size' | 'name' ;
type SortingType = 'asc' | 'desc' ;
2020-02-19 10:13:33 +00:00
interface ActiveSorting {
2020-11-12 19:29:22 +00:00
order : SortingOrder ;
type : SortingType ;
2020-02-19 10:13:33 +00:00
}
2020-11-05 16:58:23 +00:00
const ResourceTableComp = ( props : ResourceTable ) = > {
2020-09-15 14:01:07 +01:00
const theme = themeStyle ( props . themeId ) ;
2020-02-19 10:13:33 +00:00
const sortOrderEngagedMarker = ( s : SortingOrder ) = > {
return (
2020-04-08 11:12:25 +01:00
< a href = "#"
2020-09-15 14:01:07 +01:00
style = { { color : theme.urlColor } }
2020-04-08 11:12:25 +01:00
onClick = { ( ) = > props . onToggleSorting ( s ) } > {
( props . sorting . order === s && props . sorting . type === 'desc' ) ? '▾' : '▴' } < / a >
2020-02-19 10:13:33 +00:00
) ;
} ;
2020-04-08 11:12:25 +01:00
const titleCellStyle = {
2020-09-15 14:01:07 +01:00
. . . theme . textStyle ,
2020-04-08 11:12:25 +01:00
textOverflow : 'ellipsis' ,
overflowX : 'hidden' ,
maxWidth : 1 ,
width : '100%' ,
whiteSpace : 'nowrap' ,
} ;
const cellStyle = {
2020-09-15 14:01:07 +01:00
. . . theme . textStyle ,
2020-04-08 11:12:25 +01:00
whiteSpace : 'nowrap' ,
2020-09-15 14:01:07 +01:00
color : theme.colorFaded ,
2020-04-08 11:12:25 +01:00
width : 1 ,
} ;
const headerStyle = {
2020-09-15 14:01:07 +01:00
. . . theme . textStyle ,
2020-04-08 11:12:25 +01:00
whiteSpace : 'nowrap' ,
width : 1 ,
fontWeight : 'bold' ,
} ;
2024-05-21 02:14:39 -07:00
const filteredResources = props . resources . filter (
( resource : InnerResource ) = > ! props . filter || resource . title ? . includes ( props . filter ) || resource . id . includes ( props . filter ) ,
) ;
2020-04-08 17:40:32 +00:00
return (
< table style = { { width : '100%' } } >
< thead >
< tr >
< th style = { headerStyle } > { _ ( 'Title' ) } { sortOrderEngagedMarker ( 'name' ) } < / th >
< th style = { headerStyle } > { _ ( 'Size' ) } { sortOrderEngagedMarker ( 'size' ) } < / th >
< th style = { headerStyle } > { _ ( 'ID' ) } < / th >
< th style = { headerStyle } > { _ ( 'Action' ) } < / th >
2020-02-19 10:13:33 +00:00
< / tr >
2020-04-08 17:40:32 +00:00
< / thead >
< tbody >
2024-05-21 02:14:39 -07:00
{ filteredResources . map ( ( resource : InnerResource , index : number ) = >
2020-04-08 17:40:32 +00:00
< tr key = { index } >
< td style = { titleCellStyle } className = "titleCell" >
< a
2020-09-15 14:01:07 +01:00
style = { { color : theme.urlColor } }
2020-04-08 17:40:32 +00:00
href = "#"
onClick = { ( ) = > props . onResourceClick ( resource ) } > { resource . title || ` ( ${ _ ( 'Untitled' ) } ) ` }
< / a >
< / td >
< td style = { cellStyle } className = "dataCell" > { prettyBytes ( resource . size ) } < / td >
< td style = { cellStyle } className = "dataCell" > { resource . id } < / td >
< td style = { cellStyle } className = "dataCell" >
2020-09-15 14:01:07 +01:00
< button style = { theme . buttonStyle } onClick = { ( ) = > props . onResourceDelete ( resource ) } > { _ ( 'Delete' ) } < / button >
2020-04-08 17:40:32 +00:00
< / td >
2023-08-22 11:58:53 +01:00
< / tr > ,
2020-04-08 17:40:32 +00:00
) }
< / tbody >
< / table >
) ;
2020-02-19 10:13:33 +00:00
} ;
const getSortingOrderColumn = ( s : SortingOrder ) : string = > {
switch ( s ) {
case 'name' : return 'title' ;
case 'size' : return 'size' ;
}
} ;
const getNextSortingOrderType = ( s : SortingType ) : SortingType = > {
if ( s === 'asc' ) {
return 'desc' ;
} else {
return 'asc' ;
}
} ;
2024-05-21 02:14:39 -07:00
const defaultMaxResources = 10000 ;
const searchMaxResources = 1000 ;
2020-02-19 10:13:33 +00:00
class ResourceScreenComponent extends React . Component < Props , State > {
2023-03-06 14:22:01 +00:00
public constructor ( props : Props ) {
2020-02-19 10:13:33 +00:00
super ( props ) ;
this . state = {
resources : undefined ,
2024-05-21 02:14:39 -07:00
filter : '' ,
2020-02-19 10:13:33 +00:00
sorting : {
type : 'asc' ,
order : 'name' ,
} ,
isLoading : false ,
} ;
}
2024-05-21 02:14:39 -07:00
private get maxResources() {
// Use a smaller maximum when searching for performance -- results update
// when the search input changes.
if ( this . state . filter ) {
return searchMaxResources ;
} else {
return defaultMaxResources ;
}
}
private reloadResourcesCounter = 0 ;
public async reloadResources() {
2020-02-19 10:13:33 +00:00
this . setState ( { isLoading : true } ) ;
2024-05-21 02:14:39 -07:00
this . reloadResourcesCounter ++ ;
const currentCounterValue = this . reloadResourcesCounter ;
let searchOptions : Partial < LoadOptions > = { } ;
if ( this . state . filter ) {
const search = ` % ${ this . state . filter } % ` ;
searchOptions = {
where : 'id LIKE ? OR title LIKE ?' ,
whereParams : [ search , search ] ,
} ;
}
2020-02-19 10:13:33 +00:00
const resources = await Resource . all ( {
order : [ {
2024-05-21 02:14:39 -07:00
by : getSortingOrderColumn ( this . state . sorting . order ) ,
dir : this.state.sorting.type ,
2021-01-02 15:27:53 +00:00
caseInsensitive : true ,
2020-02-19 10:13:33 +00:00
} ] ,
2024-05-21 02:14:39 -07:00
limit : this.maxResources ,
2020-02-19 10:13:33 +00:00
fields : [ 'title' , 'id' , 'size' , 'file_extension' ] ,
2024-05-21 02:14:39 -07:00
. . . searchOptions ,
2020-02-19 10:13:33 +00:00
} ) ;
2024-05-21 02:14:39 -07:00
const cancelled = currentCounterValue !== this . reloadResourcesCounter ;
if ( cancelled ) return ;
2020-02-19 10:13:33 +00:00
this . setState ( { resources , isLoading : false } ) ;
}
2023-03-06 14:22:01 +00:00
public componentDidMount() {
2024-05-21 02:14:39 -07:00
void this . reloadResources ( ) ;
}
public componentDidUpdate ( _prevProps : Props , prevState : State ) {
if ( prevState . sorting !== this . state . sorting || prevState . filter !== this . state . filter ) {
void this . reloadResources ( ) ;
}
2020-02-19 10:13:33 +00:00
}
2023-03-06 14:22:01 +00:00
public onResourceDelete ( resource : InnerResource ) {
2020-04-08 11:12:25 +01:00
const ok = bridge ( ) . showConfirmMessageBox ( _ ( 'Delete attachment "%s"?' , resource . title ) , {
buttons : [ _ ( 'Delete' ) , _ ( 'Cancel' ) ] ,
defaultId : 1 ,
} ) ;
if ( ! ok ) {
return ;
}
2024-03-09 02:33:05 -08:00
Resource . delete ( resource . id , { sourceDescription : 'ResourceScreen' } )
2022-09-30 17:23:14 +01:00
// eslint-disable-next-line promise/prefer-await-to-then -- Old code before rule was applied
2020-02-19 10:13:33 +00:00
. catch ( ( error : Error ) = > {
2024-03-09 02:33:05 -08:00
console . error ( error ) ;
2020-02-19 10:13:33 +00:00
bridge ( ) . showErrorMessageBox ( error . message ) ;
} )
2022-09-30 17:23:14 +01:00
// eslint-disable-next-line promise/prefer-await-to-then -- Old code before rule was applied
2020-02-19 10:13:33 +00:00
. finally ( ( ) = > {
2024-05-21 02:14:39 -07:00
void this . reloadResources ( ) ;
2020-02-19 10:13:33 +00:00
} ) ;
}
2023-03-06 14:22:01 +00:00
public openResource ( resource : InnerResource ) {
2020-02-19 10:13:33 +00:00
const resourcePath = Resource . fullPath ( resource ) ;
2024-02-22 13:29:16 -08:00
const ok = bridge ( ) . openItem ( resourcePath ) ;
2020-02-19 10:13:33 +00:00
if ( ! ok ) {
bridge ( ) . showErrorMessageBox ( ` This file could not be opened: ${ resourcePath } ` ) ;
}
}
2023-03-06 14:22:01 +00:00
public onToggleSortOrder ( sortOrder : SortingOrder ) {
2020-02-19 10:13:33 +00:00
let newSorting = { . . . this . state . sorting } ;
if ( sortOrder === this . state . sorting . order ) {
newSorting . type = getNextSortingOrderType ( newSorting . type ) ;
} else {
newSorting = {
order : sortOrder ,
type : 'desc' ,
} ;
}
this . setState ( { sorting : newSorting } ) ;
}
2024-05-21 02:14:39 -07:00
public onFilterUpdate = ( updateEvent : React.ChangeEvent < HTMLInputElement > ) = > {
this . setState ( { filter : updateEvent.target.value } ) ;
} ;
2023-03-06 14:22:01 +00:00
public render() {
2020-02-19 10:13:33 +00:00
const style = this . props . style ;
2020-09-15 14:01:07 +01:00
const theme = themeStyle ( this . props . themeId ) ;
2020-04-08 11:12:25 +01:00
2024-04-05 12:16:49 +01:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2020-11-12 19:13:28 +00:00
const rootStyle : any = {
2020-04-08 11:12:25 +01:00
. . . style ,
overflowY : 'scroll' ,
color : theme.color ,
padding : 20 ,
boxSizing : 'border-box' ,
2020-09-15 14:01:07 +01:00
flex : 1 ,
2020-04-08 11:12:25 +01:00
} ;
2020-09-15 14:01:07 +01:00
// rootStyle.height = style.height - 35; // Minus the header height
delete rootStyle . height ;
2020-04-08 11:12:25 +01:00
delete rootStyle . width ;
2020-09-15 14:01:07 +01:00
const containerHeight = style . height ;
2020-04-08 17:40:32 +00:00
return (
2020-09-15 14:01:07 +01:00
< div style = { { . . . theme . containerStyle , fontFamily : theme.fontFamily , height : containerHeight , display : 'flex' , flexDirection : 'column' } } >
2020-04-08 17:40:32 +00:00
< div style = { rootStyle } >
< div style = { { . . . theme . notificationBox , marginBottom : 10 } } > {
_ ( 'This is an advanced tool to show the attachments that are linked to your notes. Please be careful when deleting one of them as they cannot be restored afterwards.' )
} < / div >
2024-05-21 02:14:39 -07:00
< div style = { { float : 'right' } } >
< input
style = { theme . inputStyle }
type = "search"
value = { this . state . filter }
onChange = { this . onFilterUpdate }
placeholder = { _ ( 'Search...' ) }
/ >
< / div >
2020-04-08 17:40:32 +00:00
{ this . state . isLoading && < div > { _ ( 'Please wait...' ) } < / div > }
{ ! this . state . isLoading && < div >
{ ! this . state . resources && < div >
{ _ ( 'No resources!' ) }
< / div >
}
2024-05-21 02:14:39 -07:00
{ this . state . resources && this . state . resources . length === this . maxResources &&
< div > { _ ( 'Warning: not all resources shown for performance reasons (limit: %s).' , this . maxResources ) } < / div >
2020-04-08 17:40:32 +00:00
}
2020-11-05 16:58:23 +00:00
{ this . state . resources && < ResourceTableComp
2020-09-15 14:01:07 +01:00
themeId = { this . props . themeId }
2020-04-08 17:40:32 +00:00
style = { style }
resources = { this . state . resources }
sorting = { this . state . sorting }
2024-05-21 02:14:39 -07:00
filter = { this . state . filter }
2020-04-08 17:40:32 +00:00
onToggleSorting = { ( order ) = > this . onToggleSortOrder ( order ) }
onResourceClick = { ( resource ) = > this . openResource ( resource ) }
onResourceDelete = { ( resource ) = > this . onResourceDelete ( resource ) }
/ > }
2020-02-19 10:13:33 +00:00
< / div >
}
< / div >
2020-09-15 14:01:07 +01:00
< ButtonBar
onCancelClick = { ( ) = > this . props . dispatch ( { type : 'NAV_BACK' } ) }
/ >
2020-02-19 10:13:33 +00:00
< / div >
2020-04-08 17:40:32 +00:00
) ;
2020-02-19 10:13:33 +00:00
}
}
2024-04-05 12:16:49 +01:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2020-02-19 10:13:33 +00:00
const mapStateToProps = ( state : any ) = > ( {
2020-09-15 14:01:07 +01:00
themeId : state.settings.theme ,
2020-02-19 10:13:33 +00:00
} ) ;
const ResourceScreen = connect ( mapStateToProps ) ( ResourceScreenComponent ) ;
module . exports = { ResourceScreen } ;