mirror of
https://github.com/haiwen/seahub.git
synced 2025-08-17 14:37:58 +00:00
Wiki mode optimized (#2442)
This commit is contained in:
parent
9805a33ef0
commit
6831d9e519
26
frontend/package-lock.json
generated
26
frontend/package-lock.json
generated
@ -2896,9 +2896,9 @@
|
|||||||
"integrity": "sha1-9dJgKStmDghO/0zbyfCK0yR0SLU="
|
"integrity": "sha1-9dJgKStmDghO/0zbyfCK0yR0SLU="
|
||||||
},
|
},
|
||||||
"deep-extend": {
|
"deep-extend": {
|
||||||
"version": "0.4.2",
|
"version": "0.6.0",
|
||||||
"resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.4.2.tgz",
|
"resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
|
||||||
"integrity": "sha1-SLaZwn4zS/ifEIkr5DL25MfTSn8=",
|
"integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"deep-is": {
|
"deep-is": {
|
||||||
@ -4186,9 +4186,9 @@
|
|||||||
"integrity": "sha1-2uRqnXj74lKSJYzB54CkHZXAN4I="
|
"integrity": "sha1-2uRqnXj74lKSJYzB54CkHZXAN4I="
|
||||||
},
|
},
|
||||||
"follow-redirects": {
|
"follow-redirects": {
|
||||||
"version": "1.5.8",
|
"version": "1.5.9",
|
||||||
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.8.tgz",
|
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.9.tgz",
|
||||||
"integrity": "sha512-sy1mXPmv7kLAMKW/8XofG7o9T+6gAjzdZK4AJF6ryqQYUa/hnzgiypoeUecZ53x7XiqKNEpNqLtS97MshW2nxg==",
|
"integrity": "sha512-Bh65EZI/RU8nx0wbYF9shkFZlqLP+6WT/5FnA3cE/djNSuKNHJEinGGZgu/cQEkeeb2GdFOgenAmn8qaqYke2w==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"debug": "=3.1.0"
|
"debug": "=3.1.0"
|
||||||
},
|
},
|
||||||
@ -10168,9 +10168,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"seafile-js": {
|
"seafile-js": {
|
||||||
"version": "0.2.21",
|
"version": "0.2.25",
|
||||||
"resolved": "https://registry.npmjs.org/seafile-js/-/seafile-js-0.2.21.tgz",
|
"resolved": "https://registry.npmjs.org/seafile-js/-/seafile-js-0.2.25.tgz",
|
||||||
"integrity": "sha512-zUY1X8YqoLHg2G+fmFLbNOK5w96EW9glmEc/pkNFlXi2SzHd5SasPt5J2scpnD+jyLmYWLfwqJbjqp+xLEO+Yg==",
|
"integrity": "sha512-XVJ6qvFeSv6tfihBvNscBS+rmnb+TkhyXvEYKBa13PipZbJdxbzVeWRhCXiR82fQwWLzYdAAv2rclSeURyCe5Q==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"axios": "^0.18.0",
|
"axios": "^0.18.0",
|
||||||
"form-data": "^2.3.2"
|
"form-data": "^2.3.2"
|
||||||
@ -11773,12 +11773,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"webpack-bundle-tracker": {
|
"webpack-bundle-tracker": {
|
||||||
"version": "0.3.0",
|
"version": "0.4.2-beta",
|
||||||
"resolved": "https://registry.npmjs.org/webpack-bundle-tracker/-/webpack-bundle-tracker-0.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/webpack-bundle-tracker/-/webpack-bundle-tracker-0.4.2-beta.tgz",
|
||||||
"integrity": "sha512-I0Gwkug8QX8xZS14SvmfWin1AmZDoZp/0AGvlgKqNxyw20DgkFkq1jTQ/Ml73YgjFTmQ5bATyQM7TjtYMP1nFA==",
|
"integrity": "sha512-CCyJbCQnRtjR1sk97u/H5DtJibrIcJ79MnntMyjOpc9HCmfIQYgt7ze7i/Z+DStBZ4NC4HxqGDsB///2Na1DTA==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"deep-extend": "^0.4.1",
|
"deep-extend": "^0.6.0",
|
||||||
"mkdirp": "^0.5.1",
|
"mkdirp": "^0.5.1",
|
||||||
"strip-ansi": "^2.0.1"
|
"strip-ansi": "^2.0.1"
|
||||||
},
|
},
|
||||||
|
@ -24,7 +24,7 @@
|
|||||||
"react-dom": "^16.5.2",
|
"react-dom": "^16.5.2",
|
||||||
"react-moment": "^0.7.9",
|
"react-moment": "^0.7.9",
|
||||||
"reactstrap": "^6.4.0",
|
"reactstrap": "^6.4.0",
|
||||||
"seafile-js": "^0.2.21",
|
"seafile-js": "^0.2.25",
|
||||||
"seafile-ui": "^0.1.10",
|
"seafile-ui": "^0.1.10",
|
||||||
"sw-precache-webpack-plugin": "0.11.4",
|
"sw-precache-webpack-plugin": "0.11.4",
|
||||||
"url-loader": "0.6.2",
|
"url-loader": "0.6.2",
|
||||||
@ -106,7 +106,7 @@
|
|||||||
"react-dev-utils": "^5.0.0",
|
"react-dev-utils": "^5.0.0",
|
||||||
"react-i18next": "^7.6.1",
|
"react-i18next": "^7.6.1",
|
||||||
"webpack": "3.8.1",
|
"webpack": "3.8.1",
|
||||||
"webpack-bundle-tracker": "^0.3.0",
|
"webpack-bundle-tracker": "^0.4.2-beta",
|
||||||
"webpack-dev-server": "2.9.4",
|
"webpack-dev-server": "2.9.4",
|
||||||
"webpack-manifest-plugin": "1.3.2"
|
"webpack-manifest-plugin": "1.3.2"
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,12 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
import { Modal, ModalHeader, ModalBody } from 'reactstrap';
|
import { Modal, ModalHeader, ModalBody } from 'reactstrap';
|
||||||
|
|
||||||
|
const propTypes = {
|
||||||
|
onCancelDownload: PropTypes.func.isRequired,
|
||||||
|
progress: PropTypes.string.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
class ZipDownloadDialog extends React.Component {
|
class ZipDownloadDialog extends React.Component {
|
||||||
|
|
||||||
toggle = () => {
|
toggle = () => {
|
||||||
@ -19,4 +25,6 @@ class ZipDownloadDialog extends React.Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ZipDownloadDialog.propTypes = propTypes;
|
||||||
|
|
||||||
export default ZipDownloadDialog;
|
export default ZipDownloadDialog;
|
||||||
|
125
frontend/src/components/dirent-list-view/dirent-list-item.js
Normal file
125
frontend/src/components/dirent-list-view/dirent-list-item.js
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { serviceUrl, gettext } from '../../utils/constants';
|
||||||
|
import OperationGroup from '../dirent-operation/operation-group';
|
||||||
|
|
||||||
|
const propTypes = {
|
||||||
|
isItemFreezed: PropTypes.bool.isRequired,
|
||||||
|
dirent: PropTypes.object.isRequired,
|
||||||
|
onItemClick: PropTypes.func.isRequired,
|
||||||
|
onItemMenuShow: PropTypes.func.isRequired,
|
||||||
|
onItemMenuHide: PropTypes.func.isRequired,
|
||||||
|
onItemDelete: PropTypes.func.isRequired,
|
||||||
|
onItemStarred: PropTypes.func.isRequired,
|
||||||
|
onItemDownload: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
|
class DirentListItem extends React.Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
isOperationShow: false,
|
||||||
|
highlight: false
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
//UI Interactive
|
||||||
|
onMouseEnter = () => {
|
||||||
|
if (!this.props.isItemFreezed) {
|
||||||
|
this.setState({
|
||||||
|
highlight: true,
|
||||||
|
isOperationShow: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMouseOver = () => {
|
||||||
|
if (!this.props.isItemFreezed) {
|
||||||
|
this.setState({
|
||||||
|
highlight: true,
|
||||||
|
isOperationShow: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMouseLeave = () => {
|
||||||
|
if (!this.props.isItemFreezed) {
|
||||||
|
this.setState({
|
||||||
|
highlight: false,
|
||||||
|
isOperationShow: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemMenuShow = () => {
|
||||||
|
this.props.onItemMenuShow();
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemMenuHide = () => {
|
||||||
|
this.setState({
|
||||||
|
isOperationShow: false,
|
||||||
|
highlight: ''
|
||||||
|
});
|
||||||
|
this.props.onItemMenuHide();
|
||||||
|
}
|
||||||
|
|
||||||
|
//buiness handler
|
||||||
|
onItemSelected = () => {
|
||||||
|
//todos;
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemStarred = () => {
|
||||||
|
this.props.onItemStarred(this.props.dirent);
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemClick = () => {
|
||||||
|
this.props.onItemClick(this.props.dirent);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
onItemDownload = () => {
|
||||||
|
this.props.onItemDownload(this.props.dirent);
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemDelete = () => {
|
||||||
|
this.props.onItemDelete(this.props.dirent);
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
let { dirent } = this.props;
|
||||||
|
return (
|
||||||
|
<tr className={this.state.highlight ? 'tr-highlight' : ''} onMouseEnter={this.onMouseEnter} onMouseOver={this.onMouseOver} onMouseLeave={this.onMouseLeave}>
|
||||||
|
<td className="select">
|
||||||
|
<input type="checkbox" className="vam" />
|
||||||
|
</td>
|
||||||
|
<td className="star" onClick={this.onItemStarred}>
|
||||||
|
{dirent.starred !== undefined && !dirent.starred && <i className="far fa-star empty"></i>}
|
||||||
|
{dirent.starred !== undefined && dirent.starred && <i className="fas fa-star"></i>}
|
||||||
|
</td>
|
||||||
|
<td className="icon">
|
||||||
|
<img src={dirent.type === 'dir' ? serviceUrl + '/media/img/folder-192.png' : serviceUrl + '/media/img/file/192/txt.png'} alt={gettext('file icon')}></img>
|
||||||
|
</td>
|
||||||
|
<td className="name a-simulate" onClick={this.onItemClick}>{dirent.name}</td>
|
||||||
|
<td className="operation">
|
||||||
|
{
|
||||||
|
this.state.isOperationShow &&
|
||||||
|
<OperationGroup
|
||||||
|
dirent={dirent}
|
||||||
|
onItemMenuShow={this.onItemMenuShow}
|
||||||
|
onItemMenuHide={this.onItemMenuHide}
|
||||||
|
onDownload={this.onItemDownload}
|
||||||
|
onDelete={this.onItemDelete}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
</td>
|
||||||
|
<td className="file-size">{dirent.size && dirent.size}</td>
|
||||||
|
<td className="last-update" dangerouslySetInnerHTML={{__html: dirent.mtime}}></td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DirentListItem.propTypes = propTypes;
|
||||||
|
|
||||||
|
export default DirentListItem;
|
158
frontend/src/components/dirent-list-view/dirent-list-view.js
Normal file
158
frontend/src/components/dirent-list-view/dirent-list-view.js
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { gettext, repoID } from '../../utils/constants';
|
||||||
|
import URLDecorator from '../../utils/url-decorator';
|
||||||
|
import editorUtilities from '../../utils/editor-utilties';
|
||||||
|
import { seafileAPI } from '../../utils/seafile-api';
|
||||||
|
import DirentListItem from './dirent-list-item';
|
||||||
|
import ZipDownloadDialog from '../dialog/zip-download-dialog';
|
||||||
|
|
||||||
|
const propTypes = {
|
||||||
|
filePath: PropTypes.string.isRequired,
|
||||||
|
direntList: PropTypes.array.isRequired,
|
||||||
|
onItemDelete: PropTypes.func.isRequired,
|
||||||
|
onItemClick: PropTypes.func.isRequired,
|
||||||
|
updateViewList: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
|
class DirentListView extends React.Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
isItemFreezed: false,
|
||||||
|
isProgressDialogShow: false,
|
||||||
|
progress: '0%',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemMenuShow = () => {
|
||||||
|
this.setState({isItemFreezed: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemMenuHide = () => {
|
||||||
|
this.setState({isItemFreezed: false});
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemClick = (dirent) => {
|
||||||
|
let direntPath = this.getDirentPath(dirent);
|
||||||
|
this.props.onItemClick(direntPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemDelete = (dirent) => {
|
||||||
|
let direntPath = this.getDirentPath(dirent);
|
||||||
|
this.props.onItemDelete(direntPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemStarred = (dirent) => {
|
||||||
|
let filePath = this.getDirentPath(dirent);
|
||||||
|
if (dirent.starred) {
|
||||||
|
seafileAPI.unStarFile(repoID, filePath).then(() => {
|
||||||
|
this.props.updateViewList(this.props.filePath);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
seafileAPI.starFile(repoID, filePath).then(() => {
|
||||||
|
this.props.updateViewList(this.props.filePath);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onItemDownload = (dirent) => {
|
||||||
|
if (dirent.type === 'dir') {
|
||||||
|
this.setState({isProgressDialogShow: true, progress: '0%'});
|
||||||
|
editorUtilities.zipDownload(this.props.filePath, dirent.name).then(res => {
|
||||||
|
this.zip_token = res.data['zip_token'];
|
||||||
|
this.addDownloadAnimation();
|
||||||
|
this.interval = setInterval(this.addDownloadAnimation, 1000);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
let path = this.getDirentPath(dirent);
|
||||||
|
let url = URLDecorator.getUrl({type: 'download_file_url', repoID: repoID, filePath: path});
|
||||||
|
location.href = url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addDownloadAnimation = () => {
|
||||||
|
let _this = this;
|
||||||
|
let token = this.zip_token;
|
||||||
|
editorUtilities.queryZipProgress(token).then(res => {
|
||||||
|
let data = res.data;
|
||||||
|
let progress = data.total === 0 ? '100%' : (data.zipped / data.total * 100).toFixed(0) + '%';
|
||||||
|
this.setState({progress: progress});
|
||||||
|
|
||||||
|
if (data['total'] === data['zipped']) {
|
||||||
|
this.setState({
|
||||||
|
progress: '100%'
|
||||||
|
});
|
||||||
|
clearInterval(this.interval);
|
||||||
|
location.href = URLDecorator.getUrl({type: 'download_dir_zip_url', token: token});
|
||||||
|
setTimeout(function() {
|
||||||
|
_this.setState({isProgressDialogShow: false});
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onCancelDownload = () => {
|
||||||
|
let zip_token = this.zip_token;
|
||||||
|
editorUtilities.cancelZipTask(zip_token).then(res => {
|
||||||
|
this.setState({
|
||||||
|
isProgressDialogShow: false,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getDirentPath = (dirent) => {
|
||||||
|
let path = this.props.filePath;
|
||||||
|
return path === '/' ? path + dirent.name : path + '/' + dirent.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { direntList } = this.props;
|
||||||
|
return (
|
||||||
|
<div className="table-container">
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th width="3%" className="select"><input type="checkbox" className="vam" /></th>
|
||||||
|
<th width="3%"></th>
|
||||||
|
<th width="5%"></th>
|
||||||
|
<th width="45%">{gettext('Name')}</th>
|
||||||
|
<th width="20%"></th>
|
||||||
|
<th width="11%">{gettext('Size')}</th>
|
||||||
|
<th width="13%">{gettext('Last Update')}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{
|
||||||
|
direntList.length !== 0 && direntList.map((dirent, index) => {
|
||||||
|
return (
|
||||||
|
<DirentListItem
|
||||||
|
key={index}
|
||||||
|
dirent={dirent}
|
||||||
|
isItemFreezed={this.state.isItemFreezed}
|
||||||
|
onItemMenuShow={this.onItemMenuShow}
|
||||||
|
onItemMenuHide={this.onItemMenuHide}
|
||||||
|
onItemDelete={this.onItemDelete}
|
||||||
|
onItemStarred={this.onItemStarred}
|
||||||
|
onItemDownload={this.onItemDownload}
|
||||||
|
onItemClick={this.onItemClick}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{
|
||||||
|
this.state.isProgressDialogShow &&
|
||||||
|
<ZipDownloadDialog progress={this.state.progress} onCancelDownload={this.onCancelDownload}/>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DirentListView.propTypes = propTypes;
|
||||||
|
|
||||||
|
export default DirentListView;
|
@ -1,7 +1,16 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
import { gettext } from '../../utils/constants';
|
import { gettext } from '../../utils/constants';
|
||||||
import OperationMenu from './operation-menu';
|
import OperationMenu from './operation-menu';
|
||||||
|
|
||||||
|
const propTypes = {
|
||||||
|
dirent: PropTypes.object.isRequired,
|
||||||
|
onItemMenuShow: PropTypes.func.isRequired,
|
||||||
|
onItemMenuHide: PropTypes.func.isRequired,
|
||||||
|
onDelete: PropTypes.func.isRequired,
|
||||||
|
onDownload: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
class OperationGroup extends React.Component {
|
class OperationGroup extends React.Component {
|
||||||
|
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
@ -9,7 +18,7 @@ class OperationGroup extends React.Component {
|
|||||||
this.state = {
|
this.state = {
|
||||||
isItemMenuShow: false,
|
isItemMenuShow: false,
|
||||||
menuPosition: {top: 0, left: 0 },
|
menuPosition: {top: 0, left: 0 },
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
@ -34,24 +43,28 @@ class OperationGroup extends React.Component {
|
|||||||
this.props.onDelete();
|
this.props.onDelete();
|
||||||
}
|
}
|
||||||
|
|
||||||
onItemMenuShow = (e) => {
|
onItemMenuToggle = (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.nativeEvent.stopImmediatePropagation();
|
||||||
|
|
||||||
if (!this.state.isItemMenuShow) {
|
if (!this.state.isItemMenuShow) {
|
||||||
e.stopPropagation();
|
this.onItemMenuShow(e);
|
||||||
e.nativeEvent.stopImmediatePropagation();
|
|
||||||
|
|
||||||
let left = e.clientX - 8*16;
|
|
||||||
let top = e.clientY + 15;
|
|
||||||
let position = Object.assign({},this.state.menuPosition, {left: left, top: top});
|
|
||||||
this.setState({
|
|
||||||
menuPosition: position,
|
|
||||||
isItemMenuShow: true,
|
|
||||||
});
|
|
||||||
this.props.onItemMenuShow();
|
|
||||||
} else {
|
} else {
|
||||||
this.onItemMenuHide();
|
this.onItemMenuHide();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onItemMenuShow = (e) => {
|
||||||
|
let left = e.clientX - 8*16;
|
||||||
|
let top = e.clientY + 15;
|
||||||
|
let position = Object.assign({},this.state.menuPosition, {left: left, top: top});
|
||||||
|
this.setState({
|
||||||
|
menuPosition: position,
|
||||||
|
isItemMenuShow: true,
|
||||||
|
});
|
||||||
|
this.props.onItemMenuShow();
|
||||||
|
}
|
||||||
|
|
||||||
onItemMenuHide = () => {
|
onItemMenuHide = () => {
|
||||||
this.setState({
|
this.setState({
|
||||||
isItemMenuShow: false,
|
isItemMenuShow: false,
|
||||||
@ -81,13 +94,13 @@ class OperationGroup extends React.Component {
|
|||||||
<i className="sf2-icon-delete" title={gettext('Delete')} onClick={this.onDelete}></i>
|
<i className="sf2-icon-delete" title={gettext('Delete')} onClick={this.onDelete}></i>
|
||||||
</li>
|
</li>
|
||||||
<li className="operation-group-item">
|
<li className="operation-group-item">
|
||||||
<i className="sf2-icon-caret-down sf-dropdown-toggle" title={gettext('More Operation')} onClick={this.onItemMenuShow}></i>
|
<i className="sf2-icon-caret-down sf-dropdown-toggle" title={gettext('More Operation')} onClick={this.onItemMenuToggle}></i>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
{
|
{
|
||||||
this.state.isItemMenuShow &&
|
this.state.isItemMenuShow &&
|
||||||
<OperationMenu
|
<OperationMenu
|
||||||
currentItem={this.props.item}
|
dirent={this.props.dirent}
|
||||||
menuPosition={this.state.menuPosition}
|
menuPosition={this.state.menuPosition}
|
||||||
onRename={this.onRename}
|
onRename={this.onRename}
|
||||||
onCopy={this.onCopy}
|
onCopy={this.onCopy}
|
||||||
@ -98,4 +111,6 @@ class OperationGroup extends React.Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
OperationGroup.propTypes = propTypes;
|
||||||
|
|
||||||
export default OperationGroup;
|
export default OperationGroup;
|
||||||
|
@ -1,22 +1,47 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { gettext } from '../../utils/constants';
|
import { gettext, repoID } from '../../utils/constants';
|
||||||
|
import { seafileAPI } from '../../utils/seafile-api';
|
||||||
|
import Repo from '../../models/repo';
|
||||||
|
|
||||||
const propTypes = {
|
const propTypes = {
|
||||||
currentItem: PropTypes.object.isRequired,
|
dirent: PropTypes.object.isRequired,
|
||||||
menuPosition: PropTypes.object.isRequired,
|
menuPosition: PropTypes.object.isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
class OperationMenu extends React.Component {
|
class OperationMenu extends React.Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
repo: null,
|
||||||
|
is_repo_owner: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
seafileAPI.getRepoInfo(repoID).then(res => {
|
||||||
|
let repo = new Repo(res.data);
|
||||||
|
seafileAPI.getAccountInfo().then(res => {
|
||||||
|
let user_email = res.data.email;
|
||||||
|
let is_repo_owner = repo.owner_email === user_email;
|
||||||
|
this.setState({
|
||||||
|
repo: repo,
|
||||||
|
is_repo_owner: is_repo_owner
|
||||||
|
});
|
||||||
|
})
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
getItemType() {
|
getItemType() {
|
||||||
return this.props.currentItem.type;
|
let type = this.props.dirent.is_dir ? 'dir' : 'file';
|
||||||
|
return type;
|
||||||
}
|
}
|
||||||
|
|
||||||
renderDirentDirMenu() {
|
renderDirentDirMenu() {
|
||||||
let position = this.props.menuPosition;
|
let position = this.props.menuPosition;
|
||||||
let style = {position: 'fixed', left: position.left, top: position.top, display: 'block'};
|
let style = {position: 'fixed', left: position.left, top: position.top, display: 'block'};
|
||||||
if (this.props.currentItem.permission === 'rw') {
|
if (this.props.dirent.permission === 'rw') {
|
||||||
return (
|
return (
|
||||||
<ul className="dropdown-menu operation-menu" style={style}>
|
<ul className="dropdown-menu operation-menu" style={style}>
|
||||||
<li className="dropdown-item operation-menu-item">
|
<li className="dropdown-item operation-menu-item">
|
||||||
@ -44,7 +69,7 @@ class OperationMenu extends React.Component {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.props.currentItem.permission === 'r') {
|
if (this.props.dirent.permission === 'r') {
|
||||||
return (
|
return (
|
||||||
<ul className="dropdown-menu operation-menu" style={style}>
|
<ul className="dropdown-menu operation-menu" style={style}>
|
||||||
<li className="dropdown-item operation-menu-item">
|
<li className="dropdown-item operation-menu-item">
|
||||||
@ -62,7 +87,7 @@ class OperationMenu extends React.Component {
|
|||||||
renderDirentFileMenu() {
|
renderDirentFileMenu() {
|
||||||
let position = this.props.menuPosition;
|
let position = this.props.menuPosition;
|
||||||
let style = {position: 'fixed', left: position.left, top: position.top, display: 'block'};
|
let style = {position: 'fixed', left: position.left, top: position.top, display: 'block'};
|
||||||
if (this.props.currentItem.permission === 'rw') {
|
if (this.props.dirent.permission === 'rw') {
|
||||||
return (
|
return (
|
||||||
<ul className="dropdown-menu operation-menu" style={style}>
|
<ul className="dropdown-menu operation-menu" style={style}>
|
||||||
<li className="dropdown-item operation-menu-item">
|
<li className="dropdown-item operation-menu-item">
|
||||||
@ -105,7 +130,7 @@ class OperationMenu extends React.Component {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.props.currentItem.permission === "r") {
|
if (this.props.dirent.permission === 'r') {
|
||||||
return (
|
return (
|
||||||
<ul className="dropdown-menu operation-menu" style={style}>
|
<ul className="dropdown-menu operation-menu" style={style}>
|
||||||
<li className="dropdown-item operation-menu-item">
|
<li className="dropdown-item operation-menu-item">
|
||||||
@ -130,14 +155,14 @@ class OperationMenu extends React.Component {
|
|||||||
let type = this.getItemType();
|
let type = this.getItemType();
|
||||||
let menu = null;
|
let menu = null;
|
||||||
switch(type) {
|
switch(type) {
|
||||||
case 'file':
|
case 'file':
|
||||||
menu = this.renderDirentFileMenu();
|
menu = this.renderDirentFileMenu();
|
||||||
break;
|
break;
|
||||||
case 'dir':
|
case 'dir':
|
||||||
menu = this.renderDirentDirMenu();
|
menu = this.renderDirentDirMenu();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return menu;
|
return menu;
|
||||||
}
|
}
|
||||||
|
26
frontend/src/models/dirent.js
Normal file
26
frontend/src/models/dirent.js
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import moment from 'moment';
|
||||||
|
import { Utils } from '../utils/utils';
|
||||||
|
|
||||||
|
class Dirent {
|
||||||
|
constructor(json) {
|
||||||
|
this.id = json.id;
|
||||||
|
this.name = json.name;
|
||||||
|
this.type = json.type;
|
||||||
|
this.mtime = moment.unix(json.mtime).fromNow();
|
||||||
|
this.permission = json.permission;
|
||||||
|
if (json.type === 'file') {
|
||||||
|
this.size = Utils.bytesToSize(json.size);
|
||||||
|
this.starred = json.starred;
|
||||||
|
this.is_locked = json.is_locked;
|
||||||
|
this.lock_time = moment.unix(json.lock_time).fromNow();
|
||||||
|
this.lock_owner= json.lock_owner;
|
||||||
|
this.locked_by_me = json.locked_by_me;
|
||||||
|
this.modifier_name = json.modifier_name;
|
||||||
|
this.modifier_email = json.modifier_email;
|
||||||
|
this.modifier_contact_email = json.modifier_contact_email;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Dirent;
|
21
frontend/src/models/repo.js
Normal file
21
frontend/src/models/repo.js
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { Utils } from '../utils/utils';
|
||||||
|
|
||||||
|
class Repo {
|
||||||
|
constructor(object) {
|
||||||
|
this.repo_id = object.repo_id;
|
||||||
|
this.repo_name = object.name;
|
||||||
|
this.permission = object.permission;
|
||||||
|
this.size = Utils.bytesToSize(object.size);
|
||||||
|
this.file_count = object.file_count;
|
||||||
|
this.owner_name = object.owner_name;
|
||||||
|
this.owner_email = object.owner_email;
|
||||||
|
this.owner_contact_email = object.owner_contact_email;
|
||||||
|
this.is_admin = object.is_admin;
|
||||||
|
this.is_virtual = object.is_virtual;
|
||||||
|
this.no_quota = object.no_quota;
|
||||||
|
this.has_been_shared_out = object.has_been_shared_out;
|
||||||
|
this.encrypted = object.encrypted;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Repo;
|
@ -1,9 +1,27 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
import { gettext, repoID, serviceUrl, slug, siteRoot } from '../../utils/constants';
|
import { gettext, repoID, serviceUrl, slug, siteRoot } from '../../utils/constants';
|
||||||
|
import { seafileAPI } from '../../utils/seafile-api';
|
||||||
import CommonToolbar from '../../components/toolbar/common-toolbar';
|
import CommonToolbar from '../../components/toolbar/common-toolbar';
|
||||||
import PathToolbar from '../../components/toolbar/path-toolbar';
|
import PathToolbar from '../../components/toolbar/path-toolbar';
|
||||||
import MarkdownViewer from '../../components/markdown-viewer';
|
import MarkdownViewer from '../../components/markdown-viewer';
|
||||||
import TreeDirView from '../../components/tree-dir-view/tree-dir-view';
|
import DirentListView from '../../components/dirent-list-view/dirent-list-view';
|
||||||
|
import Dirent from '../../models/dirent';
|
||||||
|
|
||||||
|
const propTypes = {
|
||||||
|
content: PropTypes.string,
|
||||||
|
lastModified: PropTypes.string,
|
||||||
|
latestContributor: PropTypes.string,
|
||||||
|
permission: PropTypes.string,
|
||||||
|
filePath: PropTypes.string.isRequired,
|
||||||
|
isFileLoading: PropTypes.bool.isRequired,
|
||||||
|
isViewFileState: PropTypes.bool.isRequired,
|
||||||
|
onMenuClick: PropTypes.func.isRequired,
|
||||||
|
onSearchedClick: PropTypes.func.isRequired,
|
||||||
|
onMainNavBarClick: PropTypes.func.isRequired,
|
||||||
|
onMainItemClick: PropTypes.func.isRequired,
|
||||||
|
onMainItemDelete: PropTypes.func.isRequired,
|
||||||
|
}
|
||||||
|
|
||||||
class MainPanel extends Component {
|
class MainPanel extends Component {
|
||||||
|
|
||||||
@ -11,20 +29,35 @@ class MainPanel extends Component {
|
|||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.state = {
|
||||||
isWikiMode: true,
|
isWikiMode: true,
|
||||||
needOperationGroup: true,
|
direntList: []
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentWillReceiveProps(nextProps) {
|
||||||
|
let node = nextProps.changedNode;
|
||||||
|
if (node && node.isDir()) {
|
||||||
|
let path = node.path;
|
||||||
|
this.updateViewList(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateViewList = (filePath) => {
|
||||||
|
seafileAPI.listDir(repoID, filePath, 48).then(res => {
|
||||||
|
let direntList = [];
|
||||||
|
res.data.forEach(item => {
|
||||||
|
let dirent = new Dirent(item);
|
||||||
|
direntList.push(dirent);
|
||||||
|
});
|
||||||
|
this.setState({
|
||||||
|
direntList: direntList,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
onMenuClick = () => {
|
onMenuClick = () => {
|
||||||
this.props.onMenuClick();
|
this.props.onMenuClick();
|
||||||
}
|
}
|
||||||
|
|
||||||
onEditClick = (e) => {
|
|
||||||
// const w=window.open('about:blank')
|
|
||||||
e.preventDefault();
|
|
||||||
window.location.href= serviceUrl + '/lib/' + repoID + '/file' + this.props.filePath + '?mode=edit';
|
|
||||||
}
|
|
||||||
|
|
||||||
onMainNavBarClick = (e) => {
|
onMainNavBarClick = (e) => {
|
||||||
this.props.onMainNavBarClick(e.target.dataset.path);
|
this.props.onMainNavBarClick(e.target.dataset.path);
|
||||||
}
|
}
|
||||||
@ -38,8 +71,12 @@ class MainPanel extends Component {
|
|||||||
this.props.switchViewMode(e.target.id);
|
this.props.switchViewMode(e.target.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
onEditClick = (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
window.location.href= serviceUrl + '/lib/' + repoID + '/file' + this.props.filePath + '?mode=edit';
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
let filePathList = this.props.filePath.split('/');
|
let filePathList = this.props.filePath.split('/');
|
||||||
let nodePath = '';
|
let nodePath = '';
|
||||||
let pathElem = filePathList.map((item, index) => {
|
let pathElem = filePathList.map((item, index) => {
|
||||||
@ -88,31 +125,32 @@ class MainPanel extends Component {
|
|||||||
<div className="path-containter">
|
<div className="path-containter">
|
||||||
<a href={siteRoot + '#common/'} className="normal">{gettext('Libraries')}</a>
|
<a href={siteRoot + '#common/'} className="normal">{gettext('Libraries')}</a>
|
||||||
<span className="path-split">/</span>
|
<span className="path-split">/</span>
|
||||||
<a href={siteRoot + 'wiki/lib/' + repoID + '/'} className="normal">{slug}</a>
|
{
|
||||||
|
this.props.filePath === '/' ?
|
||||||
|
<span>{slug}</span> :
|
||||||
|
<a className="path-link" data-path="/" onClick={this.onMainNavBarClick}>{slug}</a>
|
||||||
|
}
|
||||||
{pathElem}
|
{pathElem}
|
||||||
</div>
|
</div>
|
||||||
<PathToolbar filePath={this.props.filePath}/>
|
<PathToolbar filePath={this.props.filePath}/>
|
||||||
</div>
|
</div>
|
||||||
<div className="cur-view-content">
|
<div className="cur-view-content">
|
||||||
{ this.props.isViewFileState &&
|
{ this.props.isViewFileState ?
|
||||||
<MarkdownViewer
|
<MarkdownViewer
|
||||||
markdownContent={this.props.content}
|
markdownContent={this.props.content}
|
||||||
latestContributor={this.props.latestContributor}
|
latestContributor={this.props.latestContributor}
|
||||||
lastModified = {this.props.lastModified}
|
lastModified = {this.props.lastModified}
|
||||||
onLinkClick={this.props.onLinkClick}
|
onLinkClick={this.props.onLinkClick}
|
||||||
isFileLoading={this.props.isFileLoading}
|
isFileLoading={this.props.isFileLoading}
|
||||||
|
/> :
|
||||||
|
<DirentListView
|
||||||
|
direntList={this.state.direntList}
|
||||||
|
filePath={this.props.filePath}
|
||||||
|
onItemClick={this.props.onMainItemClick}
|
||||||
|
onItemDelete={this.props.onMainItemDelete}
|
||||||
|
updateViewList={this.updateViewList}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
{ !this.props.isViewFileState &&
|
|
||||||
<TreeDirView
|
|
||||||
node={this.props.changedNode}
|
|
||||||
onMainNodeClick={this.props.onMainNodeClick}
|
|
||||||
onDeleteItem={this.props.onDeleteNode}
|
|
||||||
onRenameItem={this.props.onRenameNode}
|
|
||||||
needOperationGroup={this.state.needOperationGroup}
|
|
||||||
>
|
|
||||||
</TreeDirView>
|
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -120,4 +158,6 @@ class MainPanel extends Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MainPanel.propTypes = propTypes;
|
||||||
|
|
||||||
export default MainPanel;
|
export default MainPanel;
|
||||||
|
@ -54,31 +54,8 @@ class Wiki extends Component {
|
|||||||
this.exitViewFileState(treeData, node);
|
this.exitViewFileState(treeData, node);
|
||||||
this.setState({isFileLoading: false});
|
this.setState({isFileLoading: false});
|
||||||
} else {
|
} else {
|
||||||
seafileAPI.getFileInfo(repoID, filePath).then((res) => {
|
this.setState({tree_data: treeData});
|
||||||
let { mtime, permission, last_modifier_name } = res.data;
|
this.initMainPanelData(filePath);
|
||||||
|
|
||||||
this.setState({
|
|
||||||
tree_data: treeData,
|
|
||||||
latestContributor: last_modifier_name,
|
|
||||||
lastModified: moment.unix(mtime).fromNow(),
|
|
||||||
permission: permission,
|
|
||||||
filePath: filePath,
|
|
||||||
isFileLoading: false
|
|
||||||
});
|
|
||||||
|
|
||||||
seafileAPI.getFileDownloadLink(repoID, filePath).then((res) => {
|
|
||||||
const downLoadUrl = res.data;
|
|
||||||
seafileAPI.getFileContent(downLoadUrl).then((res) => {
|
|
||||||
this.setState({
|
|
||||||
content: res.data,
|
|
||||||
isFileLoading: false
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
let fileUrl = serviceUrl + '/wiki/lib/' + repoID + filePath;
|
|
||||||
window.history.pushState({urlPath: fileUrl, filePath: filePath}, filePath, fileUrl);
|
|
||||||
}
|
}
|
||||||
}, () => {
|
}, () => {
|
||||||
/* eslint-disable */
|
/* eslint-disable */
|
||||||
@ -149,7 +126,14 @@ class Wiki extends Component {
|
|||||||
|
|
||||||
onpopstate = (event) => {
|
onpopstate = (event) => {
|
||||||
if (event.state && event.state.filePath) {
|
if (event.state && event.state.filePath) {
|
||||||
this.initMainPanelData(event.state.filePath);
|
let path = event.state.filePath;
|
||||||
|
if (this.isMarkdownFile(path)) {
|
||||||
|
this.initMainPanelData(path);
|
||||||
|
} else {
|
||||||
|
let changedNode = this.state.tree_data.getNodeByPath(path);
|
||||||
|
this.exitViewFileState(this.state.tree_data, changedNode);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -177,9 +161,11 @@ class Wiki extends Component {
|
|||||||
window.history.pushState({urlPath: fileUrl, filePath: node.path},node.path, fileUrl);
|
window.history.pushState({urlPath: fileUrl, filePath: node.path},node.path, fileUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
onMainNodeClick = (node) => {
|
onMainItemClick = (direntPath) => {
|
||||||
let tree = this.state.tree_data.clone();
|
let tree = this.state.tree_data.clone();
|
||||||
tree.expandNode(node);
|
let node = tree.getNodeByPath(direntPath);
|
||||||
|
let parentNode = tree.findNodeParentFromTree(node);
|
||||||
|
tree.expandNode(parentNode);
|
||||||
if (node.isMarkdown()) {
|
if (node.isMarkdown()) {
|
||||||
this.initMainPanelData(node.path);
|
this.initMainPanelData(node.path);
|
||||||
this.enterViewFileState(tree, node, node.path);
|
this.enterViewFileState(tree, node, node.path);
|
||||||
@ -192,6 +178,15 @@ class Wiki extends Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onMainItemDelete = (direntPath) => {
|
||||||
|
let node = this.state.tree_data.getNodeByPath(direntPath);
|
||||||
|
this.onDeleteNode(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
onMainItemRename = () => {
|
||||||
|
//todos:
|
||||||
|
}
|
||||||
|
|
||||||
onNodeClick = (e, node) => {
|
onNodeClick = (e, node) => {
|
||||||
if (node instanceof Node && node.isMarkdown()){
|
if (node instanceof Node && node.isMarkdown()){
|
||||||
let tree = this.state.tree_data.clone();
|
let tree = this.state.tree_data.clone();
|
||||||
@ -315,7 +310,6 @@ class Wiki extends Component {
|
|||||||
});
|
});
|
||||||
} else if (node.isDir()) {
|
} else if (node.isDir()) {
|
||||||
editorUtilities.renameDir(filePath, newName).then(res => {
|
editorUtilities.renameDir(filePath, newName).then(res => {
|
||||||
|
|
||||||
let currentFilePath = this.state.filePath;
|
let currentFilePath = this.state.filePath;
|
||||||
let currentFileNode = tree.getNodeByPath(currentFilePath);
|
let currentFileNode = tree.getNodeByPath(currentFilePath);
|
||||||
let nodePath = node.path;
|
let nodePath = node.path;
|
||||||
@ -355,11 +349,18 @@ class Wiki extends Component {
|
|||||||
onDeleteNode = (node) => {
|
onDeleteNode = (node) => {
|
||||||
let filePath = node.path;
|
let filePath = node.path;
|
||||||
if (node.isDir()) {
|
if (node.isDir()) {
|
||||||
editorUtilities.deleteDir(filePath);
|
editorUtilities.deleteDir(filePath).then(() => {
|
||||||
|
this.deleteNode(node);
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
editorUtilities.deleteFile(filePath);
|
editorUtilities.deleteFile(filePath).then(() => {
|
||||||
|
this.deleteNode(node);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteNode = (node) => {
|
||||||
|
let tree = this.state.tree_data.clone();
|
||||||
|
|
||||||
let isCurrentFile = false;
|
let isCurrentFile = false;
|
||||||
if (node.isDir()) {
|
if (node.isDir()) {
|
||||||
@ -368,9 +369,9 @@ class Wiki extends Component {
|
|||||||
isCurrentFile = this.isModifyCurrentFile(node);
|
isCurrentFile = this.isModifyCurrentFile(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
let tree = this.state.tree_data.clone();
|
|
||||||
|
|
||||||
if (this.state.isViewFileState) {
|
if (this.state.isViewFileState) {
|
||||||
|
tree.deleteNode(node);
|
||||||
|
|
||||||
if (isCurrentFile) {
|
if (isCurrentFile) {
|
||||||
let homeNode = this.getHomeNode(tree);
|
let homeNode = this.getHomeNode(tree);
|
||||||
tree.expandNode(homeNode);
|
tree.expandNode(homeNode);
|
||||||
@ -385,13 +386,14 @@ class Wiki extends Component {
|
|||||||
} else {
|
} else {
|
||||||
let parentNode = tree.getNodeByPath(this.state.filePath);
|
let parentNode = tree.getNodeByPath(this.state.filePath);
|
||||||
let isChild = tree.isNodeChild(parentNode, node);
|
let isChild = tree.isNodeChild(parentNode, node);
|
||||||
|
|
||||||
|
tree.deleteNode(node);
|
||||||
if (isChild) {
|
if (isChild) {
|
||||||
this.exitViewFileState(tree, parentNode);
|
this.exitViewFileState(tree, parentNode);
|
||||||
} else {
|
} else {
|
||||||
this.setState({tree_data: tree});
|
this.setState({tree_data: tree});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
tree.deleteNode(node);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -528,14 +530,13 @@ class Wiki extends Component {
|
|||||||
isViewFileState={this.state.isViewFileState}
|
isViewFileState={this.state.isViewFileState}
|
||||||
changedNode={this.state.changedNode}
|
changedNode={this.state.changedNode}
|
||||||
isFileLoading={this.state.isFileLoading}
|
isFileLoading={this.state.isFileLoading}
|
||||||
|
switchViewMode={this.switchViewMode}
|
||||||
onLinkClick={this.onLinkClick}
|
onLinkClick={this.onLinkClick}
|
||||||
onMenuClick={this.onMenuClick}
|
onMenuClick={this.onMenuClick}
|
||||||
onSearchedClick={this.onSearchedClick}
|
onSearchedClick={this.onSearchedClick}
|
||||||
onMainNavBarClick={this.onMainNavBarClick}
|
onMainNavBarClick={this.onMainNavBarClick}
|
||||||
onMainNodeClick={this.onMainNodeClick}
|
onMainItemClick={this.onMainItemClick}
|
||||||
switchViewMode={this.switchViewMode}
|
onMainItemDelete={this.onMainItemDelete}
|
||||||
onDeleteNode={this.onDeleteNode}
|
|
||||||
onRenameNode={this.onRenameNode}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -777,9 +777,16 @@ a.op-icon:focus {
|
|||||||
.table-container table th {
|
.table-container table th {
|
||||||
padding-top: 1rem;
|
padding-top: 1rem;
|
||||||
}
|
}
|
||||||
|
.table-container table .select,
|
||||||
|
.table-container table .star,
|
||||||
.table-container table .icon {
|
.table-container table .icon {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.table-container table .star .empty {
|
||||||
|
color: #d0d0d0;
|
||||||
|
}
|
||||||
|
|
||||||
.table-container table .icon img {
|
.table-container table .icon img {
|
||||||
width: 1.5rem;
|
width: 1.5rem;
|
||||||
height: 1.5rem;
|
height: 1.5rem;
|
||||||
|
@ -13,7 +13,8 @@ from seahub.api2.utils import api_error
|
|||||||
|
|
||||||
from seahub.base.templatetags.seahub_tags import email2nickname, \
|
from seahub.base.templatetags.seahub_tags import email2nickname, \
|
||||||
email2contact_email
|
email2contact_email
|
||||||
from seahub.utils import is_org_context
|
from seahub.utils.repo import get_repo_owner, is_repo_admin, \
|
||||||
|
repo_has_been_shared_out
|
||||||
from seahub.views import check_folder_permission
|
from seahub.views import check_folder_permission
|
||||||
|
|
||||||
from seaserv import seafile_api
|
from seaserv import seafile_api
|
||||||
@ -45,10 +46,14 @@ class RepoView(APIView):
|
|||||||
error_msg = 'Permission denied.'
|
error_msg = 'Permission denied.'
|
||||||
return api_error(status.HTTP_403_FORBIDDEN, error_msg)
|
return api_error(status.HTTP_403_FORBIDDEN, error_msg)
|
||||||
|
|
||||||
if is_org_context(request):
|
username = request.user.username
|
||||||
repo_owner = seafile_api.get_org_repo_owner(repo_id)
|
repo_owner = get_repo_owner(request, repo_id)
|
||||||
else:
|
|
||||||
repo_owner = seafile_api.get_repo_owner(repo_id)
|
try:
|
||||||
|
has_been_shared_out = repo_has_been_shared_out(request, repo_id)
|
||||||
|
except Exception as e:
|
||||||
|
has_been_shared_out = False
|
||||||
|
logger.error(e)
|
||||||
|
|
||||||
result = {
|
result = {
|
||||||
"repo_id": repo.id,
|
"repo_id": repo.id,
|
||||||
@ -62,6 +67,10 @@ class RepoView(APIView):
|
|||||||
"encrypted": repo.encrypted,
|
"encrypted": repo.encrypted,
|
||||||
"file_count": repo.file_count,
|
"file_count": repo.file_count,
|
||||||
"permission": permission,
|
"permission": permission,
|
||||||
|
"no_quota": True if seafile_api.check_quota(repo_id) < 0 else False,
|
||||||
|
"is_admin": is_repo_admin(username, repo_id),
|
||||||
|
"is_virtual": repo.is_virtual,
|
||||||
|
"has_been_shared_out": has_been_shared_out,
|
||||||
}
|
}
|
||||||
|
|
||||||
return Response(result)
|
return Response(result)
|
||||||
|
@ -75,7 +75,8 @@ from seahub.utils.repo import get_repo_owner, get_library_storages, \
|
|||||||
get_locked_files_by_dir, get_related_users_by_repo, \
|
get_locked_files_by_dir, get_related_users_by_repo, \
|
||||||
is_valid_repo_id_format, can_set_folder_perm_by_user, \
|
is_valid_repo_id_format, can_set_folder_perm_by_user, \
|
||||||
add_encrypted_repo_secret_key_to_database
|
add_encrypted_repo_secret_key_to_database
|
||||||
from seahub.utils.star import star_file, unstar_file
|
from seahub.utils.star import star_file, unstar_file, \
|
||||||
|
get_dir_starred_files
|
||||||
from seahub.utils.file_types import DOCUMENT
|
from seahub.utils.file_types import DOCUMENT
|
||||||
from seahub.utils.file_size import get_file_size_unit
|
from seahub.utils.file_size import get_file_size_unit
|
||||||
from seahub.utils.file_op import check_file_lock
|
from seahub.utils.file_op import check_file_lock
|
||||||
@ -1893,7 +1894,7 @@ def get_dir_entrys_by_id(request, repo, path, dir_id, request_type=None):
|
|||||||
"""
|
"""
|
||||||
username = request.user.username
|
username = request.user.username
|
||||||
try:
|
try:
|
||||||
dirs = seafserv_threaded_rpc.list_dir_with_perm(repo.id, path, dir_id,
|
dirs = seafile_api.list_dir_with_perm(repo.id, path, dir_id,
|
||||||
username, -1, -1)
|
username, -1, -1)
|
||||||
dirs = dirs if dirs else []
|
dirs = dirs if dirs else []
|
||||||
except SearpcError, e:
|
except SearpcError, e:
|
||||||
@ -1945,10 +1946,16 @@ def get_dir_entrys_by_id(request, repo, path, dir_id, request_type=None):
|
|||||||
if e not in nickname_dict:
|
if e not in nickname_dict:
|
||||||
nickname_dict[e] = email2nickname(e)
|
nickname_dict[e] = email2nickname(e)
|
||||||
|
|
||||||
|
starred_files = get_dir_starred_files(username, repo.id, path)
|
||||||
for e in file_list:
|
for e in file_list:
|
||||||
e['modifier_contact_email'] = contact_email_dict.get(e['modifier_email'], '')
|
e['modifier_contact_email'] = contact_email_dict.get(e['modifier_email'], '')
|
||||||
e['modifier_name'] = nickname_dict.get(e['modifier_email'], '')
|
e['modifier_name'] = nickname_dict.get(e['modifier_email'], '')
|
||||||
|
|
||||||
|
file_path = posixpath.join(path, e['name'])
|
||||||
|
e['starred'] = False
|
||||||
|
if normalize_file_path(file_path) in starred_files:
|
||||||
|
e['starred'] = True
|
||||||
|
|
||||||
dir_list.sort(lambda x, y: cmp(x['name'].lower(), y['name'].lower()))
|
dir_list.sort(lambda x, y: cmp(x['name'].lower(), y['name'].lower()))
|
||||||
file_list.sort(lambda x, y: cmp(x['name'].lower(), y['name'].lower()))
|
file_list.sort(lambda x, y: cmp(x['name'].lower(), y['name'].lower()))
|
||||||
|
|
||||||
|
@ -240,5 +240,31 @@ def is_group_repo_staff(repo_id, username):
|
|||||||
|
|
||||||
return is_staff
|
return is_staff
|
||||||
|
|
||||||
|
def repo_has_been_shared_out(request, repo_id):
|
||||||
|
|
||||||
|
has_been_shared_out = False
|
||||||
|
username = request.user.username
|
||||||
|
|
||||||
|
if is_org_context(request):
|
||||||
|
org_id = request.user.org.org_id
|
||||||
|
|
||||||
|
is_inner_org_pub_repo = False
|
||||||
|
# check if current repo is pub-repo
|
||||||
|
org_pub_repos = seafile_api.list_org_inner_pub_repos_by_owner(
|
||||||
|
org_id, username)
|
||||||
|
for org_pub_repo in org_pub_repos:
|
||||||
|
if repo_id == org_pub_repo.id:
|
||||||
|
is_inner_org_pub_repo = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if seafile_api.org_repo_has_been_shared(repo_id, including_groups=True) or is_inner_org_pub_repo:
|
||||||
|
has_been_shared_out = True
|
||||||
|
else:
|
||||||
|
if seafile_api.repo_has_been_shared(repo_id, including_groups=True) or \
|
||||||
|
(not request.cloud_mode and seafile_api.is_inner_pub_repo(repo_id)):
|
||||||
|
has_been_shared_out = True
|
||||||
|
|
||||||
|
return has_been_shared_out
|
||||||
|
|
||||||
# TODO
|
# TODO
|
||||||
from seahub.share.utils import is_repo_admin
|
from seahub.share.utils import is_repo_admin
|
||||||
|
@ -1,14 +1,11 @@
|
|||||||
# Copyright (c) 2012-2016 Seafile Ltd.
|
# Copyright (c) 2012-2016 Seafile Ltd.
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
import logging
|
import logging
|
||||||
import urllib2
|
|
||||||
|
|
||||||
from django.db import IntegrityError
|
from django.db import IntegrityError
|
||||||
|
|
||||||
from pysearpc import SearpcError
|
|
||||||
from seaserv import seafile_api
|
|
||||||
|
|
||||||
from seahub.base.models import UserStarredFiles
|
from seahub.base.models import UserStarredFiles
|
||||||
|
from seahub.utils import normalize_file_path
|
||||||
|
|
||||||
# Get an instance of a logger
|
# Get an instance of a logger
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -61,5 +58,5 @@ def get_dir_starred_files(email, repo_id, parent_dir, org_id=-1):
|
|||||||
repo_id=repo_id,
|
repo_id=repo_id,
|
||||||
path__startswith=parent_dir,
|
path__startswith=parent_dir,
|
||||||
org_id=org_id)
|
org_id=org_id)
|
||||||
return [ f.path for f in starred_files ]
|
return [ normalize_file_path(f.path) for f in starred_files ]
|
||||||
|
|
||||||
|
@ -46,7 +46,8 @@ from seahub.utils import check_filename_with_rename, EMPTY_SHA1, \
|
|||||||
from seahub.utils.star import get_dir_starred_files
|
from seahub.utils.star import get_dir_starred_files
|
||||||
from seahub.utils.file_types import IMAGE, VIDEO
|
from seahub.utils.file_types import IMAGE, VIDEO
|
||||||
from seahub.utils.file_op import check_file_lock, ONLINE_OFFICE_LOCK_OWNER
|
from seahub.utils.file_op import check_file_lock, ONLINE_OFFICE_LOCK_OWNER
|
||||||
from seahub.utils.repo import get_locked_files_by_dir, get_repo_owner
|
from seahub.utils.repo import get_locked_files_by_dir, get_repo_owner, \
|
||||||
|
repo_has_been_shared_out
|
||||||
from seahub.utils.error_msg import file_type_error_msg, file_size_error_msg
|
from seahub.utils.error_msg import file_type_error_msg, file_size_error_msg
|
||||||
from seahub.base.accounts import User
|
from seahub.base.accounts import User
|
||||||
from seahub.thumbnail.utils import get_thumbnail_src
|
from seahub.thumbnail.utils import get_thumbnail_src
|
||||||
@ -332,26 +333,8 @@ def list_lib_dir(request, repo_id):
|
|||||||
result["is_admin"] = is_repo_admin(username, repo_id)
|
result["is_admin"] = is_repo_admin(username, repo_id)
|
||||||
if repo_owner == username:
|
if repo_owner == username:
|
||||||
result["is_repo_owner"] = True
|
result["is_repo_owner"] = True
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if is_org_context(request):
|
result["has_been_shared_out"] = repo_has_been_shared_out(request, repo_id)
|
||||||
org_id = request.user.org.org_id
|
|
||||||
|
|
||||||
is_inner_org_pub_repo = False
|
|
||||||
# check if current repo is pub-repo
|
|
||||||
org_pub_repos = seafile_api.list_org_inner_pub_repos_by_owner(
|
|
||||||
org_id, username)
|
|
||||||
for org_pub_repo in org_pub_repos:
|
|
||||||
if repo_id == org_pub_repo.id:
|
|
||||||
is_inner_org_pub_repo = True
|
|
||||||
break
|
|
||||||
|
|
||||||
if seafile_api.org_repo_has_been_shared(repo_id, including_groups=True) or is_inner_org_pub_repo:
|
|
||||||
result["has_been_shared_out"] = True
|
|
||||||
else:
|
|
||||||
if seafile_api.repo_has_been_shared(repo_id, including_groups=True) or \
|
|
||||||
(not request.cloud_mode and seafile_api.is_inner_pub_repo(repo_id)):
|
|
||||||
result["has_been_shared_out"] = True
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(e)
|
logger.error(e)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user