diff --git a/client/scss/_dropzone.scss b/client/scss/_dropzone.scss
index 09b8b1bc..a3934edd 100644
--- a/client/scss/_dropzone.scss
+++ b/client/scss/_dropzone.scss
@@ -4,6 +4,7 @@
// be a flex container for children
display: flex;
flex-direction: column;
+ position: relative;
}
.dropzone {
@@ -12,6 +13,7 @@
flex: 1 0 auto;
// be a flex container for children
display: flex;
+ padding: 1em;
-webkit-flex-direction: column;
flex-direction: column;
justify-content: center;
@@ -52,8 +54,37 @@
.dropzone-preview-image {
display: block;
- padding: 1em;
- width: calc(100% - 2em);
+ width: 100%;
+}
+
+.dropzone-preview-memeify {
+ margin-top: 3em;
+}
+
+.dropzone-memeify-button {
+ background: $primary-color;
+ color: #fff;
+ cursor: pointer;
+ font-size: .8em;
+ padding: 3px 6px;
+ position: absolute;
+ right: 0;
+ top: 0;
+ z-index: 3;
+}
+
+.dropzone-memeify-saveMessage {
+ padding-top: .25em;
+ position: relative;
+ top: .5em;
+}
+
+.dropzone-memeify-toolbar {
+ /* TODO: Cleanup `!important` */
+ background: $primary-color !important;
+ left: -1em !important;
+ right: -1em !important;
+ top: -4em !important;
}
.dropzone-instructions-display__chooser-label {
diff --git a/client/src/components/Creatify/EditableFontface/index.js b/client/src/components/Creatify/EditableFontface/index.js
new file mode 100644
index 00000000..2ef131e0
--- /dev/null
+++ b/client/src/components/Creatify/EditableFontface/index.js
@@ -0,0 +1,99 @@
+import React, { Component } from 'react';
+
+const DEFAULT_TEXT_RENDER = (text) => text;
+
+export default class EditableFontface extends Component {
+ constructor(props) {
+ super(props);
+
+ this.state = {
+ blinkSelection: props.blinkSelection == false ? false : true,
+ value: props.value,
+ };
+
+ this.textInput = React.createRef();
+ }
+
+ componentDidMount() {
+ const textInput = this.textInput.current;
+
+ if(textInput) {
+ textInput.focus();
+ }
+ }
+
+ render() {
+ const me = this;
+
+ const {
+ blinkSelection,
+ value
+ } = me.state;
+
+ const {
+ editable = true,
+ fontFace,
+ preview,
+ } = me.props;
+
+ const textRender = fontFace.textRender || DEFAULT_TEXT_RENDER;
+
+ const textStyles = Object.assign({
+ ...(blinkSelection ? {
+ animation: 'textBlink 1s infinite',
+ } : {}),
+ minHeight: '20px',
+ WebkitFontSmoothing: 'antialiased',
+ MozOsxFontSmoothing: 'grayscale',
+ }, fontFace.text, preview ? fontFace.previewOverrides : {});
+
+ const fontInput = (editable === true) ? (
+ me.onKeyPress(e)} onChange={(e) => me.onChange(e)} style={{
+ ...{
+ bottom: 0,
+ opacity: 0,
+ padding: 0,
+ left: 0,
+ position: 'absolute',
+ top: 0,
+ width: '100%',
+ zIndex: 1,
+ },
+ ...(fontFace.editorStyle || {}),
+ }} />
+ ) : null;
+
+ return (
+
+
+ {fontInput}
+
{textRender(value)}
+
+ );
+ }
+
+ onKeyPress(e) {
+ this.setState({
+ blinkSelection: false,
+ value: e.target.value
+ });
+ }
+
+ onChange(e) {
+ this.setState({
+ blinkSelection: false,
+ value: e.target.value
+ });
+ }
+};
+
+export const PRESETS = {
+ 'Green Machine': require('../FontFaces/GreenMachine'),
+ 'Inferno': require('../FontFaces/Inferno'),
+ 'Lazer': require('../FontFaces/Lazer'),
+ 'Neon': require('../FontFaces/Neon'),
+ 'Old Blue': require('../FontFaces/OldBlue'),
+ 'Retro Rainbow': require('../FontFaces/RetroRainbow'),
+ 'The Special': require('../FontFaces/TheSpecial'),
+ 'Vapor Wave': require('../FontFaces/VaporWave'),
+}
diff --git a/client/src/components/Creatify/FontFaces/.gitkeep b/client/src/components/Creatify/FontFaces/.gitkeep
new file mode 100644
index 00000000..e69de29b
diff --git a/client/src/components/Creatify/FontFaces/GreenMachine.js b/client/src/components/Creatify/FontFaces/GreenMachine.js
new file mode 100644
index 00000000..60b0b462
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/GreenMachine.js
@@ -0,0 +1,15 @@
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'courier, Courier New',
+ fontWeight: 'bold',
+ fontSize: '2em',
+ },
+ text: {
+ color: '#00b700',
+ fontFamily: 'courier, Courier New',
+ fontSize: '2rem',
+ fontWeight: 'bold',
+ textShadow: '1px 1px 2px #003605',
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/Inferno.js b/client/src/components/Creatify/FontFaces/Inferno.js
new file mode 100644
index 00000000..45fc4b20
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/Inferno.js
@@ -0,0 +1,19 @@
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'helvetica, Helvetica Nue',
+ fontWeight: 'bold',
+ fontSize: '2em',
+ },
+ text: {
+ fontFamily: 'helvetica, Helvetica Nue',
+ fontWeight: 'bold',
+ fontSize: '2em',
+ color: '#fff',
+ textShadow: '0px 0px 3px #c20000, 0px 0px 3px #e0f2ff, 0px 0px 5px #532761, 0px 0px 20px #670606, 0 0 10px rgba(0, 0, 0, .8), 0 0 10px #fefcc9, 5px -5px 15px #feec85, -10px -10px 20px #ffae34, 10px -20px 25px #ec760c, -10px -30px 30px #cd4606, 0 -40px 35px #973716, 5px -45px 40px #451b0e, 0 -2px 15px rgba(255, 200, 160, .5)',
+ },
+ previewOverrides: {
+ fontSize: '1.5em',
+ padding: '0 1rem 0 1rem',
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/Lazer.js b/client/src/components/Creatify/FontFaces/Lazer.js
new file mode 100644
index 00000000..96373b4c
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/Lazer.js
@@ -0,0 +1,23 @@
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'helvetica, Helvetica Nue',
+ fontWeight: 'bold',
+ fontSize: '2em',
+ textTransform: 'uppercase',
+ whiteSpace: 'nowrap',
+ },
+ text: {
+ fontFamily: 'helvetica, Helvetica Nue',
+ fontWeight: 'bold',
+ backgroundImage: 'linear-gradient(180deg, #249bff 0%, #e1f8ff 44%, #3a006b 44%, #ff57d6 100%)',
+ backgroundClip: 'text',
+ fontSize: '2em',
+ color: 'transparent',
+ filter: 'drop-shadow(0 0 .05rem black)',
+ textTransform: 'uppercase',
+ whiteSpace: 'nowrap',
+ WebkitBackgroundClip: 'text',
+ WebkitTextStroke: '0.03em rgba(255, 255, 255, 0.6)',
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/Neon.js b/client/src/components/Creatify/FontFaces/Neon.js
new file mode 100644
index 00000000..16d0eb20
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/Neon.js
@@ -0,0 +1,21 @@
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'Helvetica, Arial',
+ fontWeight: 'bold',
+ fontSize: '2em',
+ letterSpacing: '.1em',
+ },
+ text: {
+ color: '#fff',
+ fontFamily: 'Helvetica, Arial',
+ fontSize: '2rem',
+ fontWeight: 'bold',
+ letterSpacing: '.1em',
+ textShadow: '0 0 0.05em #fff, 0 0 0.1em #fff, 0 0 0.2em #fff, 0 0 .2em #ff1de2, 0 0 .4em #ff26e3, 0 0 .5em #ff00de, 0 0 1em #ff61eb, 0 0 1.5em #ff7cee',
+ },
+ previewOverrides: {
+ fontSize: '1.8em',
+ padding: '0 1rem 0 1rem',
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/OldBlue.js b/client/src/components/Creatify/FontFaces/OldBlue.js
new file mode 100644
index 00000000..99448d87
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/OldBlue.js
@@ -0,0 +1,39 @@
+import React from 'react';
+
+const charToFullWidth = char => {
+ const c = char.charCodeAt( 0 )
+ return c >= 33 && c <= 126
+ ? String.fromCharCode( ( c - 33 ) + 65281 )
+ : char
+}
+
+const stringToFullWidth =
+
+module.exports = {
+ container: {},
+ editorStyle: {},
+ text: {
+ fontFamily: 'Segoe UI,Helvetica,Arial',
+ },
+ previewOverrides: {
+ height: '2.6rem',
+ },
+ textRender: (text) => {
+ const id = `curve-${text.replace(/[^A-Za-z0-9]/g, '')}-oceanwave`
+ return (
+
+ );
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/RetroRainbow.js b/client/src/components/Creatify/FontFaces/RetroRainbow.js
new file mode 100644
index 00000000..833c9671
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/RetroRainbow.js
@@ -0,0 +1,21 @@
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'Arial, sans-serif',
+ fontWeight: 'bold',
+ fontSize: '1.2em',
+ transform: 'scale(1, 1.5)',
+ },
+ text: {
+ fontFamily: 'Arial, sans-serif',
+ fontWeight: 'bold',
+ backgroundImage: 'linear-gradient(to right, #b306a9, #ef2667, #f42e2c, #ffa509, #fdfc00, #55ac2f, #0b13fd, #a804af)',
+ backgroundClip: 'text',
+ fontSize: '1.2em',
+ transform: 'scale(1, 1.5)',
+ color: 'transparent',
+ paddingBottom: '.25em',
+ paddingTop: '.1em',
+ WebkitBackgroundClip: 'text',
+ },
+};
diff --git a/client/src/components/Creatify/FontFaces/TheSpecial.js b/client/src/components/Creatify/FontFaces/TheSpecial.js
new file mode 100644
index 00000000..d7e5087a
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/TheSpecial.js
@@ -0,0 +1,41 @@
+import React from 'react';
+
+module.exports = {
+ container: {},
+ editorStyle: {
+ fontFamily: 'Arial, sans-serif',
+ fontWeight: 'bold',
+ fontSize: '1.4em',
+ },
+ text: {
+ fontFamily: 'Arial, sans-serif',
+ fontWeight: 'bold',
+ backgroundImage: 'linear-gradient(to right, #b306a9, #ef2667, #f42e2c, #ffa509, #fdfc00, #55ac2f, #0b13fd, #a804af)',
+ backgroundClip: 'text',
+ fontSize: '1.4em',
+ color: 'transparent',
+ paddingBottom: '.25em',
+ paddingTop: '.1em',
+ WebkitBackgroundClip: 'text',
+ },
+ textRender: (text) => {
+ text = text
+ .replace(/love [^\s.!$]+/g, 'love LBRY')
+ .replace(/LBRY/g, 'amazing LBRY')
+ .replace(/julie/gi, 'super Julie')
+ .replace(/tom/gi, 'amazing Tom')
+ .replace(/(btc|bch)/gi, 'LBC')
+ .replace(/\w+ is \w+/gi, 'LBRY is amazing');
+
+ return text.split(/chris[\d\w]*/gi).reduce((result, value, index) => {
+ if(index !== 0) {
+ result.push(
);
+ }
+ result.push({value})
+
+ return result;
+ }, [])
+ },
+};
+
+const THE_FACE = '';
diff --git a/client/src/components/Creatify/FontFaces/VaporWave.js b/client/src/components/Creatify/FontFaces/VaporWave.js
new file mode 100644
index 00000000..7d2f0575
--- /dev/null
+++ b/client/src/components/Creatify/FontFaces/VaporWave.js
@@ -0,0 +1,43 @@
+import React from 'react';
+
+const charToFullWidth = char => {
+ const c = char.charCodeAt( 0 )
+ return c >= 33 && c <= 126
+ ? String.fromCharCode( ( c - 33 ) + 65281 )
+ : char
+}
+
+const stringToFullWidth =
+
+module.exports = {
+ container: {},
+ editorStyle: {},
+ text: {
+ fontFamily: 'Segoe UI,Helvetica,Arial',
+ },
+ previewOverrides: {
+ transform: 'rotate(39deg)',
+ height: '7rem',
+ paddingLeft: '2rem',
+ margin: '-2rem 0',
+ },
+ textRender: (text) => {
+ const formattedText = text.toLowerCase().split('').map((char) => {
+ const c = char.charCodeAt( 0 )
+ return (c >= 33 && c <= 126) ? String.fromCharCode(c + 65248) : char
+ }).join('');
+
+ // TODO: Inline the path
+ const id = `curve-${text.replace(/[^A-Za-z0-9]/g, '')}-oceanwave`
+ return (
+
+ );
+ },
+};
diff --git a/client/src/components/Creatify/RichDraggable/index.js b/client/src/components/Creatify/RichDraggable/index.js
new file mode 100644
index 00000000..c0e9e462
--- /dev/null
+++ b/client/src/components/Creatify/RichDraggable/index.js
@@ -0,0 +1,60 @@
+import React, { Component } from 'react';
+import Draggable from 'react-draggable';
+
+let body;
+try {
+ body = document.body;
+} catch(e) {}
+
+export default class RichDraggable extends Component {
+ constructor(props) {
+ super(props);
+
+ this.contents = React.createRef();
+ this.state = {
+ height: 0,
+ width: 0,
+ };
+ }
+
+ componentDidMount() {
+ const height = this.contents.current.offsetHeight;
+ const width = this.contents.current.offsetWidth;
+
+ this.setState({
+ height,
+ width,
+ });
+ }
+
+ render() {
+ const me = this;
+
+ const {
+ props,
+ state,
+ } = me;
+
+ const {
+ height: bottom,
+ width: right,
+ } = props.bounds;
+
+ const bounds = {
+ //top: 0,
+ //left: 0,
+ right: right - state.width,
+ bottom: bottom - state.height,
+ };
+
+ return (
+
+
+
+ );
+ }
+};
diff --git a/client/src/components/Creatify/index.js b/client/src/components/Creatify/index.js
new file mode 100644
index 00000000..91c77c9e
--- /dev/null
+++ b/client/src/components/Creatify/index.js
@@ -0,0 +1,236 @@
+import { library } from '@fortawesome/fontawesome-svg-core'
+import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'
+
+import React, { Component } from 'react';
+import Select from 'react-select'
+
+import RichDraggable from './RichDraggable';
+import EditableFontface, { PRESETS as FontPresets } from './EditableFontface';
+
+import {
+ faFont,
+ faMinusCircle,
+ faPlusCircle,
+} from '@fortawesome/free-solid-svg-icons';
+
+const getRasterizedCanvas = (contents, width, height) => {
+ return new Promise((resolve) => {
+ // Parse to xHTML for SVG/foreignObject rendering
+ contents = new XMLSerializer().serializeToString(
+ new DOMParser().parseFromString(contents, 'text/html')
+ );
+
+ // Resolves a bug in Chrome where it renders correctly, but
+ // replaces the inline styles with an invalid `background-clip`.
+ if(/Chrome/.test(navigator.userAgent)) {
+ contents = contents.replace(/background\-clip:(\s*text\s*)[;$]/g,
+ (match, group) => (`-webkit-background-clip:text;${match}`)
+ );
+ }
+
+ // Attempt to match font kerning with the DOM.
+ const kerningAndPadding = '';
+ const svgContents = ``;
+
+ const pixelRatio = 2;
+
+ let img = document.createElement('img');
+ let canvas = document.createElement('canvas');
+
+ img.height = canvas.height = height * pixelRatio;
+ img.width = canvas.width = width * pixelRatio;
+ canvas.style.height = `${height}px`;
+ canvas.style.width = `${width}px`;
+
+ let shadowNode = document.createElement('div');
+ Object.assign(shadowNode.style, {
+ height: 0,
+ overflow: 'hidden',
+ width: 0,
+ });
+ document.body.appendChild(shadowNode);
+
+ shadowNode.appendChild(img);
+
+ var svg64 = btoa(unescape(encodeURIComponent(svgContents)));
+ var b64Start = 'data:image/svg+xml;base64,';
+ var image64 = b64Start + svg64;
+ img.addEventListener('load', () => {
+ window.requestAnimationFrame(() => {
+ // We still can't trust Firefox's %$%&* engine, add another 5ms timeout
+ // `background-clip: text` is very broken and does not always render in time.
+ setTimeout(() => {
+ let context = canvas.getContext('2d', { alpha: false });
+ context.clearRect(0, 0, canvas.width, canvas.height);
+ context.fillStyle = 'white';
+ context.imageSmoothingEnabled = false;
+ context.scale(pixelRatio, pixelRatio);
+ context.fillRect(0, 0, canvas.width, canvas.height);
+ context.drawImage(img, 0, 0);
+
+ document.body.removeChild(shadowNode);
+
+ resolve(canvas);
+ }, 10);
+ });
+ });
+ img.src = image64;
+ });
+};
+
+export default class Creatify extends Component {
+ constructor(props) {
+ super(props);
+
+ const fontKeys = Object.keys(FontPresets);
+
+ this.contents = React.createRef();
+
+ const fontOptions = fontKeys.map(
+ (fontName) => (
+ {
+ value: fontName,
+ label: (
+
+
+
+ ),
+ fontName,
+ }
+ )
+ );
+
+ this.state = {
+ activeElement: false,
+ bounds: {},
+ fontName: fontKeys[0],
+ elements: [],
+ fontOptions,
+ };
+ }
+
+ componentDidMount() {
+ // TODO: Fix bounds
+ /*
+ const bounds = this.contents.current.getBoundingClientRect();
+
+ this.setState({
+ bounds,
+ });
+
+ console.log({
+ bounds
+ })
+ */
+ }
+
+ setActiveElement(activeElement) {
+ this.setState({ activeElement });
+ }
+
+ addElement() {
+ const {
+ state
+ } = this;
+
+ const newElementKey = `element-${state.elements.length}-${Date.now()}`;
+
+ const newElement = (
+ this.setActiveElement(newElement)}>
+
+
+ );
+
+ this.setState({
+ elements: [...state.elements, newElement],
+ activeElement: newElement,
+ });
+ }
+
+ removeElement() {
+ const {
+ state
+ } = this;
+
+ const activeElementIndex = state.elements.indexOf(state.activeElement);
+
+ if(state.elements.length === 0 || activeElementIndex === -1) {
+ return;
+ }
+
+ const elements = [...state.elements];
+ elements.splice(activeElementIndex, 1)
+
+ this.setState({
+ activeElement: false,
+ elements,
+ });
+ }
+
+ async onSave() {
+ const renderedCanvas = await this.renderContentsToCanvas();
+
+ if(this.props.onSave) {
+ this.props.onSave(renderedCanvas);
+ }
+ }
+
+ async renderContentsToCanvas() {
+ const me = this;
+
+ const contentsElement = me.contents.current;
+ let contents = contentsElement.outerHTML;
+
+ // Cheap border/handles removal
+ contents = `` + contents;
+
+ const contentsWidth = contentsElement.offsetWidth;
+ const contentsHeight = contentsElement.offsetHeight;
+
+ // Fix the dimensions, fixes when flex is used.
+ contents = `${contents}
`;
+
+ return await getRasterizedCanvas(contents, contentsWidth, contentsHeight);
+ }
+
+ render() {
+ const me = this;
+ const {
+ props,
+ state,
+ } = this;
+
+ // TODO: Abstract into separate package & use CSS Modules.
+ const spacerCss = { width: '.3em' };
+ return (
+
+
+
this.addElement()} />
+
+ this.removeElement()} />
+
+
+
+
+ this.onSave()} style={{ alignItems: 'center', alignSelf: 'stretch', border: '1px solid #fff', borderRadius: '4px', color: '#fff', display: 'flex', padding: '0 0.6em' }}>
+ Save
+
+
+
+ {state.elements}
+ {props.children}
+
+
+ );
+ }
+
+ setFont(fontName) {
+ this.setState({
+ fontName,
+ });
+ }
+};
diff --git a/client/src/containers/Dropzone/view.jsx b/client/src/containers/Dropzone/view.jsx
index b524b014..f7f05a95 100644
--- a/client/src/containers/Dropzone/view.jsx
+++ b/client/src/containers/Dropzone/view.jsx
@@ -1,17 +1,34 @@
import React from 'react';
+
import { validateFile } from '../../utils/file';
+import Creatify from '@components/Creatify';
import DropzonePreviewImage from '@components/DropzonePreviewImage';
import DropzoneDropItDisplay from '@components/DropzoneDropItDisplay';
import DropzoneInstructionsDisplay from '@components/DropzoneInstructionsDisplay';
+import { library } from '@fortawesome/fontawesome-svg-core'
+import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'
+import { faEdit } from '@fortawesome/free-solid-svg-icons';
+
class Dropzone extends React.Component {
constructor (props) {
super(props);
+
this.state = {
- dragOver : false,
- mouseOver : false,
- dimPreview: false,
+ dragOver : false,
+ mouseOver : false,
+ dimPreview : false,
+ filePreview: null,
+ memeify : false,
};
+
+ if(props.file) {
+ // No side effects allowed with `getDerivedStateFromProps`, so
+ // we must use `componentDidUpdate` and `constructor` routines.
+ // Note: `FileReader` has an `onloadend` side-effect
+ this.updateFilePreview();
+ }
+
this.handleDrop = this.handleDrop.bind(this);
this.handleDragOver = this.handleDragOver.bind(this);
this.handleDragEnd = this.handleDragEnd.bind(this);
@@ -23,6 +40,25 @@ class Dropzone extends React.Component {
this.handleFileInput = this.handleFileInput.bind(this);
this.chooseFile = this.chooseFile.bind(this);
}
+
+ componentDidUpdate(prevProps) {
+ if(prevProps.file !== this.props.file) {
+ this.updateFilePreview();
+ }
+ }
+
+ updateFilePreview() {
+ if (this.props.file) {
+ const fileReader = new FileReader();
+ fileReader.readAsDataURL(this.props.file);
+ fileReader.onloadend = () => {
+ this.setState({
+ filePreview: fileReader.result
+ });
+ };
+ }
+ }
+
handleDrop (event) {
event.preventDefault();
this.setState({dragOver: false});
@@ -35,9 +71,11 @@ class Dropzone extends React.Component {
}
}
}
+
handleDragOver (event) {
event.preventDefault();
}
+
handleDragEnd (event) {
var dt = event.dataTransfer;
if (dt.items) {
@@ -48,27 +86,34 @@ class Dropzone extends React.Component {
event.dataTransfer.clearData();
}
}
+
handleDragEnter () {
this.setState({dragOver: true, dimPreview: true});
}
+
handleDragLeave () {
this.setState({dragOver: false, dimPreview: false});
}
+
handleMouseEnter () {
this.setState({mouseOver: true, dimPreview: true});
}
+
handleMouseLeave () {
this.setState({mouseOver: false, dimPreview: false});
}
+
handleClick (event) {
event.preventDefault();
document.getElementById('file_input').click();
}
+
handleFileInput (event) {
event.preventDefault();
const fileList = event.target.files;
this.chooseFile(fileList[0]);
}
+
chooseFile (file) {
if (file) {
try {
@@ -80,15 +125,72 @@ class Dropzone extends React.Component {
this.props.selectFile(file);
}
}
+
+ selectFileFromCanvas (canvas) {
+ const destinationFormat = 'image/jpeg';
+
+ canvas.toBlob((blob) => {
+ const file = new File([blob], `memeify-${Math.random().toString(36).substring(7)}.jpg`, {
+ type: destinationFormat,
+ });
+
+ this.props.selectFile(file);
+
+ // TODO: Add ability to reset.
+ this.setState({
+ memeify: false,
+ });
+ }, destinationFormat, 0.95);
+ }
+
render () {
- const { dragOver, mouseOver, dimPreview } = this.state;
+ const { dragOver, mouseOver, dimPreview, filePreview, memeify } = this.state;
const { file, thumbnail, fileError, isUpdate, sourceUrl, fileExt } = this.props;
+
+ const hasContent = !!(file || isUpdate);
+
+ const dropZoneHooks = file ? {} : {
+ onDrop: this.handleDrop,
+ onDragOver: this.handleDragOver,
+ onDragEnd: this.handleDragEnd,
+ onDragEnter: this.handleDragEnter,
+ onDragLeave: this.handleDragLeave,
+ onMouseEnter: this.handleMouseEnter,
+ onMouseLeave: this.handleMouseLeave,
+ onClick: this.handleClick,
+ };
+
+ const dropZonePreviewProps = file ? {
+ dimPreview,
+ file,
+ thumbnail,
+ } : {
+ dimPreview: true,
+ isUpdate: true,
+ sourceUrl,
+ };
+
+ const memeifyContent = memeify && file && filePreview ? (
+ this.selectFileFromCanvas(canvas)}>
+
+
![]({filePreview})
+
+
+ ) : null;
+
+ const dropZoneClassName = 'dropzone' + (dragOver ? ' dropzone--drag-over' : '');
+
return (
{isUpdate && fileExt === 'mp4' ? (
Video updates are currently disabled. This feature will be available soon. You can edit metadata.
) : (
-
+
+ { hasContent && !memeify && fileExt !== 'mp4' && (
+
this.setState({ memeify: !memeify })}>
+ Memeify
+
+ )}
-
- {file || isUpdate ? (
-
- {file ? (
-
- ) : (
-
- )}
+
+ {hasContent ? (
+
+
{ dragOver ? : null }
{ mouseOver ? (
@@ -133,15 +214,15 @@ class Dropzone extends React.Component {
message={fileExt === 'mp4' ? 'Drag & drop new thumbnail' : null}
/>
) : null }
+ {memeifyContent}
) : (
dragOver ?
: (
-
+
)
)}
+ {memeifyContent ?
{`Don't forget to save before you publish.`}
: null}
)}
diff --git a/package.json b/package.json
index 4177efb1..8f066fae 100644
--- a/package.json
+++ b/package.json
@@ -46,6 +46,9 @@
},
"homepage": "https://github.com/lbryio/spee.ch#readme",
"dependencies": {
+ "@fortawesome/fontawesome-svg-core": "^1.2.8",
+ "@fortawesome/free-solid-svg-icons": "^5.5.0",
+ "@fortawesome/react-fontawesome": "^0.1.3",
"axios": "^0.18.0",
"bcrypt": "^2.0.1",
"body-parser": "^1.18.3",
@@ -67,12 +70,14 @@
"passport-local": "^1.0.0",
"prop-types": "^15.6.2",
"react": "^16.4.2",
- "react-feather": "^1.1.4",
"react-dom": "^16.6.1",
+ "react-draggable": "^3.0.5",
+ "react-feather": "^1.1.4",
"react-ga": "^2.5.3",
"react-helmet": "^5.2.0",
"react-redux": "^5.1.1",
"react-router-dom": "^4.3.1",
+ "react-select": "^2.1.1",
"redux": "^4.0.1",
"redux-saga": "^0.16.2",
"sequelize": "^4.41.1",