다음 샘플은 그림 셰이프를 추가하는 방법을 보여 줍니다.
다음 예시는 그림의 투명도를 변경하는 방법을 보여 줍니다.
다음 예시는 그림을 자르는 방법을 보여 줍니다.
다음 예시는 그림의 밝기 및 대비를 설정하는 방법을 보여 줍니다.
다음 예시는 그림을 다시 칠하는 방법을 보여 줍니다.
다음 예시는 그림의 기하 도형 유형을 변경하는 방법을 보여 줍니다.
다음 예시는 그림의 테두리를 설정하는 방법을 보여 줍니다.
그림의 일부가 투명하면 그림의 배경색을 설정할 수 있습니다.
<template>
<div class="sample-tutorial">
<gc-spread-sheets class="sample-spreadsheets" @workbookInitialized="initSpread">
</gc-spread-sheets>
<div class="options-container" v-if="active">
<h3>Picture Format</h3>
<div class="option-row">
<label>Image</label><input class="img" type="file" accept="image/*" @change="onImgChange"></input>
</div>
<div class="option-row">
<label>Geometry Type</label>
<select class="geometry-type" v-model="geometryType" @change="onGeometryTypeChange">
<option value="1">rectangle</option>
<option value="5">rounded rectangle</option>
<option value="9">oval</option>
<option value="21">heart</option>
<option value="178">cloud</option>
</select>
</div>
<div class="option-row">
<label>Transparency</label><input class="transparency" type="number" min="0" max="1" step="0.01"
v-model.number="transparency" @change="onPictureFormatPropValueChange"></input>
</div>
<div class="divide-line"></div>
<details>
<summary>Crop</summary>
<div class="option-row">
<label>Crop Left</label><input class="crop-left" type="number" max="0.99" step="0.01"
v-model.number="crop.left" @change="onPictureFormatPropValueChange"></input>
</div>
<div class="option-row">
<label>Crop Right</label><input class="crop-right" type="number" max="0.99" step="0.01"
v-model.number="crop.right" @change="onPictureFormatPropValueChange"></input>
</div>
<div class="option-row">
<label>Crop Top</label><input class="crop-top" type="number" max="0.99" step="0.01" v-model.number="crop.top"
@change="onPictureFormatPropValueChange"></input>
</div>
<div class="option-row">
<label>Crop Bottom</label><input class="crop-bottom" type="number" max="0.99" step="0.01"
v-model.number="crop.bottom" @change="onPictureFormatPropValueChange"></input>
</div>
</details>
<div class="divide-line"></div>
<details>
<summary>Brightness/Contrast</summary>
<div class="option-row">
<label>Brightness</label><input class="brightness" type="number" min="-1" max="1" step="0.01"
v-model.number="brightness" @change="onPictureFormatPropValueChange"></input>
</div>
<div class="option-row">
<label>Contrast</label><input class="contrast" type="number" min="-1" max="1" step="0.01"
v-model.number="contrast" @change="onPictureFormatPropValueChange"></input>
</div>
</details>
<div class="divide-line"></div>
<details>
<summary>Recolor</summary>
<div class="option-row">
<label>Recolor Type</label>
<select class="recolor-type" v-model="recolorType" @change="onRecolorTypeChange">
<option value="none">none</option>
<option value="gray-scale">grayscale</option>
<option value="black-and-white">blackAndWhite</option>
<option value="duotone">duotone</option>
</select>
</div>
<div class="option-row">
<label>BlackAndWhite</label><input class="black-and-white" type="number" min="0" max="1" step="0.05"
v-model.number="blackAndWhite" @change="onBlackAndWhiteChange"></input>
</div>
<div class="option-row" v-if="isShow">
<label>Duotone Color1</label><input class="duotone-color1" type="color" v-model="duotoneColor1"
@change="onDuotoneColorChange"></input>
</div>
<div class="option-row" v-if="isShow">
<label>Duotone Color2</label><input class="duotone-color2" type="color" v-model="duotoneColor2"
@change="onDuotoneColorChange"></input>
</div>
</details>
<div class="divide-line"></div>
<details>
<summary>Style</summary>
<div class="option-row">
<label>Background Color</label><input class="back-color" type="color" v-model="backColor"
@change="onBackColorChange"></input>
</div>
<div class="option-row">
<label>Border Color</label><input class="border-color" type="color" v-model="borderColor"
@change="onBorderColorChange"></input>
</div>
<div class="option-row">
<label>Border Width</label><input class="border-width" type="number" min="-1" step="1"
v-model.number="borderWidth" @change="onBorderWidthChange"></input>
</div>
</details>
</div>
<div class="options-container" v-if="!active">Try to select a shape.</div>
</div>
</template>
<script setup>
import GC from "@mescius/spread-sheets";
import { ref } from "vue";
import "@mescius/spread-sheets-vue";
import '@mescius/spread-sheets-shapes';
import '@mescius/spread-sheets-resources-ko';
GC.Spread.Common.CultureManager.culture("ko-kr");
const spreadRef = ref(null);
const sheetRef = ref(null);
let active = ref(false);
let isShow = ref(false);
let geometryType = ref(1);
let crop = ref({
left: 0,
right: 0,
top: 0,
bottom: 0
});
let transparency = ref(0);
let brightness = ref(0);
let contrast = ref(0);
let recolorType = ref('none');
let blackAndWhite = ref(0.5);
let duotoneColor1 = ref('#000000');
let duotoneColor2 = ref('#FF0000');
let backColor = ref('');
let borderColor = ref('');
let borderWidth = ref(-1);
let isSyncState = ref(false);
const initSpread = (spread) => {
spreadRef.value = spread;
sheetRef.value = spread.getActiveSheet();
sheetRef.value.bind(GC.Spread.Sheets.Events.ShapeSelectionChanged, () => shapeSelectionChangedHandler());
addDefaultShape();
}
function addDefaultShape() {
var sheet = sheetRef.value;
var splogoImg = "";
var foodImg = "";
var pic1 = sheet.shapes.addPictureShape('pic1', splogoImg, 80, 80, 150, 150);
var pic2 = sheet.shapes.addPictureShape('pic2', splogoImg, 310, 80, 150, 150);
var pic3 = sheet.shapes.addPictureShape('pic3', splogoImg, 540, 80, 150, 150);
var pic4 = sheet.shapes.addPictureShape('pic4', foodImg, 80, 300, 285, 180);
var pic5 = sheet.shapes.addPictureShape('pic5', foodImg, 420, 300, 285, 180);
pic1.isSelected(true);
// set pic2 style
var style = pic2.style();
style.fill = { type: 1, color: '#666666' };
pic2.style(style);
// set pic3 geometry type
pic3.geometryType(GC.Spread.Sheets.Shapes.AutoShapeType.roundedRectangle);
// set pic3 style
var style = pic3.style();
style.fill = { type: 1, color: '#666666' };
pic3.style(style);
// set pic5 duotone effect
pic5.pictureFormat({ duotone: { color1: 'black', color2: 'rgb(150, 242, 21)' } });
}
function shapeSelectionChangedHandler() {
var pictures = getSelectedPictures();
var isActive = pictures.length === 1;
active.value = isActive;
if (isActive) {
syncState(pictures[0]);
}
}
function syncState(pic) {
let geometryType1 = pic.geometryType();
let format = pic.pictureFormat();
let style = pic.style();
isSyncState.value = true;
isShow.value = false;
geometryType.value = geometryType1;
crop.value = format.crop;
transparency.value = format.transparency;
brightness.value = format.brightness;
contrast.value = format.contrast;
let recolorType1 = 'none';
if (format.grayscale) {
recolorType1 = 'gray-scale';
} else if (format.blackAndWhite !== undefined) {
recolorType1 = 'black-and-white';
blackAndWhite.value = format.blackAndWhite;
} else if (format.duotone) {
recolorType1 = 'duotone';
isShow.value = true;
duotoneColor1.value = rgbToHex(format.duotone.color1);
duotoneColor2.value = rgbToHex(format.duotone.color2);
}
recolorType.value = recolorType1;
backColor.value = rgbToHex(style.fill.color);
borderColor.value = rgbToHex(style.line.color);
borderWidth.value = style.line.width;
isSyncState.value = false;
}
function onImgChange(e) {
var blob = e.target.files[0];
var fileReader = new FileReader();
fileReader.addEventListener('load', function () {
getFirstSelectedPicture().src(fileReader.result);
});
fileReader.readAsDataURL(blob);
}
function onGeometryTypeChange(e) {
getFirstSelectedPicture().geometryType(geometryType.value);
}
function onPictureFormatPropValueChange(e) {
setPictureFormat((format) => {
format.crop = crop.value;
format.transparency = transparency.value;
format.brightness = brightness.value;
format.contrast = contrast.value;
});
}
function onRecolorTypeChange(e) {
setPictureFormat((format) => {
clearRecolorAndBrightnessContrast(format);
let recolorType1 = recolorType.value;
if (recolorType1 === 'gray-scale') {
format.grayscale = true;
} else if (recolorType1 === 'black-and-white') {
format.blackAndWhite = blackAndWhite.value;
} else if (recolorType1 === 'duotone') {
format.duotone = {
color1: duotoneColor1.value,
color2: duotoneColor2.value
};
}
});
}
function onBlackAndWhiteChange(e) {
setPictureFormat((format) => {
clearRecolorAndBrightnessContrast(format);
format.blackAndWhite = blackAndWhite.value;
});
}
function onDuotoneColorChange(e) {
setPictureFormat((format) => {
clearRecolorAndBrightnessContrast(format);
format.duotone = {
color1: duotoneColor1.value,
color2: duotoneColor2.value
};
})
}
function onBackColorChange(e) {
let value = e.target.value;
var picture = getFirstSelectedPicture();
var style = picture.style();
style.fill = { type: 1, color: value, transparency: 0 };
picture.style(style);
}
function onBorderColorChange(e) {
let value = e.target.value;
var picture = getFirstSelectedPicture();
var style = picture.style();
style.line.color = value;
picture.style(style);
}
function onBorderWidthChange(e) {
let value = Number(e.target.value);
var picture = getFirstSelectedPicture();
var style = picture.style();
style.line.color = borderColor.value;
style.line.width = value;
picture.style(style);
}
function getSelectedPictures() {
var pictures = [];
var shapes = sheetRef.value.shapes.all();
for (var i = 0; i < shapes.length; i++) {
var shape = shapes[i];
if (shape.isSelected() && shape instanceof GC.Spread.Sheets.Shapes.PictureShape) {
pictures.push(shape);
}
}
return pictures;
}
function getFirstSelectedPicture() {
return getSelectedPictures()[0];
}
function rgbToHex(color) {
let rgb = color.match(/rgb\((\d+),(\d+),(\d+)\)/);
if (!rgb || rgb.length < 4) {
return '#000000';
}
return '#' + rgb.slice(1, 4).map(function (v) {
let hex = Number(v).toString(16);
return hex.length === 1 ? "0" + hex : hex;
}).join('');
}
function setPictureFormat(changeFormatFunc) {
var pic = getFirstSelectedPicture();
var format = pic.pictureFormat();
changeFormatFunc(format);
pic.pictureFormat(format);
syncState(pic);
}
function clearRecolorAndBrightnessContrast(format) {
format.grayscale = false;
format.blackAndWhite = undefined;
format.duotone = undefined;
format.brightness = 0;
format.contrast = 0;
}
</script>
<style scoped>
body {
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
}
#app {
width: 100%;
height: 100%;
}
.sample-tutorial {
position: relative;
height: 100%;
overflow: hidden;
}
.sample-spreadsheets {
width: calc(100% - 300px);
height: 100%;
overflow: hidden;
}
.options-container {
position: absolute;
top: 0px;
right: 0px;
width: 300px;
height: 100%;
padding: 12px;
box-sizing: border-box;
background: #fbfbfb;
overflow: auto;
}
h3 {
text-align: center;
}
.option-row {
background: #fbfbfb;
position: relative;
margin: 12px 0px;
}
details {
margin: 12px 0px;
}
.option-row input {
position: absolute;
box-sizing: border-box;
right: 20px;
width: 126px;
height: 26px;
}
.option-row select {
position: absolute;
right: 20px;
width: 126px;
height: 26px;
}
.divide-line {
width: 100%;
height: 1px;
background: #cbcbcb;
margin-top: 10px;
margin-bottom: 3px;
}
</style>
<!DOCTYPE html>
<html style="height:100%;font-size:14px;">
<head>
<meta charset="utf-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<title>SpreadJS VUE</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link rel="stylesheet" type="text/css"
href="$DEMOROOT$/ko/vue3/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css">
<script src="$DEMOROOT$/ko/vue3/node_modules/systemjs/dist/system.src.js"></script>
<script src="./systemjs.config.js"></script>
<script src="./compiler.js" type="module"></script>
<script>
var System = SystemJS;
System.import("./src/app.js");
System.import('$DEMOROOT$/ko/lib/vue3/license.js');
</script>
</head>
<body>
<div id="app"></div>
</body>
</html>
(function (global) {
SystemJS.config({
transpiler: 'plugin-babel',
babelOptions: {
es2015: true
},
paths: {
// paths serve as alias
'npm:': 'node_modules/'
},
packageConfigPaths: [
'./node_modules/*/package.json',
"./node_modules/@mescius/*/package.json",
"./node_modules/@babel/*/package.json",
"./node_modules/@vue/*/package.json"
],
map: {
'vue': "npm:vue/dist/vue.esm-browser.js",
'tiny-emitter': 'npm:tiny-emitter/index.js',
'plugin-babel': 'npm:systemjs-plugin-babel/plugin-babel.js',
"systemjs-babel-build": "npm:systemjs-plugin-babel/systemjs-babel-browser.js",
'@mescius/spread-sheets': 'npm:@mescius/spread-sheets/index.js',
'@mescius/spread-sheets-resources-ko': 'npm:@mescius/spread-sheets-resources-ko/index.js',
'@mescius/spread-sheets-shapes': 'npm:@mescius/spread-sheets-shapes/index.js',
'@mescius/spread-sheets-vue': 'npm:@mescius/spread-sheets-vue/index.js'
},
meta: {
'*.css': { loader: 'systemjs-plugin-css' },
'*.vue': { loader: "../plugin-vue/index.js" }
}
});
})(this);