1
0
Fork 0
mirror of https://github.com/terribleplan/next.js.git synced 2024-01-19 02:48:18 +00:00
next.js/examples/with-draft-js/pages/index.js

350 lines
8.2 KiB
JavaScript
Raw Normal View History

2018-02-09 08:40:32 +00:00
import React from 'react'
import {
Editor,
EditorState,
RichUtils,
convertToRaw,
convertFromRaw
} from 'draft-js'
2018-02-09 08:40:32 +00:00
export default class App extends React.Component {
constructor (props) {
super(props)
this.state = {
editorState: EditorState.createWithContent(convertFromRaw(initialData)),
showToolbar: false,
windowWidth: 0,
toolbarMeasures: {
w: 0,
h: 0
},
selectionMeasures: {
w: 0,
h: 0
},
selectionCoordinates: {
x: 0,
y: 0
},
toolbarCoordinates: {
x: 0,
y: 0
},
showRawData: false
}
this.focus = () => this.editor.focus()
this.onChange = editorState => this.setState({ editorState })
2018-02-09 08:40:32 +00:00
}
onClickEditor = () => {
this.focus()
this.checkSelectedText()
}
// 1- Check if some text is selected
checkSelectedText = () => {
if (typeof window !== 'undefined') {
const text = window.getSelection().toString()
if (text !== '') {
// 1-a Define the selection coordinates
this.setSelectionXY()
} else {
// Hide the toolbar if nothing is selected
this.setState({
showToolbar: false
})
}
}
}
// 2- Identify the selection coordinates
setSelectionXY = () => {
var r = window
.getSelection()
.getRangeAt(0)
.getBoundingClientRect()
2018-02-09 08:40:32 +00:00
var relative = document.body.parentNode.getBoundingClientRect()
// 2-a Set the selection coordinates in the state
this.setState(
{
selectionCoordinates: r,
windowWidth: relative.width,
selectionMeasures: {
w: r.width,
h: r.height
}
},
() => this.showToolbar()
)
2018-02-09 08:40:32 +00:00
}
// 3- Show the toolbar
showToolbar = () => {
this.setState(
{
showToolbar: true
},
() => this.measureToolbar()
)
2018-02-09 08:40:32 +00:00
}
// 4- The toolbar was hidden until now
measureToolbar = () => {
// 4-a Define the toolbar width and height, as it is now visible
this.setState(
{
toolbarMeasures: {
w: this.elemWidth,
h: this.elemHeight
}
},
() => this.setToolbarXY()
)
2018-02-09 08:40:32 +00:00
}
// 5- Now that we have all measures, define toolbar coordinates
setToolbarXY = () => {
let coordinates = {}
const {
selectionMeasures,
selectionCoordinates,
toolbarMeasures,
windowWidth
} = this.state
2018-02-09 08:40:32 +00:00
const hiddenTop = selectionCoordinates.y < toolbarMeasures.h
const hiddenRight =
windowWidth - selectionCoordinates.x < toolbarMeasures.w / 2
2018-02-09 08:40:32 +00:00
const hiddenLeft = selectionCoordinates.x < toolbarMeasures.w / 2
const normalX =
selectionCoordinates.x - toolbarMeasures.w / 2 + selectionMeasures.w / 2
2018-02-09 08:40:32 +00:00
const normalY = selectionCoordinates.y - toolbarMeasures.h
const invertedY = selectionCoordinates.y + selectionMeasures.h
const moveXToLeft = windowWidth - toolbarMeasures.w
const moveXToRight = 0
coordinates = {
x: normalX,
y: normalY
}
if (hiddenTop) {
coordinates.y = invertedY
}
if (hiddenRight) {
coordinates.x = moveXToLeft
}
if (hiddenLeft) {
coordinates.x = moveXToRight
}
this.setState({
toolbarCoordinates: coordinates
})
}
handleKeyCommand = command => {
const { editorState } = this.state
2018-02-09 08:40:32 +00:00
const newState = RichUtils.handleKeyCommand(editorState, command)
if (newState) {
this.onChange(newState)
return true
}
return false
}
toggleToolbar = inlineStyle => {
2018-02-09 08:40:32 +00:00
this.onChange(
RichUtils.toggleInlineStyle(this.state.editorState, inlineStyle)
2018-02-09 08:40:32 +00:00
)
}
render () {
const { editorState } = this.state
2018-02-09 08:40:32 +00:00
// Make sure we're not on the ssr
if (typeof window !== 'undefined') {
// Let's stick the toolbar to the selection
// when the window is resized
window.addEventListener('resize', this.checkSelectedText)
}
const toolbarStyle = {
display: this.state.showToolbar ? 'block' : 'none',
backgroundColor: 'black',
color: 'white',
position: 'absolute',
left: this.state.toolbarCoordinates.x,
top: this.state.toolbarCoordinates.y,
zIndex: 999,
padding: 10
}
return (
<div>
<div
ref={elem => {
2018-02-09 08:40:32 +00:00
this.elemWidth = elem ? elem.clientWidth : 0
this.elemHeight = elem ? elem.clientHeight : 0
}}
style={toolbarStyle}
>
<ToolBar editorState={editorState} onToggle={this.toggleToolbar} />
2018-02-09 08:40:32 +00:00
</div>
<div onClick={this.onClickEditor} onBlur={this.checkSelectedText}>
<Editor
customStyleMap={styleMap}
editorState={editorState}
handleKeyCommand={this.handleKeyCommand}
onChange={this.onChange}
placeholder='Tell a story...'
spellCheck={false}
ref={element => {
this.editor = element
}}
2018-02-09 08:40:32 +00:00
/>
</div>
<div style={{ marginTop: 40 }}>
<button
onClick={() =>
this.setState({ showRawData: !this.state.showRawData })
}
>
2018-02-09 08:40:32 +00:00
{!this.state.showRawData ? 'Show' : 'Hide'} Raw Data
</button>
<br />
{this.state.showRawData &&
JSON.stringify(convertToRaw(editorState.getCurrentContent()))}
2018-02-09 08:40:32 +00:00
</div>
</div>
)
}
}
// Custom overrides for each style
const styleMap = {
CODE: {
backgroundColor: 'rgba(0, 0, 0, 0.05)',
fontFamily: '"Inconsolata", "Menlo", "Consolas", monospace',
fontSize: 16,
padding: 4
},
BOLD: {
color: '#395296',
fontWeight: 'bold'
},
ANYCUSTOMSTYLE: {
color: '#00e400'
}
}
class ToolbarButton extends React.Component {
constructor () {
super()
this.onToggle = e => {
2018-02-09 08:40:32 +00:00
e.preventDefault()
this.props.onToggle(this.props.style)
}
}
render () {
const buttonStyle = {
padding: 10
}
return (
<span onMouseDown={this.onToggle} style={buttonStyle}>
{this.props.label}
2018-02-09 08:40:32 +00:00
</span>
)
}
}
var toolbarItems = [
{ label: 'Bold', style: 'BOLD' },
{ label: 'Italic', style: 'ITALIC' },
{ label: 'Underline', style: 'UNDERLINE' },
{ label: 'Code', style: 'CODE' },
{ label: 'Surprise', style: 'ANYCUSTOMSTYLE' }
2018-02-09 08:40:32 +00:00
]
const ToolBar = props => {
2018-02-09 08:40:32 +00:00
var currentStyle = props.editorState.getCurrentInlineStyle()
return (
<div>
{toolbarItems.map(toolbarItem => (
2018-02-09 08:40:32 +00:00
<ToolbarButton
key={toolbarItem.label}
active={currentStyle.has(toolbarItem.style)}
label={toolbarItem.label}
onToggle={props.onToggle}
style={toolbarItem.style}
/>
))}
2018-02-09 08:40:32 +00:00
</div>
)
}
const initialData = {
blocks: [
{
key: '16d0k',
text: 'You can edit this text.',
type: 'unstyled',
depth: 0,
inlineStyleRanges: [{ offset: 0, length: 23, style: 'BOLD' }],
entityRanges: [],
data: {}
},
{
key: '98peq',
text: '',
type: 'unstyled',
depth: 0,
inlineStyleRanges: [],
entityRanges: [],
data: {}
},
{
key: 'ecmnc',
text:
'Luke Skywalker has vanished. In his absence, the sinister FIRST ORDER has risen from the ashes of the Empire and will not rest until Skywalker, the last Jedi, has been destroyed.',
type: 'unstyled',
depth: 0,
inlineStyleRanges: [
{ offset: 0, length: 14, style: 'BOLD' },
{ offset: 133, length: 9, style: 'BOLD' }
],
entityRanges: [],
data: {}
},
{
key: 'fe2gn',
text: '',
type: 'unstyled',
depth: 0,
inlineStyleRanges: [],
entityRanges: [],
data: {}
},
{
key: '4481k',
text:
'With the support of the REPUBLIC, General Leia Organa leads a brave RESISTANCE. She is desperate to find her brother Luke and gain his help in restoring peace and justice to the galaxy.',
type: 'unstyled',
depth: 0,
inlineStyleRanges: [
{ offset: 34, length: 19, style: 'BOLD' },
{ offset: 117, length: 4, style: 'BOLD' },
{ offset: 68, length: 10, style: 'ANYCUSTOMSTYLE' }
],
entityRanges: [],
data: {}
}
],
entityMap: {}
}