diff --git a/package.json b/package.json
index 79b0e86..db19c0d 100644
--- a/package.json
+++ b/package.json
@@ -18,7 +18,8 @@
"react-lazy-load": "^3.0.13",
"react-redux": "^7.2.0",
"react-scripts": "3.4.1",
- "redux": "^4.0.5"
+ "redux": "^4.0.5",
+ "serialize-javascript": "^3.1.0"
},
"scripts": {
"start": "react-scripts start",
diff --git a/src/actions/index.js b/src/actions/index.js
index a7e3ef6..ad09615 100644
--- a/src/actions/index.js
+++ b/src/actions/index.js
@@ -36,3 +36,17 @@ export const loadDownloadData = (data) => ({
type: actionTypes.LOAD_DOWNLOAD_DATA,
data
})
+
+export const changeTitle = (title) => {
+ return {
+ type: actionTypes.CHANGE_TITLE,
+ title
+ }
+}
+
+export const changeIcon = (icon) => {
+ return {
+ type: actionTypes.CHANGE_ICON,
+ icon
+ }
+}
diff --git a/src/components/Qrcode.css b/src/components/Qrcode.css
index 77a1826..a0adebc 100644
--- a/src/components/Qrcode.css
+++ b/src/components/Qrcode.css
@@ -176,6 +176,7 @@
border: rgba(0,0,0,0.12) solid 2px;
width: calc((100vw - 56px) / 2);
height: calc((100vw - 56px) / 2);
+ background: rgba(255,255,255,0.5);
}
@media (min-width: 500px) {
@@ -567,11 +568,15 @@ input[type="number"]{
}
.div-btn {
- white-space: nowrap;
- overflow-x: hidden;
- display: flex;
justify-content: space-between;
- margin-bottom: 10px;
+ width: calc((100vw - 46px) / 1);
+ display: flex;
+ display: -webkit-flex; /* Safari */
+ flex-wrap: wrap;
+}
+
+.div-btn .dl-btn {
+ margin: 0 0 10px!important;
}
.ul-btn {
@@ -603,6 +608,7 @@ input[type="number"]{
background: var(--bg-color);
font-family: 'Futura', sans-serif;
color: var(--input-font-color);
+ display: inline-block;
}
@media (min-width: 500px) {
@@ -640,6 +646,7 @@ input[type="number"]{
}
@media (min-width: 500px) {
+
.div-btn {
max-width: 410px;
}
@@ -649,6 +656,12 @@ input[type="number"]{
}
}
+@media (min-width: 666px) {
+ .img-dl-btn {
+ max-width: 620px!important;
+ }
+}
+
.Qr-footer {
color: var(--footer-font-color);
margin-bottom: -1em;
diff --git a/src/components/app/PartDownload.js b/src/components/app/PartDownload.js
index 1abef19..916d02d 100644
--- a/src/components/app/PartDownload.js
+++ b/src/components/app/PartDownload.js
@@ -15,7 +15,20 @@ const WxMessage = () => {
return null
}
-const PartDownload = ({ value, downloadCount, onSvgDownload, onJpgDownload }) => {
+const ImgBox = ({ imgData }) => {
+ if (imgData.length > 0) {
+ return (
+
+
+
+
+
+ )
+ }
+ return null
+}
+
+const PartDownload = ({ value, downloadCount, onSvgDownload, onImgDownload }) => {
const [imgData, setImgData] = useState('');
return (
@@ -29,23 +42,17 @@ const PartDownload = ({ value, downloadCount, onSvgDownload, onJpgDownload }) =>
-
-
+
+
+
-
- {
- imgData.length > 0 ?
: null
- }
+
@@ -54,8 +61,9 @@ const PartDownload = ({ value, downloadCount, onSvgDownload, onJpgDownload }) =>
PartDownload.propTypes = {
value: PropTypes.string.isRequired,
+ downloadCount: PropTypes.number,
onSvgDownload: PropTypes.func.isRequired,
- onJpgDownload: PropTypes.func.isRequired,
+ onImgDownload: PropTypes.func.isRequired,
}
export default PartDownload;
diff --git a/src/components/app/PartMore.js b/src/components/app/PartMore.js
index 6c1f1a7..e25f8b1 100644
--- a/src/components/app/PartMore.js
+++ b/src/components/app/PartMore.js
@@ -51,6 +51,7 @@ const PartMore = () => {
最新消息
+
2020.9.1
新增 C3 样式、图标插入、 PNG 下载。
2020.6.29
新的反馈渠道!我们开始征集好玩的二维码设计啦,可以是推送尾图、海报等等,快来上传吧。点击提交。
@@ -80,9 +81,10 @@ const PartMore = () => {
这个工具开发的初衷之一就是便利设计师将其纳入自己的工作流程中。SVG 是一个优秀的、标准的矢量图片格式,各大设计软件如 Adobe Illustrator、Sketch 等都对 SVG
有着很好的支持。用户可以在下载 SVG 后导入这些软件进行二次加工,如删除中央的信息点 放入自己的 Logo 等。如果需要直接分享二维码图像,请直接下载 JPG 格式。
+
二维码无法识别的原因是什么?
+
二维码无法识别有很多原因。比如定位点不匹配识别模式、信息点颜色对比不够、遮挡部分太大。建议尝试调整容错率、颜色、图标大小等参数并在各种二维码扫描器中测试,以保证二维码被识别的成功率
使用遇到了问题,怎么反馈?
我们是两位大一的学生,忙于学业,可能在设计与开发的过程中有一些疏漏,敬请谅解。如果遇到浏览器兼容问题,请暂时选择更换软件或设备尝试。
-
请注意,应用并不能保证二维码时刻可被识别,需要多加测试。
编写二维码样式是一个锻炼设计与开发(JavaScript)能力的绝佳机会,如果你有兴趣和我们一起玩这个项目,欢迎添加我的微信(微信号:nhciao,请备注真实姓名)或发送邮件至 contact@qrbtf.com 联系我们!
diff --git a/src/components/app/PartParams.js b/src/components/app/PartParams.js
index 968ecda..ce46d13 100644
--- a/src/components/app/PartParams.js
+++ b/src/components/app/PartParams.js
@@ -2,6 +2,7 @@ import React from 'react';
import './App.css';
import ParamListViewer from "../../containers/param/ParamListViewer";
import ParamCorrectLevelViewer from "../../containers/param/ParamCorrectLevelViewer";
+import ParamIconViewer from "../../containers/param/disposable/ParamIconViewer";
const PartParams = () => (
@@ -12,6 +13,7 @@ const PartParams = () => (
diff --git a/src/components/param/FrameworkParam.js b/src/components/param/FrameworkParam.js
new file mode 100644
index 0000000..a0d0337
--- /dev/null
+++ b/src/components/param/FrameworkParam.js
@@ -0,0 +1,18 @@
+import React from 'react';
+import PropTypes from 'prop-types'
+import '../Qrcode.css';
+
+const FrameworkParam = ({ paramName, children, ...other }) => (
+
+
+
+ { paramName } |
+
+ { children }
+ |
+
+
+
+)
+
+export default FrameworkParam;
diff --git a/src/components/param/ParamColor.js b/src/components/param/ParamColor.js
index e8ecd08..3c83c84 100644
--- a/src/components/param/ParamColor.js
+++ b/src/components/param/ParamColor.js
@@ -42,7 +42,7 @@ const ParamColor = ({ rendererIndex, paramIndex, value, onChange }) => {
setDisplay(false)} />
(
-
-
-
- 容错率 |
-
-
- |
-
-
-
-)
-
-ParamCorrectLevel.propTypes = {
- value: PropTypes.number.isRequired,
- onChange: PropTypes.func.isRequired
-}
-
-export default ParamCorrectLevel;
diff --git a/src/components/param/ParamList.js b/src/components/param/ParamList.js
index fdb52da..6662755 100644
--- a/src/components/param/ParamList.js
+++ b/src/components/param/ParamList.js
@@ -6,6 +6,7 @@ import ParamSelectViewer from "../../containers/param/ParamSelectViewer";
import ParamColorViewer from "../../containers/param/ParamColorViewer";
import ParamUploadViewer from "../../containers/param/ParamUploadViewer";
import ParamCheckBoxViewer from "../../containers/param/ParamCheckBoxViewer";
+import FrameworkParam from "./FrameworkParam";
const mapTypeToComponent = ({
[ParamTypes.TEXT_EDITOR]: ParamTextViewer,
@@ -18,19 +19,12 @@ const mapTypeToComponent = ({
const ParamList = ({ rendererIndex, paramInfo }) => (
paramInfo.map((item, paramIndex) => {
return (
-
-
-
- {item.key} |
-
- {React.createElement(mapTypeToComponent[item.type], {
- rendererIndex: rendererIndex,
- paramIndex: paramIndex
- })}
- |
-
-
-
+
+ {React.createElement(mapTypeToComponent[item.type], {
+ rendererIndex: rendererIndex,
+ paramIndex: paramIndex
+ })}
+
);
})
)
diff --git a/src/components/param/disposable/ParamCorrectLevel.js b/src/components/param/disposable/ParamCorrectLevel.js
new file mode 100644
index 0000000..b6c3540
--- /dev/null
+++ b/src/components/param/disposable/ParamCorrectLevel.js
@@ -0,0 +1,25 @@
+import React from 'react';
+import PropTypes from 'prop-types'
+import '../../Qrcode.css';
+import FrameworkParam from "../FrameworkParam";
+
+const ParamCorrectLevel = ({value, onChange}) => (
+
+
+
+)
+
+ParamCorrectLevel.propTypes = {
+ value: PropTypes.number.isRequired,
+ onChange: PropTypes.func.isRequired
+}
+
+export default ParamCorrectLevel;
diff --git a/src/components/param/disposable/ParamIcon.js b/src/components/param/disposable/ParamIcon.js
new file mode 100644
index 0000000..ac94339
--- /dev/null
+++ b/src/components/param/disposable/ParamIcon.js
@@ -0,0 +1,60 @@
+import React from 'react';
+import PropTypes from 'prop-types'
+import '../../Qrcode.css';
+import FrameworkParam from "../FrameworkParam";
+import {getExactValue} from "../../../utils/util";
+import ParamIconSrcViewer from "../../../containers/param/disposable/ParamIconSrcViewer";
+
+const IconParams = ({ icon, onBlur, onKeyPress }) => {
+ const { enabled, src, scale } = icon;
+ const components = [];
+
+ if (getExactValue(enabled, 0) == 1) {
+ components.push(
+
+
+
+ );
+ }
+
+ if (getExactValue(enabled, 0) != 0) {
+ components.push(
+
+ onBlur({...icon, scale: e.target.value})}
+ onKeyPress={(e) => onKeyPress(e, {...icon, scale: e.target.value})}
+ />
+
+ )
+ }
+ return components;
+}
+
+const ParamIcon = ({icon, onBlur, onKeyPress}) => (
+
+
+
+
+
+
+)
+
+ParamIcon.propTypes = {
+ icon: PropTypes.object.isRequired,
+ onChange: PropTypes.func.isRequired
+}
+
+export default ParamIcon;
diff --git a/src/components/param/disposable/ParamTitle.js b/src/components/param/disposable/ParamTitle.js
new file mode 100644
index 0000000..206c73b
--- /dev/null
+++ b/src/components/param/disposable/ParamTitle.js
@@ -0,0 +1,67 @@
+import React from 'react';
+import PropTypes from 'prop-types'
+import '../../Qrcode.css';
+import FrameworkParam from "../FrameworkParam";
+import {getExactValue} from "../../../utils/util";
+import ParamTitleColorViewer from "../../../containers/param/disposable/ParamTitleColorViewer";
+
+const TitleParams = ({ title, onChange }) => {
+ const { enabled, text, color, size, align } = title;
+ if (getExactValue(enabled, 0)) {
+ return (
+
+
+ onChange({...title, text: e.target.value})}
+ />
+
+
+
+
+
+ onChange({...title, size: e.target.value})}
+ />
+
+
+
+
+
+ )
+ }
+ return null;
+}
+
+const ParamTitle = ({title, onChange}) => (
+
+
+
+
+
+
+)
+
+ParamTitle.propTypes = {
+ title: PropTypes.object.isRequired,
+ onChange: PropTypes.func.isRequired
+}
+
+export default ParamTitle;
diff --git a/src/components/renderer/Renderer25D.js b/src/components/renderer/Renderer25D.js
index 4aebef5..0ae6a5f 100644
--- a/src/components/renderer/Renderer25D.js
+++ b/src/components/renderer/Renderer25D.js
@@ -2,8 +2,15 @@ import React from "react";
import {ParamTypes} from "../../constant/ParamTypes";
import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
+import {getExactValue, getIdNum} from "../../utils/util";
-function listPoints(qrcode, params) {
+const X = [ Math.sqrt(3)/2, 1/2];
+const Y = [-Math.sqrt(3)/2, 1/2];
+const Z = [0, 0];
+
+const matrixString = 'matrix(' + String(X[0]) + ', ' + String(X[1]) + ', ' + String(Y[0]) + ', ' + String(Y[1]) + ', ' + String(Z[0]) + ', ' + String(Z[1]) + ')'
+
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
@@ -19,12 +26,6 @@ function listPoints(qrcode, params) {
let rightColor = params[4];
let id = 0;
- const X = [-Math.sqrt(3)/2, 1/2];
- const Y = [ Math.sqrt(3)/2, 1/2];
- const Z = [0, 0];
-
- const matrixString = 'matrix(' + String(X[0]) + ', ' + String(X[1]) + ', ' + String(Y[0]) + ', ' + String(Y[1]) + ', ' + String(Z[0]) + ', ' + String(Z[1]) + ')'
-
if (height <= 0) height = 1.0;
if (height2 <= 0) height2 = 1.0;
@@ -77,6 +78,149 @@ function getParamInfo() {
];
}
+let defaultDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconEnabled = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100);
+ const iconXY = (nCount - iconSize) / 2;
+
+ if (icon && iconEnabled) {
+
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push(
+
+
+
+ );
+ pointList.push(
+
+
+
+
+
+
+
+
+
+
+ );
+
+ }
+ }
+
+ return pointList;
+}
+
+let builtinDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100);
+ const iconXY = (nCount - iconSize) / 2;
+
+ const WeChatIconSmall = (
+
+
+
+
+ )
+
+ const WeChatIcon = (
+
+
+
+
+
+ )
+
+ const WeChatPayIcon = (
+
+
+
+
+ )
+
+ const AlipayIcon = (
+
+
+
+
+ )
+
+ function builtinIcon() {
+ if (iconMode === 2) {
+ return WeChatIconSmall
+ } else if (iconMode === 3) {
+ return WeChatIcon
+ } else if (iconMode === 4) {
+ return WeChatPayIcon
+ } else if (iconMode === 5) {
+ return AlipayIcon
+ }
+ }
+
+ if (icon && iconMode) {
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push(
+
+
+
+ );
+ pointList.push(
+
+
+
+
+
+
+
+
+ {builtinIcon()}
+
+
+
+ );
+ }
+ }
+
+ return pointList;
+}
+
+function drawIcon({ qrcode, icon, params }) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ if (iconMode === 1) {
+
+ // Custom
+ // default
+ return defaultDrawIcon({ qrcode, icon, params });
+
+ } else {
+
+ return builtinDrawIcon({ qrcode, icon, params });
+ }
+}
+
function viewBox(qrcode) {
if (!qrcode) return '0 0 0 0';
@@ -87,7 +231,8 @@ function viewBox(qrcode) {
const Renderer25D = createRenderer({
listPoints: listPoints,
getParamInfo: getParamInfo,
- getViewBox: viewBox
+ getViewBox: viewBox,
+ drawIcon: drawIcon
})
export default Renderer25D
diff --git a/src/components/renderer/RendererBase.js b/src/components/renderer/RendererBase.js
index f2c01aa..2164c36 100644
--- a/src/components/renderer/RendererBase.js
+++ b/src/components/renderer/RendererBase.js
@@ -1,14 +1,25 @@
import React from "react";
import {ParamTypes} from "../../constant/ParamTypes";
import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
-import {createRenderer} from "../style/Renderer";
-import {rand} from "../../utils/util";
+import {createRenderer, defaultDrawIcon} from "../style/Renderer";
+import {getExactValue, rand} from "../../utils/util";
import LinkTrace from "../link/LinkTrace";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
-
const nCount = qrcode.getModuleCount();
+
+ const iconEnabled = getExactValue(icon.enabled, 0);
+
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > .33 ? .33 : scale));
+ const iconXY = (nCount - iconSize) / 2;
+
+ function nearIcon(x, y) {
+ return Math.pow((nCount - 1) / 2 - x, 2) + Math.pow((nCount - 1) / 2 - y, 2) < Math.pow(iconSize / 2, 2);
+ }
+
const typeTable = getTypeTable(qrcode);
const pointList = new Array(nCount);
@@ -65,12 +76,14 @@ function listPoints(qrcode, params) {
}
}
else {
- if (type === 0)
+ if (type === 0) {
pointList.push()
- else if (type === 1)
+ } else if (type === 1) {
+ if (!(iconEnabled && nearIcon(x, y))) {}
pointList.push()
- else if (type === 2)
+ } else if (type === 2) {
pointList.push()
+ }
}
}
}
diff --git a/src/components/renderer/RendererCircle.js b/src/components/renderer/RendererCircle.js
index d5e6788..b0a9329 100644
--- a/src/components/renderer/RendererCircle.js
+++ b/src/components/renderer/RendererCircle.js
@@ -4,7 +4,7 @@ import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
import {rand} from "../../utils/util";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
diff --git a/src/components/renderer/RendererDSJ.js b/src/components/renderer/RendererDSJ.js
index bde7224..31ee9a5 100644
--- a/src/components/renderer/RendererDSJ.js
+++ b/src/components/renderer/RendererDSJ.js
@@ -4,7 +4,7 @@ import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
import LinkTrace from "../link/LinkTrace";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
diff --git a/src/components/renderer/RendererFunc.js b/src/components/renderer/RendererFunc.js
index 82ba1b2..52753b5 100644
--- a/src/components/renderer/RendererFunc.js
+++ b/src/components/renderer/RendererFunc.js
@@ -3,7 +3,7 @@ import {ParamTypes} from "../../constant/ParamTypes";
import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
@@ -13,7 +13,6 @@ function listPoints(qrcode, params) {
let type = params[0];
let size = params[1] / 100;
let funcType = params[1];
- let opacity = params[2] / 100;
let posType = params[3];
let id = 0;
let otherColor = params[4];
diff --git a/src/components/renderer/RendererImage.js b/src/components/renderer/RendererImage.js
index 5f9f7fe..e3f2cda 100644
--- a/src/components/renderer/RendererImage.js
+++ b/src/components/renderer/RendererImage.js
@@ -4,7 +4,7 @@ import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
import {defaultImage} from "../../constant/References";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
diff --git a/src/components/renderer/RendererImageFill.js b/src/components/renderer/RendererImageFill.js
new file mode 100644
index 0000000..1a3408f
--- /dev/null
+++ b/src/components/renderer/RendererImageFill.js
@@ -0,0 +1,60 @@
+import React from "react";
+import {ParamTypes} from "../../constant/ParamTypes";
+import {createRenderer} from "../style/Renderer";
+import {defaultImage} from "../../constant/References";
+
+function listPoints({ qrcode, params, icon }) {
+ if (!qrcode) return []
+
+ const nCount = qrcode.getModuleCount();
+ const pointList = new Array(nCount);
+
+ let color = params[1];
+ let opacity = params[2] / 100;
+ let id = 0;
+
+ pointList.push();
+ pointList.push();
+
+ for (let x = 0; x < nCount; x++) {
+ for (let y = 0; y < nCount; y++) {
+ if (!qrcode.isDark(x, y)) {
+ pointList.push();
+ }
+ }
+ }
+
+ return pointList;
+}
+
+function getParamInfo() {
+ return [
+ {
+ type: ParamTypes.UPLOAD_BUTTON,
+ key: '背景图片',
+ default: defaultImage,
+ },
+ {
+ type: ParamTypes.COLOR_EDITOR,
+ key: '覆盖颜色',
+ default: '#000000'
+ },
+ {
+ type: ParamTypes.TEXT_EDITOR,
+ key: '覆盖不透明度',
+ default: 10,
+ },
+ ];
+}
+
+const RendererImageFill = createRenderer({
+ listPoints: listPoints,
+ getParamInfo: getParamInfo,
+})
+
+
+export default RendererImageFill
+
+RendererImageFill.detail = (
+ 图像填充
+);
diff --git a/src/components/renderer/RendererLine.js b/src/components/renderer/RendererLine.js
index 72a1420..e81686f 100644
--- a/src/components/renderer/RendererLine.js
+++ b/src/components/renderer/RendererLine.js
@@ -4,7 +4,7 @@ import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {createRenderer} from "../style/Renderer";
import {rand} from "../../utils/util";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
diff --git a/src/components/renderer/RendererRandRect.js b/src/components/renderer/RendererRandRect.js
index 78571f6..7fbaf86 100644
--- a/src/components/renderer/RendererRandRect.js
+++ b/src/components/renderer/RendererRandRect.js
@@ -2,7 +2,7 @@ import React from "react";
import {rand} from "../../utils/util";
import {createRenderer} from "../style/Renderer";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
const nCount = qrcode.getModuleCount();
diff --git a/src/components/renderer/RendererResImage.js b/src/components/renderer/RendererResImage.js
index 02d6264..8453443 100644
--- a/src/components/renderer/RendererResImage.js
+++ b/src/components/renderer/RendererResImage.js
@@ -3,10 +3,11 @@ import {gamma} from "../../utils/imageUtils";
import {ParamTypes} from "../../constant/ParamTypes";
import {getTypeTable, QRPointType} from "../../utils/qrcodeHandler";
import {defaultResImage} from "../../constant/References";
+import {getExactValue, getIdNum} from "../../utils/util";
-function listPoints(qrcode, params) {
+function listPoints({ qrcode, params, icon }) {
if (!qrcode) return []
-
+console.log(icon)
const nCount = qrcode.getModuleCount();
const typeTable = getTypeTable(qrcode);
const pointList = new Array(nCount);
@@ -155,7 +156,141 @@ function getGrayPointList(params, size, black, white) {
})
}
-const RendererResImage = ({qrcode, params, setParamInfo}) => {
+let defaultDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconEnabled = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100 * 3);
+ const iconXY = (nCount*3 - iconSize) / 2;
+
+ if (icon && iconEnabled) {
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push();
+ pointList.push(
+
+
+
+
+
+
+
+
+
+
+ );
+ }
+
+ }
+
+ return pointList;
+}
+
+let builtinDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100 * 3);
+ const iconXY = (nCount*3 - iconSize) / 2;
+
+ const WeChatIconSmall = (
+
+
+
+
+ )
+
+ const WeChatIcon = (
+
+
+
+
+
+ )
+
+ const WeChatPayIcon = (
+
+
+
+
+ )
+
+ const AlipayIcon = (
+
+
+
+
+ )
+
+ function builtinIcon() {
+ if (iconMode === 2) {
+ return WeChatIconSmall
+ } else if (iconMode === 3) {
+ return WeChatIcon
+ } else if (iconMode === 4) {
+ return WeChatPayIcon
+ } else if (iconMode === 5) {
+ return AlipayIcon
+ }
+ }
+
+ if (icon && iconMode) {
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push();
+ pointList.push(
+
+
+
+
+
+
+
+
+ {builtinIcon()}
+
+
+
+ );
+ }
+ }
+
+ return pointList;
+}
+
+function drawIcon({ qrcode, icon, params }) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ if (iconMode === 1) {
+
+ // Custom
+ // default
+ return defaultDrawIcon({ qrcode, icon, params });
+
+ } else {
+
+ return builtinDrawIcon({ qrcode, icon, params });
+ }
+}
+
+const RendererResImage = ({qrcode, params, setParamInfo, icon}) => {
let otherColor = params[5];
useEffect(() => {
@@ -178,7 +313,8 @@ const RendererResImage = ({qrcode, params, setParamInfo}) => {
- {gpl.concat(listPoints(qrcode, params))}
+ {gpl.concat(listPoints({ qrcode, params, icon }))}
+ {drawIcon({ qrcode, params, icon })}
)
}
diff --git a/src/components/style/Renderer.js b/src/components/style/Renderer.js
index 8a64af6..3b53999 100644
--- a/src/components/style/Renderer.js
+++ b/src/components/style/Renderer.js
@@ -1,5 +1,5 @@
import React, {useEffect} from "react";
-import {extend} from "../../utils/util";
+import {extend, getExactValue, getIdNum} from "../../utils/util";
const Renderer = ({ rendererType, ...other }) => (
React.createElement(rendererType, other)
@@ -9,24 +9,158 @@ function areEqual(prevProps, nextProps) {
return !(prevProps.selected === true || nextProps.selected === true)
}
-export function createRenderer(renderer) {
- let defaultViewBox = function (qrcode) {
- if (!qrcode) return '0 0 0 0';
+let defaultViewBox = function (qrcode) {
+ if (!qrcode) return '0 0 0 0';
+
+ const nCount = qrcode.getModuleCount();
+ return String(-nCount / 5) + ' ' + String(-nCount / 5) + ' ' + String(nCount + nCount / 5 * 2) + ' ' + String(nCount + nCount / 5 * 2);
+}
+
+let defaultDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconEnabled = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100);
+ const iconXY = (nCount - iconSize) / 2;
+
+ if (icon && iconEnabled) {
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push();
+ pointList.push(
+
+
+
+
+
+
+
+
+
+
+ );
+ }
- const nCount = qrcode.getModuleCount();
- return String(-nCount / 5) + ' ' + String(-nCount / 5) + ' ' + String(nCount + nCount / 5 * 2) + ' ' + String(nCount + nCount / 5 * 2);
}
+ return pointList;
+}
+
+let builtinDrawIcon = function ({ qrcode, params, title, icon }) {
+ if (!qrcode) return []
+
+ let id = 0;
+ const nCount = qrcode.getModuleCount();
+ const pointList = [];
+ const sq25 = "M32.048565,-1.29480038e-15 L67.951435,1.29480038e-15 C79.0954192,-7.52316311e-16 83.1364972,1.16032014 87.2105713,3.3391588 C91.2846454,5.51799746 94.4820025,8.71535463 96.6608412,12.7894287 C98.8396799,16.8635028 100,20.9045808 100,32.048565 L100,67.951435 C100,79.0954192 98.8396799,83.1364972 96.6608412,87.2105713 C94.4820025,91.2846454 91.2846454,94.4820025 87.2105713,96.6608412 C83.1364972,98.8396799 79.0954192,100 67.951435,100 L32.048565,100 C20.9045808,100 16.8635028,98.8396799 12.7894287,96.6608412 C8.71535463,94.4820025 5.51799746,91.2846454 3.3391588,87.2105713 C1.16032014,83.1364972 5.01544207e-16,79.0954192 -8.63200256e-16,67.951435 L8.63200256e-16,32.048565 C-5.01544207e-16,20.9045808 1.16032014,16.8635028 3.3391588,12.7894287 C5.51799746,8.71535463 8.71535463,5.51799746 12.7894287,3.3391588 C16.8635028,1.16032014 20.9045808,7.52316311e-16 32.048565,-1.29480038e-15 Z";
+
+ // draw icon
+ if (icon) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ const {src, scale} = icon;
+
+ const iconSize = Number(nCount * (scale > 33 ? 33 : scale) / 100);
+ const iconXY = (nCount - iconSize) / 2;
+
+ const WeChatIconSmall = (
+
+
+
+
+ )
+
+ const WeChatIcon = (
+
+
+
+
+
+ )
+
+ const WeChatPayIcon = (
+
+
+
+
+ )
+
+ const AlipayIcon = (
+
+
+
+
+ )
+
+ function builtinIcon() {
+ if (iconMode === 2) {
+ return WeChatIconSmall
+ } else if (iconMode === 3) {
+ return WeChatIcon
+ } else if (iconMode === 4) {
+ return WeChatPayIcon
+ } else if (iconMode === 5) {
+ return AlipayIcon
+ }
+ }
+
+ if (icon && iconMode) {
+ const randomIdDefs = getIdNum();
+ const randomIdClips = getIdNum();
+ pointList.push();
+ pointList.push(
+
+
+
+
+
+
+
+
+ {builtinIcon()}
+
+
+
+ );
+ }
+ }
+
+ return pointList;
+}
+
+function drawIcon({ qrcode, icon, params }) {
+ const iconMode = getExactValue(icon.enabled, 0);
+ if (iconMode === 1) {
+
+ // Custom
+ // default
+ return defaultDrawIcon({ qrcode, icon, params });
+
+ } else {
+
+ return builtinDrawIcon({ qrcode, icon, params });
+ }
+}
+
+export function createRenderer(renderer) {
renderer = extend({
getViewBox: defaultViewBox,
- listPoints: (qrcode, params) => { return []; },
+ listPoints: ({ qrcode, params, icon }) => { return []; },
getParamInfo: () => {return []; },
beginRendering: ({ qrcode, params, setParamInfo }) => {},
beforeListing: ({ qrcode, params, setParamInfo }) => {},
- afterListing: ({ qrcode, params, setParamInfo }) => {},
+ drawIcon: drawIcon
}, renderer);
- return ({ qrcode, params, setParamInfo}) => {
+ return ({ qrcode, params, title, icon, setParamInfo}) => {
useEffect(() => {
setParamInfo(renderer.getParamInfo());
}, [setParamInfo]);
@@ -36,11 +170,12 @@ export function createRenderer(renderer) {
);
}
}
export default React.memo(Renderer, areEqual)
+export { defaultDrawIcon, defaultViewBox }
diff --git a/src/components/svg/SVG/资源 1.svg b/src/components/svg/SVG/资源 1.svg
new file mode 100644
index 0000000..5c0e3d6
--- /dev/null
+++ b/src/components/svg/SVG/资源 1.svg
@@ -0,0 +1,7 @@
+
\ No newline at end of file
diff --git a/src/components/svg/SVG/资源 2.svg b/src/components/svg/SVG/资源 2.svg
new file mode 100644
index 0000000..2093eff
--- /dev/null
+++ b/src/components/svg/SVG/资源 2.svg
@@ -0,0 +1,10 @@
+
\ No newline at end of file
diff --git a/src/components/svg/SVG/资源 3.svg b/src/components/svg/SVG/资源 3.svg
new file mode 100644
index 0000000..ae5d969
--- /dev/null
+++ b/src/components/svg/SVG/资源 3.svg
@@ -0,0 +1,8 @@
+
\ No newline at end of file
diff --git a/src/components/svg/SVG/资源 4.svg b/src/components/svg/SVG/资源 4.svg
new file mode 100644
index 0000000..385c371
--- /dev/null
+++ b/src/components/svg/SVG/资源 4.svg
@@ -0,0 +1,7 @@
+
\ No newline at end of file
diff --git a/src/constant/ActionTypes.js b/src/constant/ActionTypes.js
index 12a5c88..b2a11df 100644
--- a/src/constant/ActionTypes.js
+++ b/src/constant/ActionTypes.js
@@ -5,4 +5,6 @@ export const actionTypes = {
CREATE_PARAM: 'CREATE_PARAM',
CHANGE_PARAM: 'CHANGE_PARAM',
LOAD_DOWNLOAD_DATA: 'LOAD_DOWNLOAD_DATA',
+ CHANGE_TITLE: 'CHANGE_TITLE',
+ CHANGE_ICON: 'CHANGE_ICON'
}
diff --git a/src/constant/References.js b/src/constant/References.js
index db93576..88014c9 100644
--- a/src/constant/References.js
+++ b/src/constant/References.js
@@ -1,5 +1,5 @@
export const QRBTF_URL = 'https://qrbtf.com'
-export const defaultResImage = ""
export const defaultImage = ""
+export const defaultResImage = ""
diff --git a/src/containers/app/PartDownloadViewer.js b/src/containers/app/PartDownloadViewer.js
index 35f6c62..0d07225 100644
--- a/src/containers/app/PartDownloadViewer.js
+++ b/src/containers/app/PartDownloadViewer.js
@@ -3,7 +3,7 @@ import PartDownload from "../../components/app/PartDownload";
import {saveImg, saveSvg} from "../../utils/downloader";
import {getDownloadCount, increaseDownloadData, recordDownloadDetail} from "../../api/TcbHandler";
import {getParamDetailedValue, outerHtml} from "../../utils/util";
-import {handleDownloadJpg, handleDownloadSvg} from "../../utils/gaHelper";
+import {handleDownloadImg, handleDownloadSvg} from "../../utils/gaHelper";
function saveDB(state, type, updateDownloadData) {
return new Promise(resolve => {
@@ -45,11 +45,11 @@ const mapStateToProps = (state, ownProps) => ({
saveDB(state, 'svg', ownProps.updateDownloadData);
handleDownloadSvg(state.value);
},
- onJpgDownload: () => {
+ onImgDownload: (type) => {
return new Promise(resolve => {
- saveImg(state.value, outerHtml(state.selectedIndex), 1500, 1500).then((res) => {
- saveDB(state, 'jpg', ownProps.updateDownloadData).then(() => {
- handleDownloadJpg(state.value);
+ saveImg(state.value, outerHtml(state.selectedIndex), 1500, 1500, type).then((res) => {
+ saveDB(state, type, ownProps.updateDownloadData).then(() => {
+ handleDownloadImg(state.value, type);
resolve(res)
});
});
diff --git a/src/containers/param/ParamCorrectLevelViewer.js b/src/containers/param/ParamCorrectLevelViewer.js
index aa872ae..db00b3b 100644
--- a/src/containers/param/ParamCorrectLevelViewer.js
+++ b/src/containers/param/ParamCorrectLevelViewer.js
@@ -1,5 +1,5 @@
import {changeCorrectLevel} from "../../actions";
-import ParamCorrectLevel from "../../components/param/ParamCorrectLevel";
+import ParamCorrectLevel from "../../components/param/disposable/ParamCorrectLevel";
import {connect} from "react-redux";
const mapStateToProps = (state) => ({
diff --git a/src/containers/param/disposable/ParamIconSrcViewer.js b/src/containers/param/disposable/ParamIconSrcViewer.js
new file mode 100644
index 0000000..3c859b7
--- /dev/null
+++ b/src/containers/param/disposable/ParamIconSrcViewer.js
@@ -0,0 +1,23 @@
+import { connect } from 'react-redux';
+import {isPicture, toBase64} from "../../../utils/imageUtils";
+import ParamUpload from "../../../components/param/ParamUpload";
+
+const mapStateToProps = (state, ownProps) => ({
+ rendererIndex: -1,
+ paramIndex: -1,
+})
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ onChange: (e) => {
+ if (e.target.files.length > 0) {
+ const file = e.target.files[0];
+ if (isPicture(file)) {
+ toBase64(file, 1.0).then(res => {
+ ownProps.onChange({ ...ownProps.icon, src: res})
+ })
+ }
+ }
+ }
+})
+
+export default connect(mapStateToProps, mapDispatchToProps)(ParamUpload);
diff --git a/src/containers/param/disposable/ParamIconViewer.js b/src/containers/param/disposable/ParamIconViewer.js
new file mode 100644
index 0000000..2f54a80
--- /dev/null
+++ b/src/containers/param/disposable/ParamIconViewer.js
@@ -0,0 +1,20 @@
+import { connect } from 'react-redux';
+import ParamIcon from "../../../components/param/disposable/ParamIcon";
+import {changeIcon} from "../../../actions";
+
+const mapStateToProps = (state, ownProps) => ({
+ icon: state.icon
+})
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ onBlur: (icon) => {
+ dispatch(changeIcon(icon))
+ },
+ onKeyPress: (e, icon) => {
+ if (e.key === 'Enter') {
+ dispatch(changeIcon(icon))
+ }
+ }
+})
+
+export default connect(mapStateToProps, mapDispatchToProps)(ParamIcon);
diff --git a/src/containers/param/disposable/ParamTitleColorViewer.js b/src/containers/param/disposable/ParamTitleColorViewer.js
new file mode 100644
index 0000000..3f8f930
--- /dev/null
+++ b/src/containers/param/disposable/ParamTitleColorViewer.js
@@ -0,0 +1,16 @@
+import { connect } from 'react-redux';
+import ParamColor from "../../../components/param/ParamColor";
+
+const mapStateToProps = (state, ownProps) => ({
+ rendererIndex: -1,
+ paramIndex: -1,
+ value: state.title.color
+})
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ onChange: (color) => {
+ ownProps.onChange({ ...ownProps.title, color: color.hex })
+ }
+})
+
+export default connect(mapStateToProps, mapDispatchToProps)(ParamColor);
diff --git a/src/containers/param/disposable/ParamTitleViewer.js b/src/containers/param/disposable/ParamTitleViewer.js
new file mode 100644
index 0000000..dc50fe4
--- /dev/null
+++ b/src/containers/param/disposable/ParamTitleViewer.js
@@ -0,0 +1,15 @@
+import { connect } from 'react-redux';
+import ParamTitle from "../../../components/param/disposable/ParamTitle";
+import {changeTitle} from "../../../actions";
+
+const mapStateToProps = (state, ownProps) => ({
+ title: state.title
+})
+
+const mapDispatchToProps = (dispatch, ownProps) => ({
+ onChange: (title) => {
+ dispatch(changeTitle(title))
+ }
+})
+
+export default connect(mapStateToProps, mapDispatchToProps)(ParamTitle);
diff --git a/src/containers/style/RendererViewer.js b/src/containers/style/RendererViewer.js
index 6553625..b28787b 100644
--- a/src/containers/style/RendererViewer.js
+++ b/src/containers/style/RendererViewer.js
@@ -7,6 +7,8 @@ const mapStateToProps = (state, ownProps) => ({
rendererIndex: ownProps.index,
qrcode: state.qrcode,
params: fillEmptyWith(state.paramValue[ownProps.index].slice(), 0),
+ title: state.title,
+ icon: state.icon,
selected: state.selectedIndex === ownProps.index,
})
diff --git a/src/containers/style/StyleListViewer.js b/src/containers/style/StyleListViewer.js
index 670ca5f..e6050f8 100644
--- a/src/containers/style/StyleListViewer.js
+++ b/src/containers/style/StyleListViewer.js
@@ -12,17 +12,19 @@ import { RendererRandRound, RendererRect, RendererRound } from "../../components
import { RendererLine, RendererLine2 } from "../../components/renderer/RendererLine";
import { RendererFuncA, RendererFuncB } from "../../components/renderer/RendererFunc";
import * as React from "react";
+import RendererImageFill from "../../components/renderer/RendererImageFill";
const styles = [
{value: "A1", renderer: RendererRect},
{value: "C2", renderer: RendererResImage},
{value: "SP — 1", renderer: RendererDSJ},
+ {value: "A — a1", renderer: RendererLine},
+ {value: "SP — 3", renderer: RendererCircle},
{value: "A2", renderer: RendererRound},
{value: "A3", renderer: RendererRandRound},
- {value: "A — a1", renderer: RendererLine},
{value: "A — b2", renderer: RendererFuncB},
- {value: "SP — 3", renderer: RendererCircle},
{value: "C1", renderer: RendererImage},
+ {value: "C3", renderer: RendererImageFill},
{value: "B1", renderer: Renderer25D},
{value: "A — a2", renderer: RendererLine2},
{value: "A — b1", renderer: RendererFuncA},
diff --git a/src/reducers/index.js b/src/reducers/index.js
index 417abf0..3f46af9 100644
--- a/src/reducers/index.js
+++ b/src/reducers/index.js
@@ -13,6 +13,8 @@ const initialState = {
history: [],
downloadData: [],
qrcode: encodeData({text: QRBTF_URL, correctLevel: 0}),
+ icon: { enabled: 0, src: '', scale: 22 },
+ title: { enabled: 0, text: '', color: 'black', size: 20, align: 'middle'},
paramInfo: new Array(16).fill(new Array(16)),
paramValue: new Array(16).fill(new Array(16))
}
@@ -65,6 +67,16 @@ export default function appReducer(state = initialState, action) {
downloadData: action.data
});
}
+ case actionTypes.CHANGE_TITLE: {
+ return Object.assign({}, state, {
+ title: Object.assign({}, state.title, action.title)
+ });
+ }
+ case actionTypes.CHANGE_ICON: {
+ return Object.assign({}, state, {
+ icon: Object.assign({}, state.icon, action.icon)
+ });
+ }
default: return state
}
}
diff --git a/src/utils/downloader.js b/src/utils/downloader.js
index 67a5fec..cbbd01c 100644
--- a/src/utils/downloader.js
+++ b/src/utils/downloader.js
@@ -1,6 +1,8 @@
const svgHead = "\n " +
"\n"
+const MIME = { "jpg": "image/jpeg", "png": "image/png" };
+
export function saveSvg(value, content) {
let htmlContent = [svgHead + content]
let bl = new Blob(htmlContent, {type: "image/svg+xml"})
@@ -13,9 +15,11 @@ export function saveSvg(value, content) {
a.click()
}
-export function saveImg(value, content, width, height) {
+export function saveImg(value, content, width, height, type) {
+ if (!MIME[type]) throw "Error image type";
+
// Finish creating downloadable data
- let filename = "QRcode_" + value + ".jpg";
+ let filename = "QRcode_" + value + "." + type;
const wrap = document.createElement('div');
wrap.innerHTML = content;
@@ -40,15 +44,15 @@ export function saveImg(value, content, width, height) {
return new Promise(resolve => {
img.onload = () => {
- ctx.fillStyle = 'white'
- ctx.fillRect(0, 0, width, height)
+ ctx.fillStyle = 'white';
+ if (type === 'jpg') ctx.fillRect(0, 0, width, height);
ctx.drawImage(img, 0, 0, width, height);
// `download` attr is not well supported
// Will result in a download popup for chrome and the
// image opening in a new tab for others.
let a = document.createElement('a');
- let data = canvas.toDataURL('image/jpeg', 0.8);
+ let data = canvas.toDataURL(MIME[type], 0.8);
a.setAttribute('href', data)
a.setAttribute('target', 'download')
a.setAttribute('download', filename);
diff --git a/src/utils/gaHelper.js b/src/utils/gaHelper.js
index 8716ca5..c9eb730 100644
--- a/src/utils/gaHelper.js
+++ b/src/utils/gaHelper.js
@@ -16,10 +16,10 @@ export function handleDownloadSvg(rendererName) {
});
}
-export function handleDownloadJpg(rendererName) {
+export function handleDownloadImg(rendererName, type) {
ReactGA.event({
category: 'Style',
- action: 'DownloadJpg',
+ action: 'Download' + type.charAt(0).toUpperCase() + type.slice(1),
label: rendererName,
});
}
diff --git a/src/utils/util.js b/src/utils/util.js
index b5745ba..e1438c2 100644
--- a/src/utils/util.js
+++ b/src/utils/util.js
@@ -1,11 +1,18 @@
import {ParamTypes} from "../constant/ParamTypes";
let seed = 0;
+let idNum = 0;
+
export function rand(min, max) {
seed = (seed * 9301 + 49297) % 233280;
return min + (seed / 233280.0) * (max - min);
}
+export function getIdNum() {
+ idNum += 1
+ return idNum.toString()
+}
+
export function fillEmptyWith(arr, value) {
for (let i = 0; i < arr.length; i++)
if (!arr[i]) arr[i] = value;
diff --git a/src/未标题-2.ai b/src/未标题-2.ai
new file mode 100644
index 0000000..2b379da
--- /dev/null
+++ b/src/未标题-2.ai
@@ -0,0 +1,1319 @@
+%PDF-1.5
%
+1 0 obj
<>/OCGs[30 0 R]>>/Pages 3 0 R/Type/Catalog>>
endobj
2 0 obj
<>stream
+
+
+
+
+ application/pdf
+
+
+ 未标题-2
+
+
+ Adobe Illustrator 24.0 (Macintosh)
+ 2020-09-01T23:18:32+08:00
+ 2020-09-01T23:18:33+08:00
+ 2020-09-01T23:18:33+08:00
+
+
+
+ 256
+ 56
+ JPEG
+ /9j/4AAQSkZJRgABAgEASABIAAD/7QAsUGhvdG9zaG9wIDMuMAA4QklNA+0AAAAAABAASAAAAAEA
AQBIAAAAAQAB/+4ADkFkb2JlAGTAAAAAAf/bAIQABgQEBAUEBgUFBgkGBQYJCwgGBggLDAoKCwoK
DBAMDAwMDAwQDA4PEA8ODBMTFBQTExwbGxscHx8fHx8fHx8fHwEHBwcNDA0YEBAYGhURFRofHx8f
Hx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8f/8AAEQgAOAEAAwER
AAIRAQMRAf/EAaIAAAAHAQEBAQEAAAAAAAAAAAQFAwIGAQAHCAkKCwEAAgIDAQEBAQEAAAAAAAAA
AQACAwQFBgcICQoLEAACAQMDAgQCBgcDBAIGAnMBAgMRBAAFIRIxQVEGE2EicYEUMpGhBxWxQiPB
UtHhMxZi8CRygvElQzRTkqKyY3PCNUQnk6OzNhdUZHTD0uIIJoMJChgZhJRFRqS0VtNVKBry4/PE
1OT0ZXWFlaW1xdXl9WZ2hpamtsbW5vY3R1dnd4eXp7fH1+f3OEhYaHiImKi4yNjo+Ck5SVlpeYmZ
qbnJ2en5KjpKWmp6ipqqusra6voRAAICAQIDBQUEBQYECAMDbQEAAhEDBCESMUEFURNhIgZxgZEy
obHwFMHR4SNCFVJicvEzJDRDghaSUyWiY7LCB3PSNeJEgxdUkwgJChgZJjZFGidkdFU38qOzwygp
0+PzhJSktMTU5PRldYWVpbXF1eX1RlZmdoaWprbG1ub2R1dnd4eXp7fH1+f3OEhYaHiImKi4yNjo
+DlJWWl5iZmpucnZ6fkqOkpaanqKmqq6ytrq+v/aAAwDAQACEQMRAD8APP8AlZvnr/q7Sf8AARf8
0Zq/Hn3vmf8ALmr/AJ5+Q/U7/lZvnr/q7Sf8BF/zRj48+9f5c1f88/Ifqd/ys3z1/wBXaT/gIv8A
mjHx596/y5q/55+Q/U7/AJWb56/6u0n/AAEX/NGPjz71/lzV/wA8/Ifqd/ys3z1/1dpP+Ai/5ox8
efev8uav+efkP1O/5Wb56/6u0n/ARf8ANGPjz71/lzV/zz8h+p6X+UPmTW9bttTfVbprpoHiERYK
vEMHr9kL4ZlaaZldvUezuty54z8SXFVfpehZlPSPA/zk/wCU2l/4wQ/qOa3VfW+ee0n+NH+qGDZj
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
/mS+T//Z
+
+
+
+ proof:pdf
+ uuid:65E6390686CF11DBA6E2D887CEACB407
+ xmp.did:6da2570a-e399-4a50-9d2a-a9e576bcfbde
+ uuid:a46cd506-372d-084c-808d-7bac5b065f97
+
+ xmp.iid:5af09d4c-8b50-4d62-a7ed-db54dfca4395
+ xmp.did:5af09d4c-8b50-4d62-a7ed-db54dfca4395
+ uuid:65E6390686CF11DBA6E2D887CEACB407
+ default
+
+
+
+
+ saved
+ xmp.iid:64f9c79d-28d4-4b36-af44-34b958e5a48b
+ 2020-09-01T18:27:41+08:00
+ Adobe Illustrator 24.0 (Macintosh)
+ /
+
+
+ saved
+ xmp.iid:6da2570a-e399-4a50-9d2a-a9e576bcfbde
+ 2020-09-01T18:33:10+08:00
+ Adobe Illustrator 24.0 (Macintosh)
+ /
+
+
+
+ Web
+ AIRobin
+ Document
+ 1
+ False
+ False
+
+ 100.000039
+ 100.000000
+ Pixels
+
+
+
+ Cyan
+ Magenta
+ Yellow
+
+
+
+
+
+ 默认色板组
+ 0
+
+
+
+ 白色
+ RGB
+ PROCESS
+ 255
+ 255
+ 255
+
+
+ 黑色
+ RGB
+ PROCESS
+ 0
+ 0
+ 0
+
+
+ RGB 红
+ RGB
+ PROCESS
+ 255
+ 0
+ 0
+
+
+ RGB 黄
+ RGB
+ PROCESS
+ 255
+ 255
+ 0
+
+
+ RGB 绿
+ RGB
+ PROCESS
+ 0
+ 255
+ 0
+
+
+ RGB 青
+ RGB
+ PROCESS
+ 0
+ 255
+ 255
+
+
+ RGB 蓝
+ RGB
+ PROCESS
+ 0
+ 0
+ 255
+
+
+ RGB 洋红
+ RGB
+ PROCESS
+ 255
+ 0
+ 255
+
+
+ R=193 G=39 B=45
+ RGB
+ PROCESS
+ 193
+ 39
+ 45
+
+
+ R=237 G=28 B=36
+ RGB
+ PROCESS
+ 237
+ 28
+ 36
+
+
+ R=241 G=90 B=36
+ RGB
+ PROCESS
+ 241
+ 90
+ 36
+
+
+ R=247 G=147 B=30
+ RGB
+ PROCESS
+ 247
+ 147
+ 30
+
+
+ R=251 G=176 B=59
+ RGB
+ PROCESS
+ 251
+ 176
+ 59
+
+
+ R=252 G=238 B=33
+ RGB
+ PROCESS
+ 252
+ 238
+ 33
+
+
+ R=217 G=224 B=33
+ RGB
+ PROCESS
+ 217
+ 224
+ 33
+
+
+ R=140 G=198 B=63
+ RGB
+ PROCESS
+ 140
+ 198
+ 63
+
+
+ R=57 G=181 B=74
+ RGB
+ PROCESS
+ 57
+ 181
+ 74
+
+
+ R=0 G=146 B=69
+ RGB
+ PROCESS
+ 0
+ 146
+ 69
+
+
+ R=0 G=104 B=55
+ RGB
+ PROCESS
+ 0
+ 104
+ 55
+
+
+ R=34 G=181 B=115
+ RGB
+ PROCESS
+ 34
+ 181
+ 115
+
+
+ R=0 G=169 B=157
+ RGB
+ PROCESS
+ 0
+ 169
+ 157
+
+
+ R=41 G=171 B=226
+ RGB
+ PROCESS
+ 41
+ 171
+ 226
+
+
+ R=0 G=113 B=188
+ RGB
+ PROCESS
+ 0
+ 113
+ 188
+
+
+ R=46 G=49 B=146
+ RGB
+ PROCESS
+ 46
+ 49
+ 146
+
+
+ R=27 G=20 B=100
+ RGB
+ PROCESS
+ 27
+ 20
+ 100
+
+
+ R=102 G=45 B=145
+ RGB
+ PROCESS
+ 102
+ 45
+ 145
+
+
+ R=147 G=39 B=143
+ RGB
+ PROCESS
+ 147
+ 39
+ 143
+
+
+ R=158 G=0 B=93
+ RGB
+ PROCESS
+ 158
+ 0
+ 93
+
+
+ R=212 G=20 B=90
+ RGB
+ PROCESS
+ 212
+ 20
+ 90
+
+
+ R=237 G=30 B=121
+ RGB
+ PROCESS
+ 237
+ 30
+ 121
+
+
+ R=199 G=178 B=153
+ RGB
+ PROCESS
+ 199
+ 178
+ 153
+
+
+ R=153 G=134 B=117
+ RGB
+ PROCESS
+ 153
+ 134
+ 117
+
+
+ R=115 G=99 B=87
+ RGB
+ PROCESS
+ 115
+ 99
+ 87
+
+
+ R=83 G=71 B=65
+ RGB
+ PROCESS
+ 83
+ 71
+ 65
+
+
+ R=198 G=156 B=109
+ RGB
+ PROCESS
+ 198
+ 156
+ 109
+
+
+ R=166 G=124 B=82
+ RGB
+ PROCESS
+ 166
+ 124
+ 82
+
+
+ R=140 G=98 B=57
+ RGB
+ PROCESS
+ 140
+ 98
+ 57
+
+
+ R=117 G=76 B=36
+ RGB
+ PROCESS
+ 117
+ 76
+ 36
+
+
+ R=96 G=56 B=19
+ RGB
+ PROCESS
+ 96
+ 56
+ 19
+
+
+ R=66 G=33 B=11
+ RGB
+ PROCESS
+ 66
+ 33
+ 11
+
+
+ R=0 G=156 B=225
+ PROCESS
+ 100.000000
+ RGB
+ 0
+ 156
+ 225
+
+
+
+
+
+ 灰色
+ 1
+
+
+
+ R=0 G=0 B=0
+ RGB
+ PROCESS
+ 0
+ 0
+ 0
+
+
+ R=26 G=26 B=26
+ RGB
+ PROCESS
+ 26
+ 26
+ 26
+
+
+ R=51 G=51 B=51
+ RGB
+ PROCESS
+ 51
+ 51
+ 51
+
+
+ R=77 G=77 B=77
+ RGB
+ PROCESS
+ 77
+ 77
+ 77
+
+
+ R=102 G=102 B=102
+ RGB
+ PROCESS
+ 102
+ 102
+ 102
+
+
+ R=128 G=128 B=128
+ RGB
+ PROCESS
+ 128
+ 128
+ 128
+
+
+ R=153 G=153 B=153
+ RGB
+ PROCESS
+ 153
+ 153
+ 153
+
+
+ R=179 G=179 B=179
+ RGB
+ PROCESS
+ 179
+ 179
+ 179
+
+
+ R=204 G=204 B=204
+ RGB
+ PROCESS
+ 204
+ 204
+ 204
+
+
+ R=230 G=230 B=230
+ RGB
+ PROCESS
+ 230
+ 230
+ 230
+
+
+ R=242 G=242 B=242
+ RGB
+ PROCESS
+ 242
+ 242
+ 242
+
+
+
+
+
+ Web 颜色组
+ 1
+
+
+
+ R=63 G=169 B=245
+ RGB
+ PROCESS
+ 63
+ 169
+ 245
+
+
+ R=122 G=201 B=67
+ RGB
+ PROCESS
+ 122
+ 201
+ 67
+
+
+ R=255 G=147 B=30
+ RGB
+ PROCESS
+ 255
+ 147
+ 30
+
+
+ R=255 G=29 B=37
+ RGB
+ PROCESS
+ 255
+ 29
+ 37
+
+
+ R=255 G=123 B=172
+ RGB
+ PROCESS
+ 255
+ 123
+ 172
+
+
+ R=189 G=204 B=212
+ RGB
+ PROCESS
+ 189
+ 204
+ 212
+
+
+
+
+
+
+ Adobe PDF library 15.00
+ 21.0.0
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
endstream
endobj
3 0 obj
<>
endobj
5 0 obj
<>/Resources<>/ExtGState<>/Properties<>>>/TrimBox[0.0 0.0 100.0 100.0]/Type/Page>>
endobj
28 0 obj
<>/Resources<>/ExtGState<>/Properties<>>>/TrimBox[0.0 0.0 100.0 100.0]/Type/Page>>
endobj
29 0 obj
<>/Resources<>/ExtGState<>/Properties<>>>/TrimBox[0.0 0.0 100.0 100.0]/Type/Page>>
endobj
32 0 obj
<>/Resources<>/ExtGState<>/Properties<>>>/TrimBox[0.0 0.0 100.0 100.0]/Type/Page>>
endobj
38 0 obj
<>stream
+HlS1+qcv+qZg4*,Rb[8.EUg%ݾЛ{JB˅
+Q<{Wv/G:OvI$)CB~$4+><JxJY@ e-$Je/ªFʵ" ;l( AXDXչKU1E