Shape Fill

SpreadJS allows you to add special fill effects to the shapes such as solid fill, gradient fill, picture fill and texture fill.

The following sample shows set no fill. The following sample shows set solid fill. We support linear gradients and radial gradients. When using linear gradients, you can use angle or direction field to specify the gradient. But if you set direction and angle at the same time, direction will work. The following samples shows set gradient fill. The following samples shows set picture fill. Currently we only support pictures with base64 format. The following samples shows set texture fill.
<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'; 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$/en/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$/en/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$/en/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-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);