Android车载应用开发与分析(10)- 车载空调系统(HVAC)

news/2024/11/29 10:48:55/

1. HVAC 功能介绍

HVAC 全称:供暖通风与空气调节(Heating Ventilation and Air Conditioning)。用户可以通过他来控制整个汽车的空调系统,是汽车中非常重要的一个功能。

汽车的空调HMI虽然并不复杂,但是大多都是用符号来表示功能,对于还没有实际用过汽车空调系统的开发者来说,理解空调的各个符号表示的含义也是非常有必要。
下面就以Android 12中的HVAC来介绍空调系统中包含的最基础的功能。

1.1 双区温度调节

空调的温度调节功能,默认是华氏度,可以在系统设置修改温度单位。可调节范围是61 - 82华氏度,对应16 - 28 摄氏度。
左侧按钮用来调节主驾,右侧按钮用来调节副驾。在以往都是只有高配车型才有双区空调,现在的车上双区空调几乎已经是标配了。

1.2 空调开关


开启关闭空调的开关

1.3 内/外循环


内循环是汽车空气调节系统的一种状态。这种状态下,车内外的换气通道关闭,风机关闭时车内气流不循环,风机开启时,吸入的气流也仅来自车内,形成车辆内部的气流循环。
外循环则相反,风机开启时,吸入的气流也仅来自车外,可以更新车内的空气质量,代价是会更耗电。

1.4 风量调节


用于增大或减小空调的风量。

1.5 风向调节


从左到右分别是吹脸、吹脸+吹脚、吹脚、吹脚+吹挡风玻璃

1.6 A/C开关


A/C按键,它就是制冷开关,按下A/C按键,也就启动了压缩机,通俗地说就是开冷气。

1.7 主副驾座椅加热


左边的按钮用于调节主驾座椅加热,右边的按钮用于调节副驾座椅加热

1.8 除霜


左边的按钮是开启/关闭 前挡风玻璃加热,开启后用来除去前挡风玻璃上的雾气。右边的按钮是开启/关闭后挡风玻璃加热,开启后用来除去后挡风玻璃上的雾气。

1.9 自动模式


自动空调其实就是省略了风速、风向等调节功能,自动空调是全自动调节,只需要选择风向和设定温度。AUTO按键按下后,就会根据车内传感器来控制出风的温度,冬天热风,夏天冷风。会保持车内有较适宜的温度,如果温度过高或过低,空调也会自动改变出风口的温度及风速,调整车内温度。
以上就是车载空调系统中最基础的功能了,实际开发中我们还会遇到如座椅通风、座椅按摩、智能新风、负离子等等一些近几年才出现的空调新功能,在应用开发上无非就是多几个界面或按钮。

2. HVAC 源码结构

本文中的源码基于Android 12下HVAC APP,源码请见:https://github.com/linux-link/CarHvac
原生的Hvac App中不存在Activity、Fragment等传统意义上用来显示HMI的组件,取而代之是使用Service来显示一个Window。主要原因在于Hvac的界面层级比一般的HMI的层级要高,呼出Hvac时需要部分或全部覆盖其他的应用上(当然IVI中还是有应用比Hvac的层级要高的),这时候使用Activity就显不合适了。

需要注意的是,Havc在Android 12中虽然有一个独立的app,但是上图中Hvac其实并不是使用这个独立的app,它的HMI和相关其实都是写在SystemUI中的。
通过使用adb发送一个广播,可以调出真正的Hvac。

adb shell am broadcast -a android.car.intent.action.TOGGLE_HVAC_CONTROLS


以下是Hvac App的关键部分的源码结构图

3. HVAC 核心源码分析

3.1 AndroidManifest.xml

<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.android.car.hvac"><uses-sdkandroid:minSdkVersion="22"android:targetSdkVersion="29" /><uses-permission android:name="android.car.permission.CONTROL_CAR_CLIMATE" /><uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /><!-- Required to use the TYPE_DISPLAY_OVERLAY layout param for the overlay hvac ui--><uses-permission android:name="android.permission.INTERNAL_SYSTEM_WINDOW" /><!-- Allow Hvac to go across all users--><uses-permission android:name="android.permission.INTERACT_ACROSS_USERS" /><uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL" /><protected-broadcast android:name="android.car.intent.action.TOGGLE_HVAC_CONTROLS" /><applicationandroid:icon="@drawable/ic_launcher_hvac"android:label="@string/hvac_label"android:persistent="true"><!--用于控制空调功能的Service--><serviceandroid:name=".HvacController"android:exported="false"android:singleUser="true" /><!-- 用于显示UI的Service--><serviceandroid:name=".HvacUiService"android:exported="false"android:singleUser="true" /><!-- 监听开机广播 --><receiverandroid:name=".BootCompleteReceiver"android:exported="true"><intent-filter><action android:name="android.intent.action.BOOT_COMPLETED" /></intent-filter></receiver></application>
</manifest>

3.2 BootCompleteReceiver

用于监听开机的广播,当前收到系统的开机广播后,会将HvacUiService拉起。

public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Intent hvacUiService = new Intent(context, HvacUiService.class);context.startService(hvacUiService);}
}

3.3 HvacUiService

HvacUiService 用来托管Hvac UI的Service。从名字上也能看出,整个HvacUiService都是围绕着如何将Hvac准确的绘制出来,基本不含其他的逻辑。

@Override
public void onCreate() {...// 由于不存在从服务内部获取系统ui可见性的方法,因此我们将全屏放置一些东西,并检查其最终测量结果,作为获取该信息的黑客手段。// 一旦我们有了初始状态,我们就可以安全地从那时开始注册更改事件。View windowSizeTest = new View(this) {@Overrideprotected void onLayout(boolean changed, int left, int top, int right, int bottom) {Log.i(TAG, "onLayout: changed" + changed + ";left:" + left + ";top:" + top + ";right:" + right + ";bottom" + bottom);boolean sysUIShowing = (mDisplayMetrics.heightPixels != bottom);mInitialYOffset = (sysUIShowing) ? -mNavBarHeight : 0;Log.i(TAG, "onLayout: sysUIShowing:" + sysUIShowing + ";mInitialYOffset" + mInitialYOffset);layoutHvacUi();// 我们现在有了初始状态,因此不再需要这个空视图。mWindowManager.removeView(this);mAddedViews.remove(this);}};addViewToWindowManagerAndTrack(windowSizeTest, testparams);// 接收事件的广播IntentFilter filter = new IntentFilter();filter.addAction(CAR_INTENT_ACTION_TOGGLE_HVAC_CONTROLS);filter.addAction(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);// 注册接收器,以便任何具有CONTROL_CAR_CLIMATE权限的用户都可以调用它。registerReceiverAsUser(mBroadcastReceiver, UserHandle.ALL, filter,Car.PERMISSION_CONTROL_CAR_CLIMATE, null);
}private BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {@Overridepublic void onReceive(Context context, Intent intent) {String action = intent.getAction();Log.i(TAG, "onReceive: " + action);// 自定义广播,用于展开Hvac的HMIif (action.equals(CAR_INTENT_ACTION_TOGGLE_HVAC_CONTROLS)) {mHvacPanelController.toggleHvacUi();} else if (action.equals(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)) {// home 按键的广播,收起Hvac的HMImHvacPanelController.collapseHvacUi();}}
};// 添加View到WindowManager中
private void addViewToWindowManagerAndTrack(View view, WindowManager.LayoutParams params) {mWindowManager.addView(view, params);mAddedViews.add(view);
}

HvacUIService在onCreate()中主要完成两件事:
1.注册事件广播。这个事件实际并没有发送源,因为SystemUI中额外写了一个Hvac,不过正是这个广播让我们可以把这个单独的Hvac调出。
2.绘制UI。HvacUIService在被拉起后并没有立即开始UI的绘制,而是在屏幕上临时放置一个用于测量窗口的 windowSizeTest ,当windowSizeTestView开始测量后,通过比对View的高度和屏幕的高度,即可判断出systemUI是否已经显示,这时就可以开始着手绘制真正的Hvac的UI了,并且可以更安全的操作UI。
接下来就是绘制真正的Hvac界面:

/*** 在确定最小偏移量后调用。* 这将生成HVAC UI所需的所有组件的布局。* 启动时,折叠视图所需的所有窗口都可见,而展开视图的窗口已创建并调整大小,但不可见。*/
private void layoutHvacUi() {LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);WindowManager.LayoutParams params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.TYPE_DISPLAY_OVERLAY,WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE| WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS& ~WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,PixelFormat.TRANSLUCENT);params.packageName = this.getPackageName();params.gravity = Gravity.BOTTOM | Gravity.LEFT;params.x = 0;params.y = mInitialYOffset;params.width = mScreenWidth;params.height = mScreenBottom;params.setTitle("HVAC Container");disableAnimations(params);// required of the sysui visiblity listener is not triggered.params.hasSystemUiListeners = true;mContainer = inflater.inflate(R.layout.hvac_panel, null);mContainer.setLayoutParams(params);mContainer.setOnSystemUiVisibilityChangeListener(visibility -> {Log.i(TAG, "layoutHvacUi: visibility:" + visibility);boolean systemUiVisible = (visibility & View.SYSTEM_UI_FLAG_FULLSCREEN) == 0;int y = 0;if (systemUiVisible) {// 当systemUi可见时,窗口系统坐标从系统导航栏上方的0开始。因此,如果我们想获得屏幕底部的实际高度,我们需要将y值设置为导航栏高度的负值。y = -mNavBarHeight;}setYPosition(mDriverTemperatureBar, y);setYPosition(mPassengerTemperatureBar, y);setYPosition(mDriverTemperatureBarCollapsed, y);setYPosition(mPassengerTemperatureBarCollapsed, y);setYPosition(mContainer, y);});// 顶部填充应根据屏幕高度和扩展hvac面板的高度进行计算。由填充物定义的空间意味着可以单击以关闭hvac面板。int topPadding = mScreenBottom - mPanelFullExpandedHeight;mContainer.setPadding(0, topPadding, 0, 0);mContainer.setFocusable(false);mContainer.setFocusableInTouchMode(false);View panel = mContainer.findViewById(R.id.hvac_center_panel);panel.getLayoutParams().height = mPanelCollapsedHeight;addViewToWindowManagerAndTrack(mContainer, params);// 创建温度计barcreateTemperatureBars(inflater);// UI状态控制器,用来控制展开/收起时UI的各种状态并执行动画mHvacPanelController = new HvacPanelController(this /* context */, mContainer,mWindowManager, mDriverTemperatureBar, mPassengerTemperatureBar,mDriverTemperatureBarCollapsed, mPassengerTemperatureBarCollapsed);// 绑定 HvacController ServiceIntent bindIntent = new Intent(this /* context */, HvacController.class);if (!bindService(bindIntent, mServiceConnection, Context.BIND_AUTO_CREATE)) {Log.e(TAG, "Failed to connect to HvacController.");}
}

HvacPanelController是空调的面板控制器,在与HvacController绑定成功后,将HvacController的实例传递给HvacPanelController。

private ServiceConnection mServiceConnection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName className, IBinder service) {mHvacController = ((HvacController.LocalBinder) service).getService();final Context context = HvacUiService.this;final Runnable r = () -> {// hvac控制器从车辆刷新其值后,绑定所有值。mHvacPanelController.updateHvacController(mHvacController);};if (mHvacController != null) {mHvacController.requestRefresh(r, new Handler(context.getMainLooper()));}}@Overridepublic void onServiceDisconnected(ComponentName className) {mHvacController = null;mHvacPanelController.updateHvacController(null);//TODO:b/29126575重新启动后重新连接控制器}
};

我们接着看HvacPanelController

3.4 HvacPanelController

HvacPanelController 主要作用是初始化其他界面Controller,并从HvacController中获取数据,显示在UI上。

private FanSpeedBarController mFanSpeedBarController;
private FanDirectionButtonsController mFanDirectionButtonsController;
private TemperatureController mTemperatureController;
private TemperatureController mTemperatureControllerCollapsed;
private SeatWarmerController mSeatWarmerController;public void updateHvacController(HvacController controller) {mHvacController = controller;mFanSpeedBarController = new FanSpeedBarController(mFanSpeedBar, mHvacController);mFanDirectionButtonsController= new FanDirectionButtonsController(mFanDirectionButtons, mHvacController);mTemperatureController = new TemperatureController(mPassengerTemperatureBarExpanded,mDriverTemperatureBarExpanded,mPassengerTemperatureBarCollapsed,mDriverTemperatureBarCollapsed,mHvacController);mSeatWarmerController = new SeatWarmerController(mPassengerSeatWarmer,mDriverSeatWarmer, mHvacController);// 切换按钮不需要额外的逻辑来映射硬件和UI设置。只需使用ToggleListener来处理点击。mAcButton.setIsOn(mHvacController.getAcState());mAcButton.setToggleListener(new ToggleButton.ToggleListener() {@Overridepublic void onToggled(boolean isOn) {mHvacController.setAcState(isOn);}});...setAutoMode(mHvacController.getAutoModeState());mHvacPowerSwitch.setIsOn(mHvacController.getHvacPowerState());mHvacPowerSwitch.setToggleListener(isOn -> mHvacController.setHvacPowerState(isOn));mHvacController.registerCallback(mToggleButtonCallbacks);mToggleButtonCallbacks.onHvacPowerChange(mHvacController.getHvacPowerState());
}

Hvac界面展开和收起的动画也是在HvacPanelController 中处理的,不过关于动画部分打算以后再开个新坑讲一讲。

3.5 HvacController

HvacController是HvacApp与CarService之间的信息传输控制器,本质上也是一个Service。

public class HvacController extends Service {private final Binder mBinder = new LocalBinder();@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {return START_STICKY;}@Overridepublic IBinder onBind(Intent intent) {return mBinder;}public class LocalBinder extends Binder {HvacController getService() {return HvacController.this;}}...
}

在Hvac中的设置及获取数据的操作都是通过HvacController进行的,在HvacController启动时会获取一个Car实例,并通过connect方法连接CarService。当连接CarService成功后初始化CarHvacManager并通过CarHvacManager获取车辆支持的属性列表,以及获取界面所需的基础数据。

@Override
public void onCreate() {super.onCreate();if (getPackageManager().hasSystemFeature(PackageManager.FEATURE_AUTOMOTIVE)) {// 连接 CarServicemCarApiClient = Car.createCar(this, mCarServiceConnection);mCarApiClient.connect();}
}private final ServiceConnection mCarServiceConnection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {synchronized (mHvacManagerReady) {try {// 连接上CarService后,获取到其中的HvacManager.initHvacManager((CarHvacManager) mCarApiClient.getCarManager(Car.HVAC_SERVICE));// 连接成功后,唤醒正在等待CarHvacManager的线程mHvacManagerReady.notifyAll();} catch (CarNotConnectedException e) {Log.e(TAG, "Car not connected in onServiceConnected");}}}@Overridepublic void onServiceDisconnected(ComponentName name) {}
};

向CarService获取数据需要先得到CarHvacManager的实例,所以在连接成功后,调用mHvacManagerReady.notifyAll() 唤醒所有之前等待CarHvacManager实例的线程

// HvacUiService.java - mServiceConnection
{final Runnable r = () -> {// hvac控制器从车辆刷新其值后,绑定所有值。mHvacPanelController.updateHvacController(mHvacController);};if (mHvacController != null) {mHvacController.requestRefresh(r, new Handler(context.getMainLooper()));}
}// HvacController.java
public void requestRefresh(final Runnable r, final Handler h) {final AsyncTask<Void, Void, Void> task = new AsyncTask<Void, Void, Void>() {@Overrideprotected Void doInBackground(Void... unused) {synchronized (mHvacManagerReady) {while (mHvacManager == null) {try {mHvacManagerReady.wait();} catch (InterruptedException e) {// We got interrupted so we might be shutting down.return null;}}}// 刷新数据fetchTemperature(DRIVER_ZONE_ID);fetchTemperature(PASSENGER_ZONE_ID);fetchFanSpeed();...return null;}@Overrideprotected void onPostExecute(Void unused) {// 切换到主线程中执行runnableh.post(r);}};task.execute();
}private void fetchFanSpeed() {if (mHvacManager != null) {int zone = SEAT_ALL; //特定于汽车的解决方法。try {int speed = mHvacManager.getIntProperty(CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT, zone);mDataStore.setFanSpeed(speed);} catch (android.car.CarNotConnectedException e) {Log.e(TAG, "Car not connected in fetchFanSpeed");}}
}

上面的代码就是利用AsyncTask在子线程中等待CarHvacManager的实例,然后刷新数据并存储在DatStore中。
需要注意一点的是while (mHvacManager == null)不能替换成if(mHvacManager == null),这是因为Java有个叫“spurious wakeup”的现象,即线程在不该醒过来的时候醒过来。

A thread can wake up without being notified, interrupted, or timing out, a so-called spurious wakeup. While this will rarely occur in practice, applications must guard against it by testing for the condition that should have caused the thread to be awakened, and continuing to wait if the condition is not satisfied.
一个线程有可能会在未被通知、打断、或超时的情况下醒来,这就是所谓的“spurious wakeup”。尽管实际上这种情况很少发生,应用程序仍然必须对此有所防范,手段是检查正常的导致线程被唤醒的条件是否满足,如果不满足就继续等待。

3.6 Car API

Car是Android汽车平台最高等级的API,为外界提供汽车所有服务和数据访问的接口,提供了一系列与汽车有关的API。它不仅仅可以提供HvacManger,像车辆的速度、档位状态等等所有与汽车有关的信息都可以从Car API中获取。
Hvac中的CarHvacManager实现了CarManagerBase接口,并且只要是作为CarXXXManager, 都需要实现CarManagerBase接口,如CarCabinManagerCarSensorManager等都实现了该接口。
CarHvacManager的控制操作是通过CarPropertyManager来完成的,CarPropertyManager统一控制汽车属性相关的操作。CarHvacManager只是控制与Hvac相关的操作,在汽车中还有很多属性控制的Manager,如传感器,座舱等属性的控制,他们都是通过CarPropertyManager进行属性操作,通过在操作时传入的属性ID,属性区域以及属性值,在CarPropertyManager中会将这些参数转化为一个CarPropertyValue对象继续往CarService传递。

mHvacManager.getIntProperty(CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT, zone);private final CarPropertyManager mCarPropertyMgr;public int getIntProperty(int propertyId, int area) {return this.mCarPropertyMgr.getIntProperty(propertyId, area);
}

CarHvacManager也是通过注册一个callback来得到 Car API 的数据回调。

mHvacManager.registerCallback(mHardwareCallback);private final CarHvacManager.CarHvacEventCallback mHardwareCallback = new CarHvacManager.CarHvacEventCallback() {@Overridepublic void onChangeEvent(final CarPropertyValue val) {int areaId = val.getAreaId();switch (val.getPropertyId()) {case CarHvacManager.ID_ZONED_AC_ON:handleAcStateUpdate(getValue(val));break;case CarHvacManager.ID_ZONED_FAN_DIRECTION:handleFanPositionUpdate(areaId, getValue(val));break;case CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT:handleFanSpeedUpdate(areaId, getValue(val));break;case CarHvacManager.ID_ZONED_TEMP_SETPOINT:handleTempUpdate(val);break;case CarHvacManager.ID_WINDOW_DEFROSTER_ON:handleDefrosterUpdate(areaId, getValue(val));break;case CarHvacManager.ID_ZONED_AIR_RECIRCULATION_ON:handleAirCirculationUpdate(getValue(val));break;case CarHvacManager.ID_ZONED_SEAT_TEMP:handleSeatWarmerUpdate(areaId, getValue(val));break;case CarHvacManager.ID_ZONED_AUTOMATIC_MODE_ON:handleAutoModeUpdate(getValue(val));break;case CarHvacManager.ID_ZONED_HVAC_POWER_ON:handleHvacPowerOn(getValue(val));break;default:if (Log.isLoggable(TAG, Log.DEBUG)) {Log.d(TAG, "Unhandled HVAC event, id: " + val.getPropertyId());}}}@Overridepublic void onErrorEvent(final int propertyId, final int zone) {}
};

Hvac中每个Property对应的含义如下:

// 全局属性,只有一个
ID_MIRROR_DEFROSTER_ON  //视镜除雾
ID_STEERING_WHEEL_HEAT  //方向盘温度
ID_OUTSIDE_AIR_TEMP  //室外温度
ID_TEMPERATURE_DISPLAY_UNITS  //在使用的温度
// 区域属性,可在不同区域设置
ID_ZONED_TEMP_SETPOINT  //用户设置的温度
ID_ZONED_TEMP_ACTUAL  //区域实际温度
ID_ZONED_HVAC_POWER_ON  //HVAC系统电源开关
ID_ZONED_FAN_SPEED_SETPOINT  //风扇设置的速度
ID_ZONED_FAN_SPEED_RPM  //风扇实际的速度
ID_ZONED_FAN_DIRECTION_AVAILABLE  //风扇可设置的方向
ID_ZONED_FAN_DIRECTION  //现在风扇设置的方向
ID_ZONED_SEAT_TEMP  //座椅温度
ID_ZONED_AC_ON  //空调开关
ID_ZONED_AUTOMATIC_MODE_ON  //HVAC自动模式开关
ID_ZONED_AIR_RECIRCULATION_ON  //空气循环开关
ID_ZONED_MAX_AC_ON  //空调最大速度开关
ID_ZONED_DUAL_ZONE_ON  //双区模式开关
ID_ZONED_MAX_DEFROST_ON  //最大除雾开关
ID_ZONED_HVAC_AUTO_RECIRC_ON  //自动循环模式开关
ID_WINDOW_DEFROSTER_ON  //除雾模式开关

使用Car API时务必需要注意,注册的callback是有可能会非常频繁的产生回调的,应用层需要先将数据存储在DataStore中进行过滤,才能更新到UI上。而且也不要实时的打印日志,否则可能会导致日志缓冲区EOF,也会严重干扰其它进程的日志输出。

3.7 DataStore

DataStore 用于存储HvacController从 Car API 中获取的属性值。
用户操作IVI界面和使用硬按键,都会更新Hvac的相关属性。这两种不同的更新方式都是从不同的线程更新到当前状态。此外,在某些情况下,Hvac系统可能会发送虚假的更新,因此这个类将所有内容更新管理合并,从而确保在用户看来应用程序的界面是正常的

@GuardedBy("mFanSpeed")
private Integer mFanSpeed = 0;
private static final long COALESCE_TIME_MS = 0L;public int getFanSpeed() {synchronized (mFanSpeed) {return mFanSpeed;}
}// 仅用于主动 获取、设定 数据时更新speed数据。
public void setFanSpeed(int speed) {synchronized (mFanSpeed) {mFanSpeed = speed;mLastFanSpeedSet = SystemClock.uptimeMillis();}
}// 从callback中得到数据时,因为数据可能会刷新的很频繁,所以需要先判断时间戳,确定数据是否真的需要更新
public boolean shouldPropagateFanSpeedUpdate(int zone, int speed) {// TODO:我们暂时忽略风扇速度区域,因为我们没有多区域车。synchronized (mFanSpeed) {if (SystemClock.uptimeMillis() - mLastFanSpeedSet < COALESCE_TIME_MS) {return false;}mFanSpeed = speed;}return true;
}

HvacController中我们从callback得到数据刷新时,先通过DataStore判断以下是否需要更新数据,如果确实需要更新,再将更新后的数据回调给其他的UI控制器。

// HvacController.java
private final CarHvacManager.CarHvacEventCallback mHardwareCallback = new CarHvacManager.CarHvacEventCallback() {@Overridepublic void onChangeEvent(final CarPropertyValue val) {int areaId = val.getAreaId();switch (val.getPropertyId()) {case CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT:// 处理来自callback的数据handleFanSpeedUpdate(areaId, getValue(val));break;// ... 省略default:if (Log.isLoggable(TAG, Log.DEBUG)) {Log.d(TAG, "Unhandled HVAC event, id: " + val.getPropertyId());}}}
};private void handleFanSpeedUpdate(int zone, int speed) {// 判断是否需要更新本地的数据boolean shouldPropagate = mDataStore.shouldPropagateFanSpeedUpdate(zone, speed);if (Log.isLoggable(TAG, Log.DEBUG)) {Log.d(TAG, "Fan Speed Update, zone: " + zone + " speed: " + speed +" should propagate: " + shouldPropagate);}if (shouldPropagate) {// 将更新后的数据回调给各个UI控制器synchronized (mCallbacks) {for (int i = 0; i < mCallbacks.size(); i++) {mCallbacks.get(i).onFanSpeedChange(speed);}}}
}public void setFanSpeed(final int fanSpeed) {// 更新当前的数据mDataStore.setFanSpeed(fanSpeed);final AsyncTask<Void, Void, Void> task = new AsyncTask<Void, Void, Void>() {int newFanSpeed;protected Void doInBackground(Void... unused) {if (mHvacManager != null) {int zone = SEAT_ALL; // Car specific workaround.try {if (Log.isLoggable(TAG, Log.DEBUG)) {Log.d(TAG, "Setting fanspeed to: " + fanSpeed);}mHvacManager.setIntProperty(CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT, zone, fanSpeed);newFanSpeed = mHvacManager.getIntProperty(CarHvacManager.ID_ZONED_FAN_SPEED_SETPOINT, zone);} catch (android.car.CarNotConnectedException e) {Log.e(TAG, "Car not connected in setFanSpeed");}}return null;}};task.execute();
}

4. 总结

最后我们以一张从Car API的callback中的数据更新界面的伪时序图来把Hvac的几个核心组件串起来

以上就是车载空调部分的讲解,实际开发中,空调模块功能性需求一般不会出现什么太大的技术性困难,空调模块的技术性难度几乎都体现在复杂的动画和交互上,有关车载应用的复杂动画技术,我们以后在来细讲解决方案。


http://www.ppmy.cn/news/380562.html

相关文章

电子电气架构车载网关系列——什么是车载网关?

电子电气架构车载网关系列——什么是车载网关? 我是穿拖鞋的汉子,魔都中一个工科男! 今天是新年假期后第一个周末,连续上了7天班后终于得以休息!老规矩分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 想让对方听你的劝,最好的方式不是语言,而是你自己真的变…

车载冰箱E-mark认证要多长时间?

根据ECE Regulation要求&#xff0c;车载冰箱产品进入欧洲市场&#xff0c;需要通过电子产品 E-mark认证&#xff0c;并在产品上刻印认证标志&#xff0c;进口国海关才允许放行。ECE是欧洲经济委员会Economic Commission for Europe的简称。而欧洲经济委员会并非欧洲组织&#…

车载中间件大全

前言 软件定义汽车时代下&#xff0c;中间件的作用愈发重要。随着 EE 架构逐渐趋于集中化&#xff0c;汽车软件系统出现了多种操作系统并存的局面&#xff0c;这也导致系统的复杂性和开发成本的剧增。为了提高软件的管理性、移植性、裁剪性和质量&#xff0c;需要定义一套架构…

在诗和远方中感受别样科技魅力,一次带上英得尔车载冰箱的自驾之旅!

作为一名旅游爱好者&#xff0c;我很喜欢自驾游&#xff0c;虽然称不上资深&#xff0c;也并非发烧友&#xff0c;但是有这一种对大自然的向往。 在平时假期间&#xff0c;老胡我就曾多次和家人或朋友开车去户外自驾&#xff0c;自认为已经完全体验到了自驾游的乐趣了。但2018年…

车载冰箱不间断供电设计

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、问题提出与分析1. 问题提出2. 问题分析 二、电路设计1.电路仿真2. 原理图设计3.实际电路 三、实际调试 前言 因为家里有需要低温冷藏保存的药品&#xff0…

MATLAB文化算法

目录 文化算法 主要代码 Sphere AdjustCulture 结果 文化算法 基本概念&#xff1a;优化算法 | 详解文化算法&#xff08;附MATLAB代码&#xff09; - 知乎 不同于遗传算法只有种群进化空间&#xff0c;文化算法包含信念空间、种群空间两个进化空间&#xff0c;因此&#…

一款基于jQuery的超酷动画幻灯片

今天给大家带来一款仿步步高vivo手机网站的一款首页焦点幻灯展示特效&#xff0c;带有超酷炫的动画特效&#xff0c;动态效果丝毫不逊色于flash动画&#xff0c;具有很强的视觉冲击力&#xff0c;推荐下载学习&#xff01; 提示&#xff1a;兼容360、FireFox、Chrome、Safari、…

mysql 常用sql

1. 常见命令 连接本地数据库与远程数据库&#xff08;172.16.xx.xx:3306&#xff09;&#xff1a; mysql -h localhost -u root -p123 mysql -h 172.16.xx.xx -P 3306 -u root -p 2. DDL 数据定义语言&#xff08;Data Definition Lanuage, DDL&#xff09;定义了数据库模式&…