PteroTheme/resources/scripts/components/server/files/FileEditContainer.tsx

98 lines
3.6 KiB
TypeScript
Raw Normal View History

2019-10-20 01:35:01 +01:00
import React, { lazy, useEffect, useState } from 'react';
import { ServerContext } from '@/state/server';
2019-10-20 01:35:01 +01:00
import getFileContents from '@/api/server/files/getFileContents';
import useRouter from 'use-react-router';
2019-10-26 21:16:27 +01:00
import { Actions, useStoreState } from 'easy-peasy';
import { ApplicationStore } from '@/state';
import { httpErrorToHuman } from '@/api/http';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
import saveFileContents from '@/api/server/files/saveFileContents';
import FileManagerBreadcrumbs from '@/components/server/files/FileManagerBreadcrumbs';
2019-12-22 00:38:40 +00:00
import { useParams } from 'react-router';
import FileNameModal from '@/components/server/files/FileNameModal';
2019-10-12 23:29:45 +01:00
2019-10-19 23:31:02 +01:00
const LazyAceEditor = lazy(() => import(/* webpackChunkName: "editor" */'@/components/elements/AceEditor'));
2019-08-17 19:40:51 +01:00
export default () => {
2019-12-22 00:38:40 +00:00
const { action } = useParams();
const { history, location: { hash } } = useRouter();
const [ loading, setLoading ] = useState(action === 'edit');
2019-10-20 01:35:01 +01:00
const [ content, setContent ] = useState('');
2019-12-22 00:38:40 +00:00
const [ modalVisible, setModalVisible ] = useState(false);
2019-10-26 21:16:27 +01:00
2019-12-22 00:38:40 +00:00
const { id, uuid } = ServerContext.useStoreState(state => state.server.data!);
2019-10-26 21:16:27 +01:00
const addError = useStoreState((state: Actions<ApplicationStore>) => state.flashes.addError);
2019-10-26 21:16:27 +01:00
let fetchFileContent: null | (() => Promise<string>) = null;
2019-10-12 23:29:45 +01:00
2019-12-22 00:38:40 +00:00
if (action !== 'new') {
useEffect(() => {
getFileContents(uuid, hash.replace(/^#/, ''))
.then(setContent)
.catch(error => console.error(error))
.then(() => setLoading(false));
}, [ uuid, hash ]);
}
2019-08-17 19:40:51 +01:00
2019-12-22 00:38:40 +00:00
const save = (name?: string) => {
2019-10-26 21:16:27 +01:00
if (!fetchFileContent) {
return;
}
setLoading(true);
fetchFileContent()
.then(content => {
2019-12-22 00:38:40 +00:00
return saveFileContents(uuid, name || hash.replace(/^#/, ''), content);
})
.then(() => {
if (name) {
2019-12-22 01:43:50 +00:00
history.push(`/server/${id}/files/edit#/${name}`);
2019-12-22 00:38:40 +00:00
return;
}
return Promise.resolve();
2019-10-26 21:16:27 +01:00
})
.catch(error => {
console.error(error);
addError({ message: httpErrorToHuman(error), key: 'files' });
})
.then(() => setLoading(false));
};
2019-08-17 19:40:51 +01:00
return (
2019-10-26 21:16:27 +01:00
<div className={'mt-10 mb-4'}>
2019-12-22 00:38:40 +00:00
<FileManagerBreadcrumbs withinFileEditor={true} isNewFile={action !== 'edit'}/>
<FileNameModal
visible={modalVisible}
onDismissed={() => setModalVisible(false)}
onFileNamed={(name) => {
setModalVisible(false);
save(name);
}}
/>
2019-10-26 21:16:27 +01:00
<div className={'relative'}>
<SpinnerOverlay visible={loading}/>
<LazyAceEditor
initialModePath={hash.replace(/^#/, '') || 'plain_text'}
initialContent={content}
fetchContent={value => {
fetchFileContent = value;
}}
onContentSaved={() => null}
/>
</div>
2019-10-20 01:35:01 +01:00
<div className={'flex justify-end mt-4'}>
2019-12-22 00:38:40 +00:00
{action === 'edit' ?
<button className={'btn btn-primary btn-sm'} onClick={() => save()}>
Save Content
</button>
:
<button className={'btn btn-primary btn-sm'} onClick={() => setModalVisible(true)}>
Create File
</button>
}
2019-10-20 01:35:01 +01:00
</div>
2019-08-17 19:40:51 +01:00
</div>
);
};