arkts-graphics-display.md 17.1 KB
Newer Older
H
HelloCrease 已提交
1
# 显示图片(Image)
Z
zengyawen 已提交
2 3 4 5 6 7 8 9 10 11


开发者经常需要在应用中显示一些图片,例如:按钮中的logo、网络图片、本地图片等。在应用中显示图片需要使用Image组件实现,Image支持多种图片格式,包括png、jpg、bmp、svg和gif,具体用法请参考[Image](../reference/arkui-ts/ts-basic-components-image.md)组件。


Image通过调用接口来创建,接口调用形式如下:



```ts
12
Image(src: PixelMap | ResourceStr | DrawableDescriptor)
Z
zengyawen 已提交
13 14 15 16 17 18 19 20 21 22 23 24 25
```


该接口通过图片数据源获取图片,支持本地图片和网络图片的渲染展示。其中,src是图片的数据源,加载方式请参考[加载图片资源](#加载图片资源)


## 加载图片资源

Image支持加载存档图、多媒体像素图两种类型。


### 存档图类型数据源

26
存档图类型的数据源可以分为本地资源、网络资源、Resource资源、媒体库资源和base64。
Z
zengyawen 已提交
27 28

- 本地资源
H
HelloCrease 已提交
29

Z
zengyawen 已提交
30 31 32 33 34 35 36 37 38 39
  创建文件夹,将本地图片放入ets文件夹下的任意位置。

  Image组件引入本地图片路径,即可显示图片(根目录为ets文件夹)。

  ```ts
  Image('images/view.jpg')
  .width(200)
  ```

- 网络资源
H
HelloCrease 已提交
40

Z
zengyawen 已提交
41 42 43 44 45 46 47
  引入网络图片需申请权限ohos.permission.INTERNET,具体申请方式请参考[权限申请声明](../security/accesstoken-guidelines.md)。此时,Image组件的src参数为网络图片的链接。

  ```ts
  Image('https://www.example.com/example.JPG') // 实际使用时请替换为真实地址
  ```

- Resource资源
H
HelloCrease 已提交
48

Z
zengyawen 已提交
49 50
  使用资源格式可以跨包/跨模块引入图片,resources文件夹下的图片都可以通过$r资源接口读 取到并转换到Resource格式。

51
  **图1** resources  
Z
zengyawen 已提交
52 53 54 55 56 57 58 59 60 61 62 63 64

  ![image-resource](figures/image-resource.jpg)

  调用方式:

  ```
  Image($r('app.media.icon'))
  ```

  还可以将图片放在rawfile文件夹下。

  **图2** rawfile  

H
HelloCrease 已提交
65
  ![image-rawfile](figures/image-rawfile.jpg)
Z
zengyawen 已提交
66 67 68 69 70 71 72

  调用方式:

  ```
  Image($rawfile('snap'))
  ```

H
HelloCrease 已提交
73
- 媒体库file://data/storage
H
HelloCrease 已提交
74

H
HelloCrease 已提交
75
  支持file://路径前缀的字符串,用于访问通过媒体库提供的图片路径。
Z
zengyawen 已提交
76 77 78 79

  1. 调用接口获取图库的照片url。
      ```ts
      import picker from '@ohos.file.picker';
Y
yamila 已提交
80
      
Z
zengyawen 已提交
81 82 83
      @Entry
      @Component
      struct Index {
周沺耳 已提交
84
        @State imgDatas: string[] = [];
Z
zengyawen 已提交
85
        // 获取照片url集
周沺耳 已提交
86
        getAllImg() {
Z
zengyawen 已提交
87 88
          let result = new Array<string>();
          try {
L
lixinnan 已提交
89
            let PhotoSelectOptions:picker.PhotoSelectOptions = new picker.PhotoSelectOptions();
Z
zengyawen 已提交
90 91
            PhotoSelectOptions.MIMEType = picker.PhotoViewMIMETypes.IMAGE_TYPE;
            PhotoSelectOptions.maxSelectNumber = 5;
L
lixinnan 已提交
92 93
            let photoPicker:picker.PhotoViewPicker = new picker.PhotoViewPicker();
            photoPicker.select(PhotoSelectOptions).then((PhotoSelectResult:picker.PhotoSelectResult) => {
周沺耳 已提交
94 95
              this.imgDatas = PhotoSelectResult.photoUris;
              console.info('PhotoViewPicker.select successfully, PhotoSelectResult uri: ' + JSON.stringify(PhotoSelectResult));
L
lixinnan 已提交
96
            }).catch((err:Error) => {
Z
zengyawen 已提交
97 98 99
              console.error(`PhotoViewPicker.select failed with. Code: ${err.code}, message: ${err.message}`);
            });
          } catch (err) {
L
lixinnan 已提交
100 101 102
            let message:BusinessError = (err as BusinessError).message;
            let code:BusinessError = (err as BusinessError).code;
            console.error(`PhotoViewPicker failed with. Code: ${code}, message: ${message}`);    }
Z
zengyawen 已提交
103
        }
Y
yamila 已提交
104
      
Z
zengyawen 已提交
105 106
        // aboutToAppear中调用上述函数,获取图库的所有图片url,存在imgDatas中
        async aboutToAppear() {
周沺耳 已提交
107
          this.getAllImg();
Z
zengyawen 已提交
108 109 110
        }
        // 使用imgDatas的url加载图片。
        build() {
周沺耳 已提交
111 112
          Column() {
            Grid() {
L
lixinnan 已提交
113
              ForEach(this.imgDatas, (item:string) => {
周沺耳 已提交
114 115 116 117
                GridItem() {
                  Image(item)
                    .width(200)
                }
L
lixinnan 已提交
118
              }, ((item:string):string => JSON.stringify(item)))
周沺耳 已提交
119 120
            }
          }.width('100%').height('100%')
Z
zengyawen 已提交
121 122 123
        }
      }
      ```
Y
yamila 已提交
124

Z
zengyawen 已提交
125 126
  2. 从媒体库获取的url格式通常如下。
      ```ts
H
HelloCrease 已提交
127
      Image('file://media/Photos/5')
Z
zengyawen 已提交
128 129 130 131
      .width(200)
      ```

- base64
H
HelloCrease 已提交
132

Z
zengyawen 已提交
133 134 135 136 137 138 139
  路径格式为data:image/[png|jpeg|bmp|webp];base64,[base64 data],其中[base64 data]为Base64字符串数据。

  Base64格式字符串可用于存储图片的像素数据,在网页上使用较为广泛。


### 多媒体像素图

H
HelloCrease 已提交
140
PixelMap是图片解码后的像素图,具体用法请参考[图片开发指导](../media/image-overview.md)。以下示例将加载的网络图片返回的数据解码成PixelMap格式,再显示在Image组件上,
Z
zengyawen 已提交
141 142 143 144 145 146 147 148

1. 创建PixelMap状态变量。

   ```ts
   @State image: PixelMap = undefined;
   ```

2. 引用多媒体。
H
HelloCrease 已提交
149

Z
zengyawen 已提交
150 151 152 153 154 155 156 157 158 159 160
   请求网络图片请求,解码编码PixelMap。

   1. 引用网络权限与媒体库权限。
       ```ts
       import http from '@ohos.net.http';
       import ResponseCode from '@ohos.net.http';
       import image from '@ohos.multimedia.image';
       ```
   2. 填写网络图片地址。
       ```ts
       http.createHttp().request("https://www.example.com/xxx.png",
L
lixinnan 已提交
161
         (error:Error) => {
Z
zengyawen 已提交
162 163 164 165 166 167 168
           if (error){
             console.error(`http reqeust failed with. Code: ${error.code}, message: ${error.message}`);
           } else {
           }
         }
       )
       ```
Y
yamila 已提交
169
   3. 将网络地址成功返回的数据,编码转码成pixelMap的图片格式。   
Z
zengyawen 已提交
170
       ```ts
L
lixinnan 已提交
171
       let code:object = data.responseCode;
Y
yamila 已提交
172
       if (ResponseCode.ResponseCode.OK === code) {
L
lixinnan 已提交
173 174 175 176 177 178 179 180 181 182 183 184
         let imageSource:image = image.createImageSource(data.result);
         class tmp{
           height:number = 100
           width:number = 100
         }
         let si:tmp = new tmp()
         let options:Record<string,number|boolean|tmp> = {
           'alphaType': 0, // 透明度
           'editable': false, // 是否可编辑
           'pixelFormat': 3, // 像素格式
           'scaleMode': 1, // 缩略值
           'size': { height: 100, width: 100 }
Y
yamila 已提交
185
         } // 创建图片大小
L
lixinnan 已提交
186 187 188 189 190 191 192 193 194 195
         class imagetmp{
           image:image
           set(val:PixelMap){
             this.image = val
           }
         }
          imageSource.createPixelMap(options).then((pixelMap:PixelMap) => {
          let im = new imagetmp()
            im.set(pixelMap)
       })
Y
yamila 已提交
196
       }
Z
zengyawen 已提交
197 198 199
       ```
   4. 显示图片。
       ```ts
L
lixinnan 已提交
200 201 202 203 204 205 206 207
       class htp{
        httpRequest:Function|undefined = undefined
        set(){
          if(this.httpRequest){
          this.httpRequest()
          }
        }
      }
Z
zengyawen 已提交
208 209
       Button("获取网络图片")
         .onClick(() => {
L
lixinnan 已提交
210 211
           let sethtp = new htp()
           sethtp.set()
Z
zengyawen 已提交
212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229
         })
       Image(this.image).height(100).width(100)
       ```


## 显示矢量图

Image组件可显示矢量图(svg格式的图片),支持的svg标签为:svg、rect、circle、ellipse、path、line、polyline、polygon和animate。

svg格式的图片可以使用fillColor属性改变图片的绘制颜色。


```ts
Image($r('app.media.cloud')).width(50)
.fillColor(Color.Blue) 
```

  **图3** 原始图片  
H
HelloCrease 已提交
230

Z
zengyawen 已提交
231 232 233
![屏幕截图_20230223_141141](figures/屏幕截图_20230223_141141.png)

  **图4** 设置绘制颜色后的svg图片  
H
HelloCrease 已提交
234

Z
zengyawen 已提交
235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
![屏幕截图_20230223_141404](figures/屏幕截图_20230223_141404.png)


## 添加属性

给Image组件设置属性可以使图片显示更灵活,达到一些自定义的效果。以下是几个常用属性的使用示例,完整属性信息详见[Image](../reference/arkui-ts/ts-basic-components-image.md)


### 设置图片缩放类型

通过objectFit属性使图片缩放到高度和宽度确定的框内。


```ts
@Entry
@Component
struct MyComponent {
  scroller: Scroller = new Scroller()

  build() {
    Scroll(this.scroller) {
      Row() {
        Image($r('app.media.img_2')).width(200).height(150)
          .border({ width: 1 })
          .objectFit(ImageFit.Contain).margin(15) // 保持宽高比进行缩小或者放大,使得图片完全显示在显示边界内。
          .overlay('Contain', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.ic_img_2')).width(200).height(150)
          .border({ width: 1 })
          .objectFit(ImageFit.Cover).margin(15)
          // 保持宽高比进行缩小或者放大,使得图片两边都大于或等于显示边界。
          .overlay('Cover', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.img_2')).width(200).height(150)
          .border({ width: 1 })
            // 自适应显示。
          .objectFit(ImageFit.Auto).margin(15)
          .overlay('Auto', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
      }
      Row() {
        Image($r('app.media.img_2')).width(200).height(150)
          .border({ width: 1 })
          .objectFit(ImageFit.Fill).margin(15)
          // 不保持宽高比进行放大缩小,使得图片充满显示边界。
          .overlay('Fill', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.img_2')).width(200).height(150)
          .border({ width: 1 })
          // 保持宽高比显示,图片缩小或者保持不变。
          .objectFit(ImageFit.ScaleDown).margin(15)
          .overlay('ScaleDown', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.img_2')).width(200).height(150)
          .border({ width: 1 })
          // 保持原有尺寸显示。
          .objectFit(ImageFit.None).margin(15)
          .overlay('None', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
      }
    }
  }
}
```

H
HelloCrease 已提交
294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503
![zh-cn_image_0000001622804833](figures/zh-cn_image_0000001622804833.png)


### 图片插值

当原图分辨率较低并且放大显示时,图片会模糊出现锯齿。这时可以使用interpolation属性对图片进行插值,使图片显示得更清晰。


```ts
@Entry
@Component
struct Index {
  build() {
    Column() {
      Row() {
        Image($r('app.media.grass'))
          .width('40%')
          .interpolation(ImageInterpolation.None)
          .borderWidth(1)
          .overlay("Interpolation.None", { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
          .margin(10)
        Image($r('app.media.grass'))
          .width('40%')
          .interpolation(ImageInterpolation.Low)
          .borderWidth(1)
          .overlay("Interpolation.Low", { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
          .margin(10)
      }.width('100%')
      .justifyContent(FlexAlign.Center)

      Row() {
        Image($r('app.media.grass'))
          .width('40%')
          .interpolation(ImageInterpolation.Medium)
          .borderWidth(1)
          .overlay("Interpolation.Medium", { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
          .margin(10)
        Image($r('app.media.grass'))
          .width('40%')
          .interpolation(ImageInterpolation.High)
          .borderWidth(1)
          .overlay("Interpolation.High", { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
          .margin(10)
      }.width('100%')
      .justifyContent(FlexAlign.Center)
    }
    .height('100%')
  }
}
```

![zh-cn_image_0000001643127365](figures/zh-cn_image_0000001643127365.png)


### 设置图片重复样式

通过objectRepeat属性设置图片的重复样式方式,重复样式请参考[ImageRepeat](../reference/arkui-ts/ts-appendix-enums.md#imagerepeat)枚举说明。


```ts
@Entry
@Component
struct MyComponent {
  build() {
    Column({ space: 10 }) {
      Row({ space: 5 }) {
        Image($r('app.media.ic_public_favor_filled_1'))
          .width(110)
          .height(115)
          .border({ width: 1 })
          .objectRepeat(ImageRepeat.XY)
          .objectFit(ImageFit.ScaleDown)
          // 在水平轴和竖直轴上同时重复绘制图片
          .overlay('ImageRepeat.XY', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.ic_public_favor_filled_1'))
          .width(110)
          .height(115)
          .border({ width: 1 })
          .objectRepeat(ImageRepeat.Y)
          .objectFit(ImageFit.ScaleDown)
          // 只在竖直轴上重复绘制图片
          .overlay('ImageRepeat.Y', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.ic_public_favor_filled_1'))
          .width(110)
          .height(115)
          .border({ width: 1 })
          .objectRepeat(ImageRepeat.X)
          .objectFit(ImageFit.ScaleDown)
          // 只在水平轴上重复绘制图片
          .overlay('ImageRepeat.X', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
      }
    }.height(150).width('100%').padding(8)
  }
}
```

![zh-cn_image_0000001593444112](figures/zh-cn_image_0000001593444112.png)


### 设置图片渲染模式

通过renderMode属性设置图片的渲染模式为原色或黑白。


```ts
@Entry
@Component
struct MyComponent {
  build() {
    Column({ space: 10 }) {
      Row({ space: 50 }) {
        Image($r('app.media.example'))
          // 设置图片的渲染模式为原色 
          .renderMode(ImageRenderMode.Original)
          .width(100)
          .height(100)
          .border({ width: 1 })
            // overlay是通用属性,用于在组件上显示说明文字
          .overlay('Original', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
        Image($r('app.media.example'))
          // 设置图片的渲染模式为黑白
          .renderMode(ImageRenderMode.Template)
          .width(100)
          .height(100)
          .border({ width: 1 })
          .overlay('Template', { align: Alignment.Bottom, offset: { x: 0, y: 20 } })
      }
    }.height(150).width('100%').padding({ top: 20,right: 10 })
  }
}
```

![zh-cn_image_0000001593293100](figures/zh-cn_image_0000001593293100.png)


### 设置图片解码尺寸

通过sourceSize属性设置图片解码尺寸,降低图片的分辨率。

原图尺寸为1280\*960,该示例将图片解码为150\*100和400\*400。


```ts
@Entry
@Component
struct Index {
  build() {
    Column() {
      Row({ space: 20 }) {
        Image($r('app.media.example'))
          .sourceSize({
            width: 150,
            height: 150
          })
          .objectFit(ImageFit.ScaleDown)
          .width('25%')
          .aspectRatio(1)
          .border({ width: 1 })
          .overlay('width:150 height:150', { align: Alignment.Bottom, offset: { x: 0, y: 40 } })
        Image($r('app.media.example'))
          .sourceSize({
            width: 400,
            height: 400
          })
          .objectFit(ImageFit.ScaleDown)
          .width('25%')
          .aspectRatio(1)
          .border({ width: 1 })
          .overlay('width:400 height:400', { align: Alignment.Bottom, offset: { x: 0, y: 40 } })
      }.height(150).width('100%').padding(20)

    }
  }
}
```

![zh-cn_image_0000001593769844](figures/zh-cn_image_0000001593769844.png)


### 为图片添加滤镜效果

通过colorFilter修改图片的像素颜色,为图片添加滤镜。


```ts
@Entry
@Component
struct Index {
  build() {
    Column() {
      Row() {
        Image($r('app.media.example'))
          .width('40%')
          .margin(10)
        Image($r('app.media.example'))
          .width('40%')
          .colorFilter(
            [1, 1, 0, 0, 0,
             0, 1, 0, 0, 0,
             0, 0, 1, 0, 0,
             0, 0, 0, 1, 0])
          .margin(10)
      }.width('100%')
      .justifyContent(FlexAlign.Center)
    }
  }
}
```

![zh-cn_image_0000001643171357](figures/zh-cn_image_0000001643171357.png)
Z
zengyawen 已提交
504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522


### 同步加载图片

一般情况下,图片加载流程会异步进行,以避免阻塞主线程,影响UI交互。但是特定情况下,图片刷新时会出现闪烁,这时可以使用syncLoad属性,使图片同步加载,从而避免出现闪烁。不建议图片加载较长时间时使用,会导致页面无法响应。


```ts
Image($r('app.media.icon'))
  .syncLoad(true)
```


## 事件调用

通过在Image组件上绑定onComplete事件,图片加载成功后可以获取图片的必要信息。如果图片加载失败,也可以通过绑定onError回调来获得结果。


```ts
L
lixinnan 已提交
523 524 525 526 527 528 529 530
class tmp{
  width: number = 0
  height: number = 0
  componentWidth: number = 0
  componentHeight: number = 0
}

let msg:tmp = new tmp()
Z
zengyawen 已提交
531 532 533 534 535 536 537 538 539 540 541 542 543 544 545
@Entry
@Component
struct MyComponent {
  @State widthValue: number = 0
  @State heightValue: number = 0
  @State componentWidth: number = 0
  @State componentHeight: number = 0

  build() {
    Column() {
      Row() {
        Image($r('app.media.ic_img_2'))
          .width(200)
          .height(150)
          .margin(15)
L
lixinnan 已提交
546 547 548 549 550 551 552
          .onComplete(msg => {
            if(msg){
              this.widthValue = msg.width
              this.heightValue = msg.height
              this.componentWidth = msg.componentWidth
              this.componentHeight = msg.componentHeight
            }
Z
zengyawen 已提交
553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569
          })
            // 图片获取失败,打印结果
          .onError(() => {
            console.info('load image fail')
          })
          .overlay('\nwidth: ' + String(this.widthValue) + ', height: ' + String(this.heightValue) + '\ncomponentWidth: ' + String(this.componentWidth) + '\ncomponentHeight: ' + String(this.componentHeight), {
            align: Alignment.Bottom,
            offset: { x: 0, y: 60 }
          })
      }
    }
  }
}
```


![zh-cn_image_0000001511740460](figures/zh-cn_image_0000001511740460.png)