├── .editorconfig ├── .github └── workflows │ └── main.yml ├── .gitignore ├── .prettierignore ├── .prettierrc ├── LICENSE ├── README.md ├── config ├── config.ts ├── postcssPlugins.ts └── router.ts ├── images └── index.png ├── package.json ├── src ├── app.tsx ├── assets │ └── public │ │ ├── fiveStars.png │ │ └── upload.png ├── components │ └── Loading │ │ ├── index.less │ │ └── index.tsx ├── layouts │ └── index.tsx ├── pages │ ├── document.ejs │ └── index │ │ ├── components │ │ ├── ImageModal.tsx │ │ └── index.less │ │ ├── img.ts │ │ ├── index.less │ │ └── index.tsx ├── styles │ ├── global.less │ ├── index.less │ └── reset.less └── utils │ ├── api.ts │ ├── dev.ts │ ├── file.ts │ └── props.ts ├── tsconfig.json └── typings.d.ts /.editorconfig: -------------------------------------------------------------------------------- 1 | # http://editorconfig.org 2 | root = true 3 | 4 | [*] 5 | indent_style = space 6 | indent_size = 2 7 | end_of_line = lf 8 | charset = utf-8 9 | trim_trailing_whitespace = true 10 | insert_final_newline = true 11 | 12 | [*.md] 13 | trim_trailing_whitespace = false 14 | 15 | [Makefile] 16 | indent_style = tab 17 | -------------------------------------------------------------------------------- /.github/workflows/main.yml: -------------------------------------------------------------------------------- 1 | name: github pages 2 | 3 | on: 4 | push: 5 | branches: 6 | - main 7 | 8 | jobs: 9 | deploy: 10 | runs-on: ubuntu-20.04 11 | steps: 12 | - uses: actions/checkout@v2 13 | with: 14 | submodules: recursive 15 | fetch-depth: 0 16 | 17 | - name: Build 18 | uses: actions/setup-node@master 19 | - run: npm install 20 | - run: npm run build 21 | 22 | - name: Deploy 23 | uses: peaceiris/actions-gh-pages@v3 24 | with: 25 | github_token: ${{ secrets.ACCESS_TOKEN }} 26 | publish_dir: './dist' 27 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /npm-debug.log* 6 | /yarn-error.log 7 | /yarn.lock 8 | /package-lock.json 9 | 10 | # production 11 | /dist 12 | 13 | # misc 14 | .DS_Store 15 | 16 | # umi 17 | /src/.umi 18 | /src/.umi-production 19 | /src/.umi-test 20 | /.env.local 21 | -------------------------------------------------------------------------------- /.prettierignore: -------------------------------------------------------------------------------- 1 | **/*.md 2 | **/*.svg 3 | **/*.ejs 4 | **/*.html 5 | package.json 6 | .umi 7 | .umi-production 8 | .umi-test 9 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "singleQuote": true, 3 | "trailingComma": "all", 4 | "printWidth": 120, 5 | "overrides": [ 6 | { 7 | "files": ".prettierrc", 8 | "options": { "parser": "json" } 9 | } 10 | ] 11 | } 12 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 [Sharida] 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## 国庆专属 生成渐变红旗头像 2 | 3 | ### 特性 4 | 5 | - 纯前端实现 6 | - `React` + `TypeScript` 7 | - `canvas` 绘图 8 | - `cropperjs` 裁切头像 9 | 10 | ### 在线体验 11 | 12 | 在线体验:[渐变红旗头像生成](https://findmio.github.io/Five-star-red-flag-Avatar) 13 | 14 | 15 | ### 界面 16 | 17 | image 18 | 19 | -------------------------------------------------------------------------------- /config/config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'umi'; 2 | import routes from './router'; // 路由 3 | import extraPostCSSPlugins from './postcssPlugins'; // 额外的 postcss 配置 4 | 5 | const path = '/Five-star-red-flag-Avatar/'; 6 | 7 | export default defineConfig({ 8 | routes, 9 | extraPostCSSPlugins, 10 | alias: { 11 | config: '/config', 12 | }, 13 | 14 | publicPath: process.env.NODE_ENV === 'production' ? path : '/', 15 | base: path, 16 | 17 | fastRefresh: {}, 18 | extraBabelPlugins: [['import', { libraryName: 'antd-mobile', libraryDirectory: 'es/components', style: false }]], 19 | dynamicImport: { 20 | loading: '@/components/Loading', 21 | }, 22 | 23 | targets: { 24 | ios: 10, 25 | chrome: 79, 26 | ie: false, 27 | firefox: false, 28 | safari: false, 29 | edge: false, 30 | }, 31 | 32 | // 优化项 33 | mfsu: {}, 34 | mock: false, 35 | antd: false, 36 | hash: true, 37 | ignoreMomentLocale: true, 38 | nodeModulesTransform: { 39 | type: 'none', 40 | exclude: [], 41 | }, 42 | }); 43 | -------------------------------------------------------------------------------- /config/postcssPlugins.ts: -------------------------------------------------------------------------------- 1 | export default [ 2 | require('postcss-px-to-viewport')({ 3 | viewportWidth: 375, // 视窗的宽度,对应的是我们设计稿的宽度,一般是375 4 | unitPrecision: 3, // 指定`px`转换为视窗单位值的小数位数(很多时候无法整除) 5 | viewportUnit: 'vw', // 指定需要转换成的视窗单位,建议使用vw 6 | selectorBlackList: [], // 指定不转换为视窗单位的类,可以自定义,可以无限添加,建议定义一至两个通用的类名 7 | minPixelValue: 1, // 小于或等于`1px`不转换为视窗单位,你也可以设置为你想要的值 8 | mediaQuery: false, // 允许在媒体查询中转换`px` 9 | }), 10 | // require('autoprefixer'), 11 | ] -------------------------------------------------------------------------------- /config/router.ts: -------------------------------------------------------------------------------- 1 | export default [ 2 | { 3 | path: '/', 4 | component: '@/layouts', 5 | routes: [ 6 | { path: '/', component: '@/pages/index', title: '渐变红旗头像生成' }, 7 | ], 8 | }, 9 | ]; 10 | -------------------------------------------------------------------------------- /images/index.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/findmio/Five-star-red-flag-Avatar/8bb49ea481c598d1e11343458f67907445ada4d1/images/index.png -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "private": true, 3 | "scripts": { 4 | "start": "umi dev", 5 | "build": "umi build", 6 | "deploy": "npm run build && gh-pages -d dist", 7 | "postinstall": "umi generate tmp", 8 | "prettier": "prettier --write '**/*.{js,jsx,tsx,ts,less,md,json}'", 9 | "test": "umi-test", 10 | "test:coverage": "umi-test --coverage" 11 | }, 12 | "gitHooks": { 13 | "pre-commit": "lint-staged" 14 | }, 15 | "lint-staged": { 16 | "*.{js,jsx,less,md,json}": [ 17 | "prettier --write" 18 | ], 19 | "*.ts?(x)": [ 20 | "prettier --parser=typescript --write" 21 | ] 22 | }, 23 | "dependencies": { 24 | "@ant-design/pro-layout": "^6.5.0", 25 | "ahooks": "^3.5.0", 26 | "antd-mobile": "^5.15.1", 27 | "axios": "^0.21.4", 28 | "cropperjs": "^1.5.12", 29 | "dayjs": "^1.10.7", 30 | "lodash": "^4.17.21", 31 | "qs": "^6.10.1", 32 | "react": "17.x", 33 | "react-dom": "17.x", 34 | "react-query": "^3.24.4", 35 | "umi": "^3.5.19" 36 | }, 37 | "devDependencies": { 38 | "@types/lodash": "^4.14.175", 39 | "@types/qs": "^6.9.7", 40 | "@types/react": "^17.0.0", 41 | "@types/react-dom": "^17.0.0", 42 | "@umijs/preset-react": "1.x", 43 | "@umijs/test": "^3.5.19", 44 | "autoprefixer": "^10.3.5", 45 | "babel-plugin-import": "^1.13.3", 46 | "gh-pages": "^3.2.3", 47 | "lint-staged": "^10.0.7", 48 | "postcss-px-to-viewport": "^1.1.1", 49 | "prettier": "^2.2.0", 50 | "typescript": "^4.1.2", 51 | "yorkie": "^2.0.0" 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /src/app.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { QueryClient, QueryClientProvider } from 'react-query'; 3 | import 'antd-mobile/es/global'; 4 | import '@/styles/index.less'; 5 | 6 | export function onRouteChange(props: { matchedRoutes: any }) { 7 | const { matchedRoutes } = props; 8 | if (matchedRoutes.length) { 9 | } 10 | } 11 | 12 | const queryClient = new QueryClient({ 13 | defaultOptions: { 14 | queries: { 15 | refetchOnWindowFocus: false, // 窗口获得焦点时重新获取数据 16 | }, 17 | }, 18 | }); 19 | 20 | /** 21 | * react-dom 渲染时的根组件 22 | */ 23 | export function rootContainer(container: any) { 24 | const App = () => {container}; 25 | return React.createElement(App, null, container); 26 | } 27 | -------------------------------------------------------------------------------- /src/assets/public/fiveStars.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/findmio/Five-star-red-flag-Avatar/8bb49ea481c598d1e11343458f67907445ada4d1/src/assets/public/fiveStars.png -------------------------------------------------------------------------------- /src/assets/public/upload.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/findmio/Five-star-red-flag-Avatar/8bb49ea481c598d1e11343458f67907445ada4d1/src/assets/public/upload.png -------------------------------------------------------------------------------- /src/components/Loading/index.less: -------------------------------------------------------------------------------- 1 | .page { 2 | position: fixed; 3 | z-index: 999; 4 | background-color: #fff; 5 | width: 100vw; 6 | height: 100vh; 7 | overflow: hidden; 8 | display: flex; 9 | align-items: center; 10 | justify-content: center; 11 | } 12 | 13 | .lds-spinner, 14 | .lds-spinner div, 15 | .lds-spinner div:after { 16 | box-sizing: border-box; 17 | } 18 | 19 | .lds-spinner { 20 | color: black; 21 | display: inline-block; 22 | position: relative; 23 | width: 80px; 24 | height: 80px; 25 | transform: scale(0.6); 26 | } 27 | 28 | .lds-spinner div { 29 | transform-origin: 40px 40px; 30 | animation: lds-spinner 1.2s linear infinite; 31 | } 32 | 33 | .lds-spinner div:after { 34 | content: ' '; 35 | display: block; 36 | position: absolute; 37 | top: 3.2px; 38 | left: 36.8px; 39 | width: 4.4px; 40 | height: 12.6px; 41 | border-radius: 20%; 42 | background: currentColor; 43 | } 44 | 45 | .lds-spinner div:nth-child(1) { 46 | transform: rotate(0deg); 47 | animation-delay: -1.1s; 48 | } 49 | 50 | .lds-spinner div:nth-child(2) { 51 | transform: rotate(30deg); 52 | animation-delay: -1s; 53 | } 54 | 55 | .lds-spinner div:nth-child(3) { 56 | transform: rotate(60deg); 57 | animation-delay: -0.9s; 58 | } 59 | 60 | .lds-spinner div:nth-child(4) { 61 | transform: rotate(90deg); 62 | animation-delay: -0.8s; 63 | } 64 | 65 | .lds-spinner div:nth-child(5) { 66 | transform: rotate(120deg); 67 | animation-delay: -0.7s; 68 | } 69 | 70 | .lds-spinner div:nth-child(6) { 71 | transform: rotate(150deg); 72 | animation-delay: -0.6s; 73 | } 74 | 75 | .lds-spinner div:nth-child(7) { 76 | transform: rotate(180deg); 77 | animation-delay: -0.5s; 78 | } 79 | 80 | .lds-spinner div:nth-child(8) { 81 | transform: rotate(210deg); 82 | animation-delay: -0.4s; 83 | } 84 | 85 | .lds-spinner div:nth-child(9) { 86 | transform: rotate(240deg); 87 | animation-delay: -0.3s; 88 | } 89 | 90 | .lds-spinner div:nth-child(10) { 91 | transform: rotate(270deg); 92 | animation-delay: -0.2s; 93 | } 94 | 95 | .lds-spinner div:nth-child(11) { 96 | transform: rotate(300deg); 97 | animation-delay: -0.1s; 98 | } 99 | 100 | .lds-spinner div:nth-child(12) { 101 | transform: rotate(330deg); 102 | animation-delay: 0s; 103 | } 104 | 105 | @keyframes lds-spinner { 106 | 0% { 107 | opacity: 1; 108 | } 109 | 110 | 100% { 111 | opacity: 0; 112 | } 113 | } 114 | -------------------------------------------------------------------------------- /src/components/Loading/index.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import styles from './index.less'; 3 | 4 | 5 | const Loading = () => { 6 | return ( 7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 | ); 24 | }; 25 | export default Loading; -------------------------------------------------------------------------------- /src/layouts/index.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | 3 | const Layout: React.FC = (props) => { 4 | const { children } = props; 5 | 6 | return <>{children}; 7 | }; 8 | export default Layout; 9 | -------------------------------------------------------------------------------- /src/pages/document.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 13 |   14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | -------------------------------------------------------------------------------- /src/pages/index/components/ImageModal.tsx: -------------------------------------------------------------------------------- 1 | import Cropper from 'cropperjs'; 2 | import React, { useEffect, useRef } from 'react'; 3 | import styles from './index.less'; 4 | import { Dialog } from 'antd-mobile'; 5 | import { getDataUrl } from '@/utils/file'; 6 | import 'cropperjs/dist/cropper.css'; 7 | 8 | interface Props { 9 | imageModalVisible: boolean; 10 | onClose(visible: boolean): void; 11 | onConfirm(data: string): void; 12 | imgData: string; 13 | } 14 | 15 | interface CProps { 16 | url: string; 17 | store: any; 18 | } 19 | 20 | const Content: React.FC = (props) => { 21 | const { url, store } = props; 22 | const imageRef = useRef(null); 23 | 24 | useEffect(() => { 25 | if (imageRef.current) { 26 | store.current = new Cropper(imageRef.current, { 27 | aspectRatio: 1, 28 | autoCropArea: 1, 29 | viewMode: 1, 30 | movable: false, 31 | zoomable: false, 32 | }); 33 | } 34 | }, [url]); 35 | 36 | return ( 37 |
38 | 图片 39 |
40 | ); 41 | }; 42 | 43 | const ImageModal: React.FC = (props) => { 44 | const { imageModalVisible, onClose, imgData, onConfirm } = props; 45 | const cropperRef = useRef(null); 46 | 47 | const handleConfirm = () => { 48 | cropperRef.current.getCroppedCanvas().toBlob(async (blob: Blob) => { 49 | const data = await getDataUrl(blob); 50 | onConfirm(data); 51 | }); 52 | }; 53 | 54 | return ( 55 | { 59 | switch (values.key) { 60 | case 'cancel': 61 | onClose(false); 62 | break; 63 | case 'confirm': 64 | handleConfirm(); 65 | break; 66 | } 67 | }} 68 | content={} 69 | actions={[ 70 | [ 71 | { 72 | key: 'cancel', 73 | text: '取消', 74 | }, 75 | { 76 | key: 'confirm', 77 | text: '确认', 78 | bold: true, 79 | }, 80 | ], 81 | ]} 82 | /> 83 | ); 84 | }; 85 | export default ImageModal; 86 | -------------------------------------------------------------------------------- /src/pages/index/components/index.less: -------------------------------------------------------------------------------- 1 | .image { 2 | width: 250px; 3 | min-height: 250px; 4 | img { 5 | width: 250px; 6 | min-height: 250px; 7 | } 8 | } 9 | -------------------------------------------------------------------------------- /src/pages/index/img.ts: -------------------------------------------------------------------------------- 1 | export default '' -------------------------------------------------------------------------------- /src/pages/index/index.less: -------------------------------------------------------------------------------- 1 | .page { 2 | min-height: 100vh; 3 | width: 100vw; 4 | padding: 12px; 5 | overflow-x: hidden; 6 | .avatar { 7 | display: flex; 8 | justify-content: space-evenly; 9 | img { 10 | width: 200px; 11 | height: 200px; 12 | object-fit: cover; 13 | } 14 | .avatarRight { 15 | margin-left: 8px; 16 | display: flex; 17 | flex-direction: column; 18 | align-items: center; 19 | justify-content: space-evenly; 20 | } 21 | .avatarRightItem { 22 | display: flex; 23 | flex-direction: column; 24 | align-items: center; 25 | position: relative; 26 | background-color: #fff; 27 | div { 28 | margin-top: 6px; 29 | font-size: var(--adm-font-size-main); 30 | color: #666; 31 | } 32 | .input { 33 | position: absolute; 34 | overflow: hidden; 35 | opacity: 0; 36 | left: 0; 37 | top: 0; 38 | bottom: 0; 39 | right: 0; 40 | } 41 | } 42 | } 43 | .sliders { 44 | margin-top: 20px; 45 | .download { 46 | display: flex; 47 | align-items: center; 48 | justify-content: center; 49 | padding: 12px 0; 50 | background-color: #1677FF; 51 | color: #fff; 52 | border-radius: 4px; 53 | 54 | &:link { 55 | text-decoration: none; 56 | } 57 | } 58 | } 59 | } 60 | -------------------------------------------------------------------------------- /src/pages/index/index.tsx: -------------------------------------------------------------------------------- 1 | import ImageModal from './components/ImageModal'; 2 | import imgPlaceholder from './img'; 3 | import React, { useRef, useState, useEffect } from 'react'; 4 | import styles from './index.less'; 5 | import { Button, Form, Image, ImageViewer, Slider } from 'antd-mobile'; 6 | import { getDataUrl, dataURLtoBlob } from '@/utils/file'; 7 | import { isEmpty } from 'lodash'; 8 | import { PicturesOutline, RedoOutline } from 'antd-mobile-icons'; 9 | import { useDebounceFn } from 'ahooks'; 10 | import fiveStars from '@/assets/public/fiveStars.png'; 11 | 12 | const size = 500; 13 | const scale = 369 / 303; 14 | 15 | const initialValues = { 16 | starSize: 126, 17 | gradualChangeSize: 520, 18 | gradualChangePosition: -76, 19 | gradualChangeSection: 4, 20 | }; 21 | 22 | export default function IndexPage() { 23 | const [form] = Form.useForm(); 24 | const canvasRef = useRef(null); 25 | const inputRef = useRef(null); 26 | const imageRef = useRef(null); 27 | const fiveStarsRef = useRef(null); 28 | 29 | const [imgData, setImgData] = useState(imgPlaceholder); 30 | const [imageModalVisible, setImageModalVisible] = useState(false); 31 | const [generateImg, setGenerateImg] = useState(''); 32 | const [imageViewerVisible, setImageViewerVisible] = useState(false); 33 | 34 | useEffect(() => { 35 | generate(); 36 | }, []); 37 | 38 | const { run: sliderChange } = useDebounceFn( 39 | () => { 40 | generate(); 41 | }, 42 | { 43 | wait: 300, 44 | }, 45 | ); 46 | 47 | const generate = async (baseImg: string = imgData) => { 48 | const { starSize, gradualChangeSize, gradualChangePosition, gradualChangeSection } = form.getFieldsValue(); 49 | 50 | if (!canvasRef.current || !imageRef.current || !fiveStarsRef.current) { 51 | return; 52 | } 53 | const ctx = canvasRef.current.getContext('2d'); 54 | ctx!.clearRect(0, 0, size, size); 55 | 56 | imageRef.current.src = baseImg; 57 | imageRef.current.onload = () => { 58 | ctx!.drawImage(imageRef.current!, 0, 0, size, size); 59 | // 填充渐变 60 | const radgrad = ctx!.createRadialGradient( 61 | 0, 62 | gradualChangePosition, 63 | 0, 64 | 0, 65 | gradualChangePosition + 100, 66 | (gradualChangeSize * 1.414) / 1.5, 67 | ); 68 | radgrad.addColorStop(0, '#d80203'); 69 | radgrad.addColorStop(gradualChangeSection / 10, 'rgba(216,2,3,0.8)'); 70 | radgrad.addColorStop(1, 'rgba(255, 255, 255, 0)'); 71 | ctx!.fillStyle = radgrad; 72 | ctx!.fillRect(0, 0, size, size); 73 | 74 | fiveStarsRef.current!.src = fiveStars; 75 | fiveStarsRef.current!.onload = () => { 76 | // 填充星星 77 | ctx!.drawImage( 78 | fiveStarsRef.current!, 79 | 0, 80 | 0, 81 | fiveStarsRef.current!.width, 82 | fiveStarsRef.current!.height, 83 | 20, 84 | 20, 85 | starSize * scale, 86 | starSize, 87 | ); 88 | setGenerateImg(canvasRef.current!.toDataURL('image/png', 1)); 89 | }; 90 | }; 91 | }; 92 | 93 | const getFile = async () => { 94 | if (!isEmpty(inputRef.current?.files)) { 95 | const file = inputRef.current!.files![0]; 96 | inputRef.current!.value = ''; 97 | const base64 = await getDataUrl(file); 98 | setImgData(base64); 99 | setImageModalVisible(true); 100 | } 101 | }; 102 | 103 | const handleReset = () => { 104 | form.resetFields(); 105 | generate(); 106 | }; 107 | 108 | const onConfirm = (data: string) => { 109 | setImgData(data); 110 | generate(data); 111 | setImageModalVisible(false); 112 | }; 113 | 114 | const downloadFile = () => { 115 | window.location.href = generateImg; 116 | }; 117 | 118 | const imageModalProps = { 119 | imageModalVisible, 120 | onClose: setImageModalVisible, 121 | imgData, 122 | onConfirm, 123 | }; 124 | 125 | return ( 126 |
127 |
128 | { 131 | setImageViewerVisible(true); 132 | }} 133 | /> 134 |
135 |
136 | 137 |
更换图片
138 | getFile()} 140 | ref={inputRef} 141 | className={styles.input} 142 | type="file" 143 | accept="image/png, image/jpeg" 144 | /> 145 |
146 |
147 | 148 |
重置参数
149 |
150 |
151 |
152 | 153 |
154 |
158 | 下载 159 | 160 | } 161 | initialValues={initialValues} 162 | > 163 | 164 | 165 | 166 | 167 | 168 | 169 | 170 | 171 | 172 | 173 | 174 | 175 |
176 |
177 | 178 | 179 | 180 | { 184 | setImageViewerVisible(false); 185 | }} 186 | /> 187 | 188 | 189 | 190 | 199 | 210 |
211 | ); 212 | } 213 | -------------------------------------------------------------------------------- /src/styles/global.less: -------------------------------------------------------------------------------- 1 | .page { 2 | min-height: 100vh; 3 | width: 100vw; 4 | overflow-y: auto; 5 | } 6 | -------------------------------------------------------------------------------- /src/styles/index.less: -------------------------------------------------------------------------------- 1 | @import url('./reset.less'); 2 | @import url('./global.less'); -------------------------------------------------------------------------------- /src/styles/reset.less: -------------------------------------------------------------------------------- 1 | html, 2 | body, 3 | div, 4 | span, 5 | applet, 6 | object, 7 | iframe, 8 | h1, 9 | h2, 10 | h3, 11 | h4, 12 | h5, 13 | h6, 14 | p, 15 | blockquote, 16 | pre, 17 | a, 18 | abbr, 19 | acronym, 20 | address, 21 | big, 22 | cite, 23 | code, 24 | del, 25 | dfn, 26 | em, 27 | img, 28 | ins, 29 | kbd, 30 | q, 31 | s, 32 | samp, 33 | small, 34 | strike, 35 | strong, 36 | sub, 37 | sup, 38 | tt, 39 | var, 40 | b, 41 | u, 42 | i, 43 | center, 44 | dl, 45 | dt, 46 | dd, 47 | ol, 48 | ul, 49 | li, 50 | fieldset, 51 | form, 52 | label, 53 | legend, 54 | table, 55 | caption, 56 | tbody, 57 | tfoot, 58 | thead, 59 | tr, 60 | th, 61 | td, 62 | article, 63 | aside, 64 | canvas, 65 | details, 66 | embed, 67 | figure, 68 | figcaption, 69 | footer, 70 | header, 71 | hgroup, 72 | menu, 73 | nav, 74 | output, 75 | ruby, 76 | section, 77 | summary, 78 | time, 79 | mark, 80 | audio, 81 | video { 82 | margin: 0; 83 | padding: 0; 84 | border: 0; 85 | font-size: 100%; 86 | font: inherit; 87 | vertical-align: baseline; 88 | box-sizing: border-box; 89 | } 90 | article, 91 | aside, 92 | details, 93 | figcaption, 94 | figure, 95 | footer, 96 | header, 97 | hgroup, 98 | menu, 99 | nav, 100 | section { 101 | display: block; 102 | } 103 | body { 104 | line-height: 1; 105 | } 106 | ol, 107 | ul { 108 | list-style: none; 109 | } 110 | blockquote, 111 | q { 112 | quotes: none; 113 | } 114 | blockquote:before, 115 | blockquote:after, 116 | q:before, 117 | q:after { 118 | content: ''; 119 | content: none; 120 | } 121 | table { 122 | border-collapse: collapse; 123 | border-spacing: 0; 124 | } 125 | -------------------------------------------------------------------------------- /src/utils/api.ts: -------------------------------------------------------------------------------- 1 | export default { 2 | } 3 | -------------------------------------------------------------------------------- /src/utils/dev.ts: -------------------------------------------------------------------------------- 1 | const { NODE_ENV } = process.env; 2 | 3 | /** 4 | * 判断是否为开发环境 5 | * @returns {Boolean} true | false 6 | */ 7 | export const isDev = NODE_ENV === 'development'; -------------------------------------------------------------------------------- /src/utils/file.ts: -------------------------------------------------------------------------------- 1 | export async function getDataUrl(file: File | Blob): Promise { 2 | return new Promise(resolve => { 3 | const reader = new FileReader(); 4 | 5 | reader.readAsDataURL(file); 6 | 7 | reader.onload = () => { 8 | resolve(reader.result as string) 9 | } 10 | }) 11 | } 12 | 13 | 14 | export function dataURLtoBlob(base64Str: string) { 15 | var bstr = atob(base64Str), n = bstr.length, u8arr = new Uint8Array(n); 16 | while (n--) { 17 | u8arr[n] = bstr.charCodeAt(n); 18 | } 19 | return new Blob([u8arr], { type: 'image/png' }); 20 | } -------------------------------------------------------------------------------- /src/utils/props.ts: -------------------------------------------------------------------------------- 1 | import { assign, assignWith, isUndefined } from 'lodash'; 2 | 3 | export function mergeProps(defaultProps: D, props: P): P & D { 4 | function customizer(objValue: any, srcValue: any) { 5 | return isUndefined(srcValue) ? objValue : srcValue 6 | } 7 | return assignWith(assign({}, defaultProps), props, customizer) 8 | } 9 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "esnext", 4 | "module": "esnext", 5 | "moduleResolution": "node", 6 | "resolveJsonModule": true, 7 | "importHelpers": true, 8 | "jsx": "react-jsx", 9 | "esModuleInterop": true, 10 | "sourceMap": true, 11 | "baseUrl": "./", 12 | "strict": true, 13 | "paths": { 14 | "@/*": ["src/*"], 15 | "@@/*": ["src/.umi/*"] 16 | }, 17 | "allowSyntheticDefaultImports": true 18 | }, 19 | "include": [ 20 | "mock/**/*", 21 | "src/**/*", 22 | "config/**/*", 23 | ".umirc.ts", 24 | "typings.d.ts" 25 | ], 26 | "exclude": [ 27 | "node_modules", 28 | "lib", 29 | "es", 30 | "dist", 31 | "typings", 32 | "**/__test__", 33 | "test", 34 | "docs", 35 | "tests" 36 | ] 37 | } 38 | -------------------------------------------------------------------------------- /typings.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.css'; 2 | declare module '*.less'; 3 | declare module '*.png'; 4 | declare module '*.svg' { 5 | export function ReactComponent( 6 | props: React.SVGProps, 7 | ): React.ReactElement; 8 | const url: string; 9 | export default url; 10 | } 11 | --------------------------------------------------------------------------------