• 如何让开发者直接在应用后台控制用户的运动状态?


    酷暑终于过去,很多人伴着凉爽的秋风开启了新一轮的健身计划。当用户进行户外运动或使用跑步机、椭圆机等器械时,他们会希望在运动健康类App里点击即可开启运动并记录运动数据。而对于开发者自己开发的应用来说,用户在使用跟华为健康App绑定的运动设备运动时,可以直接在自己的App后台控制用户运动状态并获取实时数据,不需要再从华为健康App里进行操作。

    那么,对于运动健康App来说,如何实现以上功能呢?HMS Core运动健康服务的扩展能力服务开放更多实时运动和健康数据、运动和健康解决方案场景化数据。其中控制运动并获取实时运动数据能力提供了开始、暂停、恢复和结束运动的接口,开发者可以直接在应用内调用接口,后台控制运动健康App中对应运动状态,无需跳转到华为运动健康App运动界面进行操作,此时运动健康App不会弹出运动页面,而是在后台执行。

    同时华为也提供了获取实时运动数据和停止获取实时运动数据的接口,为防止数据丢失,一般在开始运动之前调用获取实时运动数据接口,在停止运动之后调用停止获取实时运动数据接口。如果用户绑定了华为穿戴设备,启动运动时,穿戴设备将自动进入运动界面;结束运动时,穿戴设备将自动结束运动。使用接口前,需要向华为申请开通权限,并获取用户授权,否则接口将调用失败。目前支持的运动类型:户外步行、户外跑步、户外骑行、室内跑步(跑步机)、椭圆机、划船机、室内单车。具体场景获取的数据类型请参考实时运动 Bundle 对象键值

    前台运动跳转设备配对页面

    Demo

    开发步骤

    开发准备

    1. 申请Health Kit服务

    申请Health Kit服务前,请先完成申请帐号服务。

    2 .集成 HMS Core SDK

    集成SDK之前,请先集成华为帐号服务SDK

    在开始开发前,请先将SDK集成到Android Studio开发环境中。Android Studio应为V3.3.2及以上版本。

    开发步骤

    1 .开始获取实时运动数据

    1. 调用HiHealthDataStore对象的registerSportData方法,开始获取实时运动数据。

    2. 通过请求参数HiSportDataCallback对象,返回查询结果,结果中数据类型参考实时运动 Bundle 对象键值

    示例代码:

    HiHealthDataStore.registerSportData(context, new HiSportDataCallback() {    
    
        @Override    
        public void onResult(int resultCode) {
            // 接口调用结果 
            Log.i(TAG, "registerSportData onResult resultCode:" + resultCode);   
        }
        @Override    
        public void onDataChanged(int state, Bundle bundle) {
            // 实时数据变化回调        
            Log.i(TAG, "registerSportData onChange state: " + state);        
            StringBuffer stringBuffer = new StringBuffer("");              
            if (state == HiHealthKitConstant.SPORT_STATUS_RUNNING) {
                Log.i(TAG, "heart rate : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_HEARTRATE));
                Log.i(TAG, "distance : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_DISTANCE));
                Log.i(TAG, "duration : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_DURATION));
                Log.i(TAG, "calorie : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_CALORIE));
                Log.i(TAG, "totalSteps : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_TOTAL_STEPS));
                Log.i(TAG, "totalCreep : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_TOTAL_CREEP));
                Log.i(TAG, "totalDescent : " + bundle.getInt(HiHealthKitConstant.BUNDLE_KEY_TOTAL_DESCENT));
            }    
        }
    });
    

    2. 停止获取实时运动数据

    1. 调用HiHealthDataStore对象的unregisterSportData方法,停止获取实时运动数据。

    2. 通过请求参数HiSportDataCallback对象,返回查询结果。

    示例代码:

    HiHealthDataStore.unregisterSportData(context, new HiSportDataCallback() {    
        JSONObject jsonResult
        @Override    
        public void onResult(int resultCode) {
            // 接口调用结果
            Log.i(TAG, "unregisterSportData onResult resultCode:" + resultCode);   
        }
        @Override    
        public void onDataChanged(int state, Bundle bundle) {
            // 此时不会被调用     
        }
    });
    

    3. 根据运动类型开始运动

    1. 调用HiHealthDataStore对象的startSport方法,启动相应类型的运动。

    2. 通过请求参数ResultCallback对象,返回查询结果。

    示例代码:

    // 室外跑步
    int sportType = HiHealthKitConstant.SPORT_TYPE_RUN;
    HiHealthDataStore.startSport(context, sportType, new ResultCallback() {
        @Override
        public void onResult(int resultCode, Object message) {
            if (resultCode == HiHealthError.SUCCESS) {
                Log.i(TAG, "start sport success");
            }
        }
    });
    
    1. 对于器械运动(例如跑步机运动、划船机运动、椭圆机运动、动感单车运动),需要区分有无配对设备的场景,例如用户想开启划船机运动:

    此时运动健康App已配对一台划船机,那么默认连接该配对设备,然后开启后台运动。

    此时运动健康App配对不止一台划船机,那么会弹框选择设备,点击开始运动后返回到应用页面,然后开启后台运动。

    此时运动健康App没有配对划船机设备,那么会跳转到运动健康App一键扫描页面配对划船机设备,如下图所示,配对成功后会返回到应用页面,然后开启后台运动。

    4. 根据设备信息开始运动

    1. 调用HiHealthDataStore对象的startSportEx方法,传入相应启动参数StartSportParam,可通过设置参数CharacteristicConstant.SportModeType控制前台还是后台启动运动。

    2. 通过请求参数ResultCallback对象,返回开启运动状态的结果。

    示例代码:

    // 划船机为例
    // Mac地址,连接符为":",例:"11:22:33:44:55:66"
    String macAddress = "11:22:33:44:55:66" ;
    // 是否支持FTMP,0 不支持,1 支持
    int isSupportedFtmp = CharacteristicConstant.FtmpType.FTMP_SUPPORTED.getFtmpTypeValue();
    // 设备类型,划船机
    int deviceType = CharacteristicConstant.DeviceType.TYPE_ROWER_INDEX.getDeviceTypeValue();
    // 运动类型,划船机
    int sportType = CharacteristicConstant.EnhanceSportType.SPORT_TYPE_ROW_MACHINE.getEnhanceSportTypeValue();
    // 构造启动参数,用于连接设备及控制运动
    StartSportParam param = new StartSportParam(macAddress, isSupportedFtmp, deviceType, sportType);
    // 启动方式,0 前台,1 后台
    param.putInt(HiHealthDataKey.IS_BACKGROUND,
        CharacteristicConstant.SportModeType.BACKGROUND_SPORT_MODE.getType());
    HiHealthDataStore.startSportEx(mContext, param, new ResultCallback() {
        @Override
        public void onResult(int resultCode, Object message) {
            
            if (resultCode == HiHealthError.SUCCESS) {
                Log.i(TAG, "start sportEx success");
            }
        }
    });
    

    5. 结束运动

    1. 调用HiHealthDataStore对象的stopSport方法,停止相应类型的运动(前台开启的运动无法通过stopSport方法停止)。

    2. 通过请求参数ResultCallback对象,返回查询结果。

    示例代码:

    HiHealthDataStore.stopSport(context, new ResultCallback() {
        @Override
        public void onResult(int resultCode, Object message) {
            if (resultCode == HiHealthError.SUCCESS) {
                Log.i(TAG, "stop sport success");
            }
        }
    });
    

    了解更多详情>>

    访问华为开发者联盟官网
    获取开发指导文档
    华为移动服务开源仓库地址:GitHubGitee

    关注我们,第一时间了解 HMS Core 最新技术资讯~

  • 相关阅读:
    【pygame】01 pygame制作游戏的最小系统
    【图形学】13 UnityShader语义(一)
    批量删除docker中tag为<none>的镜像
    Excel VLOOKUP 使用记录
    LeetCode-分割回文串(C++)
    计网第五章(运输层)(六)(TCP可靠传输的实现)
    实践了上万次,原来这些才是敏捷测试需要遵循的原则
    【Leetcode】【每日一题】【简单】2558. 从数量最多的堆取走礼物
    嵌入式linux sqlite3读写demo
    xml开发mybatis
  • 原文地址:https://www.cnblogs.com/hmscore/p/16671977.html