为什么会想到做手电筒呢?第一是因为我的骚尼手机有闪光灯却没有自带的手电筒程序,而兰大素有晚上十一点停电的恶规,没有手电筒实在不便。那为什么不去下载一个手电筒应用呢?卧槽,问题真TM多,直接开干!
(一)尝试多Activity
按理来说手电筒是一个单Activity(单屏嘛,能做几个Activity),但是由于上一个程序(计算器)也是单一Activity的,所以我现在或许应该写一个简单的多Activity应用入门,没办法,只能强行多Activity了,通过制作一个欢迎界面(类似于某宝,某信),2.5秒后跳转到主Activity,从而达到多Activity的练习
要实现多Activity,需要理解意图类Intent,我个人将之理解为Android系统的意图——沟通各个活动的桥梁,那么切换Activity一共有五种方式可供选择:
Intent intent = new Intent();
(1)intent.setClass(this,OtherActivity.class);
(2)intent.setClassName(this,“com.xiazdong.OtherActivity”);
(3)intent.setClassName(“com.xiazdong”,“com.xiazdong.OtherActivity”);
//此种方式用来激活不同应用的Activity,只需要指定第一个参数:包名为另一个应用即可;
(4)Component comp = new Component(this,OtherActivity.class);
intent.setComponent(comp);
(5)Intent intent = new Intent(this,OtherActivity.class);
理论结束,开始做我们的欢迎界面,首先,我在网上随便找了张哔哩哔哩娘的图:
将图片放入res\drawable-hdpi中
下面创建欢迎界面的Activity,Java代码如下:
package com.example.multiactivity;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
public class SplashActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_splash);
new Handler().postDelayed(new Runnable(){
@Override
public void run() {
Intent intent = new Intent(SplashActivity.this,MainActivity.class);
startActivity(intent);
SplashActivity.this.finish();
}
}, 2500);
}
}
在AndroidManifest.xml文件中将这个activity设置成启动入口,代码如下:
<activity
android:name=".SplashActivity"
android:label="@string/title_activity_splash"
android:screenOrientation="portrait" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
把欢迎界面的XML文件写好:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@drawable/splash"
tools:context="${relativePackage}.${activityClass}" >
<TextView
android:id="@+id/VersionNumber"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:layout_alignParentTop="true"
android:layout_marginRight="22dp"
android:layout_marginTop="10dp"
android:text=" 手电筒\nVersion 1.0\n By Abang" />
</RelativeLayout>
这样就可以实现跳转了,在欢迎界面的滞留时间为2.5秒
效果图:
分析:
我们先来看看 Handler().postDelayed(new Runnable(){run函数},毫秒数)意思地延迟多少毫秒将Runnable插入到消息队列中,Runnable将在handle绑定的线程中运行。下面的run()很容易理解,这里采用的是第五种intent跳转方法
Activity_splash.xml中调用背景图 android:background=“@drawable/splash” 这里解释一下@的作用,我们会发现在Android的中引用png资源方法很简单,甚至不需要加路径和后缀,其实,@已经给出了路径
好的,多Activity分析到此,其实手电筒启动要求快速,原则上是不应该设置欢迎界面的,但是为了练习intent类和欢迎界面的制作,就只能强行这样了o(╯□╰)o
(二)手电筒UI设计
废话不多说,上代码
Activity_main.xml的代码
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="${relativePackage}.${activityClass}" >
<ToggleButton
android:id="@+id/toggleButton1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_centerVertical="true"
android:checked="true"
android:textOn=""
android:textOff=""
android:background="@drawable/togglebutton" />
</RelativeLayout>
这里为了友好的UI,我对ToggleButton控件自定义了一部分,自定义的代码togglebutton.xml存放在drawable/目录下
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_checked="true"
android:drawable="@drawable/button_focused" /> <!-- pressed -->
<item android:drawable="@drawable/button_pressed" /> <!-- default/unchecked -->
</selector>
点击之后:
至此,UI已经基本搞定,由于PS水平很有限,作这两个状态的简单UI确实磨了很久,Android的UI设计时,建议去看看Google的一些设计理念
(三)手电筒核心代码
Java代码:
package com.example.multiactivity;
import android.app.Activity;
import android.hardware.Camera;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.ToggleButton;
public class MainActivity extends Activity {
private Camera camera = Camera.open();
private ToggleButton toggleButton;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
toggleButton=(ToggleButton)findViewById(R.id.toggleButton1);
toggleButton.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
// TODO 自动生成的方法存根
ToggleButton tb=(ToggleButton)v;
Camera.Parameters param=camera.getParameters();
if(!tb.isChecked()){
param.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
}
else{
param.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
}
camera.setParameters(param);
}
});
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
}
@Override
protected void onPause(){
camera.release();
super.onPause();
}
}
在Manifest中插入的代码(很重要,涉及权限)
<uses-permission android:name="android.permission.FLASHLIGHT" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" />
注意应将它们插入到 标签以外
分析:
JAVA代码:区别于第一次的计算器,这一次的监听采用了另一种方法,即
控件.setOnClickListener(new OnClickListener(){onClick函数});
对相机闪关灯的调用主要是在onclick方法中,一定注意在退出程序是将camera释放——camera.release(),否则相机将不可用
重点来了,在哪里加入此段代码,这里,我们来了解一下Android的Activity的生命周期(摘自某网友):
在系统中的Activity被一个Activity栈所管理。当一个新的Activity启动时,将被放置到栈顶,成为运行中的Activity,一个Activity保留在栈中,不再放到前台,直到新的Activity退出为止。Activity有四种本质区别的状态:
Activity有三个关键的循环:
整个的生命周期,从onCreate(Bundle)开始到onDestroy()结束。Activity在onCreate()设置所有的“全局”状态,在onDestory()释放所有的资源。例如:某个Activity有一个在后台运行的线程,用于从网络下载数据,则该Activity可以在onCreate()中创建线程,在onDestory()中停止线程。
可见的生命周期,从onStart()开始到onStop()结束。在这段时间,可以看到Activity在屏幕上,尽管有可能不在前台,不能和用户交互。在这两个接口之间,需要保持显示给用户的UI数据和资源等,例如:可以在onStart中注册一个IntentReceiver来监听数据变化导致UI的变动,当不再需要显示时候,可以在onStop()中注销它。onStart(),onStop()都可以被多次调用,因为Activity随时可以在可见和隐藏之间转换。
前台的生命周期,从onResume()开始到onPause()结束。在这段时间里,该Activity处于所有 Activity的最前面,和用户进行交互。Activity可以经常性地在resumed和paused状态之间切换,例如:当设备准备休眠时,当一个 Activity处理结果被分发时,当一个新的Intent被分发时。所以在这些接口方法中的代码应该属于非常轻量级的。
Activity的整个生命周期都定义在下面的接口方法中,所有方法都可以被重载。所有的Activity都需要实现 onCreate(Bundle)去初始化设置,大部分Activity需要实现onPause()去提交更改过的数据,当前大部分的Activity也需要实现onFreeze()接口,以便恢复在onCreate(Bundle)里面设置的状态。
public class Activity extends ApplicationContext {
protected void onCreate(Bundle icicle);
protected void onStart();
protected void onRestart();
protected void onResume();
protected void onFreeze(Bundle outIcicle);
protected void onPause();
protected void onStop();
protected void onDestroy();
}
了解生命周期后我们发现,在重写onPause()并在其中释放camera较合理
。。。。。。。。。。。。。。
最后,我们把图标和APP的名字变更一下,变更图标只需替换icon.png文件,修改App名称只需将String.xml中的app_name改为你想要的名称即可,一个手电筒就完成了!妈妈在也不用担心我在11点熄灯后看不见了