도형 채우기

SpreadJS를 사용하면 단색 채우기, 그라데이션 채우기, 그림 채우기 및 질감 채우기 등 도형에 특별한 채우기 효과를 추가할 수 있습니다.

도형을 선택한 다음 오른쪽 패널에서 채우기 설정을 변경하면 됩니다.

다음 샘플은 채우기 없음 설정을 보여줍니다. 다음 샘플은 단색 채우기 설정을 보여줍니다. 선형 그라데이션과 방사형 그라데이션을 지원합니다. 선형 그라데이션을 사용하면 각도 또는 방향 필드를 사용하여 그라데이션을 지정할 수 있습니다. 하지만 방향과 각도를 동시에 설정하면 방향이 적용됩니다. 다음 샘플은 그라데이션 채우기 설정을 보여줍니다. 다음 샘플은 그림 채우기 설정을 보여줍니다. 현재 base64 형식 그림만 지원합니다. 다음 샘플은 질감 채우기 설정을 보여줍니다.
<template> <div class="sample-tutorial"> <gc-spread-sheets class="sample-spreadsheets" @workbookInitialized="initSpread"> </gc-spread-sheets> <div id="fillProp" class="options-container" v-if="activeRef"> <div id="shapeGradientFill" class="option-row"> <label class="title">Gradient Fill</label> <label>Graient Fill Type: </label> <select class='type' v-model="typeRef"> <option :value="0">linear</option> <option :value="1">radial</option> </select> <label>Direction: </label> <select class='direction' v-model="directionRef" v-if="typeRef === 0"> <option :value="-1">none</option> <option :value="0">linearRight</option> <option :value="45">topLeftToBottomRight</option> <option :value="90">linearDown</option> <option :value="135">topRightToBottomLeft</option> <option :value="180">linearLeft</option> <option :value="225">bottomRightToTopLeft</option> <option :value="270">linearUp</option> <option :value="315">bottomLeftToTopRight</option> </select> <select class='direction' v-model="directionRef" v-if="typeRef === 1"> <option :value="0">fromCenter</option> <option :value="1">fromTopLeft</option> <option :value="2">fromTopRight</option> <option :value="3">fromBottomLeft</option> <option :value="4">fromBottomRight</option> </select> <label>Angle: </label> <input class='angle' type="number" v-model="angleRef" /> <label>Color Stops: </label> <textarea class='stops' rows="10" v-model="stopsStrRef"></textarea> <button class='setBtn' @click="setGradientFill">Set Gradient Fill</button> </div> <div class="divide-line"></div> <div id="pictureOrTextureFill" class="option-row"> <label class="title">Picture Or Texture Fill</label> <label>Image:</label> <input class="imgInput" type="file" accept="image/*" @change="onImgChange" /> <label>transparency:</label> <input class="transparency" type="number" min="0" max="1" step="0.01" v-model="transparencyRef" /> <br> tilePictureAsTexture:<input class="tilePictureAsTexture" type="checkbox" v-model="tilePictureAsTextureRef" /> <div class="divide-line"></div> <div id="pictureFillOption" v-if="!tilePictureAsTextureRef"> <label>offsetLeft:</label> <input class="offsetLeft" type="number" min="-1" max="1" step="0.01" v-model="offsetLeftRef" /> <label>offsetRight:</label> <input class="offsetRight" type="number" min="-1" max="1" step="0.01" v-model="offsetRightRef" /> <label>offsetTop:</label> <input class="offsetTop" type="number" min="-1" max="1" step="0.01" v-model="offsetTopRef" /> <label>offsetBottom:</label> <input class="offsetBottom" type="number" min="-1" max="1" step="0.01" v-model="offsetBottomRef" /> </div> <div id='textureFillOption' v-if="tilePictureAsTextureRef"> <label>offsetX:</label> <input class="offsetX" type="number" step="1" v-model="offsetXRef" /> <label>offsetY:</label> <input class="offsetY" type="number" step="1" v-model="offsetYRef" /> <label>scaleX:</label> <input class="scaleX" type="number" min="0" max="1" step="0.01" v-model="scaleXRef" /> <label>scaleY:</label> <input class="scaleY" type="number" min="0" max="1" step="0.01" v-model="scaleYRef" /> <label>alignment:</label> <select class="alignment" v-model="alignmentRef"> <option value="0">Top Left</option> <option value="1">Top </option> <option value="2">Top Right</option> <option value="3">Left</option> <option value="4">Center</option> <option value="5">Right</option> <option value="6">Bottom Left</option> <option value="7">Bottom</option> <option value="8">Bottom Right</option> </select> <label>mirrorType:</label> <select class="mirrorType" v-model="mirrorTypeRef"> <option value="0">None</option> <option value="1">Horizontal</option> <option value="2">Vertical</option> <option value="3">Both</option> </select> </div> <div class="divide-line"></div> <button class='setBtn' @click="setPictureOrTextureFill">Set Picture Or Texture Fill</button> </div> </div> <div id="tip" class="options-container" v-if="!activeRef">Try to select a shape and apply a fill effect</div> </div> </template> <script setup> import GC from "@mescius/spread-sheets"; import { ref, watch } 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 defaultSrc = ""; const spreadRef = ref(null); const sheetRef = ref(null); const shapesRef = ref(null); const activeRef = ref(false); const srcRef = ref(defaultSrc); const transparencyRef = ref(0); const typeRef = ref(0); const directionRef = ref(0); const angleRef = ref(0); const stopsStrRef = ref(JSON.stringify([ { position: 0, color: "#82BC00" }, { position: 0.5, color: "white" }, { position: 1, color: "orange" } ], null, 2)); const tilePictureAsTextureRef = ref(false); const offsetLeftRef = ref(0); const offsetRightRef = ref(0); const offsetTopRef = ref(0); const offsetBottomRef = ref(0); const offsetXRef = ref(0); const offsetYRef = ref(0); const scaleXRef = ref(1); const scaleYRef = ref(1); const alignmentRef = ref(0); const mirrorTypeRef = ref(0); const initSpread = (spread) => { spreadRef.value = spread; sheetRef.value = spread.getActiveSheet(); shapesRef.value = sheetRef.value.shapes; addDefaultShape(); sheetRef.value.bind(GC.Spread.Sheets.Events.ShapeSelectionChanged, shapeSelectionChangedHandler); } const shapeSelectionChangedHandler = () => { var selectedShape = shapesRef.value.all().filter(function (sp) { return sp.isSelected() && sp instanceof GC.Spread.Sheets.Shapes.Shape; }); activeRef.value = selectedShape.length === 1; } const addDefaultShape = () => { let shapes = shapesRef.value; let x = 30, y = 30, width = 200, height = 100, xGap = 260, yGap = 160; for (var i = 0; i < 9; i++) { var tx = x + xGap * (i % 3); var ty = y + yGap * Math.floor(i / 3); shapes.add("shape" + (i + 1), GC.Spread.Sheets.Shapes.AutoShapeType.rectangle, tx, ty, width, height); } setFill(shapes.get('shape1'), { type: GC.Spread.Sheets.Shapes.GradientFillType.linear, direction: GC.Spread.Sheets.Shapes.LinearGradientFillDirection.linearRight, stops: [ { color: '#f7a711', position: 0 }, { color: '#f7a711', position: 0.25 }, { color: '#f6c65e', position: 0.25 }, { color: '#f6c65e', position: 0.50 }, { color: '#f8e29c', position: 0.50 }, { color: '#f8e29c', position: 0.75 }, { color: '#fffada', position: 0.75 }, { color: '#fffada', position: 1 }, ] }); setFill(shapes.get('shape2'), { type: GC.Spread.Sheets.Shapes.GradientFillType.linear, direction: GC.Spread.Sheets.Shapes.LinearGradientFillDirection.linearDown, stops: [ { color: '#f7a711', position: 0 }, { color: '#ffffff', position: 1 }, ] }); setFill(shapes.get('shape3'), { type: GC.Spread.Sheets.Shapes.GradientFillType.linear, direction: GC.Spread.Sheets.Shapes.LinearGradientFillDirection.topLeftToBottomRight, stops: [ { color: '#82bc00', position: 0 }, { color: '#ffffff', position: 1 }, ] }); setFill(shapes.get('shape4'), { type: GC.Spread.Sheets.Shapes.GradientFillType.radial, direction: GC.Spread.Sheets.Shapes.RadialGradientFillDirection.fromCenter, stops: [ { color: '#82bc00', position: 0 }, { color: '#82bc00', position: 0.25 }, { color: '#afd25e', position: 0.25 }, { color: '#afd25e', position: 0.50 }, { color: '#d6e89e', position: 0.50 }, { color: '#d6e89e', position: 0.75 }, { color: '#f9ffdd', position: 0.75 }, { color: '#f9ffdd', position: 1 }, ] }); setFill(shapes.get('shape5'), { type: GC.Spread.Sheets.Shapes.GradientFillType.radial, direction: GC.Spread.Sheets.Shapes.RadialGradientFillDirection.fromCenter, stops: [ { color: '#82bc00', position: 0 }, { color: '#ffffff', position: 1 }, ] }); setFill(shapes.get('shape6'), { type: GC.Spread.Sheets.Shapes.GradientFillType.radial, direction: GC.Spread.Sheets.Shapes.RadialGradientFillDirection.fromBottomLeft, stops: [ { color: '#f7a711', position: 0 }, { color: '#ffffff', position: 1 }, ] }); loadImage("$DEMOROOT$/spread/source/images/logo.jpg", function (src) { setFill(shapes.get('shape7'), { src: src }); }); loadImage("$DEMOROOT$/spread/source/images/splogo.png", function (src) { setFill(shapes.get('shape8'), { src: src }); }); loadImage("$DEMOROOT$/spread/source/images/splogo.png", function (src) { setFill(shapes.get('shape9'), { src: src, tilePictureAsTexture: true, scaleX: 0.25, scaleY: 0.25 }); }); } const loadImage = (url, callback) => { try { var xhr = new XMLHttpRequest(); xhr.onload = function () { var blob = xhr.response; var fileReader = new FileReader(); fileReader.onload = function () { callback(fileReader.result); }; fileReader.readAsDataURL(blob); }; xhr.responseType = "blob"; xhr.open("GET", url); xhr.send(); } catch (error) { callback(defaultSrc); } } const setGradientFill = () => { setFill(getSelectedShape(), { type: typeRef.value, direction: directionRef.value === -1 ? undefined : directionRef.value, angle: angleRef.value, stops: JSON.parse(stopsStrRef.value), }) } const setPictureOrTextureFill = () => { setFill(getSelectedShape(), { src: srcRef.value, transparency: transparencyRef.value, tilePictureAsTexture: tilePictureAsTextureRef.value, offsetLeft: offsetLeftRef.value, offsetRight: offsetRightRef.value, offsetTop: offsetTopRef.value, offsetBottom: offsetBottomRef.value, offsetX: offsetXRef.value, offsetY: offsetYRef.value, scaleX: scaleXRef.value, scaleY: scaleYRef.value, alignment: alignmentRef.value, mirrorType: mirrorTypeRef.value, }); } const getSelectedShape = () => { let selectedShape = null; shapesRef.value.all().forEach(shape => { if (shape.isSelected()) { selectedShape = shape; } }); return selectedShape; } const setFill = (shape, fill) => { let style = shape.style(); style.fill = fill; shape.style(style); } const onImgChange = (e) => { var blob = e.target.files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function () { srcRef.value = fileReader.result; }); fileReader.readAsDataURL(blob); } watch(typeRef, () => { directionRef.value = 0; }) </script> <style scoped> .sample-tutorial { position: relative; height: 100%; overflow: hidden; } .sample-spreadsheets { width: calc(100% - 280px); height: 100%; overflow: hidden; float: left; } .options-container { float: right; width: 280px; padding: 12px; height: 100%; box-sizing: border-box; background: #fbfbfb; overflow: auto; } .option-row { font-size: 14px; padding-left: 5px; } .divide-line { width: 100%; height: 1px; background: #cbcbcb; margin-top: 10px; margin-bottom: 3px; } .title { text-align: center; font-weight: bold; } label { display: block; margin-top: 15px; margin-bottom: 5px; } p { padding: 2px 10px; background-color: #F4F8EB; } input { margin-left: 10px; display: inline; } input[type=button] { width: 50px; margin-left: 1px; } select { width: 160px; margin-left: 10px; display: inline; } textarea { width: 210px; margin: 0px 0px; margin-left: 10px; padding: 2px; } body { position: absolute; top: 0; bottom: 0; left: 0; right: 0; } .setBtn { width: 220px; height: 31px; margin: 12px 10px; } #app { height: 100%; } </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$/spread/source/data/inventory-tracker.js" type="text/javascript"></script> <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);