@bbc/react-transcript-editor
Version:
A React component to make transcribing audio and video easier and faster.
134 lines (107 loc) • 4.17 kB
JavaScript
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
import React from 'react';
import { // Draft,
Editor, EditorState, // ContentState,
CompositeDecorator, convertFromRaw } from 'draft-js';
import Word from './Word';
import sttJsonAdapter from './adapters/index.js';
import styles from './index.module.css';
class TimedTextEditor extends React.Component {
constructor(props) {
super(props);
_defineProperty(this, "handleDoubleClick", event => {
// nativeEvent --> React giving you the DOM event
let element = event.nativeEvent.target; // find the parent in Word that contains span with time-code start attribute
while (!element.hasAttribute('data-start') && element.parentElement) {
element = element.parentElement;
}
if (element.hasAttribute('data-start')) {
const t = parseFloat(element.getAttribute('data-start')); //TODO: prop to jump to video <-- To connect with MediaPlayer
// this.props.seek(t);
this.props.onWordClick(t); // TODO: pass current time of media to TimedTextEditor to know what text to highlight in this component
}
});
this.state = {
editorState: EditorState.createEmpty(),
transcriptData: this.props.transcriptData,
isEditable: this.props.isEditable,
sttJsonType: this.props.sttJsonType
};
this.onChange = editorState => {
// DraftJs option editable
if (this.state.isEditable) {
this.setState({
editorState
});
}
};
}
componentDidMount() {
this.loadData();
}
static getDerivedStateFromProps(nextProps, prevState) {
if (nextProps.transcriptData !== null) {
return {
transcriptData: nextProps.transcriptData,
isEditable: nextProps.isEditable
};
}
return null;
}
componentDidUpdate(prevProps, prevState) {
if (prevState.transcriptData !== this.state.transcriptData) {
this.loadData();
}
}
loadData() {
if (this.props.transcriptData !== null) {
const blocks = sttJsonAdapter(this.props.transcriptData, this.props.sttJsonType);
const entityRanges = blocks.map(block => block.entityRanges); // eslint-disable-next-line no-use-before-define
const flatEntityRanges = flatten(entityRanges);
const entityMap = {};
flatEntityRanges.forEach(data => {
entityMap[data.key] = {
type: 'WORD',
mutability: 'MUTABLE',
data
};
});
const contentState = convertFromRaw({
blocks,
entityMap
}); // eslint-disable-next-line no-use-before-define
const editorState = EditorState.createWithContent(contentState, decorator);
this.setState({
editorState
});
}
} // click on words - for navigation
// eslint-disable-next-line class-methods-use-this
render() {
return React.createElement("section", {
className: styles.editor,
onDoubleClick: event => this.handleDoubleClick(event) // onClick={ event => this.handleOnClick(event) }
}, React.createElement(Editor, {
editorState: this.state.editorState,
onChange: this.onChange,
stripPastedStyles: true
}));
}
} // converts nested arrays into one dimensional array
const flatten = list => list.reduce((a, b) => a.concat(Array.isArray(b) ? flatten(b) : b), []); // DraftJs decorator to recognize which entity is which
// and know what to apply to what component
const getEntityStrategy = mutability => (contentBlock, callback, contentState) => {
contentBlock.findEntityRanges(character => {
const entityKey = character.getEntity();
if (entityKey === null) {
return false;
}
return contentState.getEntity(entityKey).getMutability() === mutability;
}, callback);
}; // decorator definition - Draftjs
// defines what to use to render the entity
const decorator = new CompositeDecorator([{
strategy: getEntityStrategy('MUTABLE'),
component: Word
}]);
export default TimedTextEditor;