android:摇一摇的实现

一、需求

实现类似微信等的摇一摇效果
摇动时 振动手机、播放音频、弹出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"/>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,784评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,745评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,702评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,229评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,245评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,376评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,798评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,471评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,655评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,485评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,535评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,235评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,793评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,863评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,096评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,654评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,233评论 2 341

推荐阅读更多精彩内容