도형 채우기

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

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

다음 샘플은 채우기 없음 설정을 보여줍니다. 다음 샘플은 단색 채우기 설정을 보여줍니다. 선형 그라데이션과 방사형 그라데이션을 지원합니다. 선형 그라데이션을 사용하면 각도 또는 방향 필드를 사용하여 그라데이션을 지정할 수 있습니다. 하지만 방향과 각도를 동시에 설정하면 방향이 적용됩니다. 다음 샘플은 그라데이션 채우기 설정을 보여줍니다. 다음 샘플은 그림 채우기 설정을 보여줍니다. 현재 base64 형식 그림만 지원합니다. 다음 샘플은 질감 채우기 설정을 보여줍니다.
import { Component, NgModule, enableProdMode } from '@angular/core'; import { BrowserModule } from '@angular/platform-browser'; import { FormsModule } from '@angular/forms'; import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; import { SpreadSheetsModule } from '@mescius/spread-sheets-angular'; import GC from '@mescius/spread-sheets'; import "@mescius/spread-sheets-shapes"; import '@mescius/spread-sheets-resources-ko'; GC.Spread.Common.CultureManager.culture("ko-kr"); import './styles.css'; const defaultSrc = ""; @Component({ selector: 'app-component', templateUrl: 'src/app.component.html' }) export class AppComponent { spread: GC.Spread.Sheets.Workbook; sheet: GC.Spread.Sheets.Worksheet; shapes: GC.Spread.Sheets.Shapes.ShapeCollection; hostStyle = { width: 'calc(100% - 280px)', height: '100%', overflow: 'hidden', float: 'left' }; 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; shapeSelectionChangedHandler() { var selectedShape = this.shapes.all().filter(function(sp: any){ 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: string, callback: (src: string) => void) { try { var xhr = new XMLHttpRequest(); xhr.onload = function() { var blob = xhr.response; var fileReader = new FileReader(); fileReader.onload = function() { callback(fileReader.result as string); }; 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, }); } onImgChange(e: Event) { var self = this; var blob = (e.target as HTMLInputElement).files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function() { self.src = fileReader.result as string; }); fileReader.readAsDataURL(blob); } getSelectedShape(): any { let selectedShape = null; this.shapes.all().forEach((shape: any) => { if (shape.isSelected()) { selectedShape = shape; } }); return selectedShape; } setFill(shape: GC.Spread.Sheets.Shapes.Shape, fill: any) { let style = shape.style(); style.fill = fill; shape.style(style); } typeChangeHandler(evt: Event) { this.type = Number((<HTMLInputElement>evt.target).value); this.direction = 0; } directionChangeHandler(evt: Event) { this.direction = Number((<HTMLInputElement>evt.target).value); } angleChangeHandler(evt: Event) { this.angle = Number((<HTMLInputElement>evt.target).value); } alignmentChangeHandler(evt: Event) { this.alignment = Number((<HTMLInputElement>evt.target).value); } mirrorTypeChangeHandler(evt: Event) { this.mirrorType = Number((<HTMLInputElement>evt.target).value); } initSpread($event: any) { this.spread = $event.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)); } } @NgModule({ imports: [BrowserModule, SpreadSheetsModule, FormsModule], declarations: [AppComponent], exports: [AppComponent], bootstrap: [AppComponent] }) export class AppModule { } enableProdMode(); // Bootstrap application with hash style navigation and global services. platformBrowserDynamic().bootstrapModule(AppModule);
<!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/angular/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <!-- Polyfills --> <script src="$DEMOROOT$/ko/angular/node_modules/core-js/client/shim.min.js"></script> <script src="$DEMOROOT$/ko/angular/node_modules/zone.js/fesm2015/zone.min.js"></script> <!-- SystemJS --> <script src="$DEMOROOT$/ko/angular/node_modules/systemjs/dist/system.js"></script> <script src="systemjs.config.js"></script> <script> // workaround to load 'rxjs/operators' from the rxjs bundle System.import('rxjs').then(function (m) { System.import('@angular/compiler'); System.set(SystemJS.resolveSync('rxjs/operators'), System.newModule(m.operators)); System.import('$DEMOROOT$/ko/lib/angular/license.ts'); System.import('./src/app.component'); }); </script> </head> <body> <app-component></app-component> </body> </html>
<div class="sample-tutorial"> <gc-spread-sheets [hostStyle]="hostStyle" (workbookInitialized)="initSpread($event)"> </gc-spread-sheets> <div id="fillProp" class="options-container" *ngIf="active"> <div id="shapeGradientFill" class="option-row"> <label class="title">Gradient Fill</label> <label>Graient Fill Type: </label> <select class='type' [(ngModel)]="type" (change)="typeChangeHandler($event)"> <option [value]=0>linear</option> <option [value]=1>radial</option> </select> <label>Direction: </label> <select class='direction' [(ngModel)]="direction" *ngIf="type===0" (change)="directionChangeHandler($event)"> <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' [(ngModel)]="direction" *ngIf="type===1" (change)="directionChangeHandler($event)"> <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" [(ngModel)]="angle"/> <label>Color Stops: </label> <textarea class='stops' rows="10" [(ngModel)]="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($event)" /> <label>transparency:</label> <input class="transparency" type="number" min="0" max="1" step="0.01" [(ngModel)]="transparency" /> <br> tilePictureAsTexture: <input class="tilePictureAsTexture" type="checkbox" [(ngModel)]="tilePictureAsTexture" /> <div class="divide-line"></div> <div id="pictureFillOption" *ngIf="!tilePictureAsTexture"> <label>offsetLeft:</label> <input class="offsetLeft" type="number" min="-1" max="1" step="0.01" [(ngModel)]="offsetLeft" /> <label>offsetRight:</label> <input class="offsetRight" type="number" min="-1" max="1" step="0.01" [(ngModel)]="offsetRight" /> <label>offsetTop:</label> <input class="offsetTop" type="number" min="-1" max="1" step="0.01" [(ngModel)]="offsetTop" /> <label>offsetBottom:</label> <input class="offsetBottom" type="number" min="-1" max="1" step="0.01" [(ngModel)]="offsetBottom" /> </div> <div id='textureFillOption' *ngIf="tilePictureAsTexture"> <label>offsetX:</label> <input class="offsetX" type="number" step="1" [(ngModel)]="offsetX" /> <label>offsetY:</label> <input class="offsetY" type="number" step="1" [(ngModel)]="offsetY" /> <label>scaleX:</label> <input class="scaleX" type="number" min="0" max="1" step="0.01" [(ngModel)]="scaleX" /> <label>scaleY:</label> <input class="scaleY" type="number" min="0" max="1" step="0.01" [(ngModel)]="scaleY" /> <label>alignment: </label> <select class="alignment" [(ngModel)]="alignment" (change)="alignmentChangeHandler($event)"> <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" [(ngModel)]="mirrorType" (change)="mirrorTypeChangeHandler($event)"> <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" *ngIf="!active">Try selecting a shape to apply a fill effect.</div> </div>
.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; }
(function (global) { System.config({ transpiler: 'ts', typescriptOptions: { tsconfig: true }, meta: { 'typescript': { "exports": "ts" }, '*.css': { loader: 'css' } }, paths: { // paths serve as alias 'npm:': 'node_modules/' }, // map tells the System loader where to look for things map: { 'core-js': 'npm:core-js/client/shim.min.js', 'zone': 'npm:zone.js/fesm2015/zone.min.js', 'rxjs': 'npm:rxjs/dist/bundles/rxjs.umd.min.js', '@angular/core': 'npm:@angular/core/fesm2022', '@angular/common': 'npm:@angular/common/fesm2022/common.mjs', '@angular/compiler': 'npm:@angular/compiler/fesm2022/compiler.mjs', '@angular/platform-browser': 'npm:@angular/platform-browser/fesm2022/platform-browser.mjs', '@angular/platform-browser-dynamic': 'npm:@angular/platform-browser-dynamic/fesm2022/platform-browser-dynamic.mjs', '@angular/common/http': 'npm:@angular/common/fesm2022/http.mjs', '@angular/router': 'npm:@angular/router/fesm2022/router.mjs', '@angular/forms': 'npm:@angular/forms/fesm2022/forms.mjs', 'jszip': 'npm:jszip/dist/jszip.min.js', 'typescript': 'npm:typescript/lib/typescript.js', 'ts': './plugin.js', 'tslib':'npm:tslib/tslib.js', 'css': 'npm:systemjs-plugin-css/css.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-resources-ko': 'npm:@mescius/spread-sheets-resources-ko/index.js', '@mescius/spread-sheets-angular': 'npm:@mescius/spread-sheets-angular/fesm2020/mescius-spread-sheets-angular.mjs', '@grapecity/jsob-test-dependency-package/react-components': 'npm:@grapecity/jsob-test-dependency-package/react-components/index.js' }, // packages tells the System loader how to load when no filename and/or no extension packages: { src: { defaultExtension: 'ts' }, rxjs: { defaultExtension: 'js' }, "node_modules": { defaultExtension: 'js' }, "node_modules/@angular": { defaultExtension: 'mjs' }, "@mescius/spread-sheets-angular": { defaultExtension: 'mjs' }, '@angular/core': { defaultExtension: 'mjs', main: 'core.mjs' } } }); })(this);