도형 채우기

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

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

다음 샘플은 채우기 없음 설정을 보여줍니다. 다음 샘플은 단색 채우기 설정을 보여줍니다. 선형 그라데이션과 방사형 그라데이션을 지원합니다. 선형 그라데이션을 사용하면 각도 또는 방향 필드를 사용하여 그라데이션을 지정할 수 있습니다. 하지만 방향과 각도를 동시에 설정하면 방향이 적용됩니다. 다음 샘플은 그라데이션 채우기 설정을 보여줍니다. 다음 샘플은 그림 채우기 설정을 보여줍니다. 현재 base64 형식 그림만 지원합니다. 다음 샘플은 질감 채우기 설정을 보여줍니다.
import * as React from 'react'; import * as ReactDOM from 'react-dom'; import './styles.css'; import { AppFunc } from './app-func'; // import { App } from './app-class'; // 1. Functional Component sample ReactDOM.render(<AppFunc />, document.getElementById('app')); // 2. Class Component sample // ReactDOM.render(<App />, document.getElementById('app'));
import * as React from 'react'; import GC from '@mescius/spread-sheets'; import "@mescius/spread-sheets-shapes"; import { SpreadSheets } from "@mescius/spread-sheets-react"; import '@mescius/spread-sheets-resources-ko'; GC.Spread.Common.CultureManager.culture("ko-kr"); const defaultSrc = ""; let spread = null; let sheet = null; let shapes = null; export function AppFunc() { const [fillEffectOption, setFillEffectOption] = React.useState({ active: false, color: '#00A2E8', transparency: 0, type: 0, direction: 0, angle: 0, stops: [ { position: 0, color: "#82BC00" }, { position: 0.5, color: "white" }, { position: 1, color: "orange" } ], src: defaultSrc, tilePictureAsTexture: false, offsetLeft: 0, offsetRight: 0, offsetTop: 0, offsetBottom: 0, offsetX: 0, offsetY: 0, scaleX: 1, scaleY: 1, alignment: 0, mirrorType: 0, }); const initSpread = (currSpread) => { spread = currSpread; sheet = spread.getActiveSheet(); shapes = sheet.shapes; addDefaultShape(); sheet.bind(GC.Spread.Sheets.Events.ShapeSelectionChanged, shapeSelectionChangedHandler); } const shapeSelectionChangedHandler = () => { var selectedShape = shapes.all().filter(function (sp) { return sp.isSelected() && sp instanceof GC.Spread.Sheets.Shapes.Shape; }); if (selectedShape.length === 1) { setFillEffectOption({ ...fillEffectOption, active: true }); } else { setFillEffectOption({ ...fillEffectOption, active: false }); } } const addDefaultShape = () => { 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: fillEffectOption.type, direction: isNaN(fillEffectOption.direction) ? undefined : fillEffectOption.direction, angle: fillEffectOption.angle, stops: fillEffectOption.stops, }) } const setPictureOrTextureFill = () => { setFill(getSelectedShape(), { src: fillEffectOption.src, transparency: fillEffectOption.transparency, tilePictureAsTexture: fillEffectOption.tilePictureAsTexture, offsetLeft: fillEffectOption.offsetLeft, offsetRight: fillEffectOption.offsetRight, offsetTop: fillEffectOption.offsetTop, offsetBottom: fillEffectOption.offsetBottom, offsetX: fillEffectOption.offsetX, offsetY: fillEffectOption.offsetY, scaleX: fillEffectOption.scaleX, scaleY: fillEffectOption.scaleY, alignment: fillEffectOption.alignment, mirrorType: fillEffectOption.mirrorType, }); } const onImgChange = (e) => { var blob = e.target.files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function () { setFillEffectOption({ ...fillEffectOption, src: fileReader.result }) }); fileReader.readAsDataURL(blob); } const getSelectedShape = () => { let selectedShape = null; shapes.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 tipEl = <div id="tip" class="options-container">Try selecting a shape to apply a fill effect.</div>; const linearDirectionItems = <React.Fragment> <option value="undefined">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> </React.Fragment>; const radialDirectionItems = <React.Fragment> <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> </React.Fragment>; const pictureFillOption = <div id="pictureFillOption"> <label>offsetLeft:</label> <input class="offsetLeft" type="number" min="-1" max="1" step="0.01" value={fillEffectOption.offsetLeft} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetLeft: Number(e.target.value) }) }} /> <label>offsetRight:</label> <input class="offsetRight" type="number" min="-1" max="1" step="0.01" value={fillEffectOption.offsetRight} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetRight: Number(e.target.value) }) }} /> <label>offsetTop:</label> <input class="offsetTop" type="number" min="-1" max="1" step="0.01" value={fillEffectOption.offsetTop} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetTop: Number(e.target.value) }) }} /> <label>offsetBottom:</label> <input class="offsetBottom" type="number" min="-1" max="1" step="0.01" value={fillEffectOption.offsetBottom} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetBottom: Number(e.target.value) }) }} /> </div> const textureFillOption = <div id='textureFillOption'> <label>offsetX:</label> <input class="offsetX" type="number" step="1" value={fillEffectOption.offsetX} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetX: Number(e.target.value) }) }} /> <label>offsetY:</label> <input class="offsetY" type="number" step="1" value={fillEffectOption.offsetY} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, offsetY: Number(e.target.value) }) }} /> <label>scaleX:</label> <input class="scaleX" type="number" min="0" max="1" step="0.01" value={fillEffectOption.scaleX} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, scaleX: Number(e.target.value) }) }} /> <label>scaleY:</label> <input class="scaleY" type="number" min="0" max="1" step="0.01" value={fillEffectOption.scaleY} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, scaleY: Number(e.target.value) }) }} /> <label>alignment:</label> <select class="alignment" value={fillEffectOption.alignment} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, alignment: Number(e.target.value) }) }}> <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" value={fillEffectOption.mirrorType} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, mirrorType: Number(e.target.value) }) }}> <option value="0">None</option> <option value="1">Horizontal</option> <option value="2">Vertical</option> <option value="3">Both</option> </select> </div> const propEl = <div id="fillProp" class="options-container"> <div id="shapeGradientFill" class="option-row"> <label class="title">Gradient Fill</label> <label>Graient Fill Type: </label> <select class='type' value={fillEffectOption.type} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, type: Number(e.target.value), direction: 0 }) }}> <option value="0">linear</option> <option value="1">radial</option> </select> <label>Direction: </label> <select class='direction' value={fillEffectOption.direction} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, direction: Number(e.target.value) }) }}> {fillEffectOption.type === 0 ? linearDirectionItems : radialDirectionItems} </select> <label>Angle: </label> <input class='angle' type="number" value={fillEffectOption.angle} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, angle: Number(e.target.value) }) }} /> <label>Color Stops: </label> <textarea class='stops' rows="10" value={JSON.stringify(fillEffectOption.stops, null, 2)} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, stops: JSON.parse(e.target.value) }) }}></textarea> <button class='setBtn' onClick={setGradientFill}>Set Gradient Fill</button> </div> <div class="divide-line"></div> <label class="title">Picture Or Texture Fill</label> <div id="pictureOrTextureFill" class="option-row"> <label>Image: </label> <input class="imgInput" type="file" accept="image/*" onChange={(e) => { onImgChange(e) }} /> <label>transparency: </label> <input class="transparency" type="number" min="0" max="1" step="0.01" value={fillEffectOption.transparency} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, transparency: Number(e.target.value) }) }} /> <br></br> tilePictureAsTexture: <input class="tilePictureAsTexture" type="checkbox" checked={fillEffectOption.tilePictureAsTexture} onChange={(e) => { setFillEffectOption({ ...fillEffectOption, tilePictureAsTexture: e.target.checked }) }} /> <div class="divide-line"></div> {fillEffectOption.tilePictureAsTexture ? textureFillOption : pictureFillOption} <div class="divide-line"></div> <button class='setBtn' onClick={setPictureOrTextureFill}>Set Picture Or Texture Fill</button> </div> </div> return ( <div class="sample-tutorial"> <div id="ss" class="sample-spreadsheets"> <SpreadSheets workbookInitialized={spread => initSpread(spread)}> </SpreadSheets> </div> {fillEffectOption.active ? propEl : tipEl} </div> ); }
import * as React from 'react'; import GC from '@mescius/spread-sheets'; import "@mescius/spread-sheets-shapes"; import { SpreadSheets } from "@mescius/spread-sheets-react"; import '@mescius/spread-sheets-resources-ko'; GC.Spread.Common.CultureManager.culture("ko-kr"); const defaultSrc = ""; const Component = React.Component; export class App extends Component { constructor(props) { super(props); this.spread = null; this.state = { active: false, color: '#00A2E8', transparency: 0, type: 0, direction: 0, angle: 0, stops: [ { position: 0, color: "#82BC00" }, { position: 0.5, color: "white" }, { position: 1, color: "orange" } ], src: defaultSrc, tilePictureAsTexture: false, offsetLeft: 0, offsetRight: 0, offsetTop: 0, offsetBottom: 0, offsetX: 0, offsetY: 0, scaleX: 1, scaleY: 1, alignment: 0, mirrorType: 0, } this.setGradientFill = this.setGradientFill.bind(this); this.setPictureOrTextureFill = this.setPictureOrTextureFill.bind(this); this.onImgChange = this.onImgChange.bind(this); this.shapeSelectionChangedHandler = this.shapeSelectionChangedHandler.bind(this); } render() { const tipEl = <div id="tip" class="options-container">Try selecting a shape to apply a fill effect.</div>; const linearDirectionItems = <React.Fragment> <option value="undefined">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> </React.Fragment>; const radialDirectionItems = <React.Fragment> <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> </React.Fragment>; const pictureFillOption = <div id="pictureFillOption"> <label>offsetLeft:</label> <input class="offsetLeft" type="number" min="-1" max="1" step="0.01" value={this.state.offsetLeft} onChange={(e) => { this.setState({ offsetLeft: Number(e.target.value) }) }} /> <label>offsetRight:</label> <input class="offsetRight" type="number" min="-1" max="1" step="0.01" value={this.state.offsetRight} onChange={(e) => { this.setState({ offsetRight: Number(e.target.value) }) }} /> <label>offsetTop:</label> <input class="offsetTop" type="number" min="-1" max="1" step="0.01" value={this.state.offsetTop} onChange={(e) => { this.setState({ offsetTop: Number(e.target.value) }) }} /> <label>offsetBottom:</label> <input class="offsetBottom" type="number" min="-1" max="1" step="0.01" value={this.state.offsetBottom} onChange={(e) => { this.setState({ offsetBottom: Number(e.target.value) }) }} /> </div> const textureFillOption = <div id='textureFillOption'> <label>offsetX:</label> <input class="offsetX" type="number" step="1" value={this.state.offsetX} onChange={(e) => { this.setState({ offsetX: Number(e.target.value) }) }} /> <label>offsetY:</label> <input class="offsetY" type="number" step="1" value={this.state.offsetY} onChange={(e) => { this.setState({ offsetY: Number(e.target.value) }) }} /> <label>scaleX:</label> <input class="scaleX" type="number" min="0" max="1" step="0.01" value={this.state.scaleX} onChange={(e) => { this.setState({ scaleX: Number(e.target.value) }) }} /> <label>scaleY:</label> <input class="scaleY" type="number" min="0" max="1" step="0.01" value={this.state.scaleY} onChange={(e) => { this.setState({ scaleY: Number(e.target.value) }) }} /> <label>alignment:</label> <select class="alignment" value={this.state.alignment} onChange={(e) => { this.setState({ alignment: Number(e.target.value) }) }}> <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" value={this.state.mirrorType} onChange={(e) => { this.setState({ mirrorType: Number(e.target.value) }) }}> <option value="0">None</option> <option value="1">Horizontal</option> <option value="2">Vertical</option> <option value="3">Both</option> </select> </div> const propEl = <div id="fillProp" class="options-container"> <div id="shapeGradientFill" class="option-row"> <label class="title">Gradient Fill</label> <label>Graient Fill Type: </label> <select class='type' value={this.state.type} onChange={(e) => { this.setState({ type: Number(e.target.value), direction: 0 }) }}> <option value="0">linear</option> <option value="1">radial</option> </select> <label>Direction: </label> <select class='direction' value={this.state.direction} onChange={(e) => { this.setState({ direction: Number(e.target.value) }) }}> {this.state.type === 0 ? linearDirectionItems : radialDirectionItems} </select> <label>Angle: </label> <input class='angle' type="number" value={this.state.angle} onChange={(e) => { this.setState({ angle: Number(e.target.value) }) }} /> <label>Color Stops: </label> <textarea class='stops' rows="10" value={JSON.stringify(this.state.stops, null, 2)} onChange={(e) => { this.setState({ stops: JSON.parse(e.target.value) }) }}></textarea> <button class='setBtn' onClick={this.setGradientFill}>Set Gradient Fill</button> </div> <div class="divide-line"></div> <label class="title">Picture Or Texture Fill</label> <div id="pictureOrTextureFill" class="option-row"> <label>Image: </label> <input class="imgInput" type="file" accept="image/*" onChange={(e) => { this.onImgChange(e) }} /> <label>transparency: </label> <input class="transparency" type="number" min="0" max="1" step="0.01" value={this.state.transparency} onChange={(e) => { this.setState({ transparency: Number(e.target.value) }) }} /> <br></br> tilePictureAsTexture: <input class="tilePictureAsTexture" type="checkbox" checked={this.state.tilePictureAsTexture} onChange={(e) => { this.setState({ tilePictureAsTexture: e.target.checked }) }} /> <div class="divide-line"></div> {this.state.tilePictureAsTexture ? textureFillOption : pictureFillOption} <div class="divide-line"></div> <button class='setBtn' onClick={this.setPictureOrTextureFill}>Set Picture Or Texture Fill</button> </div> </div> return ( <div class="sample-tutorial"> <div id="ss" class="sample-spreadsheets"> <SpreadSheets workbookInitialized={spread => this.initSpread(spread)}> </SpreadSheets> </div> {this.state.active ? propEl : tipEl} </div> ); } 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); } shapeSelectionChangedHandler() { var selectedShape = this.shapes.all().filter(function (sp) { return sp.isSelected() && sp instanceof GC.Spread.Sheets.Shapes.Shape; }); if (selectedShape.length === 1) { this.setState({ active: true }); } else { this.setState({ active: false }); } } 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.state.type, direction: isNaN(this.state.direction) ? undefined : this.state.direction, angle: this.state.angle, stops: this.state.stops, }) } setPictureOrTextureFill() { this.setFill(this.getSelectedShape(), { src: this.state.src, transparency: this.state.transparency, tilePictureAsTexture: this.state.tilePictureAsTexture, offsetLeft: this.state.offsetLeft, offsetRight: this.state.offsetRight, offsetTop: this.state.offsetTop, offsetBottom: this.state.offsetBottom, offsetX: this.state.offsetX, offsetY: this.state.offsetY, scaleX: this.state.scaleX, scaleY: this.state.scaleY, alignment: this.state.alignment, mirrorType: this.state.mirrorType, }); } onImgChange(e) { var self = this; var blob = e.target.files[0]; var fileReader = new FileReader(); fileReader.addEventListener('load', function () { self.setState({ src: fileReader.result }) }); fileReader.readAsDataURL(blob); } 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); } }
<!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/react/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <!-- SystemJS --> <script src="$DEMOROOT$/ko/react/node_modules/systemjs/dist/system.src.js"></script> <script src="systemjs.config.js"></script> <script> System.import('$DEMOROOT$/ko/lib/react/license.js').then(function () { System.import('./src/app'); }); </script> </head> <body> <div id="app"></div> </body> </html>
.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%; }
(function (global) { System.config({ transpiler: 'plugin-babel', babelOptions: { es2015: true, react: true }, meta: { '*.css': { loader: 'css' } }, 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-shapes': 'npm:@mescius/spread-sheets-shapes/index.js', '@mescius/spread-sheets-react': 'npm:@mescius/spread-sheets-react/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', 'react': 'npm:react/umd/react.production.min.js', 'react-dom': 'npm:react-dom/umd/react-dom.production.min.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' }, // packages tells the System loader how to load when no filename and/or no extension packages: { src: { defaultExtension: 'jsx' }, "node_modules": { defaultExtension: 'js' }, } }); })(this);