1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-11 18:24:43 +02:00
joplin/ElectronClient/gui/ResourceScreen.tsx
2020-03-13 23:52:40 +00:00

194 lines
4.9 KiB
TypeScript

import * as React from 'react';
const { connect } = require('react-redux');
const { _ } = require('lib/locale.js');
const { themeStyle } = require('../theme.js');
const { bridge } = require('electron').remote.require('./bridge');
const { Header } = require('./Header.min.js');
const prettyBytes = require('pretty-bytes');
const Resource = require('lib/models/Resource.js');
interface Props {
style: any;
theme: any;
}
interface Resource {
title: string
id: string
size: number
file_extension: string
}
interface State {
resources: Resource[] | undefined
sorting: ActiveSorting
isLoading: boolean
}
interface ResourceTable {
resources: Resource[]
sorting: ActiveSorting
onResourceClick: (resource: Resource) => any
onResourceDelete: (resource: Resource) => any
onToggleSorting: (order: SortingOrder) => any
}
type SortingOrder = 'size' | 'name'
type SortingType = 'asc' | 'desc'
interface ActiveSorting {
order: SortingOrder
type: SortingType
}
const ResourceTable: React.FC<ResourceTable> = (props: ResourceTable) => {
const sortOrderEngagedMarker = (s: SortingOrder) => {
return (
<a href="#" onClick={ () => props.onToggleSorting(s) }>{
(props.sorting.order === s && props.sorting.type === 'desc') ? '▾' : '▴' }</a>
);
};
return <table style={{ width: '90%' }}>
<thead>
<tr>
<th>{_('Title')} { sortOrderEngagedMarker('name') }</th>
<th>{_('Size')} { sortOrderEngagedMarker('size') }</th>
<th>{_('ID')}</th>
<th>{_('Action')}</th>
</tr>
</thead>
<tbody>
{props.resources.map((resource: Resource, index: number) =>
<tr key={index}>
<td>
<a href="#" onClick={() => props.onResourceClick(resource)}>{resource.title}</a>
</td>
<td>{prettyBytes(resource.size)}</td>
<td>{resource.id}</td>
<td>
<button onClick={ () => props.onResourceDelete(resource) }>{_('Delete')}</button>
</td>
</tr>
)}
</tbody>
</table>;
};
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';
}
};
const MAX_RESOURCES = 10000;
class ResourceScreenComponent extends React.Component<Props, State> {
constructor(props: Props) {
super(props);
this.state = {
resources: undefined,
sorting: {
type: 'asc',
order: 'name',
},
isLoading: false,
};
}
async reloadResources(sorting: ActiveSorting) {
this.setState({ isLoading: true });
const resources = await Resource.all({
order: [{
by: getSortingOrderColumn(sorting.order),
dir: sorting.type,
}],
limit: MAX_RESOURCES,
fields: ['title', 'id', 'size', 'file_extension'],
});
this.setState({ resources, isLoading: false });
}
componentDidMount() {
this.reloadResources(this.state.sorting);
}
onResourceDelete(resource: Resource) {
Resource.delete(resource.id)
.catch((error: Error) => {
bridge().showErrorMessageBox(error.message);
})
.finally(() => {
this.reloadResources(this.state.sorting);
});
}
openResource(resource: Resource) {
const resourcePath = Resource.fullPath(resource);
const ok = bridge().openExternal(`file://${resourcePath}`);
if (!ok) {
bridge().showErrorMessageBox(`This file could not be opened: ${resourcePath}`);
}
}
onToggleSortOrder(sortOrder: SortingOrder) {
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 });
this.reloadResources(newSorting);
}
render() {
const style = this.props.style;
const theme = themeStyle(this.props.theme);
const headerStyle = Object.assign({}, theme.headerStyle, { width: style.width });
return <div>
<Header style={headerStyle} />
<div style={{ ...style, margin: '20px', overflow: 'scroll' }}>
{this.state.isLoading && <div>{_('Please wait...')}</div>}
{!this.state.isLoading && <div>
{!this.state.resources && <div>
{_('No resources!')}
</div>
}
{this.state.resources && this.state.resources.length === MAX_RESOURCES &&
<div>{_('Warning: not all resources shown for performance reasons (limit: %s).', MAX_RESOURCES)}</div>
}
{this.state.resources && <ResourceTable
resources={ this.state.resources }
sorting={ this.state.sorting }
onToggleSorting={ (order) => this.onToggleSortOrder(order) }
onResourceClick={ (resource) => this.openResource(resource) }
onResourceDelete={ (resource) => this.onResourceDelete(resource) }
/>}
</div>
}
</div>
</div>;
}
}
const mapStateToProps = (state: any) => ({
theme: state.settings.theme,
});
const ResourceScreen = connect(mapStateToProps)(ResourceScreenComponent);
module.exports = { ResourceScreen };