欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

android:摇一摇的实现

程序员文章站 2024-02-11 09:00:52
...

一、需求

实现类似微信等的摇一摇效果
摇动时 振动手机、播放音频、弹出dialog提示
摇动时不需要微信的动画效果

二、主要知识点及注意事项

1、振动需要使用振动传感器 --Vibrator
2、监听是否摇动手机需要使用 传感器监听器--SensorManager
3、播放音频使用 soundPool,用完之后需要在页面销毁时释放
4、4.4之前版本中需要添加振动权限才能实现振动,之后的版本不需要注册权限
5、音频文件放置在 res 目录下的 raw 目录中
6、触发振动器时,在onSensorChanged( ) 方法中使用 && ,当三个方向上的移动量同时满足条件时,才触发振动。使用&& 可以降低振动的灵敏度

三、详细实现代码

1、ShakeActivity

/**
 * 作者:CnPeng
 * <p>
 * 时间:2017/8/2:上午10:50
 * <p>
 * 说明:摇一摇的实现:播放音效,振动,弹出dialog
 * <p>
 * 注意:
 * 1、传感器监听要在 onResume  中注册,在 onPause 中取消注册
 * 2、振动要在清单文件中声明权限(4.4 以上不需要声明)-- <uses-permission android:name="android.permission.VIBRATE"/>
 * 3、sound 用完之后要在onDestroy 中释放并置空
 */

public class ShakeActivity extends AppCompatActivity implements SensorEventListener {
    private final int PlayAudioOver = 0;   //播放完毕
    private final int VibratorOver  = 1;   //振动结束

    private SensorManager     sensorManager;
    private Sensor            sensor;
    private SoundPool         soundPool;
    private boolean           isPlayAudio;  //是否正在播放音频
    private Vibrator          vibrator;     //振动器对象
    private boolean           isVibrator;   //是否正在振动
    private int               musicStreamId;  //通过SoundPool加载得到的音频id
    private float             playVolume;   //音量比率值
    private CustomAlertDialog customAlertDialog;

    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            if (null != msg) {
                switch (msg.what) {
                    case PlayAudioOver:
                        isPlayAudio = false;
                        break;
                    case VibratorOver:
                        isVibrator = false;
                        break;
                    default:
                        break;
                }
            }
        }
    };

    @Override
    protected void onCreate(
            @Nullable
                    Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        DataBindingUtil.setContentView(this, R.layout.activity_shake);

        sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);   //传感器管理器
        sensor = sensorManager.getDefaultSensor(TYPE_ACCELEROMETER);     //此处传入1 也可以,Sensor中加速度传感器对应的int值为1

        soundPool = new SoundPool(1, AudioManager.STREAM_MUSIC, 0);        //音效池
        musicStreamId = soundPool.load(ShakeActivity.this, R.raw.audio_shake, 1);//根据id加载音频

        AudioManager audioManager = (AudioManager) getSystemService(AUDIO_SERVICE); //用来获取音量
        int curVolume = audioManager.getStreamVolume(AudioManager.STREAM_MUSIC);    //当前音量值
        int maxVolume = audioManager.getStreamMaxVolume(AudioManager.STREAM_MUSIC);   //最大音量
        playVolume = (float) (curVolume * 1.0 / maxVolume);   //音量比率值

        vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);          //振动器

        customAlertDialog = new CustomAlertDialog(ShakeActivity.this);      //振动的时候需要展示的dialog
    }

    @Override
    protected void onResume() {
        super.onResume();
        sensorManager.registerListener(ShakeActivity.this, sensor, SensorManager.SENSOR_DELAY_GAME);

    }

    @Override
    protected void onPause() {
        super.onPause();
        sensorManager.unregisterListener(ShakeActivity.this, sensor);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        soundPool.release();    //销毁
        soundPool = null;
    }

    @Override
    public void onSensorChanged(SensorEvent event) {
        int type = event.sensor.getType();  //获取传感器类型
        if (type == 1) {   //等价于  type==TYPE_ACCELEROMETER
            float x = event.values[0];
            float y = event.values[1];
            float z = event.values[2];

            if (Math.abs(x) > 15 && Math.abs(y) > 15 && Math.abs(z) > 15) {     //摇动灵敏度取决于后面的常量值,这里定义了15
                playShakeAudio();
                vibratorPhone();
                showCusDialog();
            }
        }
    }

    /**
     * 展示dialog
     */
    private void showCusDialog() {
        customAlertDialog.setTitle("摇一摇");
        customAlertDialog.setMessage("摇一摇,摇到了外婆桥");
        customAlertDialog.setPositiveButton("关闭", null);
        customAlertDialog.show();
    }

    /**
     * 开启手机震动
     */
    private void vibratorPhone() {
        if (!isVibrator) {
            isVibrator = true;
            vibrator.vibrate(300);  //振动时长300ms

            new Timer().schedule(new TimerTask() {
                @Override
                public void run() {
                    handler.sendEmptyMessage(VibratorOver);
                }
            }, 400);        //延时时间根据振动时长决定
        }
    }

    /**
     * 播放摇一摇的音频
     */
    private void playShakeAudio() {
        if (!isPlayAudio) {

            isPlayAudio = true;
            soundPool.play(musicStreamId, playVolume, playVolume, 1, 0, 1.0f);
            LogUtils.e("摇一摇", "摇啊摇,要到了外婆桥");

            Timer timer = new Timer();
            timer.schedule(new TimerTask() {
                @Override
                public void run() {
                    handler.sendEmptyMessage(PlayAudioOver);
                }
            }, 1000);    //此处延时时间根据音频时间确定
        }
    }

    @Override
    public void onAccuracyChanged(Sensor sensor, int accuracy) {
        //灵敏度变化时调用。灵敏度级别参考:SensorManager.SENSOR_DELAY_GAME
    }
}

2、activity_shake.xml

<?xml version="1.0" encoding="utf-8"?>
<layout>
    <data class="ShakeBinding">

    </data>
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:gravity="center"
            android:text="摇动手机开启摇一摇"
            android:textSize="18sp"/>
    </RelativeLayout>
</layout>

3、振动的权限

  <!--震动的权限。4.4及以上不需要声明权限,但低于4.4 时需要声明权限-->
    <uses-permission android:name="android.permission.VIBRATE"/>