diff --git a/common/changes/@visactor/vrender-core/feat-image-enhance_2025-02-20-03-46.json b/common/changes/@visactor/vrender-core/feat-image-enhance_2025-02-20-03-46.json new file mode 100644 index 000000000..f4747e665 --- /dev/null +++ b/common/changes/@visactor/vrender-core/feat-image-enhance_2025-02-20-03-46.json @@ -0,0 +1,10 @@ +{ + "changes": [ + { + "packageName": "@visactor/vrender-core", + "comment": "feat: image support auto width height by rawImage wh", + "type": "none" + } + ], + "packageName": "@visactor/vrender-core" +} \ No newline at end of file diff --git a/common/changes/@visactor/vrender/feat-image-enhance_2025-02-20-03-46.json b/common/changes/@visactor/vrender/feat-image-enhance_2025-02-20-03-46.json new file mode 100644 index 000000000..d0d51583d --- /dev/null +++ b/common/changes/@visactor/vrender/feat-image-enhance_2025-02-20-03-46.json @@ -0,0 +1,10 @@ +{ + "changes": [ + { + "packageName": "@visactor/vrender", + "comment": "feat: image support auto width height by rawImage wh", + "type": "none" + } + ], + "packageName": "@visactor/vrender" +} \ No newline at end of file diff --git a/packages/vrender-core/src/graphic/config.ts b/packages/vrender-core/src/graphic/config.ts index edfd5638a..047bc31ca 100644 --- a/packages/vrender-core/src/graphic/config.ts +++ b/packages/vrender-core/src/graphic/config.ts @@ -370,6 +370,8 @@ export const DefaultImageAttribute: Required = { image: '', width: 0, height: 0, + maxWidth: 500, + maxHeight: 500, ...DefaultAttribute, fill: true, cornerRadius: 0, diff --git a/packages/vrender-core/src/graphic/graphic.ts b/packages/vrender-core/src/graphic/graphic.ts index 4c2564b85..4749d40fc 100644 --- a/packages/vrender-core/src/graphic/graphic.ts +++ b/packages/vrender-core/src/graphic/graphic.ts @@ -196,7 +196,7 @@ export abstract class Graphic = Partial; declare backgroundImg?: boolean; diff --git a/packages/vrender-core/src/graphic/image.ts b/packages/vrender-core/src/graphic/image.ts index 3b7cf63fd..6fd723c93 100644 --- a/packages/vrender-core/src/graphic/image.ts +++ b/packages/vrender-core/src/graphic/image.ts @@ -18,6 +18,8 @@ export class Image extends Graphic implements IImage { // 资源加载完成后回调,外部通过回调获取图片资源尺寸 successCallback?: () => void; failCallback?: () => void; + declare _actualWidth?: number; + declare _actualHeight?: number; static NOWORK_ANIMATE_ATTR = { image: 1, @@ -33,25 +35,39 @@ export class Image extends Graphic implements IImage { this.loadImage(this.attribute.image); } - get width(): number { - return this.attribute.width ?? 0; - } - set width(width: number) { - if (this.attribute.width === width) { - this.attribute.width = width; - this.addUpdateShapeAndBoundsTag(); + getImageElement(): HTMLImageElement | HTMLCanvasElement | null { + const { image } = this.attribute; + if (!image || !this.resources) { + return null; + } + const res = this.resources.get(image); + if (res.state !== 'success') { + return null; } + return res.data; } - get height(): number { - return this.attribute.height ?? 0; - } - set height(height: number) { - if (this.attribute.height === height) { - this.attribute.height = height; - this.addUpdateShapeAndBoundsTag(); - } + get width(): number { + this.tryUpdateAABBBounds(); + return this._actualWidth; } + // set width(width: number) { + // if (this.attribute.width === width) { + // this.attribute.width = width; + // this.addUpdateShapeAndBoundsTag(); + // } + // } + + get height(): number { + this.tryUpdateAABBBounds(); + return this._actualHeight; + } + // set height(height: number) { + // if (this.attribute.height === height) { + // this.attribute.height = height; + // this.addUpdateShapeAndBoundsTag(); + // } + // } get repeatX(): IRepeatType { return this.attribute.repeatX ?? 'no-repeat'; } @@ -85,6 +101,7 @@ export class Image extends Graphic implements IImage { this.successCallback(); } }); + this.addUpdateBoundTag(); } imageLoadFail(url: string, cb?: () => void): void { @@ -123,7 +140,37 @@ export class Image extends Graphic implements IImage { aabbBounds: IAABBBounds ) { if (!this.updatePathProxyAABBBounds(aabbBounds)) { - const { width = imageTheme.width, height = imageTheme.height } = attribute; + const { maxWidth = imageTheme.maxWidth, maxHeight = imageTheme.maxHeight } = attribute; + let { width, height } = attribute; + if (width == null || height == null) { + const imageElement = this.getImageElement(); + if (imageElement) { + // 如果给了width或者height,那就使用已给的width或者height来按比例缩放,否则就在maxWidth和maxHeight之间按比例缩放 + const imageWidth = imageElement.width; + const imageHeight = imageElement.height; + if (width != null) { + height = width * (imageHeight / imageWidth); + } else if (height != null) { + width = height * (imageWidth / imageHeight); + } else { + // 如果width和height都没有给,那就使用maxWidth和maxHeight来按比例缩放 + const imageRatio = imageWidth / imageHeight; + const maxWHRatio = maxWidth / maxHeight; + if (imageRatio > maxWHRatio) { + width = maxWidth; + height = maxWidth / imageRatio; + } else { + height = maxHeight; + width = maxHeight * imageRatio; + } + } + } else { + width = maxWidth; + height = maxHeight; + } + } + this._actualWidth = width; + this._actualHeight = height; aabbBounds.set(0, 0, width, height); } diff --git a/packages/vrender-core/src/interface/graphic/image.ts b/packages/vrender-core/src/interface/graphic/image.ts index b48c2320a..8e5d3021d 100644 --- a/packages/vrender-core/src/interface/graphic/image.ts +++ b/packages/vrender-core/src/interface/graphic/image.ts @@ -11,6 +11,14 @@ export type IImageAttribute = { * 高度 */ height: number; + /** + * 最大宽度 + */ + maxWidth?: number; + /** + * 最大高度 + */ + maxHeight?: number; /** * x方向的重复方式 */ diff --git a/packages/vrender-core/src/render/contributions/render/image-render.ts b/packages/vrender-core/src/render/contributions/render/image-render.ts index 02db999e0..49a8d7c9d 100644 --- a/packages/vrender-core/src/render/contributions/render/image-render.ts +++ b/packages/vrender-core/src/render/contributions/render/image-render.ts @@ -63,8 +63,6 @@ export class DefaultCanvasImageRender extends BaseRender implements IGra // const imageAttribute = graphicService.themeService.getCurrentTheme().imageAttribute; const imageAttribute = getTheme(image).image; const { - width = imageAttribute.width, - height = imageAttribute.height, repeatX = imageAttribute.repeatX, repeatY = imageAttribute.repeatY, x: originX = imageAttribute.x, @@ -89,6 +87,9 @@ export class DefaultCanvasImageRender extends BaseRender implements IGra return; } + const width = image.width; + const height = image.height; + // deal with cornerRadius let needRestore = false; if (cornerRadius === 0 || (isArray(cornerRadius) && (cornerRadius).every(num => num === 0))) { diff --git a/packages/vrender/__tests__/browser/src/pages/image.ts b/packages/vrender/__tests__/browser/src/pages/image.ts index e68b1e1a9..833ffe8a6 100644 --- a/packages/vrender/__tests__/browser/src/pages/image.ts +++ b/packages/vrender/__tests__/browser/src/pages/image.ts @@ -10,87 +10,38 @@ const base64 = ''; // const urlSvg = 'https://replace-with-svg-link.svg'; -const svg2 = ` - - - - - - - - - - - - - - - - - - - - - - - - - - - - -`; +const dogImage = 'https://lf9-dp-fe-cms-tos.byteorg.com/obj/bit-cloud/vrender/lovely_dog.jpg'; export const page = () => { const shapes = []; shapes.push( createImage({ - x: 100, - y: 100, - width: 200, - height: 200, - image: - 'https://lf9-dp-fe-cms-tos.byteorg.com/obj/bit-cloud/vchart-editor/upload-images/a85cb6c4-b494-4a52-bfec-5ca557132dde' - // repeatX: 'repeat', - // repeatY: 'repeat' - // cornerRadius: 100 + x: 10, + y: 10, + image: dogImage, + _debug_bounds: true + }) + ); + shapes.push( + createImage({ + x: 10, + y: 300, + width: 100, + image: dogImage, + _debug_bounds: true + }) + ); + shapes.push( + createImage({ + x: 200, + y: 300, + width: 100, + height: 100, + image: dogImage, + _debug_bounds: true }) ); - - const svgImage = createImage({ - x: 300, - y: 100, - width: 100, - height: 100, - image: svg, - stroke: 'green' - }); - shapes.push(svgImage); - - const image = createImage({ - x: 50, - y: 50, - width: 200, - height: 200, - lineWidth: 10, - cornerRadius: 14, - stroke: '#000000', - fill: 'transparent', - image: - 'http://lf9-dp-fe-cms-tos.byteorg.com/obj/bit-cloud/vchart-editor/upload-images/014d69b2-2731-4daa-9e60-68925fa1e54b.jpg' - }); - shapes.push(image); - - // shapes.push( - // createImage({ - // x: 100, - // y: 300, - // width: 100, - // height: 100, - // image: urlSvg - // }) - // ); const stage = createStage({ canvas: 'main', @@ -100,30 +51,9 @@ export const page = () => { viewHeight: 600 }); - addShapesToStage(stage, shapes as any, true); stage.render(); - window.updateImage1 = () => { - svgImage.setAttribute('image', svg1); - (svgImage as any).loadImage(svgImage.attribute.image); - stage.render(); - }; - - window.updateImage0 = () => { - svgImage.setAttribute('image', svg); - (svgImage as any).loadImage(svgImage.attribute.image); - stage.render(); - }; - - window.updateImage2 = () => { - image.setAttribute('image', base64); - (image as any).loadImage(image.attribute.image); - stage.render(); - }; - - window.updateImage3 = () => { - image.setAttribute('image', urlSvg); - (image as any).loadImage(image.attribute.image); - stage.render(); - }; + shapes.forEach(shape => { + stage.defaultLayer.add(shape); + }); };