mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-17 18:44:45 +02:00
3407a31cf6
* Improving CLI build * Improving CLI build * Remove requirement to build the tools * Moved Electron app one level down * Clean up Electron build * Moved tools to sub-dir * Updated root script * update root * update root * update root * update root * update root * update root * Updated build * Added doc * Update CI config * Should not lint index.js * Fixing jetify * Fixed linter errors * Fixed pod build * Fixed Windows build
194 lines
4.9 KiB
TypeScript
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 };
|