도형 채우기

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="active"> <div id="shapeGradientFill" class="option-row"> <label class="title">Gradient Fill</label> <label>Graient Fill Type: </label> <select class='type' v-model="type"> <option :value="0">linear</option> <option :value="1">radial</option> </select> <label>Direction: </label> <select class='direction' v-model="direction" v-if="type===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="direction" v-if="type===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="angle"/> <label>Color Stops: </label> <textarea class='stops' rows="10" v-model="stopsStr"></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="transparency" /> <br> tilePictureAsTexture:<input class="tilePictureAsTexture" type="checkbox" v-model="tilePictureAsTexture" /> <div class="divide-line"></div> <div id="pictureFillOption" v-if="!tilePictureAsTexture"> <label>offsetLeft:</label> <input class="offsetLeft" type="number" min="-1" max="1" step="0.01" v-model="offsetLeft" /> <label>offsetRight:</label> <input class="offsetRight" type="number" min="-1" max="1" step="0.01" v-model="offsetRight" /> <label>offsetTop:</label> <input class="offsetTop" type="number" min="-1" max="1" step="0.01" v-model="offsetTop" /> <label>offsetBottom:</label> <input class="offsetBottom" type="number" min="-1" max="1" step="0.01" v-model="offsetBottom" /> </div> <div id='textureFillOption' v-if="tilePictureAsTexture"> <label>offsetX:</label> <input class="offsetX" type="number" step="1" v-model="offsetX" /> <label>offsetY:</label> <input class="offsetY" type="number" step="1" v-model="offsetY" /> <label>scaleX:</label> <input class="scaleX" type="number" min="0" max="1" step="0.01" v-model="scaleX" /> <label>scaleY:</label> <input class="scaleY" type="number" min="0" max="1" step="0.01" v-model="scaleY" /> <label>alignment:</label> <select class="alignment" v-model="alignment"> <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="mirrorType"> <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="!active">Try to select a shape and apply a fill effect</div> </div> </template> <script> import Vue from "vue"; import "@mescius/spread-sheets-vue"; import "@mescius/spread-sheets-shapes"; import GC from "@mescius/spread-sheets"; import '@mescius/spread-sheets-resources-ko'; GC.Spread.Common.CultureManager.culture("ko-kr"); const defaultSrc = ""; let App = Vue.extend({ name: "app", data: function () { return { active: false, color: '#00A2E8', transparency: 0, type: 0, direction: 0, angle: 0, stopsStr: JSON.stringify([ { position: 0, color:"#82BC00" }, { position:0.5, color:"white" }, { position:1, color:"orange" } ], null, 2), src: defaultSrc, tilePictureAsTexture: false, offsetLeft: 0, offsetRight: 0, offsetTop: 0, offsetBottom: 0, offsetX: 0, offsetY: 0, scaleX: 1, scaleY: 1, alignment: 0, mirrorType: 0, }; }, methods: { initSpread(spread) { this.spread = spread; this.sheet = this.spread.getActiveSheet(); this.shapes = this.sheet.shapes; this.addDefaultShape(); this.sheet.bind(GC.Spread.Sheets.Events.ShapeSelectionChanged, this.shapeSelectionChangedHandler.bind(this)); }, shapeSelectionChangedHandler() { var selectedShape = this.shapes.all().filter(function(sp){ return sp.isSelected() && sp instanceof GC.Spread.Sheets.Shapes.Shape; }); this.active = selectedShape.length === 1; }, addDefaultShape() { let shapes = this.shapes, setFill = this.setFill, loadImage = this.loadImage; 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}); }); }, 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); } }, setGradientFill() { this.setFill(this.getSelectedShape(), { type: this.type, direction: this.direction === -1 ? undefined : this.direction, angle: this.angle, stops: JSON.parse(this.stopsStr), }) }, setPictureOrTextureFill() { this.setFill(this.getSelectedShape(), { src: this.src, transparency: this.transparency, tilePictureAsTexture: this.tilePictureAsTexture, offsetLeft: this.offsetLeft, offsetRight: this.offsetRight, offsetTop: this.offsetTop, offsetBottom: this.offsetBottom, offsetX: this.offsetX, offsetY: this.offsetY, scaleX: this.scaleX, scaleY: this.scaleY, alignment: this.alignment, mirrorType: this.mirrorType, }); }, getSelectedShape() { let selectedShape = null; this.shapes.all().forEach(shape => { if (shape.isSelected()) { selectedShape = shape; } }); return selectedShape; }, setFill(shape, fill) { let style = shape.style(); style.fill = fill; shape.style(style); }, onImgChange(e) { var self = this; var blob = e.target.files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function() { self.src = fileReader.result; }); fileReader.readAsDataURL(blob); } }, watch: { type: function() { this.direction = 0; } } }); new Vue({render: h => h(App)}).$mount("#app"); </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; } </style>
<!doctype html> <html style="height:100%;font-size:14px;"> <head> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <link rel="stylesheet" type="text/css" href="$DEMOROOT$/ko/vue/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <!-- SystemJS --> <script src="$DEMOROOT$/ko/vue/node_modules/systemjs/dist/system.src.js"></script> <script src="systemjs.config.js"></script> <script> System.import('./src/app.vue'); System.import('$DEMOROOT$/ko/lib/vue/license.js'); </script> </head> <body> <div id="app"></div> </body> </html>
(function (global) { System.config({ transpiler: 'plugin-babel', babelOptions: { es2015: true }, meta: { '*.css': { loader: 'css' }, '*.vue': { loader: 'vue-loader' } }, paths: { // paths serve as alias 'npm:': 'node_modules/' }, // map tells the System loader where to look for things map: { '@mescius/spread-sheets': 'npm:@mescius/spread-sheets/index.js', '@mescius/spread-sheets-vue': 'npm:@mescius/spread-sheets-vue/index.js', '@mescius/spread-sheets-shapes': 'npm:@mescius/spread-sheets-shapes/index.js', '@mescius/spread-sheets-resources-ko': 'npm:@mescius/spread-sheets-resources-ko/index.js', '@grapecity/jsob-test-dependency-package/react-components': 'npm:@grapecity/jsob-test-dependency-package/react-components/index.js', 'jszip': 'npm:jszip/dist/jszip.js', 'css': 'npm:systemjs-plugin-css/css.js', 'vue': 'npm:vue/dist/vue.min.js', 'vue-loader': 'npm:systemjs-vue-browser/index.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' }, // packages tells the System loader how to load when no filename and/or no extension packages: { src: { defaultExtension: 'js' }, rxjs: { defaultExtension: 'js' }, "node_modules": { defaultExtension: 'js' } } }); })(this);