그림 셰이프

그림 셰이프는 그림을 표시할 수 있는 셰이프입니다. 자르고 조정할 수 있으며 다른 셰이프와 함께 그림을 그룹화하거나 연결할 수 있습니다.

설명
app.component.ts
index.html
app.component.html
styles.css

다음 샘플은 그림 셰이프를 추가하는 방법을 보여 줍니다.

    // When adding a picture, the picture's source can be a base64 image or url.
    var picture = sheet.shapes.addPictureShape("pic1", "./test.png", 100, 100, 200, 200);

다음 예시는 그림의 투명도를 변경하는 방법을 보여 줍니다.

    // Set the transparency of the picture to 0.5
    picture.pictureFormat({ transparency: 0.5 });

다음 예시는 그림을 자르는 방법을 보여 줍니다.

    picture.pictureFormat({
        crop: {
            left: 0.15, // The left 15% of the image will be cropped
            right: 0.15, // The right 15% of the image will be cropped
            top: 0.2, // The top 20% of the image will be cropped
            bottom: 0.2 // The bottom 20% of the image will be cropped
        }
    });

다음 예시는 그림의 밝기 및 대비를 설정하는 방법을 보여 줍니다.

    // Set the brightness and contrast of the picture
    picture.pictureFormat({ brightness: 0.4, contrast: -0.4 });

다음 예시는 그림을 다시 칠하는 방법을 보여 줍니다.

    // Set the grayscale effect
    picture.pictureFormat({ grayscale: true });

    // Set blackAndWhite effect
    // Pixels with brightness greater than or equal to 0.5 are white, otherwise black
    picture.pictureFormat({ blackAndWhite: 0.5 });

    // Set the duotone effect
    picture.pictureFormat({ duotone: { color1: 'black', color2: 'red' } });

다음 예시는 그림의 기하 도형 유형을 변경하는 방법을 보여 줍니다.

    // Change the geometry type of the picture to oval
    picture.geometryType(GC.Spread.Sheets.Shapes.AutoShapeType.oval);

    // Change the geometry type of the picture to roundedRectangle
    picture.geometryType(GC.Spread.Sheets.Shapes.AutoShapeType.roundedRectangle)

다음 예시는 그림의 테두리를 설정하는 방법을 보여 줍니다.

    var style = picture.style();
    style.line.color = 'red';
    style.line.lineStyle = GC.Spread.Sheets.Shapes.PresetLineDashStyle.dashDot;
    style.line.width = 4;
    style.line.capType = GC.Spread.Sheets.Shapes.LineCapStyle.square;
    style.line.joinType = GC.Spread.Sheets.Shapes.LineJoinStyle.miter;
    picture.style(style);

그림의 일부가 투명하면 그림의 배경색을 설정할 수 있습니다.

    let style = picture.style();
    style.fill = { type: GC.Spread.Sheets.Shapes.ShapeFillType.solid, color: 'blue' };
    picture.style(style);
다음 샘플은 그림 셰이프를 추가하는 방법을 보여 줍니다. 다음 예시는 그림의 투명도를 변경하는 방법을 보여 줍니다. 다음 예시는 그림을 자르는 방법을 보여 줍니다. 다음 예시는 그림의 밝기 및 대비를 설정하는 방법을 보여 줍니다. 다음 예시는 그림을 다시 칠하는 방법을 보여 줍니다. 다음 예시는 그림의 기하 도형 유형을 변경하는 방법을 보여 줍니다. 다음 예시는 그림의 테두리를 설정하는 방법을 보여 줍니다. 그림의 일부가 투명하면 그림의 배경색을 설정할 수 있습니다.
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'; @Component({ selector: 'app-component', templateUrl: 'src/app.component.html' }) export class AppComponent { spread: GC.Spread.Sheets.Workbook; sheet: GC.Spread.Sheets.Worksheet; hostStyle = { width: 'calc(100% - 300px)', height: '100%', overflow: 'hidden', }; active = false; isShow = false; geometryType = 1; crop = { left: 0, right: 0, top: 0, bottom: 0 }; transparency = 0; brightness = 0; contrast = 0; recolorType = 'none'; blackAndWhite = 0.5; duotoneColor1 = '#000000'; duotoneColor2 = '#FF0000'; backColor = ''; borderColor = ''; borderWidth = -1; addDefaultShape() { var sheet = this.sheet; 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)'} }); } shapeSelectionChangedHandler() { var pictures = this.getSelectedPictures(); var active = pictures.length === 1; this.active = active; if (active) { this.syncState(pictures[0]); } } syncState(pic: GC.Spread.Sheets.Shapes.PictureShape) { let geometryType = pic.geometryType(); let format = pic.pictureFormat(); let style = pic.style(); this.geometryType = geometryType; this.crop = format.crop; this.transparency = format.transparency; this.brightness = format.brightness; this.contrast = format.contrast; this.isShow = false; let recolorType = 'none'; if (format.grayscale) { recolorType = 'gray-scale'; } else if (format.blackAndWhite !== undefined) { recolorType = 'black-and-white'; this.blackAndWhite = format.blackAndWhite; } else if (format.duotone) { recolorType = 'duotone'; this.isShow = true; this.duotoneColor1 = this.rgbToHex(format.duotone.color1); this.duotoneColor2 = this.rgbToHex(format.duotone.color2); } this.recolorType = recolorType; this.backColor = this.rgbToHex(style.fill.color); this.borderColor = this.rgbToHex(style.line.color); this.borderWidth = style.line.width; } onImgChange(e: Event) { var self = this; var blob = (e.target as HTMLInputElement).files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function() { self.getFirstSelectedPicture().src(fileReader.result as string); }); fileReader.readAsDataURL(blob); } onGeometryTypeChange(evt: Event) { let value = Number((<HTMLInputElement>evt.target).value); this.geometryType = value; this.getFirstSelectedPicture().geometryType(value); } onCropPropChange(evt: Event, propName: string) { let value = Number((<HTMLInputElement>evt.target).value); let crop = JSON.parse(JSON.stringify(this.crop)); crop[propName] = value; let pic = this.getFirstSelectedPicture(); var format = pic.pictureFormat(); format.crop = crop; pic.pictureFormat(format); } onPictureFormatChange(e: Event, propName: string) { let value = (<HTMLInputElement>e.target).value; let numberValue = Number(value); let pic = this.getFirstSelectedPicture(); var format = pic.pictureFormat(); if (propName === 'transparency') { format.transparency = numberValue; } else if (propName === 'brightness') { format.brightness = numberValue; } else if (propName === 'contrast') { format.contrast = numberValue; } else if (propName === 'recolorType') { this.clearRecolorAndBrightnessContrast(format); if (value === 'gray-scale') { format.grayscale = true; } else if (value === 'black-and-white') { format.blackAndWhite = this.blackAndWhite; } else if (value === 'duotone') { format.duotone = { color1: this.duotoneColor1, color2: this.duotoneColor2 } } } else if (propName === 'blackAndWhite') { this.clearRecolorAndBrightnessContrast(format); format.blackAndWhite = numberValue; } else if (propName === 'duotoneColor1') { this.clearRecolorAndBrightnessContrast(format); format.duotone = { color1: value, color2: this.duotoneColor2 } } else if (propName === 'duotoneColor2') { this.clearRecolorAndBrightnessContrast(format); format.duotone = { color1: this.duotoneColor1, color2: value } } pic.pictureFormat(format); this.syncState(pic); } onBackColorChange(evt: Event) { let value = (<HTMLInputElement>evt.target).value; this.backColor = value; var picture = this.getFirstSelectedPicture(); var style = picture.style(); style.fill = {type: 1, color: value, transparency: 0}; picture.style(style); } onBorderColorChange(evt: Event) { let value = (<HTMLInputElement>evt.target).value; this.borderColor = value; var picture = this.getFirstSelectedPicture(); var style = picture.style(); style.line.color = value; picture.style(style); } onBorderWidthChange(evt: Event) { let value = Number((<HTMLInputElement>evt.target).value); this.borderWidth = value; var picture = this.getFirstSelectedPicture(); var style = picture.style(); style.line.color = this.borderColor; style.line.width = value; picture.style(style); } rgbToHex(color: string) { 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(''); } getSelectedPictures() { var pictures = []; var shapes = this.sheet.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; } clearRecolorAndBrightnessContrast(format: GC.Spread.Sheets.Shapes.IPictureFormat) { format.grayscale = false; format.blackAndWhite = undefined; format.duotone = undefined; format.brightness = 0; format.contrast = 0; } getFirstSelectedPicture() { return this.getSelectedPictures()[0]; } initSpread($event: any) { this.spread = $event.spread; this.sheet = this.spread.getActiveSheet(); this.sheet.bind(GC.Spread.Sheets.Events.ShapeSelectionChanged, this.shapeSelectionChangedHandler.bind(this)); this.addDefaultShape(); } } @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 class="options-container" *ngIf="active"> <h3>Picture Format</h3> <div class="option-row"> <label>Image</label><input class="img" type="file" accept="image/*" (change)="onImgChange($event)" /> </div> <div class="option-row"> <label>Geometry Type</label> <select class="geometry-type" [(ngModel)]="geometryType" (change)="onGeometryTypeChange($event)"> <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" [(ngModel)]="transparency" (change)="onPictureFormatChange($event, 'transparency')" /> </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" [(ngModel)]="crop.left" (change)="onCropPropChange($event, 'left')" /> </div> <div class="option-row"> <label>Crop Right</label><input class="crop-right" type="number" max="0.99" step="0.01" [(ngModel)]="crop.right" (change)="onCropPropChange($event, 'right')" /> </div> <div class="option-row"> <label>Crop Top</label><input class="crop-top" type="number" max="0.99" step="0.01" [(ngModel)]="crop.top" (change)="onCropPropChange($event, 'top')" /> </div> <div class="option-row"> <label>Crop Bottom</label><input class="crop-bottom" type="number" max="0.99" step="0.01" [(ngModel)]="crop.bottom" (change)="onCropPropChange($event, 'bottom')" /> </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" [(ngModel)]="brightness" (change)="onPictureFormatChange($event, 'brightness')" /> </div> <div class="option-row"> <label>Contrast</label><input class="contrast" type="number" min="-1" max="1" step="0.01" [(ngModel)]="contrast" (change)="onPictureFormatChange($event, 'contrast')" /> </div> </details> <div class="divide-line"></div> <details> <summary>Recolor</summary> <div class="option-row"> <label>Recolor Type</label> <select class="recolor-type" [(ngModel)]="recolorType" (change)="onPictureFormatChange($event, 'recolorType')"> <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" [(ngModel)]="blackAndWhite" (change)="onPictureFormatChange($event, 'blackAndWhite')" /> </div> <div class="option-row" *ngIf="isShow"> <label>Duotone Color1</label><input class="duotone-color1" type="color" [(ngModel)]="duotoneColor1" (change)="onPictureFormatChange($event, 'duotoneColor1')" /> </div> <div class="option-row" *ngIf="isShow"> <label>Duotone Color2</label><input class="duotone-color2" type="color" [(ngModel)]="duotoneColor2" (change)="onPictureFormatChange($event, 'duotoneColor2')" /> </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" [(ngModel)]="backColor" (change)="onBackColorChange($event)" /> </div> <div class="option-row"> <label>Border Color</label><input class="border-color" type="color" [(ngModel)]="borderColor" (change)="onBorderColorChange($event)" /> </div> <div class="option-row"> <label>Border Width</label><input class="border-width" type="number" min="-1" step="1" [(ngModel)]="borderWidth" (change)="onBorderWidthChange($event)" /> </div> </details> </div> <div class="options-container" *ngIf="!active">Try to select a shape.</div> </div>
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; }
(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);