写点什么

【推荐 +1】HarmonyOS 官方模板优秀案例 (第 4 期:餐饮行业 · 美食菜谱)

  • 2025-08-28
    北京
  • 本文字数:14273 字

    阅读完需:约 47 分钟

🚀🚀🚀🚀

鸿蒙生态为开发者提供海量的 HarmonyOS 模板/组件,助力开发效率原地起飞

★一键直达生态市场组件&模板市场 , 快速应用DevEco Studio插件市场集成组件&模板 ★

好用的菜谱 APP 能够帮助饮食管理事半功倍!小编本期为大家介绍餐饮行业案例

👉覆盖 20+行业,本帖下方以汇总形式持续更新中,点击收藏!一键三连!常看常新!

 

【第 4 期】餐饮行业· 美食菜谱

一、 概述

1. 行业洞察

1) 行业诉求:

  • 传统餐饮私域流量缺失,依赖第三方平台导流,佣金成本高且较难沉淀用户。

    网络菜谱步骤描述模糊,用料配比不精确,成功率低。

    个性化不足:缺乏根据用户口味偏好、饮食限制(素食/低糖/无麸质)的智能推荐。

    购物不便:菜谱食材无法一键购买,特殊调料难以配齐。

    数据价值未挖掘:用户烹饪行为数据未形成精准画像。

  • 互动体验差:用户作品分享渠道单一,缺乏专业厨师点评指导。


2) 行业常用三方 SDK

说明:“以上三方库及链接仅为示例,三方库由三方开发者独立提供,以其官方内容为准”

SDK 链接:

支付宝 SDK 微信支付 SDK 银联 SDK 腾讯QQ SDK 新浪微博 SDK 极光PUSH SDK 友盟移动统计SDK 腾讯微信 SDK 高德地图 SDK 个推 Bugly ShareSDK 听云SDK


2. 行业案例概览(下载模板

基于以上行业分析,本期将介绍鸿蒙生态市场餐饮行业模板——美食菜谱应用模板,为行业提供常用功能的开发案例,涵盖分类搜索、作品分享、菜谱用料、热量计算、饮食计划等多个实用功能。

  • Stage 开发模型 + 声明式 UI 开发范式。

    分层架构设计+ 组件化拆分,支持开发者在开发时既可以选择完整使用模板,也可以根据需求单独选用其中的业务组件。

    集成华为账号、支付等服务,只需做少量配置和定制即可快速实现华为账号的登录、菜谱上传等功能。


本模板主要页面及核心功能如下所示:


二、 应用架构设计

1. 分层模块化设计

  • 产品定制层:专注于满足不同设备或使用场景的个性化需求,作为应用的入口,是用户直接互动的界面。

① 本实践暂时只支持直板机,为单 HAP 包形式,包含路由根节点、底部导航栏等。

  • 基础特性层:用于存放相对独立的功能 UI 和业务逻辑实现。

① 本实践的基础特性层将应用底部导航栏的每个选项拆分成一个独立的业务功能模块,包含首页、分类、热量计算和我的。

② 每个功能模块都具备高内聚、低耦合、可定制的特点,支持产品的灵活部署。

  • 公共能力层:存放公共能力,包括公共 UI 组件、数据管理、外部交互和工具库等共享功能。

① 本实践的公共能力层分为公共基础能力和行业组件,均打包为 HAR 包被基础特性层的业务模块引用。

② 公共基础能力包含账号管理、动态布局等工具,公共类型定义,网络库,以及弹窗、加载等公共组件。

③ 可分可合组件将包含行业特点、可完全自闭环的能力抽出独立的组件模块,支持开发者在开发中单独集成使用,详见业务组件设计章节。

2. 业务组件设计

为支持开发者单独获取特定场景的页面和功能,例如开发者已搭建了一个自己的美食菜谱元服务工程,只想单独取用本模板中的菜篮子或热量计算功能,本模板将功能完全自闭环的部分能力抽离出独立的行业组件模块,不依赖公共基础能力包,开发者可以单独集成,开箱即用,降低使用难度。


三、 行业场景技术方案

1. 一键搜题

1) 场景说明

支持华为账号一键登录及其他方式(账号密码登录、微信登录)。

用户登录后展示昵称和头像,点击用户信息栏可进入用户主页,查看并编辑个人信息。

2) 技术方案

  • 华为账号一键登录

① 通过 Account Kit 实现华为账号一键登录,并获取用户手机号,关联应用已有用户。

  • 微信登录

① 根据鸿蒙接入指南接入微信 SDK,可通过跳转微信并获取微信用户授权进行登录。

  • 头像修改

① 通过 Scenario Fusion Kit 提供的选择头像Button快速拉起头像选择页面,供用户完成华为账号头像或其他头像的选择与展示。


3) 代码参考

  • 部分核心代码参见个人信息实现章节。


2.分类搜索

1) 场景说明

  • 支持按菜谱分类展示和搜索菜谱,点击菜谱可以查看菜谱详细信息。

2) 技术方案

使用ListGrid组件实现分类列表的展示、连续滚动以及搜索结果展示。


3)代码参考

  • 部分核心代码参见分类搜索实现章节。


3.饮食计划

1) 场景说明

支持制定饮食计划的食材,也可以自定义食材并填写热量。

2) 技术方案

  • 列表展示

① 使用ListListItemGroup实现列表的连续滚动,并快速滚动到指定分类。

  • 自定义添加食物

① 使用bindSheet方法对当前页面绑定半模态弹框,实现自定义食物的添加。


4.视频播放

1) 场景说明

支持查看当日饮食热量摄入情况,或者查看一周热量摄入统计。

2) 技术方案

  • 使用Progress绘制环形数据统计,使用LinearGradient实现环形颜色的渐变效果。

    使用三方库@ohos/mpchart绘制柱状图,展示一周热量摄入信息。


3)代码参考

  • 部分核心代码参见热量计算实现章节。


四、 模板代码

1. 工程结构(下载模板

详细代码结构如下所示:

Recipes  ├─commons/commonlib/src/main  │  ├─ets  │  │  ├─components  │  │  │      BaseHeader.ets                 // 一级页面标题组件  │  │  │      BuildTitleBar.ets              // 二级页面标题组件  │  │  │      HeaderMenuBuilder.ets          // 标题菜单内容组件  │  │  │      MenuItemBuilder.ets            // 下拉菜单选项  │  │  ├─constants  │  │  │      CommonContants.ets             // 公共常量  │  │  │      CommonEnums.ets                // 公共枚举值  │  │  ├─types  │  │  │      Types.ets                      // 公共抽象类  │  │  └─utils  │  │         AccountUtil.ets                // 账号工具类  │  │         DialogUtil.ets                 // 弹窗工具类  │  │         FormatUtil.ets                 // 格式化工具类  │  │         Logger.ets                     // 日志工具类  │  │         PermissionUtil.ets             // 权限获取工具类  │  │         PreferenceUtil.ets             // 数据持久化工具类  │  │         RouterModule.ets               // 路由工具类  │  │         WindowUtil.ets                 // 窗口管理工具类  │  └─resources  ├─commons/network/src/main  │  ├─ets  │  │  ├─apis  │  │  │      APIList.ets                    // 网络请求API  │  │  │      HttpRequest.ets                // 网络请求  │  │  ├─mocks  │  │  │  └─MockData  │  │  │         Calories.ets                // 热量mock数据  │  │  │         Mine.ets                    // 我的mock数据  │  │  │         Notice.ets                  // 通知mock数据  │  │  │         RecipeList.ets              // 菜谱mock数据  │  │  │      AxiosMock.ets                  // mock请求  │  │  │      RequestMock.ets                // mock API  │  │  └─types  │  │         Calories.ets                   // 热量抽象类  │  │         Member.ets                     // 会员抽象类  │  │         Notice.ets                     // 通知抽象类  │  │         Recipe.ets                     // 菜谱抽象类  │  └─resources  │─components/aggregated_ads/src/main     │  ├─ets  │  │  ├─common  │  │  │      Constant.ets                   // 常量类  │  │  ├─components  │  │  │      AdServicePage.ets              // 广告服务组件  │  │  │      HwAdService.ets                // 华为广告  │  │  ├─util  │  │  │      UIUtil.ets                     // UI工具类  │  │  └─viewmodel  │  │         AggreagetedAdVM.ets            // 广告页面数据模型  │  └─resources  │─components/aggregated_login/src/main     │  ├─ets  │  │  ├─common  │  │  │      Constant.ets                   // 常量类  │  │  │      Logger.ets                     // 日志类  │  │  ├─components  │  │  │      AgreementDialog.ets            // 协议弹窗组件  │  │  │      LoginService.ets               // 登录组件  │  │  ├─model  │  │  │      Index.ets                      // 数据类型  │  │  │      WXApiWrap.ets                  // 微信登录数据类型  │  │  └─viewmodel  │  │         AggregatedLoginVM.ets          // 登录组件数据模型  │  └─resources  │─components/aggregated_payment/src/main     │  ├─ets  │  │  ├─common  │  │  │      Constant.ets                   // 常量类  │  │  ├─components  │  │  │      AggregatedPaymentPicker.ets    // 支付组件  │  │  ├─model  │  │  │      Index.ets                      // 数据类型  │  │  │      WXApiWrap.ets                  // 微信支付数据类型  │  │  └─viewmodel  │  │         AggregatedPaymentVM.ets        // 支付组件数据模型  │  └─resources  │─components/base_ui/src/main     │  ├─ets  │  │  ├─components  │  │  │      BaseTabs.ets                   // Tabs组件  │  │  └──types  │  │         Index.ets                      // 数据类型  │  └─resources  │─components/calorie_calculation/src/main     │  ├─ets  │  │  ├─components  │  │  │      BarChart.ets                   // 协议弹窗组件  │  │  │      CalorieCalculation.ets         // 热量计算组件  │  │  │      CaloriesSummary.ets            // 热量汇总组件  │  │  │      FoodDiary.ets                  // 饮食计划组件  │  │  ├─types  │  │  │      Index.ets                      // 数据类型  │  │  └─viewModels  │  │         CaloriesSummaryVM.ets          // 热量计算数据模型  │  └─resources  │─components/featured_recipes/src/main     │  ├─ets  │  │  ├─components  │  │  │      FeaturedRecipes.ets            // 菜谱瀑布流组件  │  │  │      RecommendedCard.ets            // 菜谱卡片  │  │  │─types  │  │  │      Index.ets                      // 数据类型  │  │  └─utils  │  │         LazyDataSource.ets             // 懒加载对象  │  │         Logger.ets                     // 日志工具  │  │         ObservedArray.ets              // 数组监听工具  │  └─resources  │─components/home_search/src/main     │  ├─ets  │  │  ├─components  │  │  │      HomeSearch.ets                 // 搜索组件  │  │  │      SearchKeys.ets                 // 热搜词组件  │  │  │      SearchResult.ets               // 搜索结果组件  │  │  └─types  │  │         Index.ets                      // 数据类型  │  └─resources  │─components/link_category/src/main     │  ├─ets  │  │  ├─components  │  │  │      LinkCategory.ets               // 分类列表组件  │  │  └─types  │  │         Index.ets                      // 数据类型  │  └─resources  │─components/personal_homepage/src/main     │  ├─ets  │  │  ├─components  │  │  │      BloggerInfomation.ets          // 个人介绍组件  │  │  │      PersonalHomepage.ets           // 个人中心组件  │  │  └─types  │  │         Index.ets                      // 数据类型  │  └─resources  │─components/shopping_basket/src/main     │  ├─ets  │  │  ├─components  │  │  │      IngredientList.ets             // 用料列表  │  │  │      PurchaseIngredients.ets        // 用料组件  │  │  │      ShoppingBasket.ets             // 菜篮子组件  │  │  └─types  │  │         Index.ets                      // 数据类型  │  └─resources  │─components/upload_recipe/src/main     │  ├─ets  │  │  ├─components  │  │  │      UploadRecipe.ets               // 上传菜谱组件  │  │  ├─types  │  │  │      Index.ets                      // 数据类型  │  │  └─viewmodel  │  │         UploadRecipeVM.ets             // 上传菜谱数据模型  │  └─resources  │─features/calories/src/main     │  ├─ets  │  │  ├─pages  │  │  │      CaloriesPage.ets               // 热量页面  │  │  │      DietPlanPage.ets               // 饮食计划食物列表页面  │  │  │      SearchFoodPage.ets             // 食物搜索页面  │  │  ├─types  │  │  │      Index.ets                      // 数据对象  │  │  └─viewModels  │  │         CaloriesPageVM.ets             // 热量页面数据模型  │  │         DietPlanPageVM.ets             // 食物列表页面数据模型  │  │         SearchFoodPageVM.ets           // 食物搜索页面数据模型  │  └─resources  │─features/classification/src/main     │  ├─ets  │  │  ├─constants  │  │  │      Enums.ets                      // 枚举数据  │  │  ├─pages  │  │  │      ClassificationPage.ets         // 分类页面  │  │  │      DishesPage.ets                 // 菜谱详情页面  │  │  │      ShoppingBasketPage.ets         // 菜篮子页面  │  │  ├─types  │  │  │      Index.ets                      // 菜谱数据对象  │  │  └─viewModels  │  │         ClassificationVM.ets           // 分类页面数据模型  │  │         DishesVM.ets                   // 菜谱详情页面数据模型  │  └─resources  │─features/home/src/main     │  ├─ets  │  │  ├─pages  │  │  │      BloggerProfilePage.ets         // 博主信息页面  │  │  │      FollowersPage.ets              // 博主关注页面  │  │  │      HomePage.ets                   // 首页页面  │  │  │      SearchPage.ets                 // 搜索页面  │  │  ├─types  │  │  │      Index.ets                      // 首页数据对象  │  │  └─viewModels  │  │         BloggerProfilePageVM.ets       // 博主信息页面数据模型  │  │         FollowersPageVM.ets            // 博主关注页面数据模型  │  │         HomePageVM.ets                 // 首页页面数据模型  │  │         SearchPageVM.ets               // 搜索页面数据模型  │─features/mine/src/main     │  ├─ets  │  │  ├─components  │  │  │      ConfirmDialogComponent.ets     // 确认弹窗组件  │  │  │      Recipes.ets                    // 菜谱卡片组件  │  │  ├─mapper  │  │  │      Index.ets                      // 数据映射  │  │  ├─model  │  │  │      Index.ets                      // 数据类型  │  │  ├─pages  │  │  │      BrowsingHistory.ets            // 浏览历史页面  │  │  │      MemberCenterPage.ets           // 会员中心页面  │  │  │      MinePage.ets                   // 我的页面  │  │  │      MyCollection.ets               // 我的收藏页面  │  │  │      NoticeCenterPage.ets           // 通知中心页面  │  │  │      NoticeDetailPage.ets           // 通知详情页面  │  │  │      PersonalInfo.ets               // 个人信息页面  │  │  │      PrivacyPolicyDetailPage.ets    // 用户协议页面  │  │  │      QuickLoginPage.ets             // 一键登录页面  │  │  │      SettingsPage.ets               // 设置页面  │  │  │      SideBarPage.ets                // 服务菜单页面  │  │  │      TermsOfServicePage.ets         // 隐私政策页面  │  │  │      UploadRecipe.ets               // 上传菜谱页面  │  │  ├─types  │  │  │      Index.ets                      // 抽象类  │  │  ├─util  │  │  │      MockApi.ets                    // 支付mock数据  │  │  │      OrderInfoUtil.ets              // 支付mock参数工具  │  │  │      SignUtils.ets                  // 支付mock签名工具  │  │  └─viewModels  │  │         BrowsingHistoryVM.ets          // 浏览历史页面数据模型  │  │         MemberCenterPageVM.ets         // 会员中心页面数据模型  │  │         MinePageVM.ets                 // 我的页面数据模型  │  │         MyCollectionVM.ets             // 我的收藏页面数据模型  │  │         MyRecipeVM.ets                 // 我的菜谱页面数据模型  │  │         NoticeCenterPageVM.ets         // 通知中心页面数据模型  │  │         SettingsPageVM.ets             // 设置页面数据模型  │  │         UploadRecipeVM.ets             // 上传菜谱页面数据模型  │  └─resources  └─products/entry/src/main        ├─ets     │  ├─entryability     │  │      EntryAbility.ets               // 应用程序入口     │  ├─entryformability     │  │      EntryFormAbility.ets           // 卡片程序入口     │  ├─pages     │  │      Index.ets                      // 入口页面     │  │      LaunchAdPage.ets               // 广告页面     │  │      LaunchPage.ets                 // 开屏页面     │  │      MainEntry.ets                  // 主页面     │  │      PrivacyPolicyPage.ets          // 隐私协议页面     │  ├─types     │  │      Types.ets                      // 抽象类     │  ├─viewModels     │  │      MainEntryVM.ets                // 入口页面数据     │  └─widget/pages     │         WidgetCard.ets                 // 卡片页面     └─resources
复制代码


2. 关键代码解读

本篇代码非应用的全量代码,只包括应用的部分能力的关键代码。

若需获取全量代码,请查看模板集成章节。

1) 个人信息

  • 华为账号一键登录

// components/aggregated_login/src/main/ets/components/LoginService.ets// 登录组件LoginWithHuaweiIDButton({    params: {        style: loginComponentManager.Style.BUTTON_CUSTOM,        loginType: loginComponentManager.LoginType.QUICK_LOGIN,        supportDarkMode: true,        customButtonParams: {            fontColor: loginComponentManager.FontColor.WHITE,            backgroundColor: this.loginBtnBgColor,        },    },    controller: this.controller,});
...
// 登录方法控制器controller: loginComponentManager.LoginWithHuaweiIDButtonController = new loginComponentManager.LoginWithHuaweiIDButtonController()// 需要用户同意协议才能完成华为账号登录,请先设置协议状态为NOT_ACCEPTED,当用户同意协议后设置协议状态为ACCEPTED,才可以完成华为账号登录 .setAgreementStatus(loginComponentManager.AgreementStatus.NOT_ACCEPTED) .onClickLoginWithHuaweiIDButton((error: BusinessError, response: loginComponentManager.HuaweiIDCredential) => { // 处理用户点击一键登录按钮逻辑,灰度传入undefined模拟流程,应用申请权限后,传入error this.handleLoginWithHuaweiIDButton(undefined, response);});
...
// 处理点击一键登录后的方法handleLoginWithHuaweiIDButton(error: BusinessError | undefined, response: loginComponentManager.HuaweiIDCredential) { this.enableStatus = false; // if部分内容配置好可用的调试证书和client_id后再放开 if (error) { Logger.error(this.domainId, this.logTag, `Failed to click LoginWithHuaweiIDButton. Code is ${error.code}, message is ${error.message}`); this.error = error; if (error.code === LoginErrorCode.ERROR_CODE_NETWORK_ERROR) { AlertDialog.show( { message: 'No Internet connection. Check your network settings. ', offset: { dx: 0, dy: -12 }, alignment: DialogAlignment.Bottom, autoCancel: false, confirm: { value: 'OK', action: () => { }, }, }, ); } else if (error.code === LoginErrorCode.ERROR_CODE_AGREEMENT_STATUS_NOT_ACCEPTED) { // 未同意协议,弹出协议弹框,推荐使用该回调方式 this.agreementDialog.open(); this.customerDiaLogOpen = true; } else if (error.code === LoginErrorCode.ERROR_CODE_LOGIN_OUT) { // 华为账号未登录提示 this.showToast($r('app.string.not_login_in')); } else if (error.code === LoginErrorCode.ERROR_CODE_NOT_SUPPORTED) { // 账号不支持该scopes或permissions提示 this.showToast($r('app.string.scope_not_supported')); } else if (error.code === LoginErrorCode.ERROR_CODE_NOT_REQUIRED_SCOPE_OR_PERMISSION) { // 应用没有申请scope权限 this.showToast($r('app.string.app_not_required_scopes_or_permissions')); } else { // 其他提示系统或服务异常 this.showToast($r('app.string.service_error')); } this.enableStatus = true; return; } try { if (this.isSelected) { // 配置好可用的调试证书和client_id后需要处理response Logger.info(this.domainId, this.logTag, 'Succeed in clicking LoginWithHuaweiIDButton.'); if (this.response.idToken !== undefined) { this.response = response; // 模拟登录成功后回调 this.loginFinishedCb(true, this.unionID) } // 模拟登录成功后回调 this.loginFinishedCb(false) } else { this.agreementDialog.open(); this.customerDiaLogOpen = true; } } catch (e) { Logger.info(this.domainId, this.logTag, `Failed to LoginWithHuaweiIDButton, errCode: ${e.code}, errMessage: ${e.message}`); AlertDialog.show( { message: $r('app.string.service_error'), offset: { dx: 0, dy: -12 }, alignment: DialogAlignment.Bottom, autoCancel: false, confirm: { value: 'OK', action: () => { }, }, }, ); } finally { this.enableStatus = true; }}
复制代码


2) 分类搜索

  • 分类列表

// components/link_category/src/main/ets/components/LinkCategory.ets// 列表页Row() {    List({ space: 8, scroller: this.titleItemScroller }) {        ForEach(this.recipeCategoryList, (item: RecipeCategory, index: number) => {            ListItem() {                this.leftListBuilder(item.name, index)            }        }, (item: RecipeCategory, index: number) => item.name + index)    }    .width(92)    .height('100%')    .backgroundColor('#F1F3F5')    .listDirection(Axis.Vertical) // 排列方向    .scrollBar(BarState.Off)    .contentStartOffset(12)    .contentEndOffset(12)
List({ space: 12, scroller: this.scroller }) { ForEach(this.recipeCategoryList, (item: RecipeCategory) => { ListItemGroup({ header: this.listTitleBuilder(item.name), space: 12, }) { ListItem() { Grid() { ForEach(item.recipeList, (listItem: RecipeBriefInfo) => { GridItem() { Column() { Image($r(`app.media.${listItem.thumbnail}`)).width(76).height(76).borderRadius(8) Text(listItem.title) .fontSize(14) .fontWeight(FontWeight.Medium) .fontColor($r('sys.color.font_primary')) .textAlign(TextAlign.Center) .constraintSize({ maxWidth: 76 }) .maxLines(2) .margin({ top: 4 }) .textOverflow({ overflow: TextOverflow.Ellipsis }) } }.onClick(() => { this.onRecipeClick(listItem) })
}, (listItem: RecipeBriefInfo) => `${item.id}${listItem.id}`) } .rowsGap(8) .columnsGap(8) .columnsTemplate('1fr 1fr 1fr')
}
} }, (item: RecipeCategory) => item.id.toString()) } .layoutWeight(1) .height('100%') .margin({ left: 8, right: 16 }) .scrollBar(BarState.Off) .sticky(StickyStyle.None) .contentStartOffset(12) .contentEndOffset(12) .onScrollIndex((start: number) => this.currentIndexChangeAction(start, false))}.layoutWeight(1)
...
// 下标索引处理currentIndexChangeAction(index: number, isClassify: boolean): void { if (this.currentIndex !== index) { this.changeCurrentIndex(index); if (isClassify) { this.scroller.scrollToIndex(index); } else { this.titleItemScroller.scrollToIndex(index); } }}
复制代码


3)热量计算

  • 环形图

// commons/lib_common/src/main/ets/utils/LoginUtils.ets// 环形组件Progress({ value: this.todayCalories, total: 2000, type: ProgressType.Ring })            .width(176).style({ strokeWidth: 20 })            .color(this.vm.gradientColor)
...
// components/calorie_calculation/src/main/ets/viewModels/CaloriesSummaryVM.ets// 渐变色设置gradientColor: LinearGradient = new LinearGradient([{ color: '#C6F093', offset: 0.25 }, { color: '#83CE26', offset: 0.5 }, { color: '#88DB23', offset: 0.85 }, { color: '#88DB24', offset: 1.0 }])
复制代码


  • 柱状图

// components/calorie_calculation/src/main/ets/components/BarChart.ets// 柱状图参数设置this.name = this.xData[this.xData.length - 1]this.rate = this.seriesData[this.seriesData.length - 1]// Step1:必须:初始化图表配置构建类this.model = new BarChartModel();
// Step2:配置图表指定样式,各部件间没有先后之分
// 为图表添加数据选择的监听器this.model.setOnChartValueSelectedListener(this.valueSelectedListener);// 获取图表描述部件,设置图表描述部件不可用,即图表不进行绘制描述部件let description: Description | null = this.model.getDescription()if (description) { description.setEnabled(false);}
// 获取图表图例部件,设置图表图例部件不可用let legend: Legend | null = this.model.getLegend();if (legend) { legend.setEnabled(false);}
// 设置图表数据最大的绘制数,如果超过该数值,则不进行绘制图表的数值标签this.model.setMaxVisibleValueCount(7);
// 为左Y轴设置LimitLine,可设置限制线的宽度,线段样式,限制标签的位置,标签字体大小等this.limitLine1 = new LimitLine(120, 'Upper Limit');this.limitLine1.setLineWidth(4);//设置虚线样式this.limitLine1.enableDashedLine(10, 10, 0);//设置标签位置this.limitLine1.setLabelPosition(LimitLabelPosition.RIGHT_TOP);this.limitLine1.setTextSize(10);
this.limitLine2 = new LimitLine(50, 'Lower Limit');this.limitLine2.setLineWidth(4);this.limitLine2.enableDashedLine(10, 10, 0);this.limitLine2.setLineColor(Color.Yellow);this.limitLine2.setLabelPosition(LimitLabelPosition.RIGHT_BOTTOM);this.limitLine2.setTextSize(10);
// 设置图表左Y轴信息this.leftAxis = this.model.getAxisLeft();if (this.leftAxis) { //设置绘制标签个数 this.leftAxis.setLabelCount(7, false); //设置标签位置 this.leftAxis.setPosition(YAxisLabelPosition.OUTSIDE_CHART) //设置距离顶部距离 this.leftAxis.setSpaceTop(15); //设置最小值 this.leftAxis.setAxisMinimum(0); //设置最大值 this.leftAxis.setAxisMaximum(2000); this.leftAxis.setGridDashedLine({ dash: [2, 2, 0], offset: 0 }) this.leftAxis.setTextColor('#E6000000') this.leftAxis.setTextSize(12)}
// 设置图表右Y轴信息this.rightAxis = this.model.getAxisRight();if (this.rightAxis) { this.rightAxis.setEnabled(false);}
class YAxisValueFormatter implements IAxisValueFormatter { getFormattedValue(value: number, axis: AxisBase): string { //将原本存在的对应的value转换成需要的字符串 return value.toString(); }}
this.leftAxis?.setValueFormatter(new YAxisValueFormatter())
// 设置X轴信息this.xAxis = this.model.getXAxis();if (this.xAxis) { this.xAxis.setPosition(XAxisPosition.BOTTOM); this.xAxis.setDrawGridLines(false); this.xAxis.setGranularity(1); this.xAxis.setLabelCount(this.xData.length); this.xAxis.setTextColor('#E6000000') this.xAxis.setTextSize(12)}
class XAxisValueFormatter implements IAxisValueFormatter { data: string[] = []
constructor(data: string[]) { this.data = data }
getFormattedValue(value: number, axis: AxisBase): string { //将原本存在的对应的value转换成需要的字符串 let day = new Date().getDay() - 1 if (day === value) { return '今日' } return xMap[value]; }}
this.xAxis?.setValueFormatter(new XAxisValueFormatter(this.xData))// 生成图表数据let barData: BarData = this.getBarData();// 将数据与图表配置类绑定this.model.setData(barData);// 设置图表最大的X轴显示范围,如不设置,则默认显示全部数据this.model.setVisibleXRangeMaximum(20);
...
// 柱状图组件BarChart({ model: this.model }).width('100%').height(190)
复制代码


3. 模板集成

本模板提供了两种代码集成方式,供开发者自由选用。

1) 整体集成(下载模板

开发者可以选择直接基于模板工程开发自己的应用工程。

  • 模板代码获取:

① 通过 IDE 插件创建模板工程,开发指导

② 通过生态市场下载源码, 下载模板

③ 通过开源仓访问源码,仓库地址

  • 打开模板工程,根据 README 说明中的快速入门章节,将自己的应用信息配置在模板工程内,即可运行并查看模板效果。



  • 对接开发者自己的服务器接口,转换数据结构,展示真实的云侧数据。

将 commons/lib_common/src/main/ets/httprequest/HttpRequestApi.ets 文件中的 mock 接口替换为真实的服务器接口。



在 commons/lib_common/src/main/ets/httprequest/HttpRequest.ets 文件中将云侧开发者自定义的数据结构转换为端侧数据结构。


 

根据自己的业务内容修改模板,进行定制化开发。


2) 按需集成

若开发者已搭建好自己的应用工程,但暂未实现其中的部分场景能力,可以选择取用其中的业务组件,集成在自己的工程中。

  • 组件代码获取:

① 通过 IDE 插件下载组件源码。开发指导

② 通过生态市场下载组件源码。 下载地址

  • 下载组件源码,根据 README 中的说明,将组件包配置在自己的工程中。



  • 根据 API 参考和示例代码,将组件集成在自己的对应场景中。


以上是第四期“餐饮行业-美食菜谱”行业优秀案例的内容,更多行业敬请期待~

欢迎下载使用行业模板“点击下载”,若您有体验和开发问题,或者迫不及待想了解 XX 行业的优秀案例,欢迎在评论区留言,小编会快马加鞭为您解答~

同时诚邀您添加下方二维码加入“组件模板活动社群”,精彩上新 &活动不错过!

👉本系列持续更新,欢迎收藏本帖!

第一期:HarmonyOS官方模板优秀案例 | 便捷生活行业· 购物中心

第二期:HarmonyOS官方模板优秀案例 | 新闻行业· 综合新闻

第三期:HarmonyOS官方模板优秀案例 | 教育行业· 教育备考

第四期:HarmonyOS官方模板优秀案例 | 餐饮行业· 美食菜谱

第五期:HarmonyOS官方模板优秀案例 | 工具行业· 日历应用

第六期:……(小编加急整理中,敬请期待)


👉HarmonyOS 组件模板相关推荐

  • 【活动 ing】HarmonyOS 组件/模板集成创新活动,报名时间截止 2025 年 8 月 30 日,点击查看

  • ·鸿蒙应用开发者激励计划 2025,点击查看

2025-08-28 16:4416

评论

发布
暂无评论

线上服务 CPU 100% ?一键定位 so easy!

Java小咖秀

性能 cpu 服务器 负载 紧急问题

派出所重点人员管控系统开发,建设智慧警务

13828808769

智慧组工

装双系统?不需要!教你在iMac上流畅使用Windows

懒得勤快

Mac 虚拟机 苹果 crossover

css网页布局小结

Darren

CSS

将本地maven仓库的数据恢复到Nexus仓库

白粥

工作笔记

华为帐号服务学习笔记(三):10分钟完成Authorization Code模式客户端Demo开发

Coding狙击

android HMS

亿网嘉元是做什么的?

飞亚科技

OKR实践中的痛点(5):战略缺失怎么玩OKR?

大叔杨

团队管理 OKR 敏捷 敏捷绩效

云数据库时代的新思考,这位90后大咖想邀你聊聊

华为云开发者联盟

数据库 开源 opengauss GaussDB 华为云数据库

Dubbo 学习笔记(三) Spring Boot 整合 Dubbo(官方版)

U2647

Spring Boot dubbo 4月日更

进来看看是不是你想要的效果,Android吸顶效果,并有着ViewPager左右切换

第三女神程忆难

Java android kotlin 安卓 移动开发

MySQL 事务隔离

Sakura

4月日更

【LeetCode】子集二Java题解

Albert

算法 LeetCode 4月日更

云小课 | 不了解EIP带宽计费规则?看这里!

华为云开发者联盟

带宽 弹性公网IP 带宽变更 计费模式

技术人如何调研和选型第三方 SDK?全文干货

融云 RongCloud

数据分析与数据增长核心逻辑杂谈

小飞象@木木自由

数据分析

SumSwap在市场上的强大突破是否会成为DEX领域最大的黑马?

币圈资讯

一文带你剖析LiteOS互斥锁Mutex源代码

华为云开发者联盟

mutex LiteOS 互斥锁 互斥锁结构体

2D+1D | vivo官网Web 3D应用开发与实战

vivo互联网技术

大前端 WebGL 3D数据可视化 Draco 3D

Linux rmdir 命令

一个大红包

linux命令 4月日更

Cloudreve 自建云盘实践,我说了没人能限得了我的容量和速度!

小傅哥

Java 小傅哥 Cloudreve 自建云盘

systemctl的使用

箭上有毒

linux运维 4月日更

「免费开源」基于Vue和Quasar的前端SPA项目crudapi后台管理系统实战之业务数据增删改查(七)

crudapi

Vue API crud crudapi quasar

公有云成本节省神器!京东云共享带宽包正式上线

京东科技开发者

公有云 带宽

创建索引,这些知识应该了解

Simon

MySQL 索引

智慧公安情报综合研判平台开发,助推公安信息化发展

13828808769

智慧城市

区块链电子合同技术方案,区块链电子合同存证

13828808769

区块链 区块链+

NA(Nirvana)公链“为应用而生” NAC公链领跑公链新格局!

区块链第一资讯

你的数仓函数结果不稳定,可能是属性指定错了

华为云开发者联盟

函数 GaussDB(DWS) 函数属性 函数下推 易失性级别

大意!6行代码,“报废”5片单片机!

不脱发的程序猿

程序人生 嵌入式软件 单片机 4月日更 国产MCU

划重点丨详解Java流程控制语句知识点

华为云开发者联盟

Java 流程控制语句

【推荐+1】HarmonyOS官方模板优秀案例
(第4期:餐饮行业 · 美食菜谱)_HarmonyOS_HarmonyOS_InfoQ精选文章