HarmonyOS 换肤功能怎么实现?

APP换肤如何实现?。另外整体UI变灰色怎么实现?

HarmonyOS
2024-09-03 10:56:18
浏览
收藏 0
回答 1
回答 1
按赞同
/
按时间
FengTianYa

目前有两种实现ArkTS 主题切换的方案,一种是基于ArkTS UI界面提供的深浅色模式接口,另一种是创建一个主题类来实现。

深浅色模式目前有以下不足:

1.只支持深浅色两种模式的切换,对于需要多个主题的场景不能满足。

2.image组件(除了svg这种支持.fillColor()的)目前不支持深浅色模式的热加载,想要实现按钮图标的切换可能还是需要使用三元表达式,对于图标多的场景不够方便

优点:系统规范较好

主题类实现深浅色模式有以下不足:

1.重新渲染ui的性能没有保障,还需要测试

2.资源都继承在一个文件里,如果是很大的工程可能比较难管理

3.相关代码规范问题

优点: 可以支持多种主题的切换,只需要加切换函数就够了 支持按钮图标,图片热加载,支持多种资源根据主题切换,并实时重新渲染 在page中引用资源和传统方式的代码量差不多,能比较方便的使用,使用方法简单快捷

参考demo:

Index.ets

import theme from '../pages/theme' 
 
@Entry 
@Component 
struct Index { 
  @State message: string = 'normal mode'; 
  @StorageLink('current mode') current_mode: string = 'normal_mode' 
  @StorageLink('main_theme') theme: theme = new theme(this.current_mode) 
  @StorageLink('gray mode') gray_mode: number = 0; 
  build() { 
    Column() { 
      Row() { 
        Text(this.message) 
          .fontSize(50) 
          .fontColor(this.theme.text_color) 
          .fontWeight(FontWeight.Bold) 
      }.justifyContent(FlexAlign.Center) 
      .width('100%') 
      Row() { 
        Column() { 
          Image(this.theme.phone_icon).objectFit(ImageFit.Contain) 
          Text('电话').fontColor(this.theme.text_color) 
          Image(this.theme.camera_icon).objectFit(ImageFit.Contain) 
          Text('相机').fontColor(this.theme.text_color) 
        }.width('22%').height(90) 
 
      }.width('100%').alignItems(VerticalAlign.Top).height('250vp').justifyContent(FlexAlign.SpaceEvenly) 
 
      Row() { 
        Button('color mode') 
          .width('30%') 
          .fontColor(this.theme.text_color) 
          .onClick(() => { 
          this.message = 'color mode' 
          this.theme.color_mode() 
          this.current_mode = 'color_mode' 
        }) 
        Button('normal mode') 
          .width('30%') 
          .fontColor(this.theme.text_color) 
          .onClick(() => { 
          this.message = 'normal mode' 
          this.theme.normal_mode() 
          this.current_mode = 'normal_mode' 
        }) 
        Button('simple mode') 
          .width('30%') 
          .fontColor(this.theme.text_color) 
          .onClick(() => { 
          this.message = 'simple mode' 
          this.theme.simple_mode() 
          this.current_mode = 'simple_mode' 
        }) 
      }.height(50) 
      .justifyContent(FlexAlign.SpaceAround) 
      .width('100%') 
 
      Image($r('app.media.find_service')).objectFit(ImageFit.Contain).height(200) 
      // 整体UI变灰色 
      Row() { 
        Button('mode off').fontColor(this.theme.text_color).height('30vp').onClick(() => this.gray_mode = 0) 
        Button('mode on').fontColor(this.theme.text_color).height('30vp').onClick(() => this.gray_mode = 1) 
      } 
 
    } 
    .grayscale(this.gray_mode) 
    .height('100%').backgroundColor(this.theme.background_color) 
 
  } 
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.
  • 67.
  • 68.
  • 69.
  • 70.

theme.ets

export default class theme { 
  //资源目录,将需要实现主题切换的资源变量存储在这 
  current_mode: string = 'normal_mode' 
  earphone_icon: Resource = $r('app.media.ic_device_earphone_hero_filled') 
  home_icon: Resource = $r('app.media.ic_public_home_filled') 
  picture_icon: Resource = $r('app.media.ic_public_picture_filled') 
  folder_icon: Resource = $r('app.media.ic_public_folder_filled') 
  icon2: Resource = $r('app.media.ic_public_picture') 
  color1: Resource = $r('app.color.red') 
  font_color: Resource = $r('app.color.black') 
  text_color: Resource = $r('app.color.black') 
  background_color: Resource = $r('app.color.white') 
  phone_icon: Resource = $r('app.media.dialer') 
  market_icon: Resource = $r('app.media.shopping') 
  notes_icon: Resource = $r('app.media.notes') 
  setting_icon: Resource = $r('app.media.settings') 
  camera_icon: Resource = $r('app.media.camera') 
  photos_icon: Resource = $r('app.media.gallery') 
  music_icon: Resource = $r('app.media.music') 
  video_icon: Resource = $r('app.media.video') 
 
  //通过读取current mode实现在重启应用后可以保存应用主题数据 
  constructor(current_mode: string) { 
    switch (current_mode) { 
      case 'normal_mode': 
        this.background_color = $r('app.color.background') 
        this.text_color = $r('app.color.brown') 
        this.phone_icon = $r('app.media.dialer') 
        this.market_icon = $r('app.media.shopping') 
        this.notes_icon = $r('app.media.notes') 
        this.setting_icon = $r('app.media.settings') 
        this.camera_icon = $r('app.media.camera') 
        this.photos_icon = $r('app.media.gallery') 
        this.music_icon = $r('app.media.music') 
        this.video_icon = $r('app.media.video') 
        break; 
      case 'color_mode': 
 
        this.background_color = $r('app.color.color_background') 
        this.text_color = $r('app.color.yellow') 
        this.phone_icon = $r('app.media.pwcall') 
        this.market_icon = $r('app.media.pwshop') 
        this.notes_icon = $r('app.media.pwnotes') 
        this.setting_icon = $r('app.media.pwsetting') 
        this.camera_icon = $r('app.media.pwcamera') 
        this.photos_icon = $r('app.media.pwphotos') 
        this.music_icon = $r('app.media.pwmusic') 
        this.video_icon = $r('app.media.pwvideo') 
 
        break; 
      case 'simple_mode': 
        this.background_color = $r('app.color.white') 
        this.text_color = $r('app.color.black') 
        this.phone_icon = $r('app.media.simplicityCall') 
        this.market_icon = $r('app.media.simplicityShop') 
        this.notes_icon = $r('app.media.simplicityNotes') 
        this.setting_icon = $r('app.media.simplicitySetting') 
        this.camera_icon = $r('app.media.simplicityCamera') 
        this.photos_icon = $r('app.media.simplicityPhotos') 
        this.music_icon = $r('app.media.simplicityMusic') 
        this.video_icon = $r('app.media.simplicityVideo') 
        break; 
 
      default: 
        this.current_mode = 'normal_mode' 
        this.background_color = $r('app.color.background') 
        this.text_color = $r('app.color.brown') 
        this.phone_icon = $r('app.media.dialer') 
        this.market_icon = $r('app.media.shopping') 
        this.notes_icon = $r('app.media.notes') 
        this.setting_icon = $r('app.media.settings') 
        this.camera_icon = $r('app.media.camera') 
        this.photos_icon = $r('app.media.gallery') 
        this.music_icon = $r('app.media.music') 
        this.video_icon = $r('app.media.video') 
        break; 
    } 
 
  } 
  //通过不同的主题切换函数更改主题变量,并在外部通过Appstorage实现应用内共享和画面重现渲染 
  light_mode() { 
    this.font_color = $r('app.color.black') 
    this.background_color = $r('app.color.white') 
    this.earphone_icon = $r('app.media.ic_device_earphone_hero_filled') 
    this.home_icon = $r('app.media.ic_public_home_filled') 
    this.picture_icon = $r('app.media.ic_public_picture_filled') 
    this.folder_icon = $r('app.media.ic_public_folder_filled') 
  } 
  dark_mode() { 
    this.font_color = $r('app.color.white') 
    this.background_color = $r('app.color.background') 
    this.earphone_icon = $r('app.media.ic_device_earphone_hero') 
    this.home_icon = $r('app.media.ic_public_home') 
    this.picture_icon = $r('app.media.ic_public_picture') 
    this.folder_icon = $r('app.media.ic_public_folder') 
  } 
  normal_mode() { 
    this.current_mode = 'normal_mode' 
    this.background_color = $r('app.color.background') 
    this.text_color = $r('app.color.brown') 
    this.phone_icon = $r('app.media.dialer') 
    this.market_icon = $r('app.media.shopping') 
    this.notes_icon = $r('app.media.notes') 
    this.setting_icon = $r('app.media.settings') 
    this.camera_icon = $r('app.media.camera') 
    this.photos_icon = $r('app.media.gallery') 
    this.music_icon = $r('app.media.music') 
    this.video_icon = $r('app.media.video') 
  } 
  color_mode() { 
    this.current_mode = 'color_mode' 
    this.background_color = $r('app.color.color_background') 
    this.text_color = $r('app.color.yellow') 
    this.phone_icon = $r('app.media.pwcall') 
    this.market_icon = $r('app.media.pwshop') 
    this.notes_icon = $r('app.media.pwnotes') 
    this.setting_icon = $r('app.media.pwsetting') 
    this.camera_icon = $r('app.media.pwcamera') 
    this.photos_icon = $r('app.media.pwphotos') 
    this.music_icon = $r('app.media.pwmusic') 
    this.video_icon = $r('app.media.pwvideo') 
  } 
  simple_mode() { 
    this.current_mode = 'simple_mode' 
    this.background_color = $r('app.color.white') 
    this.text_color = $r('app.color.black') 
    this.phone_icon = $r('app.media.simplicityCall') 
    this.market_icon = $r('app.media.simplicityShop') 
    this.notes_icon = $r('app.media.simplicityNotes') 
    this.setting_icon = $r('app.media.simplicitySetting') 
    this.camera_icon = $r('app.media.simplicityCamera') 
    this.photos_icon = $r('app.media.simplicityPhotos') 
    this.music_icon = $r('app.media.simplicityMusic') 
    this.video_icon = $r('app.media.simplicityVideo') 
  } 
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.
  • 67.
  • 68.
  • 69.
  • 70.
  • 71.
  • 72.
  • 73.
  • 74.
  • 75.
  • 76.
  • 77.
  • 78.
  • 79.
  • 80.
  • 81.
  • 82.
  • 83.
  • 84.
  • 85.
  • 86.
  • 87.
  • 88.
  • 89.
  • 90.
  • 91.
  • 92.
  • 93.
  • 94.
  • 95.
  • 96.
  • 97.
  • 98.
  • 99.
  • 100.
  • 101.
  • 102.
  • 103.
  • 104.
  • 105.
  • 106.
  • 107.
  • 108.
  • 109.
  • 110.
  • 111.
  • 112.
  • 113.
  • 114.
  • 115.
  • 116.
  • 117.
  • 118.
  • 119.
  • 120.
  • 121.
  • 122.
  • 123.
  • 124.
  • 125.
  • 126.
  • 127.
  • 128.
  • 129.
  • 130.
  • 131.
  • 132.
  • 133.
  • 134.
  • 135.
  • 136.

在resource目录下创建了dark/element目录、dark/media目录。

element创建 color.json资源 ,在 media中 存放图片资源。适配手机深色模式。

//整体UI变灰色 乱码部分为整体UI变灰色。

分享
微博
QQ
微信
回复
2024-09-03 16:56:14
相关问题
HarmonyOS 如何实现应用全局换肤功能
675浏览 • 1回复 待解决
HarmonyOS 换肤相关指导
768浏览 • 1回复 待解决
HarmonyOS 换肤方案有哪些?
1089浏览 • 1回复 待解决
HarmonyOS换肤方案有哪些?
1183浏览 • 1回复 待解决
HarmonyOS Overlay机制是否支持换肤
360浏览 • 1回复 待解决
grid如何怎么实现拖拽功能
1395浏览 • 1回复 待解决
js fa怎么实现录音功能呢?
6005浏览 • 1回复 待解决
怎么实现跨设备拷贝粘贴功能
187浏览 • 0回复 待解决
ArkTs怎么实现扫一扫功能
5859浏览 • 1回复 待解决
应用内整体换肤的最佳实践
1422浏览 • 1回复 待解决
怎么实现保存网络图片到相册功能
1512浏览 • 1回复 待解决
鸿蒙提供的画中画功能怎么实现
1677浏览 • 1回复 待解决
Arkts开发 api9 中怎么实现扫码功能
3873浏览 • 1回复 待解决
分布式相机功能在应用层怎么实现
3859浏览 • 1回复 待解决
HarmonyOS 实现RSA加密功能
1013浏览 • 1回复 待解决
HarmonyOS 如何实现ImagePreview功能
614浏览 • 1回复 待解决
HarmonyOS 如何实现轮询功能
888浏览 • 1回复 待解决
HarmonyOS AtomicInteger 计数功能实现
711浏览 • 1回复 待解决
HarmonyOS 虚线功能实现
721浏览 • 1回复 待解决
HarmonyOS 分享功能如何实现
899浏览 • 1回复 待解决
HarmonyOS 如何实现popupwindow功能
664浏览 • 1回复 待解决