parent
746efef9bf
commit
01cb6944b7
@ -1,132 +0,0 @@
|
||||
import $ from 'jquery';
|
||||
import _ from 'lodash';
|
||||
import { ConfirmDialog } from 'vj/components/dialog/index';
|
||||
import createHint from 'vj/components/hint';
|
||||
import Notification from 'vj/components/notification';
|
||||
import { previewFile } from 'vj/components/preview/preview.page';
|
||||
import uploadFiles from 'vj/components/upload';
|
||||
import download from 'vj/components/zipDownloader';
|
||||
import { NamedPage } from 'vj/misc/Page';
|
||||
import {
|
||||
i18n, pjax, request, tpl,
|
||||
} from 'vj/utils';
|
||||
|
||||
async function downloadProblemFilesAsArchive(type, files) {
|
||||
const { links, pdoc } = await request.post('', { operation: 'get_links', files, type });
|
||||
const targets = [];
|
||||
for (const filename of Object.keys(links)) targets.push({ filename, url: links[filename] });
|
||||
await download(`${pdoc.docId} ${pdoc.title}.zip`, targets);
|
||||
}
|
||||
|
||||
const page = new NamedPage('problem_files', () => {
|
||||
function ensureAndGetSelectedFiles(type) {
|
||||
const files = _.map(
|
||||
$(`.problem-files-${type} tbody [data-checkbox-group="${type}"]:checked`),
|
||||
(ch) => $(ch).closest('tr').attr('data-filename'),
|
||||
);
|
||||
if (files.length === 0) {
|
||||
Notification.error(i18n('Please select at least one file to perform this operation.'));
|
||||
return null;
|
||||
}
|
||||
return files;
|
||||
}
|
||||
|
||||
async function handleClickUpload(type, files) {
|
||||
if (!files) {
|
||||
const input = document.createElement('input');
|
||||
input.type = 'file';
|
||||
input.multiple = true;
|
||||
input.click();
|
||||
await new Promise((resolve) => { input.onchange = resolve; });
|
||||
files = input.files;
|
||||
}
|
||||
if (!files.length) {
|
||||
Notification.warn(i18n('No file selected.'));
|
||||
return;
|
||||
}
|
||||
await uploadFiles('', files, { type, pjax: true });
|
||||
}
|
||||
|
||||
async function handleClickDownloadSelected(type) {
|
||||
const selectedFiles = ensureAndGetSelectedFiles(type);
|
||||
if (selectedFiles === null) return;
|
||||
await downloadProblemFilesAsArchive(type, selectedFiles);
|
||||
}
|
||||
|
||||
async function handleClickRemoveSelected(type) {
|
||||
const selectedFiles = ensureAndGetSelectedFiles(type);
|
||||
if (selectedFiles === null) return;
|
||||
const action = await new ConfirmDialog({
|
||||
$body: tpl.typoMsg(i18n('Confirm to delete the selected files?')),
|
||||
}).open();
|
||||
if (action !== 'yes') return;
|
||||
try {
|
||||
await request.post('', {
|
||||
operation: 'delete_files',
|
||||
files: selectedFiles,
|
||||
type,
|
||||
});
|
||||
Notification.success(i18n('Selected files have been deleted.'));
|
||||
await pjax.request({ push: false });
|
||||
} catch (error) {
|
||||
Notification.error(error.message);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string} type
|
||||
* @param {JQuery.DragOverEvent<HTMLElement, undefined, HTMLElement, HTMLElement>} ev
|
||||
*/
|
||||
function handleDragOver(type, ev) {
|
||||
ev.preventDefault();
|
||||
// TODO display a drag-drop allowed hint
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string} type
|
||||
* @param {JQuery.DropEvent<HTMLElement, undefined, HTMLElement, HTMLElement>} ev
|
||||
*/
|
||||
function handleDrop(type, ev) {
|
||||
ev.preventDefault();
|
||||
if (!$('[name="upload_testdata"]').length) {
|
||||
Notification.error(i18n("You don't have permission to upload file."));
|
||||
return;
|
||||
}
|
||||
ev = ev.originalEvent;
|
||||
const files = [];
|
||||
if (ev.dataTransfer.items) {
|
||||
for (let i = 0; i < ev.dataTransfer.items.length; i++) {
|
||||
if (ev.dataTransfer.items[i].kind === 'file') {
|
||||
const file = ev.dataTransfer.items[i].getAsFile();
|
||||
files.push(file);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (let i = 0; i < ev.dataTransfer.files.length; i++) {
|
||||
files.push(ev.dataTransfer.files[i]);
|
||||
}
|
||||
}
|
||||
handleClickUpload(type, files);
|
||||
}
|
||||
|
||||
if ($('[name="upload_testdata"]').length) {
|
||||
$(document).on('click', '[name="upload_testdata"]', () => handleClickUpload('testdata'));
|
||||
$(document).on('click', '[name="upload_file"]', () => handleClickUpload('additional_file'));
|
||||
$(document).on('click', '[name="create_testdata"]', () => previewFile(undefined, 'testdata'));
|
||||
$(document).on('click', '[name="create_file"]', () => previewFile(undefined, 'additional_file'));
|
||||
$(document).on('click', '[name="remove_selected_testdata"]', () => handleClickRemoveSelected('testdata'));
|
||||
$(document).on('click', '[name="remove_selected_file"]', () => handleClickRemoveSelected('additional_file'));
|
||||
}
|
||||
$(document).on('dragover', '.problem-files-testdata', (ev) => handleDragOver('testdata', ev));
|
||||
$(document).on('dragover', '.problem-files-additional_file', (ev) => handleDragOver('additional_file', ev));
|
||||
$(document).on('drop', '.problem-files-testdata', (ev) => handleDrop('testdata', ev));
|
||||
$(document).on('drop', '.problem-files-additional_file', (ev) => handleDrop('additional_file', ev));
|
||||
$(document).on('click', '[name="download_selected_testdata"]', () => handleClickDownloadSelected('testdata'));
|
||||
$(document).on('click', '[name="download_selected_file"]', () => handleClickDownloadSelected('additional_file'));
|
||||
$(document).on('vjContentNew', (e) => {
|
||||
createHint('Hint::icon::testdata', $(e.target).find('[name="create_testdata"]').get(0)?.parentNode?.parentNode?.children?.[0]);
|
||||
});
|
||||
createHint('Hint::icon::testdata', $(document).find('[name="create_testdata"]').get(0)?.parentNode?.parentNode?.children?.[0]);
|
||||
});
|
||||
|
||||
export default page;
|
@ -0,0 +1,347 @@
|
||||
import $ from 'jquery';
|
||||
import { map } from 'lodash';
|
||||
import React from 'react';
|
||||
import ReactDOM from 'react-dom/client';
|
||||
import { ActionDialog, ConfirmDialog } from 'vj/components/dialog/index';
|
||||
import createHint from 'vj/components/hint';
|
||||
import Notification from 'vj/components/notification';
|
||||
import { previewFile } from 'vj/components/preview/preview.page';
|
||||
import uploadFiles from 'vj/components/upload';
|
||||
import download from 'vj/components/zipDownloader';
|
||||
import { NamedPage } from 'vj/misc/Page';
|
||||
import {
|
||||
i18n, pjax, request, tpl,
|
||||
} from 'vj/utils';
|
||||
|
||||
async function downloadProblemFilesAsArchive(type, files) {
|
||||
const { links, pdoc } = await request.post('', { operation: 'get_links', files, type });
|
||||
const targets = [];
|
||||
for (const filename of Object.keys(links)) targets.push({ filename, url: links[filename] });
|
||||
await download(`${pdoc.docId} ${pdoc.title}.zip`, targets);
|
||||
}
|
||||
|
||||
const page = new NamedPage('problem_files', () => {
|
||||
function ensureAndGetSelectedFiles(type) {
|
||||
const files = map(
|
||||
$(`.problem-files-${type} tbody [data-checkbox-group="${type}"]:checked`),
|
||||
(ch) => $(ch).closest('tr').attr('data-filename'),
|
||||
);
|
||||
if (files.length === 0) {
|
||||
Notification.error(i18n('Please select at least one file to perform this operation.'));
|
||||
return null;
|
||||
}
|
||||
return files;
|
||||
}
|
||||
|
||||
async function handleClickUpload(type, files?) {
|
||||
if (!files) {
|
||||
const input = document.createElement('input');
|
||||
input.type = 'file';
|
||||
input.multiple = true;
|
||||
input.click();
|
||||
await new Promise((resolve) => { input.onchange = resolve; });
|
||||
files = input.files;
|
||||
}
|
||||
if (!files.length) {
|
||||
Notification.warn(i18n('No file selected.'));
|
||||
return;
|
||||
}
|
||||
await uploadFiles('', files, { type, pjax: true });
|
||||
}
|
||||
|
||||
async function handleClickDownloadSelected(type) {
|
||||
const selectedFiles = ensureAndGetSelectedFiles(type);
|
||||
if (selectedFiles === null) return;
|
||||
await downloadProblemFilesAsArchive(type, selectedFiles);
|
||||
}
|
||||
|
||||
async function handleClickRename(ev, type) {
|
||||
const file = [$(ev.currentTarget).parent().parent().attr('data-filename')];
|
||||
// eslint-disable-next-line no-alert
|
||||
const newName = prompt(i18n('Enter a new name for the file: '));
|
||||
if (!newName) return;
|
||||
try {
|
||||
await request.post('./files', {
|
||||
operation: 'rename_files',
|
||||
files: file,
|
||||
newNames: [newName],
|
||||
type,
|
||||
});
|
||||
Notification.success(i18n('File have been renamed.'));
|
||||
await pjax.request({ push: false });
|
||||
} catch (error) {
|
||||
Notification.error(error.message);
|
||||
}
|
||||
}
|
||||
|
||||
let prism = null;
|
||||
const load = import('../components/highlighter/prismjs');
|
||||
load.then(({ default: p }) => {
|
||||
prism = p.Prism;
|
||||
}).catch(() => { });
|
||||
|
||||
async function handleClickRenameSelected(type) {
|
||||
const selectedFiles = ensureAndGetSelectedFiles(type);
|
||||
if (!selectedFiles?.length) return;
|
||||
let onActionButton = (_: string) => false; // eslint-disable-line @typescript-eslint/no-unused-vars
|
||||
|
||||
function Rename(props) {
|
||||
const [original, setOriginal] = React.useState('');
|
||||
const [replace, setReplace] = React.useState('');
|
||||
const [prefix, setPrefix] = React.useState('');
|
||||
const [suffix, setSuffix] = React.useState('');
|
||||
const [regexValid, setRegexValid] = React.useState(true);
|
||||
const [wantNext, setWantNext] = React.useState(false);
|
||||
const [preview, setPreview] = React.useState(false);
|
||||
const [highlight, setHighlight] = React.useState(null);
|
||||
const [newNames, setNewNames] = React.useState(props.names);
|
||||
|
||||
React.useEffect(() => {
|
||||
let s: string | RegExp = original;
|
||||
setRegexValid(true);
|
||||
setHighlight(null);
|
||||
if (original.length > 2 && original.startsWith('/')) {
|
||||
const availableFlags = ['g', 'i'];
|
||||
const flags = [];
|
||||
let copy = original.substring(1);
|
||||
while (availableFlags.includes(copy[copy.length - 1])) {
|
||||
flags.push(copy[copy.length - 1]);
|
||||
copy = copy.substring(0, copy.length - 1);
|
||||
}
|
||||
if (copy.endsWith('/')) {
|
||||
copy = copy.substring(0, copy.length - 1);
|
||||
flags.reverse();
|
||||
if (prism) setHighlight(`/${prism.highlight(copy, prism.languages.regex, 'RegExp')}/${flags.join('')}`);
|
||||
try {
|
||||
s = new RegExp(copy, flags.join(''));
|
||||
setRegexValid(true);
|
||||
} catch (e) {
|
||||
setRegexValid(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
setNewNames(selectedFiles.map((file) => {
|
||||
if (s) file = file.replace(s, replace);
|
||||
return prefix + file + suffix;
|
||||
}));
|
||||
}, [original, replace, prefix, suffix]);
|
||||
|
||||
onActionButton = (action) => {
|
||||
if (action === 'ok') {
|
||||
if (!preview) {
|
||||
if (!regexValid) return false;
|
||||
if (!original && !prefix && !suffix) {
|
||||
setWantNext(true);
|
||||
return false;
|
||||
}
|
||||
setPreview(true);
|
||||
return false;
|
||||
}
|
||||
request.post('', {
|
||||
operation: 'rename_files',
|
||||
files: selectedFiles,
|
||||
newNames,
|
||||
type,
|
||||
}).then(() => {
|
||||
Notification.success(i18n('Selected files have been renamed.'));
|
||||
pjax.request({ push: false });
|
||||
}).catch((error) => {
|
||||
Notification.error(error.message);
|
||||
});
|
||||
return true;
|
||||
}
|
||||
if (preview) {
|
||||
setPreview(false);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
};
|
||||
|
||||
const style = { fontFamily: 'var(--code-font-family)' };
|
||||
|
||||
return <div className="typo">
|
||||
{!preview ? <>
|
||||
<div className="row">
|
||||
<div className="medium-6 small-6 columns">
|
||||
<h2>{i18n('Batch replacement')}</h2>
|
||||
<label>{i18n('Original content')}
|
||||
<div style={{ position: 'relative' }}>
|
||||
<div className="textbox-container" style={{ zIndex: 1, position: 'relative' }}>
|
||||
<input
|
||||
className="textbox"
|
||||
type="text"
|
||||
style={{ ...style, ...(highlight ? { color: 'transparent', background: 'transparent', caretColor: 'black' } : {}) }}
|
||||
value={original}
|
||||
onChange={(e) => setOriginal(e.currentTarget.value)}
|
||||
></input>
|
||||
</div>
|
||||
<div className="textbox-container" style={{
|
||||
position: 'absolute', top: 0, left: 0, zIndex: 0,
|
||||
}}>
|
||||
{highlight && <span className="textbox" style={{
|
||||
...style, border: 'none', display: 'inline-flex', alignItems: 'center',
|
||||
}} dangerouslySetInnerHTML={{ __html: highlight }} />}
|
||||
</div>
|
||||
</div>
|
||||
</label>
|
||||
<label>{i18n('Replace with')}
|
||||
<div className="textbox-container">
|
||||
<input className="textbox" type="text" value={replace} onChange={(e) => setReplace(e.currentTarget.value)}></input>
|
||||
</div>
|
||||
</label>
|
||||
</div>
|
||||
<div className="medium-6 small-6 columns">
|
||||
<h2>{i18n('Add prefix/suffix')}</h2>
|
||||
<label>{i18n('Add prefix')}
|
||||
<div className="textbox-container">
|
||||
<input className="textbox" type="text" value={prefix} onChange={(e) => setPrefix(e.currentTarget.value)}></input>
|
||||
</div>
|
||||
</label>
|
||||
<label>{i18n('Add suffix')}
|
||||
<div className="textbox-container">
|
||||
<input className="textbox" type="text" value={suffix} onChange={(e) => setSuffix(e.currentTarget.value)}></input>
|
||||
</div>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
<div className="row">
|
||||
<div className="medium-12 columns">
|
||||
<p>{!regexValid ? i18n('Invalid RegExp') : wantNext ? i18n('No changes to make.') : i18n('RegExp supported, quote with "/"')}</p>
|
||||
</div>
|
||||
</div>
|
||||
</> : <div>
|
||||
<p>{i18n('Are you sure to rename the following file?')}</p>
|
||||
<ul>
|
||||
{original && <li>Replace {original} with {replace}</li>}
|
||||
{prefix && <li>Add {prefix} as prefix</li>}
|
||||
{suffix && <li>Add {suffix} as suffix</li>}
|
||||
</ul>
|
||||
<table className="data-table rename-confirm-table">
|
||||
<colgroup>
|
||||
<col className="col--origin" />
|
||||
<col className="col--new" />
|
||||
</colgroup>
|
||||
<thead>
|
||||
<tr>
|
||||
<th className="col--origin">{i18n('Original filename(s)')}</th>
|
||||
<th className="col--new">{i18n('New filename(s)')}</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody style={{ maxHeight: '60vh', overflow: 'scroll' }}>
|
||||
{selectedFiles.map((file, index) => <tr key={file}>
|
||||
<td className="col--origin">{file}</td>
|
||||
<td className="col--new">{newNames[index]}</td>
|
||||
</tr>)}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>}
|
||||
</div >;
|
||||
}
|
||||
|
||||
const promise = new ActionDialog({
|
||||
$body: tpl`<div id="rename_dialog"></div>`,
|
||||
width: '600px',
|
||||
onDispatch(action) {
|
||||
return onActionButton(action);
|
||||
},
|
||||
}).open();
|
||||
const root = ReactDOM.createRoot(document.getElementById('rename_dialog'));
|
||||
root.render(<Rename names={selectedFiles} />);
|
||||
await promise;
|
||||
root.unmount();
|
||||
}
|
||||
|
||||
async function handleClickRemove(ev, type) {
|
||||
const file = [$(ev.currentTarget).parent().parent().attr('data-filename')];
|
||||
const action = await new ConfirmDialog({
|
||||
$body: tpl.typoMsg(i18n('Confirm to delete the file?')),
|
||||
}).open();
|
||||
if (action !== 'yes') return;
|
||||
try {
|
||||
await request.post('./files', {
|
||||
operation: 'delete_files',
|
||||
files: file,
|
||||
type,
|
||||
});
|
||||
Notification.success(i18n('File have been deleted.'));
|
||||
await pjax.request({ push: false });
|
||||
} catch (error) {
|
||||
Notification.error(error.message);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleClickRemoveSelected(type) {
|
||||
const selectedFiles = ensureAndGetSelectedFiles(type);
|
||||
if (selectedFiles === null) return;
|
||||
const action = await new ConfirmDialog({
|
||||
$body: tpl.typoMsg(i18n('Confirm to delete the selected files?')),
|
||||
}).open();
|
||||
if (action !== 'yes') return;
|
||||
try {
|
||||
await request.post('', {
|
||||
operation: 'delete_files',
|
||||
files: selectedFiles,
|
||||
type,
|
||||
});
|
||||
Notification.success(i18n('Selected files have been deleted.'));
|
||||
await pjax.request({ push: false });
|
||||
} catch (error) {
|
||||
Notification.error(error.message);
|
||||
}
|
||||
}
|
||||
|
||||
function handleDragOver(type: string, ev: JQuery.DragOverEvent<Document, undefined, HTMLElement, HTMLElement>) {
|
||||
ev.preventDefault();
|
||||
// TODO display a drag-drop allowed hint
|
||||
}
|
||||
|
||||
function handleDrop(type: string, e: JQuery.DropEvent<Document, undefined, HTMLElement, HTMLElement>) {
|
||||
e.preventDefault();
|
||||
if (!$('[name="upload_testdata"]').length) {
|
||||
Notification.error(i18n("You don't have permission to upload file."));
|
||||
return;
|
||||
}
|
||||
const ev = e.originalEvent;
|
||||
const files = [];
|
||||
if (ev.dataTransfer.items) {
|
||||
for (let i = 0; i < ev.dataTransfer.items.length; i++) {
|
||||
if (ev.dataTransfer.items[i].kind === 'file') {
|
||||
const file = ev.dataTransfer.items[i].getAsFile();
|
||||
files.push(file);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (let i = 0; i < ev.dataTransfer.files.length; i++) {
|
||||
files.push(ev.dataTransfer.files[i]);
|
||||
}
|
||||
}
|
||||
handleClickUpload(type, files);
|
||||
}
|
||||
|
||||
if ($('[name="upload_testdata"]').length) {
|
||||
$(document).on('click', '[name="upload_testdata"]', () => handleClickUpload('testdata'));
|
||||
$(document).on('click', '[name="upload_file"]', () => handleClickUpload('additional_file'));
|
||||
$(document).on('click', '[name="create_testdata"]', () => previewFile(undefined, 'testdata'));
|
||||
$(document).on('click', '[name="create_file"]', () => previewFile(undefined, 'additional_file'));
|
||||
$(document).on('click', '[name="testdata__rename"]', (ev) => handleClickRename(ev, 'testdata'));
|
||||
$(document).on('click', '[name="additional_file__rename"]', (ev) => handleClickRename(ev, 'additional_file'));
|
||||
$(document).on('click', '[name="rename_selected_testdata"]', () => handleClickRenameSelected('testdata'));
|
||||
$(document).on('click', '[name="rename_selected_file"]', () => handleClickRenameSelected('additional_file'));
|
||||
$(document).on('click', '[name="testdata__delete"]', (ev) => handleClickRemove(ev, 'testdata'));
|
||||
$(document).on('click', '[name="additional_file__delete"]', (ev) => handleClickRemove(ev, 'additional_file'));
|
||||
$(document).on('click', '[name="remove_selected_testdata"]', () => handleClickRemoveSelected('testdata'));
|
||||
$(document).on('click', '[name="remove_selected_file"]', () => handleClickRemoveSelected('additional_file'));
|
||||
}
|
||||
$(document).on('dragover', '.problem-files-testdata', (ev) => handleDragOver('testdata', ev));
|
||||
$(document).on('dragover', '.problem-files-additional_file', (ev) => handleDragOver('additional_file', ev));
|
||||
$(document).on('drop', '.problem-files-testdata', (ev) => handleDrop('testdata', ev));
|
||||
$(document).on('drop', '.problem-files-additional_file', (ev) => handleDrop('additional_file', ev));
|
||||
$(document).on('click', '[name="download_selected_testdata"]', () => handleClickDownloadSelected('testdata'));
|
||||
$(document).on('click', '[name="download_selected_file"]', () => handleClickDownloadSelected('additional_file'));
|
||||
$(document).on('vjContentNew', (e) => {
|
||||
createHint('Hint::icon::testdata', $(e.target).find('[name="create_testdata"]').get(0)?.parentNode?.parentNode?.children?.[0]);
|
||||
});
|
||||
createHint('Hint::icon::testdata', $(document).find('[name="create_testdata"]').get(0)?.parentNode?.parentNode?.children?.[0]);
|
||||
});
|
||||
|
||||
export default page;
|
Loading…
Reference in New Issue