import React, { useCallback, useEffect, useState } from 'react'; import { Upload, Tooltip, Popover, Modal, Progress, message, Spin, Result } from 'antd'; import CopyToClipboard from 'react-copy-to-clipboard'; import { injectIntl } from 'react-intl'; import * as AntdIcons from '@ant-design/icons'; const allIcons: { [key: string]: any } = AntdIcons; const { Dragger } = Upload; interface AntdIconClassifier { load: Function; predict: Function; } declare global { interface Window { antdIconClassifier: AntdIconClassifier; } } interface PicSearcherProps { intl: any; } interface PicSearcherState { loading: boolean; modalOpen: boolean; popoverVisible: boolean; icons: iconObject[]; fileList: any[]; error: boolean; modelLoaded: boolean; } interface iconObject { type: string; score: number; } const PicSearcher: React.FC = ({ intl }) => { const { messages } = intl; const [state, setState] = useState({ loading: false, modalOpen: false, popoverVisible: false, icons: [], fileList: [], error: false, modelLoaded: false, }); const predict = (imgEl: HTMLImageElement) => { try { let icons: any[] = window.antdIconClassifier.predict(imgEl); if (gtag && icons.length) { gtag('event', 'icon', { event_category: 'search-by-image', event_label: icons[0].className, }); } icons = icons.map(i => ({ score: i.score, type: i.className.replace(/\s/g, '-') })); setState(prev => ({ ...prev, loading: false, error: false, icons })); } catch { setState(prev => ({ ...prev, loading: false, error: true })); } }; // eslint-disable-next-line class-methods-use-this const toImage = (url: string) => new Promise(resolve => { const img = new Image(); img.setAttribute('crossOrigin', 'anonymous'); img.src = url; img.onload = () => { resolve(img); }; }); const uploadFile = useCallback((file: File) => { setState(prev => ({ ...prev, loading: true })); const reader = new FileReader(); reader.onload = () => { toImage(reader.result as string).then(predict); setState(prev => ({ ...prev, fileList: [{ uid: 1, name: file.name, status: 'done', url: reader.result }], })); }; reader.readAsDataURL(file); }, []); const onPaste = useCallback((event: ClipboardEvent) => { const items = event.clipboardData && event.clipboardData.items; let file = null; if (items && items.length) { for (let i = 0; i < items.length; i++) { if (items[i].type.includes('image')) { file = items[i].getAsFile(); break; } } } if (file) { uploadFile(file); } }, []); const toggleModal = useCallback(() => { setState(prev => ({ ...prev, modalOpen: !prev.modalOpen, popoverVisible: false, fileList: [], icons: [], })); if (!localStorage.getItem('disableIconTip')) { localStorage.setItem('disableIconTip', 'true'); } }, []); // eslint-disable-next-line class-methods-use-this const onCopied = useCallback((text: string) => { message.success( {text} copied 🎉 , ); }, []); useEffect(() => { const script = document.createElement('script'); script.onload = async () => { await window.antdIconClassifier.load(); setState(prev => ({ ...prev, modelLoaded: true })); document.addEventListener('paste', onPaste); }; script.src = 'https://cdn.jsdelivr.net/gh/lewis617/antd-icon-classifier@0.0/dist/main.js'; document.head.appendChild(script); setState(prev => ({ ...prev, popoverVisible: !localStorage.getItem('disableIconTip') })); return () => { document.removeEventListener('paste', onPaste); }; }, []); return (
{state.modelLoaded || (
)} {state.modelLoaded && ( uploadFile(o.file as File)} fileList={state.fileList} showUploadList={{ showPreviewIcon: false, showRemoveIcon: false }} >

{messages['app.docs.components.icon.pic-searcher.upload-text']}

{messages['app.docs.components.icon.pic-searcher.upload-hint']}

)}
{state.icons.length > 0 && (
{messages['app.docs.components.icon.pic-searcher.result-tip']}
)} {state.icons.length > 0 && ( )} {state.icons.map(icon => { const { type } = icon; const iconName = `${type .split('-') .map(str => `${str[0].toUpperCase()}${str.slice(1)}`) .join('')}Outlined`; return ( ); })}
{messages['app.docs.components.icon.pic-searcher.th-icon']} {messages['app.docs.components.icon.pic-searcher.th-score']}
`} onCopy={onCopied}> {React.createElement(allIcons[iconName])}
{state.error && ( )}
); }; export default injectIntl(PicSearcher);