体感感应器,微信摇一摇

郏实
2023-12-01

Activity内的代码

private static final String TAG = "MainActivity";
private static final int START_SHAKE = 0x1;
private static final int AGAIN_SHAKE = 0x2;
private static final int END_SHAKE = 0x3;

private SensorManager mSensorManager;
private Sensor mAccelerometerSensor;
private Vibrator mVibrator;//手机震动
private SoundPool mSoundPool;//摇一摇音效

//记录摇动状态
private boolean isShake = false;


private LinearLayout mTopLayout;
private LinearLayout mBottomLayout;
private ImageView mTopLine;
private ImageView mBottomLine;
private TextView mTextView;

private MyHandler mHandler;
private int mWeiChatAudio;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    //设置只竖屏
    setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
    setContentView(R.layout.activity_main);
    //初始化View
    initView();
    mHandler = new MyHandler(this);
    //初始化SoundPool
    mSoundPool = new SoundPool(1, AudioManager.STREAM_SYSTEM, 5);
    mWeiChatAudio = mSoundPool.load(this, R.raw.mus, 1);

    //获取Vibrator震动服务
    mVibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
}

private void initView() {
    mTopLayout = (LinearLayout) findViewById(R.id.main_linear_top);
    mBottomLayout = ((LinearLayout) findViewById(R.id.main_linear_bottom));
    mTopLine = (ImageView) findViewById(R.id.main_shake_top_line);
    mBottomLine = (ImageView) findViewById(R.id.main_shake_bottom_line);
    mTextView = (TextView) findViewById(R.id.tv);



    //默认
    mTopLine.setVisibility(View.GONE);
    mBottomLine.setVisibility(View.GONE);


}
@Override
protected void onStart() {
    super.onStart();
    //获取 SensorManager 负责管理传感器
    mSensorManager = ((SensorManager) getSystemService(SENSOR_SERVICE));
    if (mSensorManager != null) {
        //获取加速度传感器
        mAccelerometerSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);

        if (mAccelerometerSensor != null) {
            mSensorManager.registerListener(this, mAccelerometerSensor, SensorManager.SENSOR_DELAY_UI);
        }
    }
}
@Override
protected void onPause() {
    // 务必要在pause中注销 mSensorManager
    // 否则会造成界面退出后摇一摇依旧生效的bug
    if (mSensorManager != null) {
        mSensorManager.unregisterListener(this);
    }
    super.onPause();
}

@Override
public void onSensorChanged(SensorEvent event) {

    int type = event.sensor.getType();

    if (type == Sensor.TYPE_ACCELEROMETER) {
        //获取三个方向值
        float[] values = event.values;
        float x = values[0];
        float y = values[1];
        float z = values[2];
        Log.i(TAG, "onSensorChanged:==x "+x);
        Log.i(TAG, "onSensorChanged: ==z"+z);
        Log.i(TAG, "onSensorChanged: ==y"+y);




        if ((Math.abs(x) > 17 || Math.abs(y) > 17 || Math
                .abs(z) > 17) && !isShake) {
            isShake = true;
            // TODO: 2016/10/19 实现摇动逻辑, 摇动后进行震动
            Thread thread = new Thread() {
                @Override
                public void run() {


                    super.run();
                    try {
                        Log.d(TAG, "onSensorChanged: 摇动");

                        //开始震动 发出提示音 展示动画效果
                        mHandler.obtainMessage(START_SHAKE).sendToTarget();
                        Thread.sleep(500);
                        //再来一次震动提示
                        mHandler.obtainMessage(AGAIN_SHAKE).sendToTarget();
                        Thread.sleep(500);
                        mHandler.obtainMessage(END_SHAKE).sendToTarget();


                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            };
            thread.start();
        }
    }
}

@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {

}
private static class MyHandler extends Handler {
    private WeakReference<MainActivity> mReference;
    private MainActivity mActivity;
    public MyHandler(MainActivity activity) {
        mReference = new WeakReference<MainActivity>(activity);
        if (mReference != null) {
            mActivity = mReference.get();
        }
    }
    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
        switch (msg.what) {
            case START_SHAKE:
                //This method requires the caller to hold the permission VIBRATE.
                mActivity.mVibrator.vibrate(300);
                //发出提示音
                mActivity.mSoundPool.play(mActivity.mWeiChatAudio, 1, 1, 0, 0, 1);
                mActivity.mTopLine.setVisibility(View.VISIBLE);
                mActivity.mBottomLine.setVisibility(View.VISIBLE);
                mActivity.startAnimation(false);//参数含义: (不是回来) 也就是说两张图片分散开的动画

                break;
            case AGAIN_SHAKE:
                mActivity.mTextView.setText(R.string.app_b);
                mActivity.mVibrator.vibrate(300);
                break;
            case END_SHAKE:
                //整体效果结束, 将震动设置为false
                mActivity.mTextView.setText(R.string.app_c);
                mActivity.isShake = false;
                // 展示上下两种图片回来的效果
                mActivity.startAnimation(true);

                break;
        }
    }
}
*

开启 摇一摇动画 * * @param isBack 是否是返回初识状态

private void startAnimation(boolean isBack) {
    //动画坐标移动的位置的类型是相对自己的
    int type = Animation.RELATIVE_TO_SELF;

    float topFromY;
    float topToY;
    float bottomFromY;
    float bottomToY;
    if (isBack) {
        topFromY = -0.5f;
        topToY = 0;
        bottomFromY = 0.5f;
        bottomToY = 0;
    } else {
        topFromY = 0;
        topToY = -0.5f;
        bottomFromY = 0;
        bottomToY = 0.5f;
    }

上面图片的动画效果

    TranslateAnimation topAnim = new TranslateAnimation(
            type, 0, type, 0, type, topFromY, type, topToY
    );
    topAnim.setDuration(200);
    动画终止时停留在最后一帧~不然会回到没有执行之前的状态
    topAnim.setFillAfter(true);

   底部的动画效果
    TranslateAnimation bottomAnim = new TranslateAnimation(
            type, 0, type, 0, type, bottomFromY, type, bottomToY
    );
    bottomAnim.setDuration(200);
    bottomAnim.setFillAfter(true);

    大家一定不要忘记, 当要回来时, 我们中间的两根线需要GONE掉
    if (isBack) {
        bottomAnim.setAnimationListener(new Animation.AnimationListener() {
            @Override
            public void onAnimationStart(Animation animation) {}
            @Override
            public void onAnimationRepeat(Animation animation) {}
            @Override
            public void onAnimationEnd(Animation animation) {
                //当动画结束后 , 将中间两条线GONE掉, 不让其占位
                mTopLine.setVisibility(View.GONE);
                mBottomLine.setVisibility(View.GONE);
            }
        });
    }
   设置动画
    mTopLayout.startAnimation(topAnim);
    mBottomLayout.startAnimation(bottomAnim);

}

}

Xml文件内的布局

<RelativeLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <!--摇一摇中心图片-->
    <TextView
        android:id="@+id/tv"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:textSize="30sp"
        android:textColor="@color/colorRed"
        android:text="@string/app_c"/>
    <LinearLayout
        android:gravity="center"
        android:orientation="vertical"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_alignParentTop="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true">
        <!--顶部的横线和图片-->
        <LinearLayout
            android:gravity="center_horizontal|bottom"
            android:id="@+id/main_linear_top"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">
            <ImageView
                android:src="@mipmap/shang"
                android:id="@+id/main_shake_top"
                android:layout_width="wrap_content"
                android:layout_height="100dp"/>
            <ImageView
                android:background="@mipmap/shake_top_line"
                android:id="@+id/main_shake_top_line"
                android:layout_width="match_parent"
                android:layout_height="5dp"/>
        </LinearLayout>

        <!--底部的横线和图片-->
        <LinearLayout
            android:gravity="center_horizontal|bottom"
            android:id="@+id/main_linear_bottom"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">

            <ImageView
                android:background="@mipmap/shake_bottom_line"
                android:id="@+id/main_shake_bottom_line"
                android:layout_width="match_parent"
                android:layout_height="5dp"/>
            <ImageView
                android:src="@mipmap/xia"
                android:id="@+id/main_shake_bottom"
                android:layout_width="wrap_content"
                android:layout_height="100dp"/>

        </LinearLayout>
    </LinearLayout>
</RelativeLayout>
 类似资料: