/* eslint jsx-a11y/no-noninteractive-element-interactions: 0 */ import React from 'react'; import ReactDOM from 'react-dom'; import { FormattedMessage, injectIntl } from 'react-intl'; import CopyToClipboard from 'react-copy-to-clipboard'; import classNames from 'classnames'; import LZString from 'lz-string'; import { Tooltip, Alert, Badge } from 'antd'; import { SnippetsOutlined, CheckOutlined, ThunderboltOutlined } from '@ant-design/icons'; import stackblitzSdk from '@stackblitz/sdk'; import CodePreview from './CodePreview'; import EditButton from '../EditButton'; import BrowserFrame from '../../BrowserFrame'; import CodeSandboxIcon from './CodeSandboxIcon'; import CodePenIcon from './CodePenIcon'; import RiddleIcon from './RiddleIcon'; const { ErrorBoundary } = Alert; function compress(string) { return LZString.compressToBase64(string) .replace(/\+/g, '-') // Convert '+' to '-' .replace(/\//g, '_') // Convert '/' to '_' .replace(/=+$/, ''); // Remove ending '=' } class Demo extends React.Component { iframeRef = React.createRef(); codeSandboxIconRef = React.createRef(); riddleIconRef = React.createRef(); codepenIconRef = React.createRef(); state = { codeExpand: false, copied: false, copyTooltipVisible: false, }; componentDidMount() { const { meta, location } = this.props; if (meta.id === location.hash.slice(1)) { this.anchor.click(); } } shouldComponentUpdate(nextProps, nextState) { const { codeExpand, copied, copyTooltipVisible } = this.state; const { expand, theme, showRiddleButton } = this.props; return ( (codeExpand || expand) !== (nextState.codeExpand || nextProps.expand) || copied !== nextState.copied || copyTooltipVisible !== nextState.copyTooltipVisible || nextProps.theme !== theme || nextProps.showRiddleButton !== showRiddleButton ); } getSourceCode() { const { highlightedCodes } = this.props; if (typeof document !== 'undefined') { const div = document.createElement('div'); div.innerHTML = highlightedCodes.jsx; return div.textContent; } return ''; } handleCodeExpand = demo => { const { codeExpand } = this.state; this.setState({ codeExpand: !codeExpand }); this.track({ type: 'expand', demo, }); }; saveAnchor = anchor => { this.anchor = anchor; }; handleCodeCopied = demo => { this.setState({ copied: true }); this.track({ type: 'copy', demo, }); }; onCopyTooltipVisibleChange = visible => { if (visible) { this.setState({ copyTooltipVisible: visible, copied: false, }); return; } this.setState({ copyTooltipVisible: visible, }); }; // eslint-disable-next-line class-methods-use-this track({ type, demo }) { if (!window.gtag) { return; } window.gtag('event', 'demo', { event_category: type, event_label: demo, }); } handleIframeReady = () => { const { theme, setIframeTheme } = this.props; if (this.iframeRef.current) { setIframeTheme(this.iframeRef.current, theme); } }; render() { const { state } = this; const { props } = this; const { meta, src, content, preview, highlightedCodes, style, highlightedStyle, expand, utils, intl: { locale }, theme, showRiddleButton, } = props; const { copied, copyTooltipVisible } = state; if (!this.liveDemo) { this.liveDemo = meta.iframe ? (