Android系统移植与调试之------->增加一个双击物理按键打开和关闭闪光灯并将闪光灯状态同步到下拉菜单中

简介: 最近有一个客户有这样的需求:1、在【设置】--->【无障碍】中添加一个开关按钮。            如果打开开关的话,双击某个物理按键的时候,打开闪光灯,再双击该物理按键的时候,关闭闪光灯。

最近有一个客户有这样的需求:

1、在【设置】--->【无障碍】中添加一个开关按钮。

            如果打开开关的话,双击某个物理按键的时候,打开闪光灯,再双击该物理按键的时候,关闭闪光灯。

             如果关闭开关的话,双击该物理按键的时候,不做任何处理。

2、打开关闭闪光灯的状态,在下拉菜单中也要同步。

如下图所示的效果:

                                        

当打开上面添加的开关后,如果双击POC键(POC键是我们公司自定义的一个物理按键)的时候,就打开闪光灯,并将下拉菜单中的闪光灯图标点亮,如下图所示

                                              

如果再次双击POC键(POC键是我们公司自定义的一个物理按键)的时候,就关闭闪光灯,并将下拉菜单中的闪光灯图标熄灭,如下图所示

                                                    

====================================================================================================================================

下面来具体说说怎么实现该功能:

第一步:添加开关按钮布局

在packages/apps/Settings/res/xml/accessibility_settings.xml 文件中添加一个开关,开关位置放在【按电源按钮结束通话】和【自动选择屏幕】中间,代码如下

        <SwitchPreference
                android:key="toggle_power_button_ends_call_preference"
                android:title="@string/accessibility_power_button_ends_call_prerefence_title"
                android:persistent="false"/>
        <!-- added by ouyang start [2015-11-3] -->
        <SwitchPreference
                android:key="toggle_camera_button__start_flashlight_preference"
                android:title="@string/accessibility_camera_button_start_flashlight_preference_title"
                android:summary="@string/accessibility_camera_button_start_flashlight_preference_subtitle"
                android:persistent="false"/>
 		<!-- added by ouyang end   [2015-11-3] -->
 		
        <SwitchPreference
                android:key="toggle_lock_screen_rotation_preference"
                android:title="@string/accelerometer_title"
                android:persistent="false"/>


在packages/apps/Settings/res/values-zh-rCN/strings.xml文件中添加相应的字符串

    <!-- added by ouyang start [2015-11-3] -->
    <string name="accessibility_camera_button_start_flashlight_preference_title">"双击[相机键]打开闪光灯"</string>
    <string name="accessibility_camera_button_start_flashlight_preference_title_E580">"双击[POC键]打开闪光灯"</string>
    <string name="accessibility_camera_button_start_flashlight_preference_subtitle">"开启此功能后,摄像机暂时无法使用"</string>
    <!-- added by ouyang end [2015-11-3] -->

 

packages/apps/Settings/res/values/strings.xml文件中添加相应的字符串

    <!-- added by ouyang start [2015-11-3] -->
    <string name="accessibility_camera_button_start_flashlight_preference_title">"Camera button twice,open flash light"</string>
    <string name="accessibility_camera_button_start_flashlight_preference_title_E580">"POC button twice,open flash light"</string>
    <string name="accessibility_camera_button_start_flashlight_preference_subtitle">"After open this feature, the camera can not be used"</string>
    <!-- added by ouyang end [2015-11-3] -->


 

第二步:对该开关按钮做逻辑处理,在packages/apps/Settings/src/com/android/settings/accessibility/AccessibilitySettings.java文件中

先定义一个字符串常量,该常量为添加的开关的key,定义该按钮开关常量和SharedPreences常量,用来保存该按钮打开关闭状态值

 
    // added by ouyang start [2015-11-3] 
    private static final String TOGGLE_CAMER_BUTTON_START_FLASHLIGHT_PREFERENCE =
            "toggle_camera_button__start_flashlight_preference";
    // added by ouyang end [2015-11-3] 
    
    // added by ouyang start [2015-11-3] 
	private SwitchPreference mToggleCameraButtonStartFlashlightPreference;
	private SharedPreferences preferences;
	private SharedPreferences.Editor editor;
	// added by ouyang end   [2015-11-3] 


 

通过该字符串找到定义的开关,并做些处理,如从保存好的Perference中读取开关打开关闭状态并显示,在initializeAllPreferences方法中,添加如下代码

// added by ouyang start [2015-11-3] 
        mToggleCameraButtonStartFlashlightPreference =
                (SwitchPreference) findPreference(TOGGLE_CAMER_BUTTON_START_FLASHLIGHT_PREFERENCE);
        if (android.os.SystemProperties.isHanbangVersion()) {
        	Log.e(TAG, "is HanbangVersion()");
            mToggleCameraButtonStartFlashlightPreference.setEnabled(false);
        }else{
        	Log.e(TAG, "is not HanbangVersion()");
            mToggleCameraButtonStartFlashlightPreference.setEnabled(true);
        }
        Log.e(TAG, "mToggleCameraButtonStartFlashlightPreference.isEbable="+  mToggleCameraButtonStartFlashlightPreference.isEnabled());
        if (android.os.SystemProperties.isDT800Project()) {
        	mToggleCameraButtonStartFlashlightPreference.setTitle(R.string.accessibility_camera_button_start_flashlight_preference_title);
		}else if (android.os.SystemProperties.isE580Project()) {
			mToggleCameraButtonStartFlashlightPreference.setTitle(R.string.accessibility_camera_button_start_flashlight_preference_title_E580);
		}
        preferences= getActivity().getApplicationContext().getSharedPreferences("FlashLight",
                            Context.MODE_MULTI_PROCESS
                           +Context.MODE_WORLD_READABLE 
                           +Context.MODE_WORLD_WRITEABLE);
        editor=preferences.edit();
        boolean flashChecked=preferences.getBoolean("flashChecked", false);
        mToggleCameraButtonStartFlashlightPreference.setChecked(flashChecked);
        Log.e(TAG, "flashChecked="+flashChecked);
        // added by ouyang end [2015-11-3] 


这段代码主要是对该开关进行初始化,并对不同的项目进行判断,DT800项目和E580项目加载不同的字符串来描述开关。这些方法可以在/frameworks/base/core/java/android/os/SystemProperties.java文件中自己添加,代码如下:

       public static boolean isDT800Project() {
		return SystemProperties.get("ro.custom.build.version").contains("DT800");
       }   

        public static boolean isE580Project() {
		return SystemProperties.get("ro.custom.build.version").contains("E580");
        }

        public static boolean isHanbangVersion(){
		return SystemProperties.get("ro.product.model").contains("Hanbang");
        }
  

在packages/apps/Settings/src/com/android/settings/accessibility/AccessibilitySettings.java文件中的onPreferenceTreeClick方法中,要添加对该开关的点击事件处理。

代码如下,添加几行代码:

else if (mTogglePowerButtonEndsCallPreference == preference) {
            handleTogglePowerButtonEndsCallPreferenceClick();
            return true;
        } 
        // added by ouyang start [2015-11-3] 
        else if (mToggleCameraButtonStartFlashlightPreference == preference) {
            handleToggleCameraButtonStartFlashlightPreferenceClick();
            return true;
        } 
        // added by ouyang end [2015-11-3] 
        else if (mToggleLockScreenRotationPreference == preference) {
            handleLockScreenRotationPreferenceClick();
            return true;
        } 


其中handleToggleCameraButtonStartFlashlightPreferenceClick()方法自己自定义的,代码如下:

   // added by ouyang start [2015-11-3] 
    private void handleToggleCameraButtonStartFlashlightPreferenceClick(){
    	if (mToggleCameraButtonStartFlashlightPreference.isChecked()) {
            editor.putBoolean("flashChecked", true);
            editor.commit();
	}else{
	   editor.putBoolean("flashChecked", false);
	   editor.commit();
	}
    }
    // added by ouyang end [2015-11-3] 


该方法主要功能是保存该开关是打开状态还是关闭状态,这个状态值可以让其他的进程读取。

 

第三步:定义一个BroadcastReceiver,并动态注册该广播。代码文件在frameworks/base/packages/SystemUI/src/com/android/systemui/qs/tiles/FlashlightTile.java

,在该文件中添加一个BroadcastReceiver

 //added by ouyang
    private final BroadcastReceiver mFlashBroadcastReceiver =new BroadcastReceiver(){
    	boolean isDoublePress = false;
    	@Override
    	public void onReceive(Context context, Intent intent) {
    		String action = intent.getAction();
    		Log.d(TAG, "action= " + action);
    		Log.d(TAG, "isDoublePress= " + isDoublePress);
    		Log.d(TAG, "now time = " + new SimpleDateFormat("yyyy-MM-dd hh:mm:ss.SSS").format(new Date()));
    		if ((action == "com.runbo.camera.key.up") || (action == "com.runbo.poc.key.up")) {
    			Handler checkHandler = new Handler();
    			if (!isDoublePress) {
    				isDoublePress = true;
    				Log.i(TAG, "<----===============Single Press the Camera Key---->");
    				Runnable CheckDoubleRunnable = new Runnable() {
    					@Override
    					public void run() {
    						isDoublePress = false;
    					}
    				};
    				checkHandler.postDelayed(CheckDoubleRunnable, 500);
    			} else {
    				isDoublePress = false;
    				Log.i(TAG, "<----============Double Press the Camera Key ---->");
    				boolean handlerReceiver = false;  
    				boolean flashChecked=false;
    				try {
    					Context myContext= context.createPackageContext("com.android.settings",
    							Context.CONTEXT_IGNORE_SECURITY);
    					SharedPreferences sharedPreferences = myContext.getSharedPreferences("FlashLight",
    							Context.MODE_WORLD_READABLE | Context.MODE_MULTI_PROCESS);
        				flashChecked = sharedPreferences.getBoolean("flashChecked", false);
					} catch (Exception e) {
						Log.d(TAG, Log.getStackTraceString(e));
					}				
    				if (action == "com.runbo.camera.key.up") {
    					if (android.os.SystemProperties.isDT800Project()
    							&& !android.os.SystemProperties.isHanbangVersion()) {
    						handlerReceiver = true;
    					}
    				} else if (action == "com.runbo.poc.key.up") {
    					if (android.os.SystemProperties.isE580Project()) {
    						handlerReceiver = true;
    					}
    				}
    				Log.d(TAG, "handlerReceiver= " + handlerReceiver);
    				Log.d(TAG, "flashChecked= " + flashChecked);
    				if (handlerReceiver && flashChecked) {
    					Log.i(TAG, "============mState.value="+mState.value);
    					if (!mState.value) {
    						boolean newState = !mState.value;
    						mFlashlightController.setFlashlight(newState);
    					    refreshState(newState ? UserBoolean.USER_TRUE : UserBoolean.USER_FALSE);
    						Log.i(TAG, "============setFlashlight(true);============");
						}else{
							boolean newState = !mState.value;
    						mFlashlightController.setFlashlight(newState);
    					    refreshState(newState ? UserBoolean.USER_TRUE : UserBoolean.USER_FALSE);
							Log.i(TAG, "============setFlashlight(false);============");
						}
    				}
    			}
    		}
    	}
    };


在FlashlightTile的构造方法中进行注册,代码如下:

  public FlashlightTile(Host host) {
        super(host);
        mFlashlightController = host.getFlashlightController();
        mFlashlightController.addListener(this);
        
        //added by ouyang start
        IntentFilter pocFilter = new IntentFilter();
        pocFilter.addAction("com.runbo.poc.key.up");
        pocFilter.addAction("com.runbo.camera.key.up");
        this.mContext.registerReceiver(mFlashBroadcastReceiver, pocFilter);
        Log.d(TAG, "registerReceiver(mFlashBroadcastReceiver, pocFilter)");
        //added by ouyang end
    }

 

该BroadcastReceiver接收通过按POC键后释放而发送的广播“com.runbo.poc.key.up”和通过按Camera键后释放而发送的广播“com.runbo.camera.key.up”,这两个广播是自定义的广播。然后判断是单击还是双击,当双击后再通过读取SharedPreferences的值来判断定义的开关是否是打开的,如果开关是打开的再判断闪光灯是打开还是关闭,如果是打开的,双击后就关闭闪光灯。如果是关闭的,就打开闪光灯。并更新下拉菜单的图标和相应的状态。如果开关是关闭的,不进行打开关闭闪光灯的操作。

 

这两个广播是在frameworks/base/policy/src/com/android/internal/policy/impl/PhoneWindowManager.java文件中的interceptKeyBeforeQueueing(KeyEvent event, int policyFlags)方法中额外添加的代码,发送广播出来的。


 

====================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址http://blog.csdn.net/ouyang_peng

====================================================================================



相关文章
|
2月前
|
人工智能 搜索推荐 物联网
Android系统版本演进与未来展望####
本文深入探讨了Android操作系统从诞生至今的发展历程,详细阐述了其关键版本迭代带来的创新特性、用户体验提升及对全球移动生态系统的影响。通过对Android历史版本的回顾与分析,本文旨在揭示其成功背后的驱动力,并展望未来Android可能的发展趋势与面临的挑战,为读者呈现一个既全面又具深度的技术视角。 ####
|
2月前
|
IDE Java 开发工具
移动应用与系统:探索Android开发之旅
在这篇文章中,我们将深入探讨Android开发的各个方面,从基础知识到高级技术。我们将通过代码示例和案例分析,帮助读者更好地理解和掌握Android开发。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和技巧。让我们一起开启Android开发的旅程吧!
|
1月前
|
监控 Java Android开发
深入探索Android系统的内存管理机制
本文旨在全面解析Android系统的内存管理机制,包括其工作原理、常见问题及其解决方案。通过对Android内存模型的深入分析,本文将帮助开发者更好地理解内存分配、回收以及优化策略,从而提高应用性能和用户体验。
|
1月前
|
存储 安全 Android开发
探索Android系统的最新安全特性
在数字时代,智能手机已成为我们生活中不可或缺的一部分。随着技术的不断进步,手机操作系统的安全性也越来越受到重视。本文将深入探讨Android系统最新的安全特性,包括其设计理念、实施方式以及对用户的影响。通过分析这些安全措施如何保护用户免受恶意软件和网络攻击的威胁,我们希望为读者提供对Android安全性的全面了解。
|
2月前
|
监控 Java Android开发
深入探讨Android系统的内存管理机制
本文将深入分析Android系统的内存管理机制,包括其内存分配、回收策略以及常见的内存泄漏问题。通过对这些方面的详细讨论,读者可以更好地理解Android系统如何高效地管理内存资源,从而提高应用程序的性能和稳定性。
91 16
|
2月前
|
前端开发 数据处理 Android开发
Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍
本文深入探讨了Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍,以及具体操作步骤、常见问题解决、高级调试技巧、团队协作中的调试应用和未来发展趋势,旨在帮助开发者提高调试效率,提升应用质量。
60 8
|
2月前
|
安全 Android开发 iOS开发
深入探讨Android与iOS系统的差异及未来发展趋势
本文旨在深入分析Android和iOS两大移动操作系统的核心技术差异、用户体验以及各自的市场表现,进一步探讨它们在未来技术革新中可能的发展方向。通过对比两者的开放性、安全性、生态系统等方面,本文揭示了两大系统在移动设备市场中的竞争态势和潜在变革。
|
2月前
|
算法 JavaScript Android开发
|
2月前
|
安全 搜索推荐 程序员
深入探索Android系统的碎片化问题及其解决方案
在移动操作系统的世界中,Android以其开放性和灵活性赢得了广泛的市场份额。然而,这种开放性也带来了一个众所周知的问题——系统碎片化。本文旨在探讨Android系统碎片化的现状、成因以及可能的解决方案,为开发者和用户提供一种全新的视角来理解这一现象。通过分析不同版本的Android系统分布、硬件多样性以及更新机制的影响,我们提出了一系列针对性的策略,旨在减少碎片化带来的影响,提升用户体验。
|
2月前
|
安全 Android开发 iOS开发
深入探索iOS与Android系统的差异性及优化策略
在当今数字化时代,移动操作系统的竞争尤为激烈,其中iOS和Android作为市场上的两大巨头,各自拥有庞大的用户基础和独特的技术特点。本文旨在通过对比分析iOS与Android的核心差异,探讨各自的优势与局限,并提出针对性的优化策略,以期为用户提供更优质的使用体验和为开发者提供有价值的参考。