Files
lucko-paste/src/components/EditorControls.js

160 lines
3.5 KiB
JavaScript
Raw Normal View History

2021-03-26 22:00:12 +00:00
import { useState, useEffect } from 'react';
2021-03-27 11:49:46 +00:00
import styled from 'styled-components';
2021-03-26 22:00:12 +00:00
import { gzip } from 'pako';
import history from 'history/browser';
import copy from 'copy-to-clipboard';
2021-03-27 11:49:46 +00:00
import { languageIds } from '../highlighting';
import themes from '../style/themes';
2021-03-27 11:59:56 +00:00
export default function EditorControls({ code, setCode, language, setLanguage, theme, setTheme }) {
2021-03-26 22:00:12 +00:00
const [saving, setSaving] = useState(false);
const [recentlySaved, setRecentlySaved] = useState(false);
useEffect(() => {
setRecentlySaved(false);
}, [code, language])
function save() {
if (!code || recentlySaved) {
return;
}
setSaving(true);
saveToBytebin(code, language).then((pasteId) => {
setSaving(false);
setRecentlySaved(true);
history.replace({
2021-03-27 11:59:56 +00:00
pathname: pasteId
2021-03-26 22:00:12 +00:00
});
copy(window.location.href);
document.title = 'paste | ' + pasteId;
});
}
2021-03-27 11:59:56 +00:00
function reset() {
setCode('');
setLanguage('plain');
history.replace({
pathname: '/'
});
document.title = 'paste';
}
2021-03-27 11:49:46 +00:00
return (
<Header>
<Section>
2021-03-27 11:59:56 +00:00
<Button onClick={reset}>[new]</Button>
2021-03-27 11:49:46 +00:00
<Button onClick={save}>
{recentlySaved
? '[link copied!]'
: saving ? '[saving...]' : '[save]'
}
</Button>
<MenuButton label="language" value={language} setValue={setLanguage} ids={languageIds} />
</Section>
<Section>
2021-03-27 11:59:56 +00:00
2021-03-27 11:49:46 +00:00
<MenuButton label="theme" value={theme} setValue={setTheme} ids={Object.keys(themes)} />
<Button as="a" href="https://github.com/lucko/paste" target="_blank" rel="noreferrer">[about]</Button>
</Section>
</Header>
)
}
const Header = styled.header`
position: fixed;
z-index: 2;
width: 100%;
height: 2em;
color: ${props => props.theme.primary};
background: ${props => props.theme.secondary};
display: flex;
justify-content: space-between;
`;
const Section = styled.div`
display: flex;
align-items: center;
`;
const Button = styled.div`
cursor: pointer;
height: 100%;
display: flex;
align-items: center;
padding: 0 .25em;
color: inherit;
text-decoration: none;
:hover {
background: ${props => props.theme.highlight};
}
`;
const Menu = styled.ul`
position: absolute;
top: 2em;
margin: 0;
padding: 0;
list-style: none;
background-color: ${props => props.theme.highlight};
> li {
padding: .15em .5em;
2021-03-26 22:00:12 +00:00
}
2021-03-27 11:49:46 +00:00
> li:hover {
background-color: ${props => props.theme.secondary};
}
`;
const MenuButton = ({ label, ids, value, setValue }) => {
const [open, setOpen] = useState(false);
function toggleOpen() {
setOpen(!open);
}
function select(e, id) {
2021-03-26 22:00:12 +00:00
e.stopPropagation();
2021-03-27 11:49:46 +00:00
setOpen(false);
setValue(id);
2021-03-26 22:00:12 +00:00
}
return (
2021-03-27 11:49:46 +00:00
<Button onClick={toggleOpen}>
[{label}: {value}]
{open && (
<Menu>
{ids.map(id => <li key={id} onClick={e => select(e, id)}>{id}</li>)}
</Menu>
)}
</Button>
2021-03-26 22:00:12 +00:00
)
}
async function saveToBytebin(code, language) {
try {
const compressed = gzip(code);
const contentType = 'text/' + language;
const resp = await fetch('https://bytebin.lucko.me/post', {
method: 'POST',
headers: {
'Content-Type': contentType,
'Content-Encoding': 'gzip',
'Accept': 'application/json'
},
body: compressed
});
if (resp.ok) {
const json = await resp.json();
return json.key;
}
} catch (e) {
console.log(e);
}
return null;
}