Open SiennaHan opened 5 years ago
$ yarn add react-tabs-redux or $ npm install react-tabs-redux
이걸로 탭 installation하고 써야 됩니당
html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr, acronym, address, big, cite, code, del, dfn, em, img, ins, kbd, q, s, samp, small, strike, strong, sub, sup, tt, var, b, u, i, center, dl, dt, dd, ol, ul, li, fieldset, legend, table, caption, tbody, tfoot, thead, tr, th, td, article, aside, canvas, details, embed, figure, figcaption, footer, header, hgroup, menu, nav, output, ruby, section, summary, time, mark, audio, video {
margin: 0; padding: 0; border: 0; font-size: 100%; font: inherit; vertical-align: baseline; } .app{ margin: auto; padding: auto; } body { margin: 0; padding: 0; font-family: 'Noto Sans KR', sans-serif; color: #000000; background-color: #f7f7f7; } li{ list-style: none; } a:hover, a:visited, a{ text-decoration: none; }
/ heading / h1,h2,h3,h4,h5,h6{ font-family: 'Bree Serif', serif; text-align: center; line-height:60px; } .h_white{ color:#000; border-bottom:1px solid #d8d8d8; } .h_black{ background:#393939; color:#fff; border-bottom:1px solid #d8d8d8; }
/ layout / .wrap{ width:100%; max-width: 1080px; margin:0 auto }
.col3 > div{ float:left; }
.col3 > .aside{ width:24%; }
.col3 > .main{ width:48%; margin:20px 2%; }
.main{ width:48%; margin:20px auto; }
header { width: 100%; background-image: linear-gradient(to right, #de5736, #6414d2), linear-gradient(to right, #bb2929, #a83685 34%, #513ab0 66%, #1964ac); font-family: 'Noto Sans KR', sans-serif; } .logo{ width: 135px; height: 40px; margin-top: 18px; background: url("./login_logo.png") no-repeat; background-size:120px; float:left; display:inline-block; } .navBar{ text-align:right; } .navBar li { display:inline-block; list-style: none; font-size: 14px; color: #fff; } .navBar a { box-sizing: border-box; display:block; padding:20px 40px; color: #fff;
}
.clear:after{ content:''; display: block; height:0; clear:both; }
section > div{ margin:20px 0; / margin: 20px 0 20px 0 / padding-bottom : 50px; background:#fff; box-sizing: border-box; }
.content{ padding:20px; }
.Button-Field{ box-sizing: border-box; font-size: 12px; color: black; padding: 10px 0px 10px 0px; margin: 10px 60px 10px 60px;
}
.Button-Field-Side{
box-sizing: border-box; font-size: 12px; color: black; padding: 10px 0px 10px 0px; margin: 10px 0px 10px 0px; }
button{
width: 100%; height: 40px; border-radius: 21px; border: none; box-shadow: 2px 3px 7px 0 rgba(243, 138, 138, 0.5); background-color: #c13a3a; color: #fff; } *:focus { outline: none; } .button_white{ border-radius: 21px; box-shadow: 2px 4px 7px 0 rgba(244, 137, 137, 0.5); border: solid 2px #c13a3a; background-color: #fff; color: #c13a3a; } select { width: 100%; height: 30px; padding: 2px 8px; font-size: 12px; color: #999; border: 1px solid #ddd; background: url("./selector.png") no-repeat right 6px center; apperance: none; -webkit-appearance: none; } select::-ms-expand {display:none;}
/tab/
width: 100%;
display: inline-block;
margin-right: 30px;
vertical-align: top;
}
margin: 0;
padding: 0;
box-shadow: 0px;
border: 1px solid #ccc;
border-bottom: 1px solid #868686;
height: 35px;
position: relative;
top: 5px;
}
width: 20%;
border-radius: 2px;
font-size: 13px;
margin: 2px 10px;
cursor: pointer;
border: 1px solid transparent;
color: #666;
box-shadow: none;
background: transparent;
display: inline;
position: relative;
height: 40px;
line-height: 41px;
top: -4px;
left: -1px;
outline: none;
}
color: #666464;
}
color: #c13a3a;
border: 1px solid transparent;
border-bottom: 1px solid white;
background-color: transparent;
text-decoration: underline;
}
padding: 15px;
border: 1px solid #868686;
border-top: 1px solid transparent;
}
}
/ Login Page css /
.Text-Field{
line-height: 20px; font-size: 10px; color: black; margin: 0px 60px; padding: 10px 0px;
}
.label{ display:inline-block; width: 100%; margin: 20px 0px; padding: 20px; background-color: rgba(237,156,156,0.5); font-size: 10px; }
input{ box-sizing: border-box; display:block; width: 100%; height: 40px; border-radius: 21px; border: solid 2px #e0e0e0; margin: 0px; padding: 0px 15px; font-size: 16px; background-color: #fff }
input:focus{ border: solid 2px #C13A3A; outline: none; background-color: #fff }
/ SignUp Page css /
border: none; width: 135px; height: 19px; position: absolute; top: 20px; left: 30px; background: url("./login_logo.png") no-repeat; background-size: contain; } / 프로필 사진 크기 조정 / .PROFILEIMG{ width: 50px; }
width: 200px; }
/ 일반적인 버튼 및 textarea등 태그에 대한 css / .bordered_textarea{ box-sizing: border-box; color: #999999; display:block; width: 100%; height: 40px; border-radius: 21px; border: solid 2px #e0e0e0; margin: 0px; padding: 6px 15px; font-size: 16px; }
hr{ color: rgb(237,30,39); }
img{ width: 700px; }
*{ margin: 0; padding: 0; }
.clear_fix{ overflow: auto; }
.reset_save a{ font-size: 12px; color: #ffff00;
}
/ ==================== [3] t-shirt design area ==================== /
/ ==================== [4] right help area ==================== / .help_area{ display: inline-block; background-color: #f1f1f1; padding: 6px 20px; text-align: center; position: absolute; top: 100px; right: 20px; }
.help_window_wrapper{ position: relative; }
.help_window{ visibility: hidden; padding: 9px 14px; position: absolute; top: -8px; right: 45px; width: 250px; background-color: #fff; border: 1px solid #CCCCCC; border-radius: 4px; box-shadow: 0 1px 9px 1px #CCCCCC; }
.help_window::before{ content: ""; position: absolute; top: 22px; right: -20px; margin-left: -5px; border-width: 10px; border-style: solid; border-color: transparent transparent transparent #555; }
.help_window p{ text-align: left; }
.help_window p span{ font-weight: bold; margin-left: 5px; }
/ ==================== [5] left design tool ==================== / .design_tool{ position: absolute; display: inline-block; top: 100px; left: 20px; border: 1px solid #CCCCCC; border-radius: 4px; box-shadow: 0 1px 9px 1px #CCCCCC; }
.design_tool ul{ list-style-type: none; margin: 0; padding: 0; width: 200px; background-color: #f1f1f1; }
.design_tool ul li { border-bottom: 1px solid #CCCCCC; }
.design_tool ul li:last-child { border-bottom: none; }
.design_tool ul li a { display: block; color: #000; padding: 8px 16px; text-decoration: none; }
.design_tool ul li a span{ margin-left: 16px; }
.design_tool .add_text{ position: relative; }
.text_tool_window{ padding: 9px 14px; visibility: hidden; position: absolute; top: -38px; left: 45px; width: 250px; background-color: #fff; border: 1px solid #CCCCCC; border-radius: 4px; box-shadow: 0 1px 9px 1px #CCCCCC; }
/ Popup arrow / .text_tool_window::before { content: ""; position: absolute; top: 45px; left: -15px; margin-left: -5px; border-width: 10px; border-style: solid; border-color: transparent #555 transparent transparent; }
/ Toggle this class - hide and show the popup / .help_area .show, .add_text .show { visibility: visible; -webkit-animation: fadeIn 1s; animation: fadeIn 1s; }
.help_area a, .help_area a:hover{ color: #000; text-decoration: none; }
/ animation (fade in the popup) / @-webkit-keyframes fadeIn { from {opacity: 0;} to {opacity: 1;} }
@Keyframes fadeIn { from {opacity: 0;} to {opacity:1 ;} }
.help_window .header, .text_tool_window .header{ border-bottom: 1px solid #9E9E9E; padding-bottom: 5px; } .help_window .title, .text_tool_window .title{ float: left; }
.help_window #close_help_window, .text_tool_window #close_window{ float: right; }
.text_tool_window .font_area{ float: left; }
.text_tool_window .color_area{ float: right; }
.font_area p, .color_area p, .font_style p{ margin-bottom: 10px; }
.font_style, .font_size { margin-top: 25px; }
.text_tool_window #text_area{ overflow: hidden; height: 50px; width: 100%; padding: 5px; border-radius: 3px; border: 1px solid #BDBDBD; }
.text_tool_window select{ padding: 5px; background: none repeat scroll 0 0 #FFFFFF; border: 1px solid #9C9D98; border-radius: 3px; box-shadow: 0 2px 2px #CFD0CB inset, 0 2px 2px rgba(0, 0, 0, 0.2); }
.color_area input{ width: 20px; height: 30px; background: #fff; padding: 2px; border-radius: 3px; }
/ ==================== [6] front-back button section ==================== / .change_side{ text-align: center; margin-top: 20px; }
.change_side button{ background-color: #e7e7e7; color: black; border: 0; padding: 16px 32px; margin: 4px 2px; }
/ ==================== [7] product modal ==================== / .product_details{ float: left; width: 270px; text-align: center; }
.product_details img{ width: 250px; }
/ ==================== [8] preview image modal ==================== /
text-align: center; }
margin-bottom: 20px; }
margin: 0 auto; }
margin-top: 40px; margin-bottom: 40px; }
text-decoration: none; color: #fff; background-color: #EF6C00; padding: 11px 16px; border-radius: 3px; }
margin-left: 5px; }
/ styles for tab browsers smaller than 800px; / @media (max-width: 800px){ .design_tool ul li a span{ display: none; } .design_tool ul{ width: auto; } }
/ Canvas /
canvas{
padding-left: 0; padding-right: 0; margin-left: auto; margin-right: auto; display: block; } .canvas-container{
padding-left: 0; padding-right: 0; margin-left: auto; margin-right: auto; display: block;
} .canvas-bg{
width: 430px; height: 460px; padding-left: 0; padding-right: 0; margin-left: auto; margin-right: auto; background: url('./images/bg-canvas.png'); background-repeat: repeat; border:1px solid #ececec;
}
canvas #front-canvas{ }
canvas #back-canvas{ }
.circle-picker{ margin-left: auto; margin-right: auto; }
/grouppage/
.SearchingBar { margin-bottom: 20px; }
.searching_input{ box-sizing: border-box; display:inline-block; width: 67%; height: 40px; border-radius: 21px; border: solid 2px #e0e0e0; margin: 0px; padding: 0px 15px; font-size: 16px; background-color: #fff }
.searching_button{ box-sizing: border-box; display:inline-block; width: 30%; height: 40px; padding: 0px 15px; margin-left: 10px;
}
.Group-List-Field{ box-sizing: border-box; color: black; }
.Group-List-Field > ul { padding: 0px; margin: 20px 0px; }
.Group-Button-Field{ box-sizing: border-box; font-size: 12px; color: black; margin: 10px 0px 10px 0px;
} .button_small{ width: 30%; height: 24px; display: inline-block; float: right; border: 1px solid #FFC50A; border-radius: 4px; box-shadow: 2px 3px 7px 0 rgba(148, 60, 208, 0.1); background-color: #FFB73B; color: #fff; }
.button_admin{ width: 100%; height: 20px; display: inline-block; float: right; border: 1px solid #dcdcdc; border-radius: 4px; box-shadow: 1px 2px 3px 0 rgba(160, 160, 160, 0.1); background-color: #fff; color: #ababab; }
/frontend/src/css 폴더 안에 넣으시면됩니다
/tab/
width: 100%;
display: inline-block;
margin-right: 30px;
vertical-align: top;
}
margin: 0;
padding: 0;
box-shadow: 0px;
border: 1px solid #ccc;
border-bottom: 1px solid #868686;
height: 35px;
position: relative;
top: 5px;
}
width: 20%;
border-radius: 2px;
font-size: 13px;
margin: 2px 10px;
cursor: pointer;
border: 1px solid transparent;
color: #666;
box-shadow: none;
background: transparent;
display: inline;
position: relative;
height: 40px;
line-height: 41px;
top: -4px;
left: -1px;
outline: none;
}
color: #666464;
}
color: #c13a3a;
border: 1px solid transparent;
border-bottom: 1px solid white;
background-color: transparent;
text-decoration: underline;
}
padding: 15px;
border: 1px solid #868686;
border-top: 1px solid transparent;
}
}
import React from 'react';
import {connect} from 'react-redux';
import {fabric} from 'fabric';
import {CirclePicker, SketchPicker} from 'react-color';
import {Tabs, TabContent, TabLink} from 'react-tabs-redux';
//import ThreeScene from './ThreeScene';
//import FabricCanvas from './FabricCanvas';
import MyGroupList from '../GroupPage/MyGroupList';
//import ImageUploader from 'react-images-upload';
import { toSaveDesign, toNewDesign } from '../../actions/index.js';
//import { tsImportEqualsDeclaration } from '@babel/types';
//import logo from './images/templates/templatelist';
class DesignPage extends React.Component {
constructor(props){
console.log("DesignPage - constructor")
super(props);
this.state = {
design : {
body: this.props.now_design.design.body,
sleeve: this.props.now_design.design.sleeve,
banding: this.props.now_design.design.banding,
stripe: this.props.now_design.design.stripe,
button: this.props.now_design.design.button
},
text: {
frontchest: this.props.now_design.text.frontchest,
rightarm: this.props.now_design.text.rightarm,
upperback: this.props.now_design.text.upperback,
middleback: this.props.now_design.text.middleback,
lowerback: this.props.now_design.text.lowerback,
},
logo : {
front: {
src : '',
left: 250,
top: 110,
},
arm_right: {
src : '',
left: 50,
top: 120,
},
arm_left: {
src : '',
left: 155,
top: 120,
},
back: {
src : '',
left: 135,
top: 125,
},
},
image: {
front: this.props.now_design.image.front,
back: this.props.now_design.image.back,
},
element: null,
designClickedWhat: "body",
textClickedWhat: null,
logoClickedWhat: null,
displayTextColor: false,
displayBorderColor: false,
};
this.handleElementChange = this.handleElementChange.bind(this);
this.handleDesignChange = this.handleDesignChange.bind(this);
this.handleTextChange = this.handleTextChange.bind(this);
this.handleTextColorChange = this.handleTextColorChange.bind(this);
this.handleStrokeColorChange = this.handleStrokeColorChange.bind(this);
this.handleLogoChange = this.handleLogoChange.bind(this);
this.designElementToImage = this.designElementToImage.bind(this);
this.textElementToImage = this.textElementToImage.bind(this);
this.logoElementToImage = this.logoElementToImage.bind(this);
this.updateFrontCanvas = this.updateFrontCanvas.bind(this);
this.updateBackCanvas = this.updateBackCanvas.bind(this);
this.clickedDesignPopButton = this.clickedDesignPopButton.bind(this);
this.clickedTextPopButton = this.clickedTextPopButton.bind(this);
this.clickedLogoPopButton = this.clickedLogoPopButton.bind(this);
this.moveHandler = this.moveHandler.bind(this);
this.onClickSave = this.onClickSave.bind(this);
this.getDataUrl = this.getDataUrl.bind(this);
}
componentWillMount() {
console.log("DesignPage - componentWillMount")
this.design_color = {
body: ["#f29c9f", "#fff45c", "#80c269", "#00b7ee", "#aa89bd", "#910000", "#f39800",
"#097c25", "#0075a9", "#601986", "#580b0b", "#cfcfcf", "#626262", "#001c58", "#232323"],
sleeve: ["#f29c9f", "#fff45c", "#80c269", "#00b7ee", "#aa89bd", "#910000", "#f39800",
"#097c25", "#0075a9", "#601986", "#580b0b", "#fcfcfc", "#626262", "#001c58", "#232323"],
stripe: ["#f29c9f", "#fff45c", "#80c269", "#00b7ee", "#aa89bd", "#910000", "#f39800",
"#097c25", "#0075a9", "#601986", "#580b0b", "#fcfcfc", "#cfcfcf", "#001c58", "#232323"],
banding: ["#f29c9f", "#fff45c", "#80c269", "#00b7ee", "#aa89bd", "#910000", "#f39800",
"#097c25", "#0075a9", "#601986", "#580b0b", "#cfcfcf", "#626262", "#001c58", "#232323"],
button: ["#f29c9f", "#fff45c", "#80c269", "#00b7ee", "#aa89bd", "#910000", "#f39800",
"#097c25", "#0075a9", "#601986", "#580b0b", "#fcfcfc", "#cfcfcf", "#001c58", "#232323"]
}
this.design_element = ["body", "sleeve", "stripe", "banding", "button"]
this.text_element = ["frontchest", "rightarm", "upperback", "middleback", "lowerback"]
this.logo_element = ["front", "arm_right", "arm_left", "back"]
}
componentDidMount() {
console.log("DesignPage - componentDidMount")
this.the_front_canvas = new fabric.Canvas('front-canvas', {
preserveObjectStacking: true,
height:460,
width:430,
});
this.the_back_canvas = new fabric.Canvas('back-canvas', {
preserveObjectStacking: true,
height:460,
width:430,
});
this.the_front_canvas.on({
'object:moved': this.moveHandler,
})
this.the_back_canvas.on({
'object:moved': this.moveHandler,
})
this.the_front_canvas.add(this.designElementToImage(this.state.design.body, "front_body", 0))
this.the_back_canvas.add(this.designElementToImage(this.state.design.body, "back_body", 0))
this.the_front_canvas.add(this.designElementToImage(this.state.design.sleeve, "front_sleeve", 0))
this.the_back_canvas.add(this.designElementToImage(this.state.design.sleeve, "back_sleeve", 0))
this.the_front_canvas.add(this.designElementToImage(this.state.design.banding, "front_banding", 0))
this.the_back_canvas.add(this.designElementToImage(this.state.design.banding, "back_banding", 0))
this.the_front_canvas.add(this.designElementToImage(this.state.design.stripe, "front_stripe", 3))
this.the_back_canvas.add(this.designElementToImage(this.state.design.stripe, "back_stripe", 3))
this.the_front_canvas.add(this.designElementToImage(this.state.design.button, "front_button", 3))
this.the_front_canvas.add(this.textElementToImage(this.state.text.frontchest, "frontchest"))
this.the_front_canvas.add(this.textElementToImage(this.state.text.rightarm, "rightarm"))
this.the_back_canvas.add(this.textElementToImage(this.state.text.upperback, "upperback"))
this.the_back_canvas.add(this.textElementToImage(this.state.text.middleback, "middleback"))
this.the_back_canvas.add(this.textElementToImage(this.state.text.lowerback, "lowerback"))
this.the_front_canvas.add(this.logoElementToImage(this.state.logo.front, "front"))
this.the_front_canvas.add(this.logoElementToImage(this.state.logo.arm_right, "arm_right"))
this.the_back_canvas.add(this.logoElementToImage(this.state.logo.arm_left, "arm_left"))
this.the_back_canvas.add(this.logoElementToImage(this.state.logo.back, "back"))
}
componentWillUpdate = (nextProps, nextState) => {
console.log("DesignPage - componentWillUpdate nextState: ", nextState)
// If Updated Item is not the same as the old one
// => Update the canvas with newer item
//update for design element
for(let element of this.design_element){
if(nextState.design[element] !== this.state.design[element]) {
if(element === "stripe" ) {
this.updateFrontCanvas(this.designElementToImage(nextState.design[element], 'front_'+element, 3))
this.updateBackCanvas(this.designElementToImage(nextState.design[element], 'back_'+element, 3))
}
else if (element === "button") {
this.updateFrontCanvas(this.designElementToImage(nextState.design[element], 'front_'+element, 3))
}
else {
this.updateFrontCanvas(this.designElementToImage(nextState.design[element], 'front_'+element, 0))
this.updateBackCanvas(this.designElementToImage(nextState.design[element], 'back_'+element, 0))
}
}
}
//update for text element
for(let element of this.text_element){
if(nextState.text[element] !== this.state.text[element]) {
if(element === "frontchest" || element === "rightarm") {
this.updateFrontCanvas(this.textElementToImage(nextState.text[element], element))
}
else {
this.updateBackCanvas(this.textElementToImage(nextState.text[element], element))
}
}
}
//update for logo element
for (let element of this.logo_element) {
if(nextState.logo[element] !== this.state.logo[element]) {
if(element === "front" || element === "arm_right"|| element === "arm_left") {
this.updateFrontCanvas(this.logoElementToImage(nextState.logo[element], element))
}
else {
this.updateBackCanvas(this.logoElementToImage(nextState.logo[element], element))
}
}
}
this.the_front_canvas.renderAll();
this.the_back_canvas.renderAll();
}
// componentDidUpdate(nextProps, nextState) {
// this.the_front_canvas.renderAll();
// this.the_back_canvas.renderAll();
// }
handleElementChange(e){
console.log("DesignPage - handleElementChange target: ", e.target)
let id = e.target.id;
let value = e.target.value;
if(id === "design_element") {
this.setState({designClickedWhat: value});
}
else if(id === "text_element") {
this.setState({textClickedWhat: value});
}
else if (id == "logo_element") {
this.setState({logoClickedWhat: value});
}
}
handleDesignChange(color) {
console.log("DesignPage - handleDesignChange")
let design_element = document.getElementById("design_element").value;
this.setState({design : ({...this.state.design, [design_element]: color.hex})})
}
handleTextChange(e) {
let text_element = document.getElementById("text_element").value;
console.log("DesignPage - handleTextChange e.target: ", e.target, " text element: ", text_element)
this.setState({text : ({...this.state.text,
[text_element]: ({...this.state.text[text_element], [e.target.name]:e.target.value})
})});
}
handleTextColorChange(color) {
let text_element = document.getElementById("text_element").value;
console.log("DesignPage - handleTextColorChange", color)
this.setState({text : ({...this.state.text,
[text_element]: ({...this.state.text[text_element], fill: color.hex})
})});
}
handleStrokeColorChange(color) {
let text_element = document.getElementById("text_element").value;
console.log("DesignPage - handleTextColorChange", color)
this.setState({text : ({...this.state.text,
[text_element]: ({...this.state.text[text_element], stroke: color.hex})
})});
}
// componentDidUpdate(nextProps, nextState) {
// this.the_front_canvas.renderAll();
// this.the_back_canvas.renderAll();
// }
handleLogoChange = (e) => {
e.preventDefault();
let logo_element = document.getElementById("logo_element").value;
const scope = this;
//var img = document.createElement("img");
var file = document.getElementById('input').files[0];
let reader = new FileReader();
reader.addEventListener("load", function() {
//var image= new Image();
//image.src = reader.result;
console.log(reader.result);
scope.setState({logo: ({...scope.state.logo,
[logo_element]: ({...scope.state.logo[logo_element], src :reader.result})
}) });
})
if (file) {
reader.readAsDataURL(file);
}
}
getDataUrl = (img) => {
var canvas = document.createElement('canvas')
var ctx = canvas.getContext('2d')
canvas.width = img.width
canvas.height = img.height
ctx.drawImage(img, 0, 0)
// If the image is not png, the format
// must be specified here
return canvas.toDataURL()
}
designElementToImage(color, type, z_Index) {
console.log("DesignPage - designElementToImage - color: ", color, "type: ", type)
var imgElement = document.createElement("img");
const scope = this
imgElement.addEventListener('load', function(event){
var dataUrl = scope.getDataUrl(event.currentTarget)
var img = document.createElement("img");
img.src = dataUrl;
var imgInstance = new fabric.Image(img, {
width: 430,
height: 460,
the_type: type ,
zIndex: z_Index
});
scope.setState({element: imgInstance});
})
var src = './images/templates/' + type + '/' + type + color.substring(1)+'.png';
// console.log("src: ", src)
imgElement.setAttribute("src", require(src));
// console.log(this.state.element);
var imgInstance = new fabric.Image(imgElement, {
width: 430,
height: 460,
the_type: type ,
zIndex: z_Index
});
return imgInstance;
}
textElementToImage(text, type) {
console.log("DesignPage - textElementToImage", text, type)
// const scope = this
// imgElement.addEventListener('load', function(event){
// var dataUrl = scope.getDataUrl(event.currentTarget)
// var img = document.createElement("img");
// img.src = dataUrl;
// var imgInstance = new fabric.Image(img, {
// width: 430,
// height: 460,
// the_type: type ,
// zIndex: z_Index
// });
// scope.setState({element: imgInstance});
// })
let imgInstance = new fabric.IText(text.textvalue, {
fontFamily: text.fontFamily,
fill: text.fill,
fontStyle: text.fontStyle,
fontSize: text.fontSize,
stroke: text.stroke,
strokeWidth: text.strokeWidth,
the_type: type,
zIndex: 10,
left: text.left,
top: text.top,
textAlign: "center"
})
// console.log("text imgInstance: ", imgInstance)
return imgInstance;
}
logoElementToImage(logo, type) {
console.log("FabricCanvas - textElementToImage")
//let img = document.createElement("img");
//img.setAttribute("src", require(logo.src));
let img = new Image();
img.src = logo.src;
let imgInstance;
imgInstance = new fabric.Image(img, {
width: 899,
height:959,
the_type: type,
zIndex: 10,
left: logo.left,
top: logo.top
});
imgInstance.set({
scaleY: 0.1,
scaleX: 0.1,
originX: "center",
originY: "center"
});
return imgInstance;
}
updateFrontCanvas = (next) => {
console.log("DesignPage - updateFrontCanvas next: ", next)
if(next){
let to_remove;
// Find the same kind of element
this.the_front_canvas.forEachObject( (object) => {
if(object.the_type === next.the_type){
console.log("obcject.the_type: ", object.the_type, " next.the_type: ", next.the_type)
to_remove = object;
this.the_front_canvas.remove(to_remove);
}
} );
this.the_front_canvas.add(next);
this.the_front_canvas.moveTo(next, next.zIndex);
this.the_front_canvas.renderAll();
}
}
updateBackCanvas = (next) => {
console.log("DesignPage - updateBackCanvas next: ", next)
if(next){
let to_remove;
// Find the same kind of element
this.the_back_canvas.forEachObject( (object) => {
if(object.the_type === next.the_type){
to_remove = object;
this.the_back_canvas.remove(to_remove);
}
} );
this.the_back_canvas.add(next);
this.the_back_canvas.moveTo(next, next.zIndex);
this.the_back_canvas.renderAll();
}
}
clickedDesignPopButton = () => {
console.log("clicked", this.state.designClickedWhat)
this.state.designClickedWhat
? this.setState({designClickedWhat: null})
: this.setState({designClickedWhat: "body"});
}
clickedTextPopButton = () => {
this.state.textClickedWhat
? this.setState({textClickedWhat: null})
: this.setState({textClickedWhat: "frontchest"});
}
clickedLogoPopButton = () => {
this.state.logoClickedWhat
? this.setState({logoClickedWhat: null})
: this.setState({logoClickedWhat: "front"});
}
moveHandler = (e) =>{
let movingObject = e.target;
console.log(movingObject)
console.log("left: ", movingObject.get('left'), " top: ", movingObject.get('top'))
this.setState({text : ({...this.state.text,
[movingObject.the_type]: ({...this.state.text[movingObject.the_type],
left:movingObject.get('left'),
top: movingObject.get('top')})
})});
}
saveToCanvas = () => {
console.log("DesignPage - saveToCanvas")
let link = document.createElement("a");
link.href = this.the_canvas.toDataURL({format: 'png'});
link.download = "design.png";
link.click();
}
fileChangedHandler = (event) => {
const file = event.target.files[0];
this.setState({selectedFile: file});
}
onClickSave = () => {
console.log("clickSave")
let image = {
front: this.the_front_canvas.toDataURL({format:'png'}),
back: this.the_back_canvas.toDataURL({format: 'png'})
}
this.setState({image: image})
this.props.onSave(this.props.now_design.id, this.state.design, this.state.text, image)
}
render() {
console.log("DesignPage - render state: ", this.state)
const designClickedWhat = this.state.designClickedWhat;
const textClickedWhat = this.state.textClickedWhat;
const logoClickedWhat = this.state.logoClickedWhat;
let colorPicker;
let textPicker;
let logoPicker;
const popover = {
position: 'absolute',
zIndex: '2',
}
const cover = {
position: 'fixed',
top: '0px',
right: '0px',
bottom: '0px',
left: '0px',
}
colorPicker = designClickedWhat
? <center>
<select id="design_element" onChange={(e)=>this.handleElementChange(e)}>
<option value = "body">body</option>
<option value = "sleeve">sleeve</option>
<option value = "banding">banding</option>
<option value = "stripe">stripe</option>
<option value = "button">button</option>
</select>
<br/><br/>
<CirclePicker width="220" id="design_colour"
onChangeComplete={this.handleDesignChange} colors={this.design_color[designClickedWhat]}/>
<br/>
</center>
: <div />
textPicker = textClickedWhat
? <center>
<select id="text_element" onChange={(e)=>this.handleElementChange(e)}>
<option value="frontchest">Front Chest</option>
<option value="rightarm">Right Arm</option>
<option value="upperback">Upper Back</option>
<option value="middleback">Middle Back</option>
<option value="lowerback">Lower Back</option>
</select>
<textarea id="text_area" placeholder={this.state.text[this.state.textClickedWhat].textvalue}
name="textvalue" onChange={(e)=>this.handleTextChange(e)}/>
<p>Font</p>
<select id="text_font" name="fontFamily" onChange={(e)=>this.handleTextChange(e)}>
<option>arial</option>
<option>tahoma</option>
<option>times new roman</option>
<option>anton</option>
<option>Akronim</option>
<option>Alex Brush</option>
<option>Aguafina Script</option>
<option>mistral</option>
</select>
<p>Style</p>
<select id="text_style" name="fontStyle" onChange={(e)=>this.handleTextChange(e)}>
<option>normal</option>
<option>italic</option>
<option>oblique</option>
<option>bold</option>
</select>
<p>Size</p>
<input type="range" min="0" max="200" defaultValue="100" id="text_size"
name="fontSize" onChange={(e)=>this.handleTextChange(e)}/>
<p>Color</p>
<div onClick={()=>{this.setState({displayTextColor: !this.state.displayTextColor})}}>
<button>pick color</button>
</div>
{ this.state.displayTextColor ? <div style={popover}> <div style={cover} onClick={()=>{this.setState({displayTextColor: false})}}/>
<SketchPicker color={ this.state.text[document.getElementById("text_element").value].fill } onChange={this.handleTextColorChange} />
</div> : null }
<p>Border</p>
<input type="range" min="0" max="10" defaultValue="2" id="stroke_width"
name="strokeWidth" onChange={(e)=>this.handleTextChange(e)}/>
<div onClick={()=>{this.setState({displayBorderColor: !this.state.displayBorderColor})}}>
<button>pick color</button>
</div>
{ this.state.displayBorderColor ? <div style={popover}> <div style={cover} onClick={()=>{this.setState({displayBorderColor: false})}}/>
<SketchPicker color={ this.state.text[document.getElementById("text_element").value].stroke } onChange={this.handleStrokeColorChange} />
</div> : null }
</center>
: <div/>
logoPicker = logoClickedWhat
? <center>
<select id="logo_element" onChange={(e)=>this.handleElementChange(e)}>
<option value="front">Front Chest</option>
<option value="arm_right">Right Arm</option>
<option value="arm_left">Left Arm </option>
<option value="back">Lower Back</option>
</select>
<input type = "file" id = "input" onChange = {this.handleLogoChange} />
</center>
: <div/>
return (
<section className="wrap clear col3">
{/*<!--========================================
LEFT SIDE BAR
=========================================-->*/}
<div className="aside">
<h2 className="h_white">SELECT STYLE</h2>
<div className="content">
{/*<!--========================================
Design section
=========================================-->*/}
<h1>Design</h1>
<button onClick={this.clickedDesignPopButton}>pop</button>
{colorPicker}
{/*<!--========================================
Text section
=========================================-->*/}
<h1>Text</h1>
<button onClick={this.clickedTextPopButton}>pop</button>
{textPicker}
{/*<!--========================================
Image Upload Section
=========================================-->*/}
<h1>Logo</h1>
<button onClick={this.clickedLogoPopButton}>pop</button>
{logoPicker}
</div>
</div>
{/*<!--========================================
CENTER DESIGN SECTION
=========================================-->*/}
<div className="main">
<h2 className="h_white">SAMPLE VIEW</h2>
<div className="content">
{/*<!--========================================
Fabric Canvas Section
=========================================-->*/}
{/*<ThreeScene/>*/}
<div className= "main-canvas-container">
<button class="front_btn" type="button">Front</button>
<canvas id='front-canvas'/>
<button class="back_btn" type="button">Back</button>
<canvas id='back-canvas'/>
</div>
{/*<!--========================================
NEW & SAVE Button Section
=========================================-->*/}
{this.props.isLoggedIn ?
(<div>
<button className="new_btn" type="button" onClick={() => this.props.onNew()}>NEW</button>
{/* <button className="save_btn" type="button" onClick={() => this.props.onSave(this.props.now_design.id, this.state.design, this.state.text)}>SAVE</button> */}
<button className="save_btn" type="button" onClick={() => this.onClickSave()}>SAVE</button>
</div>)
: <div>
<button className="new_btn" type="button" onClick={() => this.props.onNew()}>NEW</button>
</div>
}
</div>
</div>
{/*<!--========================================
RIGHT SIDE BAR
=========================================-->*/}
<div className="aside">
<h2 className="h_black">MY GROUP</h2>
<div className="content">
{this.props.isLoggedIn? <MyGroupList /> : <p>로그인을 해주세요</p>}
</div>
</div>
</section>
);
}
}
const mapStateToProps = (state) => ({
isLoggedIn: state.authorization,
now_design: state.now_design,
my_groups: state.my_groups,
})
const mapDispatchToProps = (dispatch) => ({
onNew: () => dispatch(toNewDesign()),
onSave: (designid, design, text, image) => dispatch(toSaveDesign(designid, design, text, image)),
})
export default connect (mapStateToProps, mapDispatchToProps)(DesignPage)
<!!!!!link href="https://fonts.googleapis.com/css?family=Bree+Serif|Noto+Sans+KR|Alfa+Slab+One|Damion|Teko:500&display=swap" rel="stylesheet">
import React from 'react'; import {connect} from 'react-redux' import {fabric} from 'fabric'; import {CirclePicker} from 'react-color';
import { Tabs, TabContent, TabLink } from 'react-tabs-redux'; //import ThreeScene from './ThreeScene';
import FabricCanvas from './FabricCanvas'; import MyGroupList from '../GroupPage/MyGroupList';
//import ImageUploader from 'react-images-upload';
import { toSaveDesign, toNewDesign } from '../../actions/index.js';
//import ThreeScene from './ThreeScene';
//the templates are imported as images and passed as porps to the TemplateList components. //if the user chooses any of the properties, the state gets updated in the DesignPage component //it gets passed onto FabricCanvas as prop //FabricCanvas uses lifecycle method ComponentWillReceiveProps() to update the canvas //about saveToCanvas: use a method from fabric named TODataUrl()
// export default class DesignPage extends React.Component { class DesignPage extends React.Component { constructor(props){ console.log("DesignPage - constructor") super(props);
})();
}
const mapStateToProps = (state) => ({ isLoggedIn: state.authorization, now_design: state.now_design, my_groups: state.my_groups, })
const mapDispatchToProps = (dispatch) => ({ onNew: () => dispatch(toNewDesign()), onSave: (designid, design, text) => dispatch(toSaveDesign(designid, design, text)), })
export default connect (mapStateToProps, mapDispatchToProps)(DesignPage)