init

Signed-off-by: cjybyjk <cjybyjk@gmail.com>
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..9c0becf
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,9 @@
+*.iml
+.gradle
+/local.properties
+.DS_Store
+/build
+/captures
+.externalNativeBuild
+.cxx
+/.idea
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..8103e29
--- /dev/null
+++ b/README.md
@@ -0,0 +1,14 @@
+# GamingMode
+
+exTHmUI 游戏模式
+
+## 性能调节器说明
+当进入游戏模式或者用户调节 SeekBar 时, 游戏模式会设置 `sys.performance.level` 这个属性。\
+设备维护者需要在代码中添加对这个属性变化的监听。
+
+### 属性值说明
+- 0-6 : 数值越大性能越高
+-  -1 : 恢复默认性能 
+
+## 使用的第三方组件
+- [EasyDanmaku](https://github.com/LittleFogCat/EasyDanmaku)
diff --git a/app/.gitignore b/app/.gitignore
new file mode 100644
index 0000000..796b96d
--- /dev/null
+++ b/app/.gitignore
@@ -0,0 +1 @@
+/build
diff --git a/app/build.gradle b/app/build.gradle
new file mode 100644
index 0000000..01d3dc8
--- /dev/null
+++ b/app/build.gradle
@@ -0,0 +1,51 @@
+apply plugin: 'com.android.application'
+
+android {
+    compileSdkVersion 29
+    buildToolsVersion "29.0.3"
+
+    defaultConfig {
+        applicationId "org.exthmui.game"
+        minSdkVersion 29
+        targetSdkVersion 29
+        versionCode 1
+        versionName "1.0"
+
+        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
+    }
+
+    buildTypes {
+        release {
+            minifyEnabled false
+            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
+        }
+    }
+    compileOptions {
+        sourceCompatibility JavaVersion.VERSION_1_8
+        targetCompatibility JavaVersion.VERSION_1_8
+    }
+}
+
+dependencies {
+    compileOnly fileTree(dir: 'system_libs/', include: ['*.jar'])
+
+    implementation fileTree(dir: "libs", include: ["*.jar"])
+
+    implementation 'androidx.appcompat:appcompat:1.2.0'
+    implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
+    implementation 'androidx.preference:preference:1.1.1'
+    implementation 'com.google.android.material:material:1.1.0'
+    implementation 'androidx.localbroadcastmanager:localbroadcastmanager:1.0.0'
+    testImplementation 'junit:junit:4.12'
+    androidTestImplementation 'androidx.test.ext:junit:1.1.1'
+    androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
+
+}
+
+allprojects{
+    gradle.projectsEvaluated {
+        tasks.withType(JavaCompile) {
+            options.compilerArgs.add('-Xbootclasspath/p:app/system_libs/framework.jar')
+        }
+    }
+}
\ No newline at end of file
diff --git a/app/keystore.properties b/app/keystore.properties
new file mode 100644
index 0000000..5c4b578
--- /dev/null
+++ b/app/keystore.properties
@@ -0,0 +1,4 @@
+keyAlias=platform
+keyPassword=android
+storeFile=platform.jks
+storePassword=android
diff --git a/app/platform.jks b/app/platform.jks
new file mode 100644
index 0000000..7ca31cf
--- /dev/null
+++ b/app/platform.jks
Binary files differ
diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro
new file mode 100644
index 0000000..481bb43
--- /dev/null
+++ b/app/proguard-rules.pro
@@ -0,0 +1,21 @@
+# Add project specific ProGuard rules here.
+# You can control the set of applied configuration files using the
+# proguardFiles setting in build.gradle.
+#
+# For more details, see
+#   http://developer.android.com/guide/developing/tools/proguard.html
+
+# If your project uses WebView with JS, uncomment the following
+# and specify the fully qualified class name to the JavaScript interface
+# class:
+#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
+#   public *;
+#}
+
+# Uncomment this to preserve the line number information for
+# debugging stack traces.
+#-keepattributes SourceFile,LineNumberTable
+
+# If you keep the line number information, uncomment this to
+# hide the original source file name.
+#-renamesourcefileattribute SourceFile
\ No newline at end of file
diff --git a/app/src/androidTest/java/org/exthmui/game/ExampleInstrumentedTest.java b/app/src/androidTest/java/org/exthmui/game/ExampleInstrumentedTest.java
new file mode 100644
index 0000000..6d029f1
--- /dev/null
+++ b/app/src/androidTest/java/org/exthmui/game/ExampleInstrumentedTest.java
@@ -0,0 +1,26 @@
+package org.exthmui.game;
+
+import android.content.Context;
+
+import androidx.test.platform.app.InstrumentationRegistry;
+import androidx.test.ext.junit.runners.AndroidJUnit4;
+
+import org.junit.Test;
+import org.junit.runner.RunWith;
+
+import static org.junit.Assert.*;
+
+/**
+ * Instrumented test, which will execute on an Android device.
+ *
+ * @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
+ */
+@RunWith(AndroidJUnit4.class)
+public class ExampleInstrumentedTest {
+    @Test
+    public void useAppContext() {
+        // Context of the app under test.
+        Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
+        assertEquals("org.exthmui.game", appContext.getPackageName());
+    }
+}
\ No newline at end of file
diff --git a/app/src/main/Android.bp b/app/src/main/Android.bp
new file mode 100644
index 0000000..ad773ec
--- /dev/null
+++ b/app/src/main/Android.bp
@@ -0,0 +1,32 @@
+android_app {
+    name: "GamingMode",
+
+    resource_dirs: ["res"],
+
+    srcs: ["java/**/*.java"],
+
+    static_libs: [
+        "androidx.annotation_annotation",
+        "androidx.appcompat_appcompat",
+        "androidx.preference_preference",
+        "androidx-constraintlayout_constraintlayout",
+        "com.google.android.material_material",
+        "androidx.localbroadcastmanager_localbroadcastmanager",
+        "org.lineageos.platform.internal",
+    ],
+
+    platform_apis: true,
+    product_specific: true,
+    certificate: "platform",
+
+    required: ["privapp_whitelist_org.exthmui.game.xml"],
+
+}
+
+prebuilt_etc {
+    name: "privapp_whitelist_org.exthmui.game.xml",
+
+    src: "privapp_whitelist_org.exthmui.game.xml",
+    sub_dir: "permissions",
+
+}
diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..b46f012
--- /dev/null
+++ b/app/src/main/AndroidManifest.xml
@@ -0,0 +1,41 @@
+<?xml version="1.0" encoding="utf-8"?>
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+    package="org.exthmui.game"
+    android:sharedUserId="android.uid.system">
+
+    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
+    <uses-permission android:name="android.permission.MEDIA_CONTENT_CONTROL" />
+    <uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL" />
+
+    <application
+        android:allowBackup="true"
+        android:icon="@mipmap/ic_launcher"
+        android:label="@string/app_name"
+        android:roundIcon="@mipmap/ic_launcher_round"
+        android:supportsRtl="true"
+        android:theme="@style/AppTheme">
+        <receiver
+            android:name=".receiver.GamingBroadcastReceiver"
+            android:enabled="true">
+            <intent-filter>
+                <action android:name="exthmui.intent.action.GAMING_MODE_ON" />
+            </intent-filter>
+        </receiver>
+
+        <service
+            android:name=".services.OverlayService"
+            android:enabled="true"
+            android:exported="false" />
+        <service
+            android:name=".services.GamingService"
+            android:enabled="true" />
+        <service
+            android:name=".services.DanmakuService"
+            android:permission="android.permission.BIND_NOTIFICATION_LISTENER_SERVICE">
+            <intent-filter>
+                <action android:name="android.service.notification.NotificationListenerService" />
+            </intent-filter>
+        </service>
+    </application>
+
+</manifest>
\ No newline at end of file
diff --git a/app/src/main/java/org/exthmui/game/misc/Constants.java b/app/src/main/java/org/exthmui/game/misc/Constants.java
new file mode 100644
index 0000000..76822ed
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/misc/Constants.java
@@ -0,0 +1,114 @@
+package org.exthmui.game.misc;
+
+import android.provider.Settings;
+
+public class Constants {
+
+    // Notification
+    public final static String CHANNEL_GAMING_MODE_STATUS = "gaming_mode_status";
+
+    public final static String PROP_GAMING_PERFORMANCE = "sys.performance.level";
+
+    public static class Broadcasts {
+        public static final String SYS_BROADCAST_GAMING_MODE_ON = "exthmui.intent.action.GAMING_MODE_ON";
+        public static final String SYS_BROADCAST_GAMING_MODE_OFF = "exthmui.intent.action.GAMING_MODE_OFF";
+
+        // Local broadcast
+        public static final String BROADCAST_CONFIG_CHANGED = "org.exthmui.game.CONFIG_CHANGED";
+        public static final String BROADCAST_NEW_DANMAKU = "org.exthmui.game.NEW_DANMAKU";
+
+        /*
+        * GAMING_ACTION 需要携带的extra: String target, value
+        */
+        public static final String BROADCAST_GAMING_ACTION = "org.exthmui.game.GAMING_ACTION";
+        public static final String BROADCAST_GAMING_MENU_CONTROL = "org.exthmui.game.GAMING_MENU_CONTROL";
+    }
+
+    public static class ConfigKeys {
+        // 显示弹幕
+        public static final String SHOW_DANMAKU = Settings.System.GAMING_MODE_SHOW_DANMAKU;
+        // 弹幕速度
+        public static final String DANMAKU_SPEED_HORIZONTAL = Settings.System.GAMING_MODE_DANMAKU_SPEED_HORIZONTAL;
+        public static final String DANMAKU_SPEED_VERTICAL = Settings.System.GAMING_MODE_DANMAKU_SPEED_VERTICAL;
+        // 弹幕大小
+        public static final String DANMAKU_SIZE_HORIZONTAL = Settings.System.GAMING_MODE_DANMAKU_SIZE_HORIZONTAL;
+        public static final String DANMAKU_SIZE_VERTICAL = Settings.System.GAMING_MODE_DANMAKU_SIZE_VERTICAL;
+
+        // 动态过滤通知
+        public static final String DYNAMIC_NOTIFICATION_FILTER = Settings.System.GAMING_MODE_DANMAKU_DYNAMIC_NOTIFICATION_FILTER;
+        // 通知黑名单
+        public static final String NOTIFICATION_APP_BLACKLIST = Settings.System.GAMING_MODE_DANMAKU_APP_BLACKLIST;
+
+        // 禁用自动亮度
+        public static final String DISABLE_AUTO_BRIGHTNESS = Settings.System.GAMING_MODE_DISABLE_AUTO_BRIGHTNESS;
+
+        // 请勿打扰模式
+        public static final String DISABLE_RINGTONE = Settings.System.GAMING_MODE_DISABLE_RINGTONE;
+
+        // 屏蔽按键
+        public static final String DISABLE_HW_KEYS = Settings.System.GAMING_MODE_DISABLE_HW_KEYS;
+
+        // 屏蔽手势
+        public static final String DISABLE_GESTURE = Settings.System.GAMING_MODE_DISABLE_GESTURE;
+
+        // 性能配置
+        public static final String CHANGE_PERFORMANCE_LEVEL = Settings.System.GAMING_MODE_CHANGE_PERFORMANCE_LEVEL;
+        public static final String PERFORMANCE_LEVEL = Settings.System.GAMING_MODE_PERFORMANCE_LEVEL;
+
+        // 快速启动app
+        public static final String QUICK_START_APPS = Settings.System.GAMING_MODE_QS_APP_LIST;
+    }
+
+    public static class ConfigDefaultValues {
+        // 显示弹幕
+        public static final boolean SHOW_DANMAKU = true;
+        // 弹幕速度
+        public static final int DANMAKU_SPEED_HORIZONTAL = 300;
+        public static final int DANMAKU_SPEED_VERTICAL = 300;
+        // 弹幕大小
+        public static final int DANMAKU_SIZE_HORIZONTAL = 36;
+        public static final int DANMAKU_SIZE_VERTICAL = 36;
+
+        // 动态过滤通知
+        public static final boolean DYNAMIC_NOTIFICATION_FILTER = true;
+
+        // 请勿打扰模式
+        public static final boolean DISABLE_RINGTONE = true;
+
+        // 关闭自动亮度
+        public static final boolean DISABLE_AUTO_BRIGHTNESS = true;
+
+        // 屏蔽手势
+        public static final boolean DISABLE_GESTURE = true;
+
+        // 屏蔽按键
+        public static final boolean DISABLE_HW_KEYS = false;
+
+        // 性能配置
+        public static final boolean CHANGE_PERFORMANCE_LEVEL = true;
+        public static final int PERFORMANCE_LEVEL = 5;
+    }
+
+    public static class GamingActionTargets {
+        // 显示弹幕
+        public static final String SHOW_DANMAKU = ConfigKeys.SHOW_DANMAKU;
+        // 请勿打扰模式
+        public static final String DISABLE_RINGTONE = ConfigKeys.DISABLE_RINGTONE;
+        // 屏蔽按键
+        public static final String DISABLE_HW_KEYS = ConfigKeys.DISABLE_HW_KEYS;
+        // 屏蔽手势
+        public static final String DISABLE_GESTURE = ConfigKeys.DISABLE_GESTURE;
+        // 自动亮度
+        public static final String DISABLE_AUTO_BRIGHTNESS = ConfigKeys.DISABLE_AUTO_BRIGHTNESS;
+        // 性能配置
+        public static final String PERFORMANCE_LEVEL = ConfigKeys.PERFORMANCE_LEVEL;
+    }
+
+    public static class LocalConfigKeys {
+        // 悬浮球位置
+        public static final String FLOATING_BUTTON_COORDINATE_HORIZONTAL_X = "floating_button_horizontal_x";
+        public static final String FLOATING_BUTTON_COORDINATE_HORIZONTAL_Y = "floating_button_horizontal_y";
+        public static final String FLOATING_BUTTON_COORDINATE_VERTICAL_X = "floating_button_vertical_x";
+        public static final String FLOATING_BUTTON_COORDINATE_VERTICAL_Y = "floating_button_vertical_y";
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/AppTile.java b/app/src/main/java/org/exthmui/game/qs/AppTile.java
new file mode 100644
index 0000000..cdafbd9
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/AppTile.java
@@ -0,0 +1,21 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+import android.content.pm.ApplicationInfo;
+import android.content.pm.PackageManager;
+
+public class AppTile extends TileBase {
+
+    public AppTile(Context context, String packageName) {
+        super(context, packageName, packageName, null);
+        PackageManager pm = context.getPackageManager();
+        try {
+            ApplicationInfo applicationInfo = pm.getApplicationInfo(packageName, PackageManager.GET_META_DATA);
+            qsText.setText(applicationInfo.loadLabel(pm));
+            qsIcon.setImageDrawable(applicationInfo.loadIcon(pm));
+        } catch (PackageManager.NameNotFoundException e) {
+            e.printStackTrace();
+        }
+
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/AutoBrightnessTile.java b/app/src/main/java/org/exthmui/game/qs/AutoBrightnessTile.java
new file mode 100644
index 0000000..f1c76bb
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/AutoBrightnessTile.java
@@ -0,0 +1,13 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class AutoBrightnessTile extends TileBase {
+    public AutoBrightnessTile(Context context) {
+        super(context, "自动亮度", Constants.GamingActionTargets.DISABLE_AUTO_BRIGHTNESS, R.drawable.ic_qs_auto_brightness);
+        setNeedInverse(true);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/DNDTile.java b/app/src/main/java/org/exthmui/game/qs/DNDTile.java
new file mode 100644
index 0000000..eddda11
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/DNDTile.java
@@ -0,0 +1,17 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+import android.content.Intent;
+import android.os.Bundle;
+import android.view.View;
+
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class DNDTile extends TileBase {
+    public DNDTile(Context context) {
+        super(context, "勿扰模式", Constants.GamingActionTargets.DISABLE_RINGTONE, R.drawable.ic_qs_dnd);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/DanmakuTile.java b/app/src/main/java/org/exthmui/game/qs/DanmakuTile.java
new file mode 100644
index 0000000..695fad5
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/DanmakuTile.java
@@ -0,0 +1,12 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class DanmakuTile extends TileBase {
+    public DanmakuTile(Context context) {
+        super(context, "通知弹幕", Constants.GamingActionTargets.SHOW_DANMAKU, R.drawable.ic_qs_danmaku);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/LockGestureTile.java b/app/src/main/java/org/exthmui/game/qs/LockGestureTile.java
new file mode 100644
index 0000000..8418fcd
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/LockGestureTile.java
@@ -0,0 +1,12 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class LockGestureTile extends TileBase {
+    public LockGestureTile(Context context) {
+        super(context, "屏蔽手势", Constants.GamingActionTargets.DISABLE_GESTURE, R.drawable.ic_qs_disable_gesture);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/LockHwKeysTile.java b/app/src/main/java/org/exthmui/game/qs/LockHwKeysTile.java
new file mode 100644
index 0000000..d628d5a
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/LockHwKeysTile.java
@@ -0,0 +1,12 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class LockHwKeysTile extends TileBase {
+    public LockHwKeysTile(Context context) {
+        super(context, "屏蔽按键", Constants.GamingActionTargets.DISABLE_HW_KEYS, R.drawable.ic_qs_disable_hw_key);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/ScreenCaptureTile.java b/app/src/main/java/org/exthmui/game/qs/ScreenCaptureTile.java
new file mode 100644
index 0000000..5e2988f
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/ScreenCaptureTile.java
@@ -0,0 +1,45 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+import android.content.Intent;
+import android.os.Bundle;
+import android.os.Handler;
+import android.os.RemoteException;
+import android.util.Log;
+import android.view.View;
+import android.view.WindowManagerGlobal;
+
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class ScreenCaptureTile extends TileBase {
+
+    private static final String TAG = "ScreenCaptureTile";
+
+    private Intent hideMenuIntent = new Intent(Constants.Broadcasts.BROADCAST_GAMING_MENU_CONTROL).putExtra("cmd", "hide");
+    private Handler mHandler = new Handler();
+
+    public ScreenCaptureTile(Context context) {
+        super(context, "屏幕截图", "", R.drawable.ic_qs_screenshot);
+        qsIcon.setSelected(true);
+    }
+
+    @Override
+    public void setConfig(Bundle bundle) {
+
+    }
+
+    @Override
+    public void onClick(View v) {
+        LocalBroadcastManager.getInstance(mContext).sendBroadcastSync(hideMenuIntent);
+        mHandler.postDelayed(() -> {
+            try {
+                WindowManagerGlobal.getWindowManagerService().takeScreenshot(1);
+            } catch (RemoteException e) {
+                Log.e(TAG, "Error while trying to take screenshot.", e);
+            }
+        }, 500);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/qs/TileBase.java b/app/src/main/java/org/exthmui/game/qs/TileBase.java
new file mode 100644
index 0000000..30e8034
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/qs/TileBase.java
@@ -0,0 +1,82 @@
+package org.exthmui.game.qs;
+
+import android.content.Context;
+import android.content.Intent;
+import android.graphics.drawable.Drawable;
+import android.os.Bundle;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.ImageView;
+import android.widget.LinearLayout;
+import android.widget.TextView;
+
+import androidx.annotation.Nullable;
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class TileBase extends LinearLayout implements View.OnClickListener {
+    ImageView qsIcon;
+    TextView qsText;
+
+    Context mContext;
+
+    private boolean isSelected;
+    private String key;
+    Intent intent;
+    private boolean needInverse;
+
+    public TileBase(Context context, CharSequence text, String key, int iconRes) {
+        this(context, null);
+        qsIcon.setImageResource(iconRes);
+        qsText.setText(text);
+        this.key = key;
+        intent = new Intent(Constants.Broadcasts.BROADCAST_GAMING_ACTION).putExtra("target", key);
+    }
+
+    public TileBase(Context context, CharSequence text, String key, Drawable icon) {
+        this(context, null);
+        qsIcon.setImageDrawable(icon);
+        qsText.setText(text);
+        this.key = key;
+        intent = new Intent(Constants.Broadcasts.BROADCAST_GAMING_ACTION).putExtra("target", key);
+    }
+
+    public TileBase(Context context, @Nullable AttributeSet attrs) {
+        this(context, attrs, 0);
+    }
+
+    public TileBase(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
+        this(context, attrs, defStyleAttr, 0);
+    }
+
+    public TileBase(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+        mContext = context;
+        LayoutInflater.from(context).inflate(R.layout.gaming_qs_view, this, true);
+        qsIcon = findViewById(R.id.qs_icon);
+        qsText = findViewById(R.id.qs_text);
+        this.setOnClickListener(this);
+    }
+
+    public void setConfig(Bundle bundle) {
+        isSelected = bundle.getBoolean(key, isSelected);
+        qsIcon.setSelected(needInverse ? !isSelected : isSelected);
+    }
+
+    public String getKey() {
+        return key;
+    }
+
+    public void setNeedInverse(boolean val) {
+        needInverse = val;
+    }
+
+    @Override
+    public void onClick(View v) {
+        intent.putExtra("value", !isSelected);
+        LocalBroadcastManager.getInstance(mContext).sendBroadcast(intent);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/receiver/GamingBroadcastReceiver.java b/app/src/main/java/org/exthmui/game/receiver/GamingBroadcastReceiver.java
new file mode 100644
index 0000000..e9f85ec
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/receiver/GamingBroadcastReceiver.java
@@ -0,0 +1,24 @@
+package org.exthmui.game.receiver;
+
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.os.UserHandle;
+import android.util.Log;
+
+import org.exthmui.game.misc.Constants;
+import org.exthmui.game.services.GamingService;
+
+public class GamingBroadcastReceiver extends BroadcastReceiver {
+
+    @Override
+    public void onReceive(Context context, Intent intent) {
+        if (intent != null) {
+            if (Constants.Broadcasts.SYS_BROADCAST_GAMING_MODE_ON.equals(intent.getAction())) {
+                Intent serviceIntent = new Intent(context, GamingService.class);
+                serviceIntent.putExtras(intent);
+                context.startForegroundServiceAsUser(serviceIntent, UserHandle.CURRENT);
+            }
+        }
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/services/DanmakuService.java b/app/src/main/java/org/exthmui/game/services/DanmakuService.java
new file mode 100644
index 0000000..fb0cf55
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/services/DanmakuService.java
@@ -0,0 +1,140 @@
+package org.exthmui.game.services;
+
+import android.app.Notification;
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.content.IntentFilter;
+import android.os.Bundle;
+import android.service.notification.NotificationListenerService;
+import android.service.notification.StatusBarNotification;
+import android.text.TextUtils;
+
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.misc.Constants;
+
+import java.util.HashMap;
+import java.util.Map;
+
+public class DanmakuService extends NotificationListenerService {
+
+    private Map<String, String> mLastNotificationMap = new HashMap<>();
+
+    private int filterThreshold = 3;
+    private boolean showDanmaku = Constants.ConfigDefaultValues.SHOW_DANMAKU;
+    private boolean useFilter = Constants.ConfigDefaultValues.DYNAMIC_NOTIFICATION_FILTER;
+    private String[] mNotificationBlacklist = new String[]{};
+    private BroadcastReceiver configReceiver = new BroadcastReceiver() {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            if (Constants.Broadcasts.BROADCAST_CONFIG_CHANGED.equals(intent.getAction())) {
+                updateConfig(intent);
+            }
+        }
+    };
+
+    public DanmakuService() {
+    }
+
+    @Override
+    public void onListenerConnected() {
+        super.onListenerConnected();
+        LocalBroadcastManager.getInstance(this).registerReceiver(configReceiver, new IntentFilter(Constants.Broadcasts.BROADCAST_CONFIG_CHANGED));
+    }
+
+    @Override
+    public void onListenerDisconnected() {
+        unregisterReceiver(configReceiver);
+        super.onListenerDisconnected();
+    }
+
+    @Override
+    public void onNotificationPosted(StatusBarNotification sbn) {
+        if (!showDanmaku) return;
+        Bundle extras = sbn.getNotification().extras;
+        if (!isInBlackList(sbn.getPackageName())){
+            String lastNotification = mLastNotificationMap.getOrDefault(sbn.getPackageName(), "");
+            String title = extras.getString(Notification.EXTRA_TITLE);
+            if (TextUtils.isEmpty(title)) title = extras.getString(Notification.EXTRA_TITLE_BIG);
+            String text = extras.getString(Notification.EXTRA_TEXT);
+            StringBuilder builder = new StringBuilder();
+            if (!TextUtils.isEmpty(title)) {
+                builder.append("[");
+                builder.append(title);
+                builder.append("] ");
+            }
+            if (!TextUtils.isEmpty(text)) {
+                builder.append(text);
+            }
+            String danmakuText = builder.toString();
+            if (!TextUtils.isEmpty(danmakuText) && (!useFilter || compareDanmaku(danmakuText, lastNotification))) {
+                sendDanmaku(danmakuText);
+            }
+            mLastNotificationMap.put(sbn.getPackageName(), danmakuText);
+        }
+    }
+
+    private boolean isInBlackList(String packageName) {
+        if (mNotificationBlacklist == null) return false;
+        for (String str : mNotificationBlacklist) {
+            if (TextUtils.equals(str, packageName)) {
+                return true;
+            }
+        }
+        return false;
+    }
+
+    private void sendDanmaku(String text) {
+        Intent intent = new Intent(Constants.Broadcasts.BROADCAST_NEW_DANMAKU);
+        intent.putExtra("danmaku_text", text);
+        LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
+    }
+
+    private boolean compareDanmaku(String a, String b) {
+        String tA = a.replaceAll("[0-9]", "");
+        String tB = b.replaceAll("[0-9]", "");
+        if (TextUtils.isEmpty(tA) || TextUtils.isEmpty(tB)) {
+            return levenshtein(a, b) > filterThreshold;
+        } else {
+            return levenshtein(tA, tB) > filterThreshold;
+        }
+    }
+
+    private void updateConfig(Intent intent) {
+        if (intent != null) {
+            showDanmaku = intent.getBooleanExtra(Constants.ConfigKeys.SHOW_DANMAKU, showDanmaku);
+            useFilter = intent.getBooleanExtra(Constants.ConfigKeys.DYNAMIC_NOTIFICATION_FILTER, useFilter);
+            mNotificationBlacklist = intent.getStringArrayExtra(Constants.ConfigKeys.NOTIFICATION_APP_BLACKLIST);
+        }
+    }
+
+    // 最小编辑距离
+    public static int levenshtein(CharSequence a, CharSequence b) {
+        if (TextUtils.isEmpty(a)) {
+            return TextUtils.isEmpty(b) ? 0 : b.length();
+        } else if (TextUtils.isEmpty(b)) {
+            return TextUtils.isEmpty(a) ? 0 : a.length();
+        }
+        final int lenA = a.length(), lenB = b.length();
+        int[][] dp = new int[lenA+1][lenB+1];
+        int flag = 0;
+        for (int i = 0; i <= lenA; i++) {
+            for (int j = 0; j <= lenB; j++) dp[i][j] = lenA + lenB;
+        }
+        for(int i=1; i <= lenA; i++) dp[i][0] = i;
+        for(int j=1; j <= lenB; j++) dp[0][j] = j;
+        for (int i = 1; i <= lenA; i++) {
+            for (int j = 1; j <= lenB; j++) {
+                if (a.charAt(i-1) == b.charAt(j-1)) {
+                    flag = 0;
+                } else {
+                    flag = 1;
+                }
+                dp[i][j] = Math.min(dp[i-1][j-1] + flag, Math.min(dp[i-1][j] + 1, dp[i][j-1] + 1));
+            }
+        }
+        return dp[lenA][lenB];
+    }
+
+}
diff --git a/app/src/main/java/org/exthmui/game/services/GamingService.java b/app/src/main/java/org/exthmui/game/services/GamingService.java
new file mode 100644
index 0000000..2a5f246
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/services/GamingService.java
@@ -0,0 +1,291 @@
+package org.exthmui.game.services;
+
+import android.app.Notification;
+import android.app.NotificationChannel;
+import android.app.NotificationManager;
+import android.app.PendingIntent;
+import android.app.Service;
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.content.IntentFilter;
+import android.media.AudioManager;
+import android.os.Bundle;
+import android.os.IBinder;
+import android.os.RemoteException;
+import android.os.ServiceManager;
+import android.os.SystemProperties;
+import android.os.UserHandle;
+import android.provider.Settings;
+import android.telecom.TelecomManager;
+import android.telephony.PhoneStateListener;
+import android.telephony.TelephonyManager;
+import android.text.TextUtils;
+import android.widget.Toast;
+
+import androidx.annotation.Nullable;
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import com.android.internal.statusbar.IStatusBarService;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+import lineageos.hardware.LineageHardwareManager;
+
+public class GamingService extends Service {
+
+    private static final int NOTIFICATION_ID = 1;
+
+    private Intent mOverlayServiceIntent;
+    private Notification mGamingNotification;
+
+    private String mCurrentPackage;
+
+    private Bundle mCurrentConfig = new Bundle();
+
+    private AudioManager mAudioManager;
+    private IStatusBarService mStatusBarService;
+    private LineageHardwareManager mLineageHardware;
+
+    private BroadcastReceiver mGamingModeOffReceiver = new BroadcastReceiver() {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            if (Constants.Broadcasts.SYS_BROADCAST_GAMING_MODE_OFF.equals(intent.getAction())) {
+                stopSelf();
+            }
+        }
+    };
+
+    private BroadcastReceiver mGamingActionReceiver = new BroadcastReceiver() {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            String target = intent.getStringExtra("target");
+            Intent configChangedIntent = new Intent(Constants.Broadcasts.BROADCAST_CONFIG_CHANGED);
+            if (Constants.GamingActionTargets.DISABLE_AUTO_BRIGHTNESS.equals(target)) {
+                setDisableAutoBrightness(intent.getBooleanExtra("value", Constants.ConfigDefaultValues.DISABLE_AUTO_BRIGHTNESS), false);
+            } else if (Constants.GamingActionTargets.DISABLE_GESTURE.equals(target)) {
+                setDisableGesture(intent.getBooleanExtra("value", Constants.ConfigDefaultValues.DISABLE_GESTURE));
+            } else if (Constants.GamingActionTargets.DISABLE_HW_KEYS.equals(target)) {
+                setDisableHwKeys(intent.getBooleanExtra("value", Constants.ConfigDefaultValues.DISABLE_HW_KEYS), false);
+            } else if (Constants.GamingActionTargets.DISABLE_RINGTONE.equals(target)) {
+                setDisableRingtone(intent.getBooleanExtra("value", Constants.ConfigDefaultValues.DISABLE_RINGTONE));
+            } else if (Constants.GamingActionTargets.SHOW_DANMAKU.equals(target)) {
+                setShowDanmaku(intent.getBooleanExtra("value", Constants.ConfigDefaultValues.SHOW_DANMAKU));
+            } else if (Constants.GamingActionTargets.PERFORMANCE_LEVEL.equals(target)) {
+                setPerformanceLevel(intent.getIntExtra("value", Constants.ConfigDefaultValues.PERFORMANCE_LEVEL));
+            } else {
+                return;
+            }
+            configChangedIntent.putExtras(mCurrentConfig);
+            LocalBroadcastManager.getInstance(GamingService.this).sendBroadcast(configChangedIntent);
+        }
+    };
+
+
+    public GamingService() {
+    }
+
+    @Override
+    public void onCreate() {
+        super.onCreate();
+        createNotificationChannel(this, Constants.CHANNEL_GAMING_MODE_STATUS, getString(R.string.channel_gaming_mode_status), NotificationManager.IMPORTANCE_LOW);
+
+        mAudioManager = (AudioManager) getSystemService(AUDIO_SERVICE);
+        mStatusBarService = IStatusBarService.Stub.asInterface(ServiceManager.getService(Context.STATUS_BAR_SERVICE));
+        try {
+            mLineageHardware = LineageHardwareManager.getInstance(this);
+        } catch (Error e) {
+            e.printStackTrace();
+        }
+
+        registerReceiver(mGamingModeOffReceiver, new IntentFilter(Constants.Broadcasts.SYS_BROADCAST_GAMING_MODE_OFF));
+        LocalBroadcastManager.getInstance(this).registerReceiver(mGamingActionReceiver, new IntentFilter(Constants.Broadcasts.BROADCAST_GAMING_ACTION));
+
+        mOverlayServiceIntent = new Intent(this, OverlayService.class);
+
+        PendingIntent stopGamingIntent = PendingIntent.getBroadcast(this, 0, new Intent(Constants.Broadcasts.SYS_BROADCAST_GAMING_MODE_OFF), 0);
+        Notification.Builder builder = new Notification.Builder(this, Constants.CHANNEL_GAMING_MODE_STATUS);
+        Notification.Action.Builder actionBuilder = new Notification.Action.Builder(null, getString(R.string.action_stop_gaming_mode), stopGamingIntent);
+        builder.addAction(actionBuilder.build());
+        builder.setContentText(getString(R.string.gaming_mode_running));
+        builder.setSmallIcon(R.drawable.ic_launcher_foreground);
+
+        mGamingNotification = builder.build();
+        startForeground(NOTIFICATION_ID, mGamingNotification);
+
+        Toast.makeText(this, R.string.gaming_mode_on, Toast.LENGTH_SHORT).show();
+    }
+
+    @Override
+    public int onStartCommand(Intent intent, int flags, int startId) {
+        if (!TextUtils.equals(intent.getStringExtra("package"), mCurrentPackage)) {
+            mCurrentPackage = intent.getStringExtra("package");
+            updateConfig();
+        }
+
+        mOverlayServiceIntent.putExtras(mCurrentConfig);
+        startServiceAsUser(mOverlayServiceIntent, UserHandle.CURRENT);
+        Settings.System.putInt(getContentResolver(), Settings.System.GAMING_MODE_ACTIVE, 1);
+        return super.onStartCommand(intent, flags, startId);
+    }
+
+    private void updateConfig() {
+        // danmaku
+        mCurrentConfig.putBoolean(Constants.ConfigKeys.SHOW_DANMAKU, getBooleanSetting(Constants.ConfigKeys.SHOW_DANMAKU, Constants.ConfigDefaultValues.SHOW_DANMAKU));
+        mCurrentConfig.putInt(Constants.ConfigKeys.DANMAKU_SPEED_HORIZONTAL, getIntSetting(Constants.ConfigKeys.DANMAKU_SPEED_HORIZONTAL, Constants.ConfigDefaultValues.DANMAKU_SPEED_HORIZONTAL));
+        mCurrentConfig.putInt(Constants.ConfigKeys.DANMAKU_SPEED_VERTICAL, getIntSetting(Constants.ConfigKeys.DANMAKU_SPEED_VERTICAL, Constants.ConfigDefaultValues.DANMAKU_SPEED_VERTICAL));
+        mCurrentConfig.putInt(Constants.ConfigKeys.DANMAKU_SIZE_HORIZONTAL, getIntSetting(Constants.ConfigKeys.DANMAKU_SIZE_HORIZONTAL, Constants.ConfigDefaultValues.DANMAKU_SIZE_HORIZONTAL));
+        mCurrentConfig.putInt(Constants.ConfigKeys.DANMAKU_SIZE_VERTICAL, getIntSetting(Constants.ConfigKeys.DANMAKU_SIZE_VERTICAL, Constants.ConfigDefaultValues.DANMAKU_SIZE_VERTICAL));
+        mCurrentConfig.putBoolean(Constants.ConfigKeys.DYNAMIC_NOTIFICATION_FILTER, getBooleanSetting(Constants.ConfigKeys.DYNAMIC_NOTIFICATION_FILTER, Constants.ConfigDefaultValues.DYNAMIC_NOTIFICATION_FILTER));
+        mCurrentConfig.putStringArray(Constants.ConfigKeys.NOTIFICATION_APP_BLACKLIST, getStringArraySetting(Constants.ConfigKeys.NOTIFICATION_APP_BLACKLIST));
+
+        // performance
+        boolean changePerformance = getBooleanSetting(Constants.ConfigKeys.CHANGE_PERFORMANCE_LEVEL, Constants.ConfigDefaultValues.CHANGE_PERFORMANCE_LEVEL);
+        int performanceLevel = getIntSetting(Constants.ConfigKeys.PERFORMANCE_LEVEL, Constants.ConfigDefaultValues.PERFORMANCE_LEVEL);
+        if (changePerformance) {
+            setPerformanceLevel(performanceLevel);
+        } else {
+            mCurrentConfig.putInt(Constants.ConfigKeys.PERFORMANCE_LEVEL, performanceLevel);
+        }
+
+        // hw keys & gesture
+        boolean disableHwKeys = getBooleanSetting(Constants.ConfigKeys.DISABLE_HW_KEYS, Constants.ConfigDefaultValues.DISABLE_HW_KEYS);
+        boolean disableGesture = getBooleanSetting(Constants.ConfigKeys.DISABLE_GESTURE, Constants.ConfigDefaultValues.DISABLE_GESTURE);
+        setDisableHwKeys(disableHwKeys, false);
+        setDisableGesture(disableGesture);
+
+        // quick-start apps
+        mCurrentConfig.putStringArray(Constants.ConfigKeys.QUICK_START_APPS, getStringArraySetting(Constants.ConfigKeys.QUICK_START_APPS));
+        setAutoRotation(false);
+
+        // misc
+        boolean disableRingtone = getBooleanSetting(Constants.ConfigKeys.DISABLE_RINGTONE, Constants.ConfigDefaultValues.DISABLE_RINGTONE);
+        setDisableRingtone(disableRingtone);
+        boolean disableAutoBrightness = getBooleanSetting(Constants.ConfigKeys.DISABLE_AUTO_BRIGHTNESS, Constants.ConfigDefaultValues.DISABLE_AUTO_BRIGHTNESS);
+        setDisableAutoBrightness(disableAutoBrightness, false);
+
+        Intent intent = new Intent(Constants.Broadcasts.BROADCAST_CONFIG_CHANGED);
+        intent.putExtras(mCurrentConfig);
+        LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
+    }
+
+    private void setDisableHwKeys(boolean disable, boolean restore) {
+        if (mLineageHardware == null) return;
+        if (!mCurrentConfig.containsKey("old_disable_hw_keys")) {
+            boolean oldValue = mLineageHardware.get(LineageHardwareManager.FEATURE_KEY_DISABLE);
+            mCurrentConfig.putBoolean("old_disable_hw_keys", oldValue);
+        }
+        if (!restore) {
+            mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_HW_KEYS, disable);
+            mLineageHardware.set(LineageHardwareManager.FEATURE_KEY_DISABLE, disable);
+        } else {
+            boolean oldValue = mCurrentConfig.getBoolean("old_disable_hw_keys");
+            mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_HW_KEYS, oldValue);
+            mLineageHardware.set(LineageHardwareManager.FEATURE_KEY_DISABLE, oldValue);
+        }
+    }
+
+    private void setDisableGesture(boolean disable) {
+        mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_GESTURE, disable);
+        try {
+            mStatusBarService.setBlockedGesturalNavigation(disable);
+        } catch (RemoteException e) {
+            e.printStackTrace();
+        }
+    }
+
+    private void setShowDanmaku(boolean show) {
+        mCurrentConfig.putBoolean(Constants.ConfigKeys.SHOW_DANMAKU, show);
+    }
+
+    private void setPerformanceLevel(int level) {
+        SystemProperties.set(Constants.PROP_GAMING_PERFORMANCE, String.valueOf(level));
+        mCurrentConfig.putInt(Constants.ConfigKeys.PERFORMANCE_LEVEL, level);
+    }
+
+    private void setDisableRingtone(boolean disable) {
+        if (!mCurrentConfig.containsKey("old_ringer_mode")) {
+            mCurrentConfig.putInt("old_ringer_mode", mAudioManager.getRingerMode());
+        }
+        int oldRingerMode = mCurrentConfig.getInt("old_ringer_mode", AudioManager.RINGER_MODE_NORMAL);
+        mAudioManager.setRingerMode(disable ? AudioManager.RINGER_MODE_VIBRATE : oldRingerMode);
+        mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_RINGTONE, disable);
+    }
+
+    private void setDisableAutoBrightness(boolean disable, boolean restore) {
+        if (!mCurrentConfig.containsKey("old_auto_brightness")) {
+            int oldValue = getIntSetting(Settings.System.SCREEN_BRIGHTNESS_MODE, Settings.System.SCREEN_BRIGHTNESS_MODE_AUTOMATIC);
+            mCurrentConfig.putInt("old_auto_brightness", oldValue);
+        }
+        if (!restore) {
+            if (disable) {
+                Settings.System.putInt(getContentResolver(), Settings.System.SCREEN_BRIGHTNESS_MODE, Settings.System.SCREEN_BRIGHTNESS_MODE_MANUAL);
+            } else {
+                Settings.System.putInt(getContentResolver(), Settings.System.SCREEN_BRIGHTNESS_MODE, Settings.System.SCREEN_BRIGHTNESS_MODE_AUTOMATIC);
+            }
+            mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_AUTO_BRIGHTNESS, disable);
+        } else {
+            int oldValue = mCurrentConfig.getInt("old_auto_brightness");
+            mCurrentConfig.putBoolean(Constants.ConfigKeys.DISABLE_AUTO_BRIGHTNESS, oldValue == Settings.System.SCREEN_BRIGHTNESS_MODE_MANUAL);
+            Settings.System.putInt(getContentResolver(), Settings.System.SCREEN_BRIGHTNESS_MODE, oldValue);
+        }
+    }
+
+    private void setAutoRotation(boolean restore) {
+        if (!mCurrentConfig.containsKey("old_auto_rotation")) {
+            mCurrentConfig.putInt("old_auto_rotation", getIntSetting(Settings.System.ACCELEROMETER_ROTATION, 0));
+        }
+        if (!restore) {
+            Settings.System.putInt(getContentResolver(), Settings.System.ACCELEROMETER_ROTATION, 1);
+        } else {
+            int oldValue = mCurrentConfig.getInt("old_auto_rotation");
+            Settings.System.putInt(getContentResolver(), Settings.System.ACCELEROMETER_ROTATION, oldValue);
+        }
+    }
+
+    @Override
+    public void onDestroy() {
+        unregisterReceiver(mGamingModeOffReceiver);
+        stopServiceAsUser(mOverlayServiceIntent, UserHandle.CURRENT);
+        setDisableGesture(false);
+        setDisableHwKeys(false, true);
+        setDisableAutoBrightness(false, true);
+        setDisableRingtone(false);
+        setPerformanceLevel(-1);
+        setAutoRotation(true);
+        Settings.System.putInt(getContentResolver(), Settings.System.GAMING_MODE_ACTIVE, 0);
+        Toast.makeText(this, R.string.gaming_mode_off, Toast.LENGTH_SHORT).show();
+        super.onDestroy();
+    }
+
+    @Nullable
+    @Override
+    public IBinder onBind(Intent intent) {
+        return null;
+    }
+
+    private static void createNotificationChannel(Context context, String channelId, String channelName, int importance) {
+        NotificationChannel channel = new NotificationChannel(channelId, channelName, importance);
+        NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
+        notificationManager.createNotificationChannel(channel);
+    }
+
+    private boolean getBooleanSetting(String key, boolean def) {
+        return Settings.System.getInt(getContentResolver(), key, def ? 1 : 0) != 0;
+    }
+
+    private int getIntSetting(String key, int def) {
+        return Settings.System.getInt(getContentResolver(), key, def);
+    }
+
+    private String[] getStringArraySetting(String key) {
+        String val = Settings.System.getString(getContentResolver(), key);
+        if (!TextUtils.isEmpty(val)) {
+            return val.split(";");
+        } else {
+            return null;
+        }
+    }
+
+}
diff --git a/app/src/main/java/org/exthmui/game/services/OverlayService.java b/app/src/main/java/org/exthmui/game/services/OverlayService.java
new file mode 100644
index 0000000..31053bc
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/services/OverlayService.java
@@ -0,0 +1,341 @@
+package org.exthmui.game.services;
+
+import android.app.Service;
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.content.IntentFilter;
+import android.content.SharedPreferences;
+import android.graphics.PixelFormat;
+import android.os.Bundle;
+import android.os.IBinder;
+import android.preference.PreferenceManager;
+import android.provider.Settings;
+import android.text.TextUtils;
+import android.view.Gravity;
+import android.view.LayoutInflater;
+import android.view.MotionEvent;
+import android.view.View;
+import android.view.ViewGroup;
+import android.view.WindowManager;
+import android.widget.FrameLayout;
+import android.widget.LinearLayout;
+import android.widget.ScrollView;
+
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+import org.exthmui.game.ui.GamingPerformanceView;
+import org.exthmui.game.ui.QuickStartAppView;
+import org.exthmui.game.ui.QuickSettingsView;
+
+import top.littlefogcat.danmakulib.danmaku.Danmaku;
+import top.littlefogcat.danmakulib.danmaku.DanmakuManager;
+import top.littlefogcat.danmakulib.utils.ScreenUtil;
+
+public class OverlayService extends Service {
+
+    private View mGamingFloatingButton;
+    private LinearLayout mGamingOverlayView;
+    private ScrollView mGamingMenu;
+    private FrameLayout mDanmakuContainer;
+    private QuickSettingsView mQSView;
+    private QuickStartAppView mQSAppView;
+
+    private DanmakuManager mDanmakuManager;
+
+    private WindowManager mWindowManager;
+    private WindowManager.LayoutParams mGamingFBLayoutParams;
+
+    private GamingPerformanceView performanceController;
+
+    private Bundle configBundle;
+
+    private OMReceiver mOMReceiver = new OMReceiver();
+    private BroadcastReceiver mSysConfigChangedReceiver = new BroadcastReceiver() {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            if (Intent.ACTION_CONFIGURATION_CHANGED.equals(intent.getAction())) updateConfig();
+        }
+    };
+
+    private SharedPreferences mPreferences;
+
+    public OverlayService() {
+    }
+
+    @Override
+    public IBinder onBind(Intent intent) {
+        return null;
+    }
+
+    @Override
+    public void onCreate() {
+        super.onCreate();
+        configBundle = new Bundle();
+
+        if (Settings.canDrawOverlays(this)) {
+            initView();
+        }
+
+        IntentFilter intentFilter = new IntentFilter();
+        intentFilter.addAction(Constants.Broadcasts.BROADCAST_NEW_DANMAKU);
+        intentFilter.addAction(Constants.Broadcasts.BROADCAST_CONFIG_CHANGED);
+        intentFilter.addAction(Constants.Broadcasts.BROADCAST_GAMING_MENU_CONTROL);
+        LocalBroadcastManager.getInstance(this).registerReceiver(mOMReceiver, intentFilter);
+
+        registerReceiver(mSysConfigChangedReceiver, new IntentFilter(Intent.ACTION_CONFIGURATION_CHANGED));
+
+        mPreferences = PreferenceManager.getDefaultSharedPreferences(this);
+    }
+
+    @Override
+    public int onStartCommand(Intent intent, int flags, int startId) {
+        if (intent.getExtras() != null) configBundle.putAll(intent.getExtras());
+        updateConfig();
+        return super.onStartCommand(intent, flags, startId);
+    }
+
+    private void initView() {
+        mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
+        initGamingMenu();
+        initFloatingButton();
+        initDanmaku();
+    }
+
+    private void updateConfig() {
+        ScreenUtil.init(this);
+
+        if (mQSView != null) {
+            mQSView.setConfig(configBundle);
+        }
+
+        if (configBundle.containsKey(Constants.ConfigKeys.QUICK_START_APPS)) {
+            mQSAppView.setConfig(configBundle);
+        }
+
+        // 悬浮球位置调整
+        if (mGamingFloatingButton != null && mGamingFBLayoutParams != null) {
+            int defaultX = ((int) getResources().getDimension(R.dimen.game_button_size) - ScreenUtil.getScreenWidth()) / 2;
+            if (ScreenUtil.isPortrait()) {
+                mGamingFBLayoutParams.x = mPreferences.getInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_VERTICAL_X, defaultX);
+                mGamingFBLayoutParams.y = mPreferences.getInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_VERTICAL_Y, 10);
+            } else {
+                mGamingFBLayoutParams.x = mPreferences.getInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_HORIZONTAL_X, defaultX);
+                mGamingFBLayoutParams.y = mPreferences.getInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_HORIZONTAL_Y, 10);
+            }
+            if (mWindowManager != null) {
+                mWindowManager.updateViewLayout(mGamingFloatingButton, mGamingFBLayoutParams);
+            }
+            mQSAppView.setFloatingButtonCoordinate(mGamingFBLayoutParams.x, mGamingFBLayoutParams.y);
+        }
+
+        // 弹幕设置
+        mDanmakuManager.setMaxDanmakuSize(20); // 设置同屏最大弹幕数
+        DanmakuManager.Config config = mDanmakuManager.getConfig(); // 弹幕相关设置
+        config.setScrollSpeed(ScreenUtil.isPortrait() ?
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SPEED_VERTICAL, Constants.ConfigDefaultValues.DANMAKU_SPEED_VERTICAL) :
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SPEED_HORIZONTAL, Constants.ConfigDefaultValues.DANMAKU_SPEED_HORIZONTAL));
+        config.setLineHeight(ScreenUtil.autoSize(ScreenUtil.isPortrait() ?
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SIZE_VERTICAL,Constants.ConfigDefaultValues.DANMAKU_SIZE_VERTICAL) + 4 :
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SIZE_HORIZONTAL,Constants.ConfigDefaultValues.DANMAKU_SIZE_HORIZONTAL) + 4)); // 设置行高
+        config.setMaxScrollLine(ScreenUtil.getScreenHeight() / 2 / config.getLineHeight());
+
+        // 性能配置
+        if (performanceController != null) {
+            performanceController.setLevel(configBundle.getInt(Constants.ConfigKeys.PERFORMANCE_LEVEL, Constants.ConfigDefaultValues.PERFORMANCE_LEVEL));
+        }
+    }
+
+    private WindowManager.LayoutParams getBaseLayoutParams() {
+        return new WindowManager.LayoutParams(
+                WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,
+                WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
+                        |WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
+                        | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL,
+                PixelFormat.TRANSLUCENT);
+    }
+
+    private void initGamingMenu() {
+        if (mGamingOverlayView == null && mWindowManager != null) {
+            mGamingOverlayView = (LinearLayout) LayoutInflater.from(this).inflate(R.layout.gaming_overlay_layout, null);
+            mGamingOverlayView.setVisibility(View.GONE);
+            WindowManager.LayoutParams mGamingViewLayoutParams = getBaseLayoutParams();
+            mWindowManager.addView(mGamingOverlayView, mGamingViewLayoutParams);
+
+            mGamingMenu = mGamingOverlayView.findViewById(R.id.gaming_menu);
+            mQSView = mGamingOverlayView.findViewById(R.id.gaming_qs);
+            mQSAppView = mGamingOverlayView.findViewById(R.id.gaming_qsapp);
+
+            performanceController = mGamingOverlayView.findViewById(R.id.performance_controller);
+            mGamingOverlayView.setOnClickListener(v -> showHideGamingMenu(0));
+        }
+    }
+
+    private void initFloatingButton() {
+        if (mGamingFloatingButton == null && mWindowManager != null) {
+            mGamingFloatingButton = LayoutInflater.from(this).inflate(R.layout.gaming_button_layout, null);
+
+            mGamingFBLayoutParams = getBaseLayoutParams();
+            mGamingFBLayoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
+            mGamingFBLayoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
+
+            mGamingFloatingButton.setOnClickListener(v -> showHideGamingMenu(0));
+            mGamingFloatingButton.setOnTouchListener(new View.OnTouchListener() {
+                private int origX;
+                private int origY;
+                private int touchX;
+                private int touchY;
+                private boolean isMoved;
+
+                @Override
+                public boolean onTouch(View v, MotionEvent event) {
+                    int x = (int) event.getRawX();
+                    int y = (int) event.getRawY();
+
+                    switch (event.getAction()) {
+                        case MotionEvent.ACTION_DOWN:
+                            origX = mGamingFBLayoutParams.x;
+                            origY = mGamingFBLayoutParams.y;
+                            touchX = x;
+                            touchY = y;
+                            break;
+                        case MotionEvent.ACTION_MOVE:
+                            isMoved = true;
+                            mGamingFBLayoutParams.x = origX + x - touchX;
+                            mGamingFBLayoutParams.y = origY + y - touchY;
+                            if (mWindowManager != null) {
+                                mWindowManager.updateViewLayout(mGamingFloatingButton, mGamingFBLayoutParams);
+                            }
+                            break;
+                        case MotionEvent.ACTION_UP:
+                            if (!isMoved) {
+                                v.performClick();
+                            } else {
+                                if (ScreenUtil.isPortrait()) {
+                                    mPreferences.edit()
+                                            .putInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_VERTICAL_X, mGamingFBLayoutParams.x)
+                                            .putInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_VERTICAL_Y, mGamingFBLayoutParams.y)
+                                            .apply();
+                                } else {
+                                    mPreferences.edit()
+                                            .putInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_HORIZONTAL_X, mGamingFBLayoutParams.x)
+                                            .putInt(Constants.LocalConfigKeys.FLOATING_BUTTON_COORDINATE_HORIZONTAL_Y, mGamingFBLayoutParams.y)
+                                            .apply();
+                                }
+                                if (mQSAppView != null) {
+                                    mQSAppView.setFloatingButtonCoordinate(mGamingFBLayoutParams.x, mGamingFBLayoutParams.y);
+                                }
+                            }
+                            isMoved = false;
+                            break;
+                        default:
+                            return false;
+                    }
+                    return true;
+                }
+            });
+
+            mWindowManager.addView(mGamingFloatingButton, mGamingFBLayoutParams);
+        }
+    }
+
+    /*
+     * mode: 0=auto, 1=show, 2=hide
+     */
+    private void showHideGamingMenu(int mode) {
+        if (mGamingOverlayView.getVisibility() == View.VISIBLE && mode != 1) {
+            // hide
+            mGamingOverlayView.setVisibility(View.GONE);
+            mGamingFloatingButton.setVisibility(View.VISIBLE);
+        } else if (mode != 2) {
+            // show
+            int gravity = 0;
+            if (mGamingFBLayoutParams.x > 0) {
+                gravity |= Gravity.RIGHT;
+            } else {
+                gravity |= Gravity.LEFT;
+            }
+            if (mGamingFBLayoutParams.y > 0) {
+                gravity |= Gravity.BOTTOM;
+            } else {
+                gravity |= Gravity.TOP;
+            }
+
+            mGamingFloatingButton.setVisibility(View.GONE);
+            mGamingOverlayView.setGravity(gravity);
+            ViewGroup.LayoutParams gamingMenuLayoutParams =  mGamingMenu.getLayoutParams();
+            if (ScreenUtil.isPortrait()) {
+                gamingMenuLayoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
+            } else {
+                gamingMenuLayoutParams.width = ScreenUtil.getScreenWidth() / 2;
+            }
+            mGamingMenu.setLayoutParams(gamingMenuLayoutParams);
+
+            mGamingOverlayView.setVisibility(View.VISIBLE);
+        }
+    }
+
+    private void initDanmaku() {
+        if (mWindowManager != null && mDanmakuContainer == null) {
+            mDanmakuContainer = new FrameLayout(this);
+            WindowManager.LayoutParams danmakuParams = getBaseLayoutParams();
+            danmakuParams.flags |= WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
+            mWindowManager.addView(mDanmakuContainer, danmakuParams);
+
+            mDanmakuManager = DanmakuManager.getInstance();
+            mDanmakuManager.init(this, mDanmakuContainer);
+        }
+    }
+
+    private void sendDanmaku(String danmakuText) {
+        if (mDanmakuManager == null) return;
+        Danmaku danmaku = new Danmaku();
+        danmaku.text = danmakuText;
+        danmaku.mode = Danmaku.Mode.scroll;
+        danmaku.size = ScreenUtil.autoSize(
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SIZE_HORIZONTAL,Constants.ConfigDefaultValues.DANMAKU_SIZE_HORIZONTAL),
+                configBundle.getInt(Constants.ConfigKeys.DANMAKU_SIZE_VERTICAL,Constants.ConfigDefaultValues.DANMAKU_SIZE_VERTICAL));
+        mDanmakuManager.send(danmaku);
+    }
+
+    @Override
+    public void onDestroy() {
+        LocalBroadcastManager.getInstance(this).unregisterReceiver(mOMReceiver);
+        unregisterReceiver(mSysConfigChangedReceiver);
+        if (mWindowManager != null) {
+            if (mGamingFloatingButton != null) mWindowManager.removeViewImmediate(mGamingFloatingButton);
+            if (mDanmakuContainer != null) mWindowManager.removeViewImmediate(mDanmakuContainer);
+            if (mGamingOverlayView != null) mWindowManager.removeViewImmediate(mGamingOverlayView);
+        }
+        super.onDestroy();
+    }
+
+    private class OMReceiver extends BroadcastReceiver {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            String action = intent.getAction();
+            if (action == null) return;
+            switch (action) {
+                case Constants.Broadcasts.BROADCAST_NEW_DANMAKU:
+                    String danmaku = intent.getStringExtra("danmaku_text");
+                    if (TextUtils.isEmpty(danmaku)) return;
+                    sendDanmaku(danmaku);
+                    break;
+                case Constants.Broadcasts.BROADCAST_CONFIG_CHANGED:
+                    if (intent.getExtras() == null) return;
+                    configBundle.putAll(intent.getExtras());
+                    updateConfig();
+                    break;
+                case Constants.Broadcasts.BROADCAST_GAMING_MENU_CONTROL:
+                    if ("hide".equals(intent.getStringExtra("cmd"))) {
+                        showHideGamingMenu(2);
+                    } else if ("show".equals(intent.getStringExtra("cmd"))) {
+                        showHideGamingMenu(1);
+                    }
+                    break;
+            }
+        }
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/ui/GamingPerformanceView.java b/app/src/main/java/org/exthmui/game/ui/GamingPerformanceView.java
new file mode 100644
index 0000000..93b6422
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/ui/GamingPerformanceView.java
@@ -0,0 +1,56 @@
+package org.exthmui.game.ui;
+
+import android.content.Context;
+import android.content.Intent;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.widget.LinearLayout;
+import android.widget.RadioGroup;
+import android.widget.SeekBar;
+
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+
+public class GamingPerformanceView extends LinearLayout implements SeekBar.OnSeekBarChangeListener {
+
+    private Context mContext;
+
+    private Intent mPerformanceIntent = new Intent(Constants.Broadcasts.BROADCAST_GAMING_ACTION)
+            .putExtra("target", Constants.GamingActionTargets.PERFORMANCE_LEVEL);
+
+    private SeekBar mSeekBar;
+
+    public GamingPerformanceView(Context context) {
+        this(context, null);
+    }
+
+    public GamingPerformanceView(Context context, AttributeSet attrs) {
+        super(context, attrs);
+        mContext = context;
+        LayoutInflater.from(context).inflate(R.layout.gaming_perofrmance_layout, this, true);
+        mSeekBar = findViewById(R.id.performance_seek);
+        mSeekBar.setOnSeekBarChangeListener(this);
+    }
+
+    public void setLevel(int level) {
+        mSeekBar.setProgress(level);
+    }
+
+    @Override
+    public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
+
+    }
+
+    @Override
+    public void onStartTrackingTouch(SeekBar seekBar) {
+
+    }
+
+    @Override
+    public void onStopTrackingTouch(SeekBar seekBar) {
+        mPerformanceIntent.putExtra("value", mSeekBar.getProgress());
+        LocalBroadcastManager.getInstance(mContext).sendBroadcast(mPerformanceIntent);
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/ui/MusicControllerView.java b/app/src/main/java/org/exthmui/game/ui/MusicControllerView.java
new file mode 100644
index 0000000..16fcc49
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/ui/MusicControllerView.java
@@ -0,0 +1,189 @@
+package org.exthmui.game.ui;
+
+import android.content.Context;
+import android.graphics.Bitmap;
+import android.graphics.BitmapFactory;
+import android.media.MediaMetadata;
+import android.media.session.MediaController;
+import android.media.session.MediaSessionManager;
+import android.media.session.PlaybackState;
+import android.net.Uri;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.ImageView;
+import android.widget.LinearLayout;
+import android.widget.TextView;
+
+import androidx.annotation.Nullable;
+
+import org.exthmui.game.R;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.util.List;
+
+public class MusicControllerView extends LinearLayout implements View.OnClickListener {
+    private MediaSessionManager mMediaSessionManager;
+    private MediaController mMediaController;
+
+    private TextView mediaTitle;
+    private TextView mediaArtist;
+    private ImageView mediaAlbumImg;
+    private ImageView prevButton;
+    private ImageView nextButton;
+    private ImageView playPauseButton;
+
+    private Context mContext;
+
+    private MediaController.Callback mMediaCallback = new MediaController.Callback() {
+        @Override
+        public void onPlaybackStateChanged(@Nullable PlaybackState state) {
+            super.onPlaybackStateChanged(state);
+            if (state != null) updatePlayPauseStatus(state.getState());
+        }
+
+        @Override
+        public void onSessionDestroyed() {
+            MusicControllerView.this.setVisibility(GONE);
+            super.onSessionDestroyed();
+        }
+
+        @Override
+        public void onMetadataChanged(@Nullable MediaMetadata metadata) {
+            updateMetaData(metadata);
+        }
+    };
+
+    private MediaSessionManager.OnActiveSessionsChangedListener onActiveSessionsChangedListener = new MediaSessionManager.OnActiveSessionsChangedListener() {
+        @Override
+        public void onActiveSessionsChanged(@Nullable List<MediaController> controllers) {
+            if (mMediaController != null) mMediaController.unregisterCallback(mMediaCallback);
+            MusicControllerView.this.setVisibility(GONE);
+            if (controllers == null) return;
+            for (MediaController controller : controllers) {
+                mMediaController = controller;
+                if (getMediaControllerPlaybackState(controller) == PlaybackState.STATE_PLAYING) {
+                    break;
+                }
+            }
+            if (mMediaController != null) {
+                mMediaController.registerCallback(mMediaCallback);
+                mMediaCallback.onMetadataChanged(mMediaController.getMetadata());
+                mMediaCallback.onPlaybackStateChanged(mMediaController.getPlaybackState());
+                MusicControllerView.this.setVisibility(VISIBLE);
+            }
+        }
+    };
+
+    public MusicControllerView(Context context) {
+        this(context, null);
+    }
+
+    public MusicControllerView(Context context, @Nullable AttributeSet attrs) {
+        this(context, attrs, 0);
+    }
+
+    public MusicControllerView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
+        this(context, attrs, defStyleAttr, 0);
+    }
+
+    public MusicControllerView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+
+        mMediaSessionManager = (MediaSessionManager) context.getSystemService(Context.MEDIA_SESSION_SERVICE);
+        LayoutInflater.from(context).inflate(R.layout.music_control_layout, this, true);
+
+        mContext = context;
+
+        mediaTitle = findViewById(R.id.music_title);
+        mediaArtist = findViewById(R.id.music_artist);
+        mediaAlbumImg = findViewById(R.id.music_cover);
+
+        prevButton = findViewById(R.id.prev_button);
+        nextButton = findViewById(R.id.next_button);
+        playPauseButton = findViewById(R.id.play_pause_button);
+
+        prevButton.setOnClickListener(this);
+        nextButton.setOnClickListener(this);
+        playPauseButton.setOnClickListener(this);
+
+        mMediaSessionManager.addOnActiveSessionsChangedListener(onActiveSessionsChangedListener, null);
+        onActiveSessionsChangedListener.onActiveSessionsChanged(mMediaSessionManager.getActiveSessions(null));
+    }
+
+    private int getMediaControllerPlaybackState(MediaController controller) {
+        if (controller != null) {
+            final PlaybackState playbackState = controller.getPlaybackState();
+            if (playbackState != null) {
+                return playbackState.getState();
+            }
+        }
+        return PlaybackState.STATE_NONE;
+    }
+
+    private void updatePlayPauseStatus(int state) {
+        if (state == PlaybackState.STATE_PLAYING) {
+            playPauseButton.setImageResource(R.drawable.ic_music_pause);
+        } else {
+            playPauseButton.setImageResource(R.drawable.ic_music_play);
+        }
+    }
+
+    private void updateMetaData(MediaMetadata mediaMetadata) {
+        if (mediaMetadata == null) return;
+        mediaTitle.setText(mediaMetadata.getText(MediaMetadata.METADATA_KEY_TITLE));
+        mediaArtist.setText(mediaMetadata.getString(MediaMetadata.METADATA_KEY_ARTIST));
+        // Try to get album cover
+        if (!(setAlbumImgFromBitmap(mediaMetadata.getBitmap(MediaMetadata.METADATA_KEY_ART)) ||
+                setAlbumImgFromBitmap(mediaMetadata.getBitmap(MediaMetadata.METADATA_KEY_ALBUM_ART)) ||
+                setAlbumImgFromUri(mediaMetadata.getString(MediaMetadata.METADATA_KEY_ART_URI)) ||
+                setAlbumImgFromUri(mediaMetadata.getString(MediaMetadata.METADATA_KEY_ALBUM_ART_URI)))) {
+            mediaAlbumImg.setImageResource(R.drawable.default_album_cover);
+        }
+    }
+
+    private boolean setAlbumImgFromBitmap(Bitmap bitmap) {
+        if (bitmap == null) return false;
+        mediaAlbumImg.setImageBitmap(bitmap);
+        return true;
+    }
+
+    private boolean setAlbumImgFromUri(String uri) {
+        try {
+            Uri albumUri = Uri.parse(uri);
+            InputStream is = mContext.getContentResolver().openInputStream(albumUri);
+            if (is != null) {
+                mediaAlbumImg.setImageBitmap(BitmapFactory.decodeStream(is));
+                is.close();
+            }
+        } catch (NullPointerException | IOException e) {
+            e.printStackTrace();
+            return false;
+        }
+        return true;
+    }
+
+    @Override
+    public void onClick(View v) {
+        if (mMediaController == null) return;
+        if (v == prevButton) {
+            mMediaController.getTransportControls().skipToPrevious();
+        } else if (v == nextButton) {
+            mMediaController.getTransportControls().skipToNext();
+        } else if (v == playPauseButton) {
+            if (getMediaControllerPlaybackState(mMediaController) == PlaybackState.STATE_PLAYING) {
+                mMediaController.getTransportControls().pause();
+            } else {
+                mMediaController.getTransportControls().play();
+            }
+        }
+    }
+
+    @Override
+    public void onDetachedFromWindow() {
+        mMediaSessionManager.removeOnActiveSessionsChangedListener(onActiveSessionsChangedListener);
+        if (mMediaController != null) mMediaController.unregisterCallback(mMediaCallback);
+        super.onDetachedFromWindow();
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/ui/QuickSettingsView.java b/app/src/main/java/org/exthmui/game/ui/QuickSettingsView.java
new file mode 100644
index 0000000..989654a
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/ui/QuickSettingsView.java
@@ -0,0 +1,61 @@
+package org.exthmui.game.ui;
+
+import android.content.Context;
+import android.os.Bundle;
+import android.util.AttributeSet;
+import android.widget.LinearLayout;
+
+import androidx.annotation.Nullable;
+
+import org.exthmui.game.R;
+import org.exthmui.game.qs.AutoBrightnessTile;
+import org.exthmui.game.qs.DNDTile;
+import org.exthmui.game.qs.DanmakuTile;
+import org.exthmui.game.qs.LockGestureTile;
+import org.exthmui.game.qs.LockHwKeysTile;
+import org.exthmui.game.qs.ScreenCaptureTile;
+import org.exthmui.game.qs.TileBase;
+
+public class QuickSettingsView extends LinearLayout {
+
+    private TileBase[] qsTiles;
+
+    public QuickSettingsView(Context context) {
+        this(context, null);
+    }
+
+    public QuickSettingsView(Context context, @Nullable AttributeSet attrs) {
+        this(context, attrs, 0);
+    }
+
+    public QuickSettingsView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
+        this(context, attrs, defStyleAttr, 0);
+    }
+
+    public QuickSettingsView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+
+        this.setDividerDrawable(context.getDrawable(R.drawable.qs_divider));
+        this.setShowDividers(SHOW_DIVIDER_MIDDLE);
+        this.setPadding(0,0, 0,8);
+
+        qsTiles = new TileBase[]{
+                new ScreenCaptureTile(context),
+                new DanmakuTile(context),
+                new DNDTile(context),
+                new LockHwKeysTile(context),
+                new LockGestureTile(context),
+                new AutoBrightnessTile(context)
+        };
+
+        for (TileBase tileBase : qsTiles) {
+            addView(tileBase);
+        }
+    }
+
+    public void setConfig(Bundle config) {
+        for (TileBase tileBase : qsTiles) {
+            tileBase.setConfig(config);
+        }
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/ui/QuickStartAppView.java b/app/src/main/java/org/exthmui/game/ui/QuickStartAppView.java
new file mode 100644
index 0000000..05d2e21
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/ui/QuickStartAppView.java
@@ -0,0 +1,106 @@
+package org.exthmui.game.ui;
+
+import android.app.ActivityOptions;
+import android.app.WindowConfiguration;
+import android.content.Context;
+import android.content.Intent;
+import android.content.pm.PackageManager;
+import android.content.res.Configuration;
+import android.graphics.Rect;
+import android.os.Bundle;
+import android.os.UserHandle;
+import android.util.AttributeSet;
+import android.util.DisplayMetrics;
+import android.view.View;
+import android.widget.LinearLayout;
+
+import androidx.annotation.Nullable;
+import androidx.localbroadcastmanager.content.LocalBroadcastManager;
+
+import org.exthmui.game.R;
+import org.exthmui.game.misc.Constants;
+import org.exthmui.game.qs.AppTile;
+
+public class QuickStartAppView extends LinearLayout implements View.OnClickListener {
+
+    private Context mContext;
+    private PackageManager mPackageManager;
+    private ActivityOptions mActivityOptions;
+    private Intent mHideMenuIntent = new Intent(Constants.Broadcasts.BROADCAST_GAMING_MENU_CONTROL).putExtra("cmd", "hide");
+    int left, right, top, bottom;
+
+    public QuickStartAppView(Context context) {
+        this(context, null);
+    }
+
+    public QuickStartAppView(Context context, @Nullable AttributeSet attrs) {
+        this(context, attrs, 0);
+    }
+
+    public QuickStartAppView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
+        this(context, attrs, defStyleAttr, 0);
+    }
+
+    public QuickStartAppView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+
+        mContext = context;
+        mPackageManager = mContext.getPackageManager();
+
+        this.setDividerDrawable(context.getDrawable(R.drawable.qs_divider));
+        this.setShowDividers(SHOW_DIVIDER_MIDDLE);
+        this.setPadding(0,0, 0,8);
+        this.setVisibility(GONE);
+
+        mActivityOptions = ActivityOptions.makeBasic();
+        mActivityOptions.setLaunchWindowingMode(WindowConfiguration.WINDOWING_MODE_FREEFORM);
+
+    }
+
+    public void setConfig(Bundle config) {
+        this.removeAllViewsInLayout();
+        this.setVisibility(GONE);
+        String[] apps = config.getStringArray(Constants.ConfigKeys.QUICK_START_APPS);
+        
+        if (apps == null || apps.length == 0) return;
+        this.setVisibility(VISIBLE);
+        for (String app : apps) {
+            AppTile appTile = new AppTile(mContext, app);
+            appTile.setOnClickListener(this);
+            this.addView(appTile);
+        }
+    }
+
+    public void setFloatingButtonCoordinate(int x, int y) {
+        DisplayMetrics m = mContext.getResources().getDisplayMetrics();
+        int width, height;
+        if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT) {
+            // 竖屏
+            width = m.widthPixels - 200;
+            height = width * 4 / 3;
+        } else {
+            // 横屏
+            height = m.heightPixels - 200;
+            width = height * 4 / 3;
+        }
+        left = 100;
+        right = left + width;
+        top = 150;
+        bottom = top + height;
+    }
+
+    @Override
+    public void onClick(View v) {
+        if (v instanceof AppTile) {
+            AppTile appTile = (AppTile) v;
+            LocalBroadcastManager.getInstance(mContext).sendBroadcastSync(mHideMenuIntent);
+            Intent startAppIntent = mPackageManager.getLaunchIntentForPackage(appTile.getKey());
+            startAppIntent.addFlags(Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT | Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
+            startAppIntent.setPackage(null);
+            Rect appWindowRect = new Rect(left, top, right, bottom);
+            mActivityOptions.setLaunchBounds(appWindowRect);
+            mContext.startActivityAsUser(startAppIntent, mActivityOptions.toBundle(), UserHandle.CURRENT);
+
+        }
+    }
+}
diff --git a/app/src/main/java/org/exthmui/game/ui/TimeAndBatteryView.java b/app/src/main/java/org/exthmui/game/ui/TimeAndBatteryView.java
new file mode 100644
index 0000000..33a2ba0
--- /dev/null
+++ b/app/src/main/java/org/exthmui/game/ui/TimeAndBatteryView.java
@@ -0,0 +1,95 @@
+package org.exthmui.game.ui;
+
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.content.IntentFilter;
+import android.os.BatteryManager;
+import android.text.format.DateFormat;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.widget.LinearLayout;
+import android.widget.TextView;
+
+import androidx.annotation.Nullable;
+
+import org.exthmui.game.R;
+
+import static android.content.Context.BATTERY_SERVICE;
+
+public class TimeAndBatteryView extends LinearLayout {
+
+    private Context mContext;
+
+    private TextView currentTime;
+    private TextView currentDate;
+    private TextView currentBattery;
+
+    private TimeChangeReceiver timeChangeReceiver = new TimeChangeReceiver();
+    private BatteryChangeReceiver batteryChangeReceiver = new BatteryChangeReceiver();
+
+    public TimeAndBatteryView(Context context) {
+        this(context, null);
+    }
+
+    public TimeAndBatteryView(Context context, @Nullable AttributeSet attrs) {
+        this(context, attrs, 0);
+    }
+
+    public TimeAndBatteryView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
+        this(context, attrs, defStyleAttr, 0);
+    }
+
+    public TimeAndBatteryView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+        mContext = context;
+
+        LayoutInflater.from(context).inflate(R.layout.time_battery_layout, this, true);
+
+        currentBattery = findViewById(R.id.current_battery);
+        currentDate = findViewById(R.id.current_date);
+        currentTime = findViewById(R.id.current_time);
+
+        mContext.registerReceiver(timeChangeReceiver, new IntentFilter(Intent.ACTION_TIME_TICK));
+        mContext.registerReceiver(batteryChangeReceiver, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));
+        updateTime();
+
+        BatteryManager batteryManager = (BatteryManager) mContext.getSystemService(BATTERY_SERVICE);
+        currentBattery.setText(mContext.getString(R.string.battery_format, batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY)));
+
+    }
+
+    private void updateTime() {
+        long sysTime = System.currentTimeMillis();
+        currentDate.setText(DateFormat.format(mContext.getString(R.string.date_format), sysTime));
+        currentTime.setText(DateFormat.format(mContext.getString(R.string.time_format), sysTime));
+    }
+
+    @Override
+    public void onDetachedFromWindow() {
+        mContext.unregisterReceiver(timeChangeReceiver);
+        mContext.unregisterReceiver(batteryChangeReceiver);
+        super.onDetachedFromWindow();
+    }
+
+    private class TimeChangeReceiver extends BroadcastReceiver {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            if (Intent.ACTION_TIME_TICK.equals(intent.getAction())) {
+                updateTime();
+            }
+        }
+    }
+
+    private class BatteryChangeReceiver extends BroadcastReceiver {
+        @Override
+        public void onReceive(Context context, Intent intent) {
+            if (Intent.ACTION_BATTERY_CHANGED.equals(intent.getAction())) {
+                int level=intent.getIntExtra(BatteryManager.EXTRA_LEVEL,0);
+                int scale=intent.getIntExtra(BatteryManager.EXTRA_SCALE,0);
+                int percent = (int)(((float)level / scale) * 100);
+                currentBattery.setText(mContext.getString(R.string.battery_format, percent));
+            }
+        }
+    }
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/CachedDanmakuViewPool.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/CachedDanmakuViewPool.java
new file mode 100644
index 0000000..dff116f
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/CachedDanmakuViewPool.java
@@ -0,0 +1,134 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.util.Log;
+
+import java.util.LinkedList;
+import java.util.concurrent.Executors;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.concurrent.TimeUnit;
+
+import top.littlefogcat.danmakulib.utils.EasyL;
+
+/**
+ * 一个简化版的DanmakuViewPool
+ */
+public class CachedDanmakuViewPool implements Pool<DanmakuView> {
+    private static final String TAG = "CachedDanmakuViewPool";
+
+    /**
+     * 缓存DanmakuView队列。显示已经完毕的DanmakuView会被添加到缓存中进行复用。
+     * 在一定的时间{@link CachedDanmakuViewPool#mKeepAliveTime}过后,没有被访问到的DanmakuView会被回收。
+     */
+    private LinkedList<DanmakuViewWithExpireTime> mCache = new LinkedList<>();
+
+    /**
+     * 缓存存活时间
+     */
+    private long mKeepAliveTime;
+    /**
+     * 定时清理缓存
+     */
+    private ScheduledExecutorService mChecker = Executors.newSingleThreadScheduledExecutor();
+    /**
+     * 创建新DanmakuView的Creator
+     */
+    private ViewCreator<DanmakuView> mCreator;
+    /**
+     * 最大DanmakuView数量。
+     * 这个数量包含了正在显示的DanmakuView和已经显示完毕进入缓存等待复用的DanmakuView之和。
+     */
+    private int mMaxSize;
+    /**
+     * 正在显示的弹幕数量。
+     */
+    private int mInUseSize;
+
+    /**
+     * @param creator 生成一个DanmakuView
+     */
+    CachedDanmakuViewPool(long keepAliveTime, int maxSize, ViewCreator<DanmakuView> creator) {
+        mKeepAliveTime = keepAliveTime;
+        mMaxSize = maxSize;
+        mCreator = creator;
+        mInUseSize = 0;
+
+        scheduleCheckUnusedViews();
+    }
+
+    /**
+     * 每隔一秒检查并清理掉空闲队列中超过一定时间没有被使用的DanmakuView
+     */
+    private void scheduleCheckUnusedViews() {
+        mChecker.scheduleWithFixedDelay(() -> {
+            EasyL.v(TAG, "scheduleCheckUnusedViews: mInUseSize=" + mInUseSize + ", mCacheSize=" + mCache.size());
+            long current = System.currentTimeMillis();
+            while (!mCache.isEmpty()) {
+                DanmakuViewWithExpireTime first = mCache.getFirst();
+                if (current > first.expireTime) {
+                    mCache.remove(first);
+                } else {
+                    break;
+                }
+            }
+        }, 1000, 1000, TimeUnit.MILLISECONDS);
+    }
+
+    @Override
+    public DanmakuView get() {
+        DanmakuView view;
+
+        if (mCache.isEmpty()) { // 缓存中没有View
+            if (mInUseSize >= mMaxSize) {
+                return null;
+            }
+            view = mCreator.create();
+        } else { // 有可用的缓存,从缓存中取
+            view = mCache.poll().danmakuView;
+        }
+
+        view.addOnEnterListener(DanmakuView::clearScroll);
+        view.addOnExitListener(v -> {
+            long expire = System.currentTimeMillis() + mKeepAliveTime;
+            v.restore();
+            DanmakuViewWithExpireTime item = new DanmakuViewWithExpireTime();
+            item.danmakuView = v;
+            item.expireTime = expire;
+            mCache.offer(item);
+            mInUseSize--;
+        });
+        mInUseSize++;
+
+        return view;
+    }
+
+    @Override
+    public void release() {
+        mCache.clear();
+    }
+
+    /**
+     * @return 使用中的DanmakuView和缓存中的DanmakuView数量之和
+     */
+    @Override
+    public int count() {
+        return mCache.size() + mInUseSize;
+    }
+
+    @Override
+    public void setMaxSize(int max) {
+        mMaxSize = max;
+    }
+
+    /**
+     * 一个包裹类,保存一个DanmakuView和它的过期时间。
+     */
+    private class DanmakuViewWithExpireTime {
+        private DanmakuView danmakuView; // 缓存的DanmakuView
+        private long expireTime; // 超过这个时间没有被访问的缓存将被丢弃
+    }
+
+    public interface ViewCreator<T> {
+        T create();
+    }
+
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Danmaku.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Danmaku.java
new file mode 100644
index 0000000..aaca739
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Danmaku.java
@@ -0,0 +1,40 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.graphics.Color;
+
+/**
+ * Created by LittleFogCat.
+ */
+
+public class Danmaku {
+    public static final int DEFAULT_TEXT_SIZE = 24;
+
+    public String text;// 文字
+    public int size = DEFAULT_TEXT_SIZE;// 字号
+    public Mode mode = Mode.scroll;// 模式:滚动、顶部、底部
+    public int color = Color.WHITE;// 默认白色
+
+    public enum Mode {
+        scroll, top, bottom
+    }
+
+    public Danmaku() {
+    }
+
+    public Danmaku(String text, int textSize, Mode mode, int color) {
+        this.text = text;
+        this.size = textSize;
+        this.mode = mode;
+        this.color = color;
+    }
+
+    @Override
+    public String toString() {
+        return "Danmaku{" +
+                "text='" + text + '\'' +
+                ", textSize=" + size +
+                ", mode=" + mode +
+                ", color='" + color + '\'' +
+                '}';
+    }
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuManager.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuManager.java
new file mode 100644
index 0000000..b5c67af
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuManager.java
@@ -0,0 +1,278 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.content.Context;
+import android.graphics.Color;
+import android.util.TypedValue;
+import android.view.ViewGroup;
+import android.widget.FrameLayout;
+
+import org.exthmui.game.misc.Constants;
+
+import java.lang.ref.WeakReference;
+
+import top.littlefogcat.danmakulib.utils.EasyL;
+import top.littlefogcat.danmakulib.utils.ScreenUtil;
+
+/**
+ * 用法示例:
+ * DanmakuManager dm = DanmakuManager.getInstance();
+ * dm.init(getContext());
+ * dm.show(new Danmaku("test"));
+ * <p>
+ * Created by LittleFogCat.
+ */
+@SuppressWarnings("unused")
+public class DanmakuManager {
+    private static final String TAG = DanmakuManager.class.getSimpleName();
+    private static final int RESULT_OK = 0;
+    private static final int RESULT_NULL_ROOT_VIEW = 1;
+    private static final int RESULT_FULL_POOL = 2;
+    private static final int TOO_MANY_DANMAKU = 2;
+
+    private static DanmakuManager sInstance;
+
+    /**
+     * 弹幕容器
+     */
+    WeakReference<FrameLayout> mDanmakuContainer;
+    /**
+     * 弹幕池
+     */
+    private Pool<DanmakuView> mDanmakuViewPool;
+
+    private Config mConfig;
+
+    private DanmakuPositionCalculator mPositionCal;
+
+    private DanmakuManager() {
+    }
+
+    public static DanmakuManager getInstance() {
+        if (sInstance == null) {
+            sInstance = new DanmakuManager();
+        }
+        return sInstance;
+    }
+
+    /**
+     * 初始化。在使用之前必须调用该方法。
+     */
+    public void init(Context context, FrameLayout container) {
+        if (mDanmakuViewPool == null) {
+            mDanmakuViewPool = new CachedDanmakuViewPool(
+                    60000, // 缓存存活时间:60秒
+                    100, // 最大弹幕数:100
+                    () -> DanmakuViewFactory.createDanmakuView(context, container));
+        }
+        setDanmakuContainer(container);
+        ScreenUtil.init(context);
+
+        mConfig = new Config();
+        mPositionCal = new DanmakuPositionCalculator(this);
+    }
+
+    public Config getConfig() {
+        if (mConfig == null) {
+            mConfig = new Config();
+        }
+        return mConfig;
+    }
+
+    private DanmakuPositionCalculator getPositionCalculator() {
+        if (mPositionCal == null) {
+            mPositionCal = new DanmakuPositionCalculator(this);
+        }
+        return mPositionCal;
+    }
+
+    public void setDanmakuViewPool(Pool<DanmakuView> pool) {
+        if (mDanmakuViewPool != null) {
+            mDanmakuViewPool.release();
+        }
+        mDanmakuViewPool = pool;
+    }
+
+    /**
+     * 设置允许同时出现最多的弹幕数,如果屏幕上显示的弹幕数超过该数量,那么新出现的弹幕将被丢弃,
+     * 直到有旧的弹幕消失。
+     *
+     * @param max 同时出现的最多弹幕数,-1无限制
+     */
+    public void setMaxDanmakuSize(int max) {
+        if (mDanmakuViewPool == null) {
+            return;
+        }
+        mDanmakuViewPool.setMaxSize(max);
+    }
+
+    /**
+     * 设置弹幕的容器,所有的弹幕都在这里面显示
+     */
+    public void setDanmakuContainer(final FrameLayout root) {
+        if (root == null) {
+            throw new NullPointerException("Danmaku container cannot be null!");
+        }
+        mDanmakuContainer = new WeakReference<>(root);
+    }
+
+    /**
+     * 发送一条弹幕
+     */
+    public int send(Danmaku danmaku) {
+        if (mDanmakuViewPool == null) {
+            throw new NullPointerException("Danmaku view pool is null. Did you call init() first?");
+        }
+
+        DanmakuView view = mDanmakuViewPool.get();
+
+        if (view == null) {
+            EasyL.w(TAG, "show: Too many danmaku, discard");
+            return RESULT_FULL_POOL;
+        }
+        if (mDanmakuContainer == null || mDanmakuContainer.get() == null) {
+            EasyL.w(TAG, "show: Root view is null.");
+            return RESULT_NULL_ROOT_VIEW;
+        }
+
+        view.setDanmaku(danmaku);
+
+        // 字体大小
+        int textSize = danmaku.size;
+        view.setTextSize(TypedValue.COMPLEX_UNIT_PX, textSize);
+
+        // 字体颜色
+        try {
+            view.setTextColor(danmaku.color);
+        } catch (Exception e) {
+            e.printStackTrace();
+            view.setTextColor(Color.WHITE);
+        }
+
+        // 计算弹幕距离顶部的位置
+        DanmakuPositionCalculator dpc = getPositionCalculator();
+        int marginTop = dpc.getMarginTop(view);
+
+        if (marginTop == -1) {
+            // 屏幕放不下了
+            EasyL.d(TAG, "send: screen is full, too many danmaku [" + danmaku + "]");
+            return TOO_MANY_DANMAKU;
+        }
+        FrameLayout.LayoutParams p = (FrameLayout.LayoutParams) view.getLayoutParams();
+        if (p == null) {
+            p = new FrameLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
+        }
+        p.topMargin = marginTop;
+        view.setLayoutParams(p);
+        view.setMinHeight((int) (getConfig().getLineHeight() * 1.35));
+        int duration = getDisplayDuration(danmaku);
+        if (danmaku.mode == Danmaku.Mode.scroll) {
+            duration = (view.getTextLength() + dpc.getParentWidth()) / duration * 1000;
+        }
+        view.show(mDanmakuContainer.get(), duration);
+        return RESULT_OK;
+    }
+
+    /**
+     * @return 返回这个弹幕显示时长 (对于滚动弹幕返回速度)
+     */
+    int getDisplayDuration(Danmaku danmaku) {
+        Config config = getConfig();
+        int duration;
+        switch (danmaku.mode) {
+            case top:
+                duration = config.getDurationTop();
+                break;
+            case bottom:
+                duration = config.getDurationBottom();
+                break;
+            case scroll:
+            default:
+                duration = config.getScrollSpeed();
+                break;
+        }
+        return duration;
+    }
+
+    /**
+     * 一些配置
+     */
+    public static class Config {
+
+        /**
+         * 行高,单位px
+         */
+        private int lineHeight;
+
+        /**
+         * 滚动弹幕速度 (px/s)
+         */
+        private int scrollSpeed;
+        /**
+         * 顶部弹幕显示时长
+         */
+        private int durationTop;
+        /**
+         * 底部弹幕的显示时长
+         */
+        private int durationBottom;
+
+        /**
+         * 滚动弹幕的最大行数
+         */
+        private int maxScrollLine;
+
+        public int getLineHeight() {
+            return lineHeight;
+        }
+
+        public void setLineHeight(int lineHeight) {
+            this.lineHeight = lineHeight;
+        }
+
+        public int getMaxScrollLine() {
+            return maxScrollLine;
+        }
+
+        public int getScrollSpeed() {
+            return scrollSpeed;
+        }
+
+        public void setScrollSpeed(int durationScroll) {
+            this.scrollSpeed = durationScroll;
+        }
+
+        public int getDurationTop() {
+            if (durationTop == 0) {
+                durationTop = 5000;
+            }
+            return durationTop;
+        }
+
+        public void setDurationTop(int durationTop) {
+            this.durationTop = durationTop;
+        }
+
+        public int getDurationBottom() {
+            if (durationBottom == 0) {
+                durationBottom = 5000;
+            }
+            return durationBottom;
+        }
+
+        public void setDurationBottom(int durationBottom) {
+            this.durationBottom = durationBottom;
+        }
+
+        public int getMaxDanmakuLine() {
+            if (maxScrollLine == 0) {
+                maxScrollLine = 12;
+            }
+            return maxScrollLine;
+        }
+
+        public void setMaxScrollLine(int maxScrollLine) {
+            this.maxScrollLine = maxScrollLine;
+        }
+    }
+
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuPositionCalculator.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuPositionCalculator.java
new file mode 100644
index 0000000..d5ef334
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuPositionCalculator.java
@@ -0,0 +1,168 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.view.ViewGroup;
+
+import java.util.ArrayList;
+import java.util.List;
+
+import top.littlefogcat.danmakulib.utils.EasyL;
+
+/**
+ * 用于计算弹幕位置,来保证弹幕不重叠又不浪费空间。
+ */
+class DanmakuPositionCalculator {
+    private static final String TAG = "DanPositionCalculator";
+    private DanmakuManager mDanmakuManager;
+    private List<DanmakuView> mLastDanmakus = new ArrayList<>();// 保存每一行最后一个弹幕消失的时间
+    private boolean[] mTops;
+    private boolean[] mBottoms;
+
+    DanmakuPositionCalculator(DanmakuManager danmakuManager) {
+        mDanmakuManager = danmakuManager;
+
+        int maxLine = danmakuManager.getConfig().getMaxDanmakuLine();
+        mTops = new boolean[maxLine];
+        mBottoms = new boolean[maxLine];
+    }
+
+    private int getLineHeightWithPadding() {
+        return (int) (1.35f * mDanmakuManager.getConfig().getLineHeight());
+    }
+
+    int getMarginTop(DanmakuView view) {
+        switch (view.getDanmaku().mode) {
+            case scroll:
+                return getScrollY(view);
+            case top:
+                return getTopY(view);
+            case bottom:
+                return getBottomY(view);
+        }
+        return -1;
+    }
+
+    private int getScrollY(DanmakuView view) {
+        if (mLastDanmakus.size() == 0) {
+            mLastDanmakus.add(view);
+            return 0;
+        }
+
+        int i;
+        for (i = 0; i < mLastDanmakus.size(); i++) {
+            DanmakuView last = mLastDanmakus.get(i);
+            int timeDisappear = calTimeDisappear(last); // 最后一条弹幕还需多久消失
+            int timeArrive = calTimeArrive(view); // 这条弹幕需要多久到达屏幕边缘
+            boolean isFullyShown = isFullyShown(last);
+            EasyL.d(TAG, "getScrollY: 行: " + i + ", 消失时间: " + timeDisappear + ", 到达时间: " + timeArrive + ", isFullyshown: " + isFullyShown);
+            if (timeDisappear <= timeArrive && isFullyShown) {
+                // 如果最后一个弹幕在这个弹幕到达之前消失,并且最后一个字已经显示完毕,
+                // 那么新的弹幕就可以在这一行显示
+                mLastDanmakus.set(i, view);
+                return i * getLineHeightWithPadding();
+            }
+        }
+        int maxLine = mDanmakuManager.getConfig().getMaxDanmakuLine();
+        if (maxLine == 0 || i < maxLine) {
+            mLastDanmakus.add(view);
+            return i * getLineHeightWithPadding();
+        }
+
+        return -1;
+    }
+
+    private int getTopY(DanmakuView view) {
+        for (int i = 0; i < mTops.length; i++) {
+            boolean isShowing = mTops[i];
+            if (!isShowing) {
+                final int finalI = i;
+                mTops[finalI] = true;
+                view.addOnExitListener(view1 -> mTops[finalI] = false);
+                return i * getLineHeightWithPadding();
+            }
+        }
+        return -1;
+    }
+
+    private int getBottomY(DanmakuView view) {
+        for (int i = 0; i < mBottoms.length; i++) {
+            boolean isShowing = mBottoms[i];
+            if (!isShowing) {
+                final int finalI = i;
+                mBottoms[finalI] = true;
+                view.addOnExitListener(view1 -> mBottoms[finalI] = false);
+                return getParentHeight() - (i + 1) * getLineHeightWithPadding();
+            }
+        }
+        return -1;
+    }
+
+    /**
+     * 这条弹幕是否已经全部出来了。如果没有的话,
+     * 后面的弹幕不能出来,否则就重叠了。
+     */
+    private boolean isFullyShown(DanmakuView view) {
+        if (view == null) {
+            return true;
+        }
+        int scrollX = view.getScrollX();
+        int textLength = view.getTextLength();
+        EasyL.d(TAG, "isFullyShown? scrollX=" + scrollX + ", textLength=" + textLength + ", parentWidth=" + getParentWidth());
+        return textLength - scrollX < getParentWidth();
+    }
+
+    /**
+     * 这条弹幕还有多少毫秒彻底消失。
+     */
+    private int calTimeDisappear(DanmakuView view) {
+        if (view == null) {
+            return 0;
+        }
+        float speed = calSpeed(view);
+        int scrollX = view.getScrollX();
+        int textLength = view.getTextLength();
+        int wayToGo = textLength - scrollX;
+
+        return (int) (wayToGo / speed);
+    }
+
+    /**
+     * 这条弹幕还要多少毫秒抵达屏幕边缘。
+     */
+    private int calTimeArrive(DanmakuView view) {
+        float speed = calSpeed(view);
+        int wayToGo = getParentWidth();
+        return (int) (wayToGo / speed);
+    }
+
+    /**
+     * 这条弹幕的速度。单位:px/ms
+     */
+    private float calSpeed(DanmakuView view) {
+        int textLength = view.getTextLength();
+        int width = getParentWidth();
+        float s = textLength + width + 0.0f;
+        int t = mDanmakuManager.getDisplayDuration(view.getDanmaku());
+        if (view.getDanmaku().mode == Danmaku.Mode.scroll) {
+            return t / 1000.0f;
+        } else {
+            return s / t;
+        }
+    }
+
+    public int getParentHeight() {
+        ViewGroup parent = mDanmakuManager.mDanmakuContainer.get();
+        if (parent == null || parent.getHeight() == 0) {
+            return 1080;
+        }
+        return parent.getHeight();
+    }
+
+    public int getParentWidth() {
+        ViewGroup parent = mDanmakuManager.mDanmakuContainer.get();
+        if (parent == null || parent.getWidth() == 0) {
+            return 1920;
+        }
+        return parent.getWidth();
+    }
+
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuView.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuView.java
new file mode 100644
index 0000000..58f467f
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuView.java
@@ -0,0 +1,236 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.content.Context;
+import android.util.AttributeSet;
+import android.view.Gravity;
+import android.view.ViewGroup;
+import android.view.animation.LinearInterpolator;
+import android.widget.Scroller;
+import android.widget.TextView;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.concurrent.CopyOnWriteArrayList;
+
+import top.littlefogcat.danmakulib.utils.ScreenUtil;
+
+/**
+ * DanmakuView的基类,继承自TextView,一个弹幕对应一个DanmakuView。
+ * 这里实现了一些通用的功能。
+ * <p>
+ * Created by LittleFogCat.
+ */
+@SuppressWarnings("unused")
+public class DanmakuView extends TextView {
+
+    /**
+     * 弹幕内容
+     */
+    private Danmaku mDanmaku;
+
+    /**
+     * 监听
+     */
+    private ListenerInfo mListenerInfo;
+
+    private class ListenerInfo {
+        private ArrayList<OnEnterListener> mOnEnterListeners;
+
+        private List<OnExitListener> mOnExitListener;
+    }
+
+    /**
+     * 弹幕进场时的监听
+     */
+    public interface OnEnterListener {
+        void onEnter(DanmakuView view);
+    }
+
+    /**
+     * 弹幕离场后的监听
+     */
+    public interface OnExitListener {
+        void onExit(DanmakuView view);
+    }
+
+    /**
+     * 显示时长 ms
+     */
+    private int mDuration;
+
+    public DanmakuView(Context context) {
+        super(context);
+    }
+
+    public DanmakuView(Context context, AttributeSet attrs) {
+        super(context, attrs);
+    }
+
+    public DanmakuView(Context context, AttributeSet attrs, int defStyleAttr) {
+        super(context, attrs, defStyleAttr);
+    }
+
+    /**
+     * 设置弹幕内容
+     */
+    public void setDanmaku(Danmaku danmaku) {
+        mDanmaku = danmaku;
+        setText(danmaku.text);
+        switch (danmaku.mode) {
+            case top:
+            case bottom:
+                setGravity(Gravity.CENTER);
+                break;
+            case scroll:
+            default:
+                setGravity(Gravity.START | Gravity.CENTER_VERTICAL);
+                break;
+        }
+    }
+
+    public Danmaku getDanmaku() {
+        return mDanmaku;
+    }
+
+    /**
+     * 显示弹幕
+     */
+    public void show(final ViewGroup parent, int duration) {
+        mDuration = duration;
+        switch (mDanmaku.mode) {
+            case top:
+            case bottom:
+                showFixedDanmaku(parent, duration);
+                break;
+            case scroll:
+            default:
+                showScrollDanmaku(parent, duration);
+                break;
+        }
+
+        if (hasOnEnterListener()) {
+            for (OnEnterListener listener : getListenerInfo().mOnEnterListeners) {
+                listener.onEnter(this);
+            }
+        }
+        postDelayed(() -> {
+            setVisibility(GONE);
+            if (hasOnExitListener()) {
+                for (OnExitListener listener : getListenerInfo().mOnExitListener) {
+                    listener.onExit(DanmakuView.this);
+                }
+            }
+            parent.removeView(DanmakuView.this);
+        }, duration);
+    }
+
+    private void showScrollDanmaku(ViewGroup parent, int duration) {
+        int screenWidth = ScreenUtil.getScreenWidth();
+        int textLength = getTextLength();
+        scrollTo(-screenWidth, 0);
+        parent.addView(this);
+        smoothScrollTo(textLength, 0, duration);
+    }
+
+    private void showFixedDanmaku(ViewGroup parent, int duration) {
+        setGravity(Gravity.CENTER);
+        parent.addView(this);
+    }
+
+    private ListenerInfo getListenerInfo() {
+        if (mListenerInfo == null) {
+            mListenerInfo = new ListenerInfo();
+        }
+        return mListenerInfo;
+    }
+
+    public void addOnEnterListener(OnEnterListener l) {
+        ListenerInfo li = getListenerInfo();
+        if (li.mOnEnterListeners == null) {
+            li.mOnEnterListeners = new ArrayList<>();
+        }
+        if (!li.mOnEnterListeners.contains(l)) {
+            li.mOnEnterListeners.add(l);
+        }
+    }
+
+    public void clearOnEnterListeners() {
+        ListenerInfo li = getListenerInfo();
+        if (li.mOnEnterListeners == null || li.mOnEnterListeners.size() == 0) {
+            return;
+        }
+        li.mOnEnterListeners.clear();
+    }
+
+    public void addOnExitListener(OnExitListener l) {
+        ListenerInfo li = getListenerInfo();
+        if (li.mOnExitListener == null) {
+            li.mOnExitListener = new CopyOnWriteArrayList<>();
+        }
+        if (!li.mOnExitListener.contains(l)) {
+            li.mOnExitListener.add(l);
+        }
+    }
+
+    public void clearOnExitListeners() {
+        ListenerInfo li = getListenerInfo();
+        if (li.mOnExitListener == null || li.mOnExitListener.size() == 0) {
+            return;
+        }
+        li.mOnExitListener.clear();
+    }
+
+    public boolean hasOnEnterListener() {
+        ListenerInfo li = getListenerInfo();
+        return li.mOnEnterListeners != null && li.mOnEnterListeners.size() != 0;
+    }
+
+    public boolean hasOnExitListener() {
+        ListenerInfo li = getListenerInfo();
+        return li.mOnExitListener != null && li.mOnExitListener.size() != 0;
+    }
+
+    public int getTextLength() {
+        return (int) getPaint().measureText(getText().toString());
+    }
+
+    public int getDuration() {
+        return mDuration;
+    }
+
+    /**
+     * 恢复初始状态
+     */
+    public void restore() {
+        clearOnEnterListeners();
+        clearOnExitListeners();
+        setVisibility(VISIBLE);
+    }
+
+    public void clearScroll() {
+        setScrollX(0);
+        setScrollY(0);
+    }
+
+    private Scroller mScroller;
+
+    public void smoothScrollTo(int x, int y, int duration) {
+        if (mScroller == null) {
+            mScroller = new Scroller(getContext(), new LinearInterpolator());
+            setScroller(mScroller);
+        }
+
+        int sx = getScrollX();
+        int sy = getScrollY();
+        mScroller.startScroll(sx, sy, x - sx, y - sy, duration);
+    }
+
+    @Override
+    public void computeScroll() {
+        if (mScroller != null && mScroller.computeScrollOffset()) {
+//            EasyL.v(TAG, "computeScroll: " + mScroller.getCurrX());
+            scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
+            postInvalidate();
+        }
+    }
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuViewFactory.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuViewFactory.java
new file mode 100644
index 0000000..12ba3de
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/DanmakuViewFactory.java
@@ -0,0 +1,24 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+import android.annotation.SuppressLint;
+import android.content.Context;
+import android.view.LayoutInflater;
+import android.view.ViewGroup;
+
+import org.exthmui.game.R;
+
+/**
+ * Created by LittleFogCat.
+ */
+class DanmakuViewFactory {
+    @SuppressLint("InflateParams")
+    static DanmakuView createDanmakuView(Context context) {
+        return (DanmakuView) LayoutInflater.from(context)
+                .inflate(R.layout.danmaku_view, null, false);
+    }
+
+    static DanmakuView createDanmakuView(Context context, ViewGroup parent) {
+        return (DanmakuView) LayoutInflater.from(context)
+                .inflate(R.layout.danmaku_view, parent, false);
+    }
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Pool.java b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Pool.java
new file mode 100644
index 0000000..5f356e7
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/danmaku/Pool.java
@@ -0,0 +1,23 @@
+package top.littlefogcat.danmakulib.danmaku;
+
+/**
+ * Created by LittleFogCat.
+ */
+public interface Pool<T> {
+    /**
+     * 从缓存中获取一个T的实例
+     */
+    T get();
+
+    /**
+     * 释放缓存
+     */
+    void release();
+
+    /**
+     * @return 缓存中T实例的数量
+     */
+    int count();
+
+    void setMaxSize(int max);
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/utils/EasyL.java b/app/src/main/java/top/littlefogcat/danmakulib/utils/EasyL.java
new file mode 100644
index 0000000..84df5ac
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/utils/EasyL.java
@@ -0,0 +1,54 @@
+package top.littlefogcat.danmakulib.utils;
+
+import android.util.Log;
+
+/**
+ * Created by LittleFogCat.
+ */
+public class EasyL {
+    private static boolean sEnabled = false;
+
+    public static void setEnabled(boolean enabled) {
+        sEnabled = enabled;
+    }
+
+    public static void v(String tag, String msg) {
+        if (sEnabled) {
+            Log.v(tag, msg);
+        }
+    }
+
+    public static void d(String tag, String msg) {
+        if (sEnabled) {
+            Log.d(tag, msg);
+        }
+    }
+
+    public static void i(String tag, String msg) {
+        if (sEnabled) {
+            Log.i(tag, msg);
+        }
+    }
+
+    public static void w(String tag, String msg) {
+        if (sEnabled) {
+            Log.w(tag, msg);
+        }
+    }
+
+    public static void e(String tag, String msg) {
+        if (sEnabled) {
+            Log.e(tag, msg);
+        }
+    }
+
+    public static void d(String tag, Object... args) {
+        if (sEnabled) {
+            StringBuilder msg = new StringBuilder();
+            for (Object arg : args) {
+                msg.append(arg).append(" ");
+            }
+            Log.d(tag, msg.toString());
+        }
+    }
+}
diff --git a/app/src/main/java/top/littlefogcat/danmakulib/utils/ScreenUtil.java b/app/src/main/java/top/littlefogcat/danmakulib/utils/ScreenUtil.java
new file mode 100644
index 0000000..0c461a3
--- /dev/null
+++ b/app/src/main/java/top/littlefogcat/danmakulib/utils/ScreenUtil.java
@@ -0,0 +1,138 @@
+package top.littlefogcat.danmakulib.utils;
+
+import android.content.Context;
+import android.util.DisplayMetrics;
+
+/**
+ * Created by LittleFogCat.
+ * <p>
+ * 自动适配屏幕像素的工具类。
+ * 需要先调用{@link ScreenUtil#init(Context)}才能正常使用。如果屏幕旋转,
+ * 那么需要再次调用{@link ScreenUtil#init(Context)}以更新。
+ */
+
+@SuppressWarnings({"unused", "WeakerAccess", "SuspiciousNameCombination"})
+public class ScreenUtil {
+    private static final String TAG = "ScreenUtil";
+    /**
+     * 屏幕宽度,在调用init()之后通过{@link ScreenUtil#getScreenWidth()}获取
+     */
+    private static int sScreenWidth = 1920;
+
+    /**
+     * 屏幕高度,在调用init()之后通过{@link ScreenUtil#getScreenHeight()} ()}获取
+     */
+    private static int sScreenHeight = 1080;
+
+    /**
+     * 设计宽度。用于{@link ScreenUtil#autoWidth(int)}
+     */
+    private static int sDesignWidth = 1080;
+
+    /**
+     * 设计高度。用于{@link ScreenUtil#autoHeight(int)} (int)}
+     */
+    private static int sDesignHeight = 1920;
+
+    /**
+     * 初始化ScreenUtil。在屏幕旋转之后,需要再次调用这个方法,否则计算将会出错。
+     */
+    public static void init(Context context) {
+        DisplayMetrics m = context.getResources().getDisplayMetrics();
+
+        sScreenWidth = m.widthPixels;
+        sScreenHeight = m.heightPixels;
+
+        if (sDesignWidth > sDesignHeight != sScreenWidth > sScreenHeight) {
+            int tmp = sDesignWidth;
+            sDesignWidth = sDesignHeight;
+            sDesignHeight = tmp;
+        }
+    }
+
+    public static void setDesignWidthAndHeight(int width, int height) {
+        sDesignWidth = width;
+        sDesignHeight = height;
+    }
+
+    /**
+     * 根据实际屏幕和设计图的比例,自动缩放像素大小。
+     * <p>
+     * 例如设计图大小是1920像素x1080像素,实际屏幕是2560像素x1440像素,那么对于一个设计图中100x100像素的方形,
+     * 实际屏幕中将会缩放为133像素x133像素。这有可能导致图形的失真(当实际的横竖比和设计图不同时)
+     *
+     * @param origin 设计图上的像素大小
+     * @return 实际屏幕中的尺寸
+     */
+    public static int autoSize(int origin) {
+        return autoWidth(origin);
+    }
+
+    /**
+     * 对于在横屏和竖屏下尺寸不同的物体,分别给出设计图的像素,返回实际屏幕中应有的像素。
+     *
+     * @param land 在横屏设计图中的像素大小
+     * @param port 在竖屏设计图中的像素大小
+     */
+    public static int autoSize(int land, int port) {
+        return isPortrait() ? autoSize(port) : autoSize(land);
+    }
+
+    /**
+     * 根据屏幕分辨率自适应宽度。
+     *
+     * @param origin 设计图中的宽度,像素
+     * @return 实际屏幕中的宽度,像素
+     */
+    public static int autoWidth(int origin) {
+        if (sScreenWidth == 0 || sDesignWidth == 0) {
+            return origin;
+        }
+        int autoSize = origin * sScreenWidth / sDesignWidth;
+        if (origin != 0 && autoSize == 0) {
+            return 1;
+        }
+        return autoSize;
+    }
+
+    /**
+     * 根据屏幕分辨率自适应高度
+     *
+     * @param origin 设计图中的高度,像素
+     * @return 实际屏幕中的高度,像素
+     */
+    public static int autoHeight(int origin) {
+        if (sScreenHeight == 0 || sDesignHeight == 0) {
+            return origin;
+        }
+        int auto = origin * sScreenHeight / sDesignHeight;
+        if (origin != 0 && auto == 0) {
+            return 1;
+        }
+        return auto;
+    }
+
+    public static int getScreenWidth() {
+        return sScreenWidth;
+    }
+
+    public static void setScreenWidth(int w) {
+        sScreenWidth = w;
+    }
+
+    public static int getScreenHeight() {
+        return sScreenHeight;
+    }
+
+    public static void setScreenHeight(int h) {
+        sScreenHeight = h;
+    }
+
+    /**
+     * 是否是竖屏
+     */
+    public static boolean isPortrait() {
+        return getScreenHeight() > getScreenWidth();
+    }
+
+}
diff --git a/app/src/main/privapp_whitelist_org.exthmui.game.xml b/app/src/main/privapp_whitelist_org.exthmui.game.xml
new file mode 100644
index 0000000..a7cb2fb
--- /dev/null
+++ b/app/src/main/privapp_whitelist_org.exthmui.game.xml
@@ -0,0 +1,23 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+     Copyright (C) 2017-2020 The LineageOS Project
+
+     Licensed under the Apache License, Version 2.0 (the "License");
+     you may not use this file except in compliance with the License.
+     You may obtain a copy of the License at
+
+          http://www.apache.org/licenses/LICENSE-2.0
+
+     Unless required by applicable law or agreed to in writing, software
+     distributed under the License is distributed on an "AS IS" BASIS,
+     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+     See the License for the specific language governing permissions and
+     limitations under the License.
+-->
+<permissions>
+    <privapp-permissions package="org.exthmui.game">
+        <permission name="android.permission.SYSTEM_ALERT_WINDOW"/>
+        <permission name="android.permission.MEDIA_CONTENT_CONTROL"/>
+        <permission name="android.permission.INTERACT_ACROSS_USERS_FULL"/>
+    </privapp-permissions>
+</permissions>
diff --git a/app/src/main/res/color/qs_background_colors.xml b/app/src/main/res/color/qs_background_colors.xml
new file mode 100644
index 0000000..bb7c365
--- /dev/null
+++ b/app/src/main/res/color/qs_background_colors.xml
@@ -0,0 +1,4 @@
+<selector xmlns:android="http://schemas.android.com/apk/res/android">
+    <item android:color="@*android:color/accent_device_default_light" android:state_selected="true" />
+    <item android:color="@color/gamingMenuBackground" android:state_selected="false" />
+</selector>
\ No newline at end of file
diff --git a/app/src/main/res/drawable/default_album_cover.xml b/app/src/main/res/drawable/default_album_cover.xml
new file mode 100644
index 0000000..014e40d
--- /dev/null
+++ b/app/src/main/res/drawable/default_album_cover.xml
@@ -0,0 +1,8 @@
+<vector android:height="96dp" android:tint="?attr/colorControlNormal"
+    android:viewportHeight="24" android:viewportWidth="24"
+    android:width="96dp" xmlns:android="http://schemas.android.com/apk/res/android">
+    <path android:fillColor="@android:color/black" android:fillAlpha="200"
+        android:pathData="M0,0 V96 H96 V0 H0" />
+
+    <path android:fillColor="@android:color/white" android:pathData="M12,3v10.55c-0.59,-0.34 -1.27,-0.55 -2,-0.55 -2.21,0 -4,1.79 -4,4s1.79,4 4,4 4,-1.79 4,-4V7h4V3h-6z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_game_button.xml b/app/src/main/res/drawable/ic_game_button.xml
new file mode 100644
index 0000000..c8e4320
--- /dev/null
+++ b/app/src/main/res/drawable/ic_game_button.xml
@@ -0,0 +1,17 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="108dp"
+    android:height="108dp"
+    android:viewportWidth="108"
+    android:viewportHeight="108">
+  <path
+      android:fillColor="#55000000"
+      android:pathData="M4,54 A50,50 0 1 0 4,53.99999"/>
+  <group android:scaleX="2.94408"
+      android:scaleY="2.94408"
+      android:translateX="18.67104"
+      android:translateY="18.67104">
+    <path
+        android:fillColor="@android:color/white"
+        android:pathData="M15,7.5V2H9v5.5l3,3 3,-3zM7.5,9H2v6h5.5l3,-3 -3,-3zM9,16.5V22h6v-5.5l-3,-3 -3,3zM16.5,9l-3,3 3,3H22V9h-5.5z"/>
+  </group>
+</vector>
diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml
new file mode 100644
index 0000000..fcaf719
--- /dev/null
+++ b/app/src/main/res/drawable/ic_launcher_background.xml
@@ -0,0 +1,74 @@
+<?xml version="1.0" encoding="utf-8"?>
+<vector
+    android:height="108dp"
+    android:width="108dp"
+    android:viewportHeight="108"
+    android:viewportWidth="108"
+    xmlns:android="http://schemas.android.com/apk/res/android">
+    <path android:fillColor="@*android:color/accent_device_default_light"
+          android:pathData="M0,0h108v108h-108z"/>
+    <path android:fillColor="#00000000" android:pathData="M9,0L9,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,0L19,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M29,0L29,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M39,0L39,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M49,0L49,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M59,0L59,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M69,0L69,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M79,0L79,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M89,0L89,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M99,0L99,108"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,9L108,9"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,19L108,19"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,29L108,29"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,39L108,39"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,49L108,49"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,59L108,59"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,69L108,69"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,79L108,79"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,89L108,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M0,99L108,99"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,29L89,29"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,39L89,39"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,49L89,49"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,59L89,59"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,69L89,69"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M19,79L89,79"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M29,19L29,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M39,19L39,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M49,19L49,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M59,19L59,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M69,19L69,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+    <path android:fillColor="#00000000" android:pathData="M79,19L79,89"
+          android:strokeColor="#33FFFFFF" android:strokeWidth="0.8"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_launcher_foreground.xml b/app/src/main/res/drawable/ic_launcher_foreground.xml
new file mode 100644
index 0000000..9fce646
--- /dev/null
+++ b/app/src/main/res/drawable/ic_launcher_foreground.xml
@@ -0,0 +1,15 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="108dp"
+    android:height="108dp"
+    android:viewportWidth="108"
+    android:viewportHeight="108"
+    android:tint="#FFFFFF">
+  <group android:scaleX="2.00448"
+      android:scaleY="2.00448"
+      android:translateX="29.94624"
+      android:translateY="29.94624">
+    <path
+        android:fillColor="@android:color/white"
+        android:pathData="M15,7.5V2H9v5.5l3,3 3,-3zM7.5,9H2v6h5.5l3,-3 -3,-3zM9,16.5V22h6v-5.5l-3,-3 -3,3zM16.5,9l-3,3 3,3H22V9h-5.5z"/>
+  </group>
+</vector>
diff --git a/app/src/main/res/drawable/ic_music_next.xml b/app/src/main/res/drawable/ic_music_next.xml
new file mode 100644
index 0000000..4fff247
--- /dev/null
+++ b/app/src/main/res/drawable/ic_music_next.xml
@@ -0,0 +1,10 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24"
+    android:tint="?attr/colorControlNormal">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M6,18l8.5,-6L6,6v12zM16,6v12h2V6h-2z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_music_pause.xml b/app/src/main/res/drawable/ic_music_pause.xml
new file mode 100644
index 0000000..13d6d2e
--- /dev/null
+++ b/app/src/main/res/drawable/ic_music_pause.xml
@@ -0,0 +1,10 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24"
+    android:tint="?attr/colorControlNormal">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M6,19h4L10,5L6,5v14zM14,5v14h4L18,5h-4z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_music_play.xml b/app/src/main/res/drawable/ic_music_play.xml
new file mode 100644
index 0000000..13c137a
--- /dev/null
+++ b/app/src/main/res/drawable/ic_music_play.xml
@@ -0,0 +1,10 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24"
+    android:tint="?attr/colorControlNormal">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M8,5v14l11,-7z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_music_previous.xml b/app/src/main/res/drawable/ic_music_previous.xml
new file mode 100644
index 0000000..1805b7d
--- /dev/null
+++ b/app/src/main/res/drawable/ic_music_previous.xml
@@ -0,0 +1,10 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24"
+    android:tint="?attr/colorControlNormal">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M6,6h2v12L6,18zM9.5,12l8.5,6L18,6z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_auto_brightness.xml b/app/src/main/res/drawable/ic_qs_auto_brightness.xml
new file mode 100644
index 0000000..5d6c10e
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_auto_brightness.xml
@@ -0,0 +1,10 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24"
+    android:tint="?attr/colorControlNormal">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M10.85,12.65h2.3L12,9l-1.15,3.65zM20,8.69V4h-4.69L12,0.69 8.69,4H4v4.69L0.69,12 4,15.31V20h4.69L12,23.31 15.31,20H20v-4.69L23.31,12 20,8.69zM14.3,16l-0.7,-2h-3.2l-0.7,2H7.8L11,7h2l3.2,9h-1.9z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_danmaku.xml b/app/src/main/res/drawable/ic_qs_danmaku.xml
new file mode 100644
index 0000000..c940942
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_danmaku.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M20,2L4,2c-1.1,0 -1.99,0.9 -1.99,2L2,22l4,-4h14c1.1,0 2,-0.9 2,-2L22,4c0,-1.1 -0.9,-2 -2,-2z M16,8H4V6H16 M19,11.5H8V9.5H19 M24,15H14V13H24"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_disable_gesture.xml b/app/src/main/res/drawable/ic_qs_disable_gesture.xml
new file mode 100644
index 0000000..b1fe495
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_disable_gesture.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M4.59,6.89c0.7,-0.71 1.4,-1.35 1.71,-1.22 0.5,0.2 0,1.03 -0.3,1.52 -0.25,0.42 -2.86,3.89 -2.86,6.31 0,1.28 0.48,2.34 1.34,2.98 0.75,0.56 1.74,0.73 2.64,0.46 1.07,-0.31 1.95,-1.4 3.06,-2.77 1.21,-1.49 2.83,-3.44 4.08,-3.44 1.63,0 1.65,1.01 1.76,1.79 -3.78,0.64 -5.38,3.67 -5.38,5.37 0,1.7 1.44,3.09 3.21,3.09 1.63,0 4.29,-1.33 4.69,-6.1L21,14.88v-2.5h-2.47c-0.15,-1.65 -1.09,-4.2 -4.03,-4.2 -2.25,0 -4.18,1.91 -4.94,2.84 -0.58,0.73 -2.06,2.48 -2.29,2.72 -0.25,0.3 -0.68,0.84 -1.11,0.84 -0.45,0 -0.72,-0.83 -0.36,-1.92 0.35,-1.09 1.4,-2.86 1.85,-3.52 0.78,-1.14 1.3,-1.92 1.3,-3.28C8.95,3.69 7.31,3 6.44,3 5.12,3 3.97,4 3.72,4.25c-0.36,0.36 -0.66,0.66 -0.88,0.93l1.75,1.71zM13.88,18.55c-0.31,0 -0.74,-0.26 -0.74,-0.72 0,-0.6 0.73,-2.2 2.87,-2.76 -0.3,2.69 -1.43,3.48 -2.13,3.48z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_disable_hw_key.xml b/app/src/main/res/drawable/ic_qs_disable_hw_key.xml
new file mode 100644
index 0000000..937e047
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_disable_hw_key.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M20,5L4,5c-1.1,0 -1.99,0.9 -1.99,2L2,17c0,1.1 0.9,2 2,2h16c1.1,0 2,-0.9 2,-2L22,7c0,-1.1 -0.9,-2 -2,-2zM11,8h2v2h-2L11,8zM11,11h2v2h-2v-2zM8,8h2v2L8,10L8,8zM8,11h2v2L8,13v-2zM7,13L5,13v-2h2v2zM7,10L5,10L5,8h2v2zM16,17L8,17v-2h8v2zM16,13h-2v-2h2v2zM16,10h-2L14,8h2v2zM19,13h-2v-2h2v2zM19,10h-2L17,8h2v2z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_dnd.xml b/app/src/main/res/drawable/ic_qs_dnd.xml
new file mode 100644
index 0000000..c6e6505
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_dnd.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24"
+    android:viewportHeight="24">
+  <path
+      android:fillColor="@android:color/white"
+      android:pathData="M7,11v2h10v-2L7,11zM12,2C6.48,2 2,6.48 2,12s4.48,10 10,10 10,-4.48 10,-10S17.52,2 12,2zM12,20c-4.41,0 -8,-3.59 -8,-8s3.59,-8 8,-8 8,3.59 8,8 -3.59,8 -8,8z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_qs_screenshot.xml b/app/src/main/res/drawable/ic_qs_screenshot.xml
new file mode 100644
index 0000000..375c0f9
--- /dev/null
+++ b/app/src/main/res/drawable/ic_qs_screenshot.xml
@@ -0,0 +1,30 @@
+<!--
+Copyright (C) 2018 The Android Open Source Project
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+    you may not use this file except in compliance with the License.
+    You may obtain a copy of the License at
+
+         http://www.apache.org/licenses/LICENSE-2.0
+
+    Unless required by applicable law or agreed to in writing, software
+    distributed under the License is distributed on an "AS IS" BASIS,
+    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+    See the License for the specific language governing permissions and
+    limitations under the License.
+-->
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24.0dp"
+    android:height="24.0dp"
+    android:viewportWidth="24.0"
+    android:viewportHeight="24.0">
+    <path
+        android:fillColor="#ffffff"
+        android:pathData="M17,1.01L7,1C5.9,1 5,1.9 5,3v18c0,1.1 0.9,2 2,2h10c1.1,0 2,-0.9 2,-2V3C19,1.9 18.1,1.01 17,1.01zM17,21H7l0,-1h10V21zM17,18H7V6h10V18zM17,4H7V3h10V4z"/>
+    <path
+        android:fillColor="#ffffff"
+        android:pathData="M9.5,8.5l2.5,0l0,-1.5l-2.5,0l-1.5,0l0,1.5l0,2.5l1.5,0z"/>
+    <path
+        android:fillColor="#ffffff"
+        android:pathData="M12,17l2.5,0l1.5,0l0,-1.5l0,-2.5l-1.5,0l0,2.5l-2.5,0z"/>
+</vector>
diff --git a/app/src/main/res/drawable/menu_divider.xml b/app/src/main/res/drawable/menu_divider.xml
new file mode 100644
index 0000000..e8f5262
--- /dev/null
+++ b/app/src/main/res/drawable/menu_divider.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8"?>
+<shape
+    xmlns:android="http://schemas.android.com/apk/res/android"
+    android:shape="rectangle">
+    <solid android:color="@android:color/transparent" />
+    <size android:height="6dp"/>
+</shape>
diff --git a/app/src/main/res/drawable/qs_background.xml b/app/src/main/res/drawable/qs_background.xml
new file mode 100644
index 0000000..efc75ee
--- /dev/null
+++ b/app/src/main/res/drawable/qs_background.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="108dp"
+    android:height="108dp"
+    android:viewportWidth="108"
+    android:viewportHeight="108">
+    <path
+        android:fillColor="@color/qs_background_colors"
+        android:pathData="M8,54 A46,46 0 1 0 8,53.99999"/>
+</vector>
diff --git a/app/src/main/res/drawable/qs_divider.xml b/app/src/main/res/drawable/qs_divider.xml
new file mode 100644
index 0000000..6c92631
--- /dev/null
+++ b/app/src/main/res/drawable/qs_divider.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8"?>
+<shape
+    xmlns:android="http://schemas.android.com/apk/res/android"
+    android:shape="rectangle">
+    <solid android:color="@android:color/transparent" />
+    <size android:width="6dp"/>
+</shape>
diff --git a/app/src/main/res/layout/danmaku_view.xml b/app/src/main/res/layout/danmaku_view.xml
new file mode 100644
index 0000000..51f5837
--- /dev/null
+++ b/app/src/main/res/layout/danmaku_view.xml
@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="utf-8"?>
+<top.littlefogcat.danmakulib.danmaku.DanmakuView xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:ellipsize="none"
+    android:shadowColor="#000000"
+    android:shadowDx="0"
+    android:shadowDy="0"
+    android:shadowRadius="2.5"
+    android:singleLine="true" />
\ No newline at end of file
diff --git a/app/src/main/res/layout/gaming_button_layout.xml b/app/src/main/res/layout/gaming_button_layout.xml
new file mode 100644
index 0000000..fc686a3
--- /dev/null
+++ b/app/src/main/res/layout/gaming_button_layout.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="wrap_content" android:layout_height="wrap_content"
+    android:background="@android:color/transparent">
+    <ImageView
+        android:id="@+id/floating_button"
+        android:layout_width="@dimen/game_button_size"
+        android:layout_height="@dimen/game_button_size"
+        android:contentDescription="@null"
+        android:scaleType="fitCenter"
+        android:src="@drawable/ic_game_button" />
+</FrameLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/gaming_overlay_layout.xml b/app/src/main/res/layout/gaming_overlay_layout.xml
new file mode 100644
index 0000000..118954a
--- /dev/null
+++ b/app/src/main/res/layout/gaming_overlay_layout.xml
@@ -0,0 +1,67 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout
+    xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="match_parent"
+    android:background="@android:color/transparent"
+    android:theme="@style/AppTheme"
+    android:gravity="bottom">
+
+    <ScrollView
+        android:id="@+id/gaming_menu"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:background="@color/gamingMenuBackground">
+
+        <LinearLayout
+            android:layout_width="match_parent"
+            android:layout_height="wrap_content"
+            android:clickable="true"
+            android:divider="@drawable/menu_divider"
+            android:orientation="vertical"
+            android:padding="@dimen/gaming_menu_padding"
+            android:showDividers="middle">
+
+            <org.exthmui.game.ui.TimeAndBatteryView
+                android:layout_width="match_parent"
+                android:layout_height="match_parent" />
+
+            <!--
+            <org.exthmui.game.ui.MusicControllerView
+                android:layout_width="match_parent"
+                android:layout_height="match_parent">
+
+            </org.exthmui.game.ui.MusicControllerView>
+            -->
+
+            <HorizontalScrollView
+                android:layout_width="match_parent"
+                android:layout_height="match_parent"
+                android:padding="4dp">
+
+                <org.exthmui.game.ui.QuickSettingsView
+                    android:id="@+id/gaming_qs"
+                    android:layout_width="match_parent"
+                    android:layout_height="match_parent" />
+            </HorizontalScrollView>
+
+            <HorizontalScrollView
+                android:layout_width="match_parent"
+                android:layout_height="match_parent"
+                android:padding="4dp">
+
+                <org.exthmui.game.ui.QuickStartAppView
+                    android:id="@+id/gaming_qsapp"
+                    android:layout_width="match_parent"
+                    android:layout_height="match_parent" />
+            </HorizontalScrollView>
+
+            <org.exthmui.game.ui.GamingPerformanceView
+                android:id="@+id/performance_controller"
+                android:layout_width="match_parent"
+                android:layout_height="match_parent" />
+
+        </LinearLayout>
+    </ScrollView>
+
+</LinearLayout>
diff --git a/app/src/main/res/layout/gaming_perofrmance_layout.xml b/app/src/main/res/layout/gaming_perofrmance_layout.xml
new file mode 100644
index 0000000..b2f0845
--- /dev/null
+++ b/app/src/main/res/layout/gaming_perofrmance_layout.xml
@@ -0,0 +1,40 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent" android:layout_height="match_parent"
+    android:orientation="vertical">
+    <TextView
+        android:layout_width="match_parent"
+        android:layout_height="wrap_content"
+        android:text="@string/performance_title"
+        android:textAppearance="@style/TextAppearance.AppCompat.Subhead" />
+
+    <SeekBar
+        android:id="@+id/performance_seek"
+        style="@android:style/Widget.Material.SeekBar.Discrete"
+        android:layout_width="match_parent"
+        android:layout_height="wrap_content"
+        android:max="6"
+        android:progress="3" />
+
+    <LinearLayout
+        android:layout_width="match_parent"
+        android:layout_height="match_parent"
+        android:orientation="horizontal">
+
+        <TextView
+            android:layout_width="0dp"
+            android:layout_height="wrap_content"
+            android:layout_weight="1"
+            android:gravity="left"
+            android:text="@string/performance_text_for_powersave" />
+
+        <TextView
+            android:layout_width="0dp"
+            android:layout_height="wrap_content"
+            android:layout_weight="1"
+            android:gravity="right"
+            android:text="@string/performance_text_for_performance" />
+
+    </LinearLayout>
+
+</LinearLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/gaming_qs_view.xml b/app/src/main/res/layout/gaming_qs_view.xml
new file mode 100644
index 0000000..7c36e33
--- /dev/null
+++ b/app/src/main/res/layout/gaming_qs_view.xml
@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    xmlns:app="http://schemas.android.com/apk/res-auto"
+    android:layout_width="wrap_content"
+    android:layout_height="wrap_content"
+    android:gravity="center"
+    android:orientation="vertical">
+
+    <ImageView
+        android:id="@+id/qs_icon"
+        android:layout_width="wrap_content"
+        android:layout_height="@dimen/gaming_qs_icon_size"
+        android:adjustViewBounds="true"
+        android:background="@drawable/qs_background"
+        android:cropToPadding="false"
+        android:padding="@dimen/gaming_qs_icon_padding"
+        android:scaleType="fitCenter" />
+
+    <TextView
+        android:id="@+id/qs_text"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:gravity="center"
+        android:maxEms="4"
+        android:maxLines="2" />
+</LinearLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/music_control_layout.xml b/app/src/main/res/layout/music_control_layout.xml
new file mode 100644
index 0000000..63ed79d
--- /dev/null
+++ b/app/src/main/res/layout/music_control_layout.xml
@@ -0,0 +1,67 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    xmlns:app="http://schemas.android.com/apk/res-auto"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:gravity="center"
+    android:orientation="horizontal">
+
+    <ImageView
+        android:id="@+id/music_cover"
+        android:layout_width="@dimen/music_cover_size"
+        android:layout_height="@dimen/music_cover_size"
+        android:scaleType="fitCenter"
+        android:src="@drawable/default_album_cover" />
+
+    <LinearLayout
+        android:layout_width="match_parent"
+        android:layout_height="match_parent"
+        android:layout_marginHorizontal="@dimen/music_info_margin_horizontal"
+        android:orientation="vertical">
+
+        <TextView
+            android:id="@+id/music_title"
+            android:layout_width="match_parent"
+            android:layout_height="wrap_content"
+            android:ellipsize="end"
+            android:maxLines="1"
+            android:maxLength="24"
+            android:textAppearance="@style/TextAppearance.AppCompat.Subhead" />
+
+        <TextView
+            android:id="@+id/music_artist"
+            android:layout_width="match_parent"
+            android:layout_height="wrap_content"
+            android:ellipsize="end"
+            android:maxLines="1" />
+
+        <LinearLayout
+            android:layout_width="match_parent"
+            android:layout_height="match_parent"
+            android:gravity="bottom"
+            android:orientation="horizontal">
+
+            <ImageView
+                android:id="@+id/prev_button"
+                android:layout_width="wrap_content"
+                android:layout_height="wrap_content"
+                android:layout_weight="1"
+                android:src="@drawable/ic_music_previous" />
+
+            <ImageView
+                android:id="@+id/play_pause_button"
+                android:layout_width="wrap_content"
+                android:layout_height="wrap_content"
+                android:layout_weight="1"
+                android:src="@drawable/ic_music_play" />
+
+            <ImageView
+                android:id="@+id/next_button"
+                android:layout_width="wrap_content"
+                android:layout_height="wrap_content"
+                android:layout_weight="1"
+                android:src="@drawable/ic_music_next" />
+        </LinearLayout>
+    </LinearLayout>
+
+</LinearLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/time_battery_layout.xml b/app/src/main/res/layout/time_battery_layout.xml
new file mode 100644
index 0000000..e9b0f99
--- /dev/null
+++ b/app/src/main/res/layout/time_battery_layout.xml
@@ -0,0 +1,29 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    xmlns:app="http://schemas.android.com/apk/res-auto"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:orientation="horizontal" >
+
+    <TextView
+        android:id="@+id/current_time"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:text="18:00"
+        android:textSize="32sp" />
+
+    <TextView
+        android:id="@+id/current_date"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_marginHorizontal="@dimen/time_battery_margin_horizontal"
+        android:text="2020年7月22日" />
+
+    <TextView
+        android:id="@+id/current_battery"
+        android:layout_width="match_parent"
+        android:layout_height="wrap_content"
+        android:gravity="end"
+        android:text="电量: 100%" />
+
+</LinearLayout>
\ No newline at end of file
diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
new file mode 100644
index 0000000..bbd3e02
--- /dev/null
+++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
@@ -0,0 +1,5 @@
+<?xml version="1.0" encoding="utf-8"?>
+<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
+    <background android:drawable="@drawable/ic_launcher_background"/>
+    <foreground android:drawable="@drawable/ic_launcher_foreground"/>
+</adaptive-icon>
\ No newline at end of file
diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml
new file mode 100644
index 0000000..eca70cf
--- /dev/null
+++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml
@@ -0,0 +1,5 @@
+<?xml version="1.0" encoding="utf-8"?>
+<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
+    <background android:drawable="@drawable/ic_launcher_background" />
+    <foreground android:drawable="@drawable/ic_launcher_foreground" />
+</adaptive-icon>
\ No newline at end of file
diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml
new file mode 100644
index 0000000..557bbb1
--- /dev/null
+++ b/app/src/main/res/values-zh-rCN/strings.xml
@@ -0,0 +1,21 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+    <string name="app_name">游戏模式核心</string>
+
+    <string name="channel_gaming_mode_status">游戏模式状态</string>
+    <string name="action_stop_gaming_mode">停止</string>
+    <string name="gaming_mode_running">游戏模式正在运行</string>
+
+
+    <string name="gaming_mode_on">游戏模式已启动</string>
+    <string name="gaming_mode_off">已退出游戏模式</string>
+
+    <string name="gaming_mode">游戏模式</string>
+    <string name="date_format">yyyy年MM月dd日</string>
+    <string name="time_format">HH:mm</string>
+    <string name="battery_format">电量: %1$d%%</string>
+
+    <string name="performance_title">性能配置</string>
+    <string name="performance_text_for_powersave">更长续航</string>
+    <string name="performance_text_for_performance">最高性能</string>
+</resources>
\ No newline at end of file
diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml
new file mode 100644
index 0000000..6599688
--- /dev/null
+++ b/app/src/main/res/values/colors.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+    <color name="colorPrimary">#6200EE</color>
+    <color name="colorPrimaryDark">#3700B3</color>
+    <color name="colorAccent">#03DAC5</color>
+    <color name="gamingMenuBackground">#66000000</color>
+</resources>
\ No newline at end of file
diff --git a/app/src/main/res/values/dimens.xml b/app/src/main/res/values/dimens.xml
new file mode 100644
index 0000000..156c92b
--- /dev/null
+++ b/app/src/main/res/values/dimens.xml
@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+    <dimen name="game_button_size">48dp</dimen>
+    <dimen name="music_cover_size">64dp</dimen>
+    <dimen name="gaming_qs_icon_size">56dp</dimen>
+    <dimen name="gaming_qs_icon_padding">16dp</dimen>
+    <dimen name="music_info_margin_horizontal">6dp</dimen>
+    <dimen name="time_battery_margin_horizontal">6dp</dimen>
+    <dimen name="gaming_menu_padding">12dp</dimen>
+</resources>
\ No newline at end of file
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
new file mode 100644
index 0000000..b1961bc
--- /dev/null
+++ b/app/src/main/res/values/strings.xml
@@ -0,0 +1,20 @@
+<resources>
+    <string name="app_name">GamingMode Core</string>
+
+    <string name="channel_gaming_mode_status">Gaming mode status</string>
+    <string name="action_stop_gaming_mode">Stop</string>
+    <string name="gaming_mode_running">Gaming mode is running</string>
+
+
+    <string name="gaming_mode_on">Gaming mode turned on</string>
+    <string name="gaming_mode_off">Gaming mode turned off</string>
+
+    <string name="gaming_mode">Gaming Mode</string>
+    <string name="date_format">yyyy-MM-dd</string>
+    <string name="time_format">HH:mm</string>
+    <string name="battery_format">Battery: %1$d%%</string>
+
+    <string name="performance_title">Performance</string>
+    <string name="performance_text_for_powersave">Powersave</string>
+    <string name="performance_text_for_performance">Performance</string>
+</resources>
\ No newline at end of file
diff --git a/app/src/main/res/values/styles.xml b/app/src/main/res/values/styles.xml
new file mode 100644
index 0000000..8e3341b
--- /dev/null
+++ b/app/src/main/res/values/styles.xml
@@ -0,0 +1,11 @@
+<resources>
+    <!-- Base application theme. -->
+    <style name="AppTheme" parent="Theme.AppCompat">
+        <!-- Customize your theme here. -->
+        <item name="colorPrimary">@color/colorPrimary</item>
+        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
+        <item name="colorAccent">@color/colorAccent</item>
+        <item name="android:textColor">#ffffffff</item>
+        <item name="android:textColorSecondary">#ffe0e0e0</item>
+    </style>
+</resources>
\ No newline at end of file
diff --git a/app/src/test/java/org/exthmui/game/ExampleUnitTest.java b/app/src/test/java/org/exthmui/game/ExampleUnitTest.java
new file mode 100644
index 0000000..b517f88
--- /dev/null
+++ b/app/src/test/java/org/exthmui/game/ExampleUnitTest.java
@@ -0,0 +1,17 @@
+package org.exthmui.game;
+
+import org.junit.Test;
+
+import static org.junit.Assert.*;
+
+/**
+ * Example local unit test, which will execute on the development machine (host).
+ *
+ * @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
+ */
+public class ExampleUnitTest {
+    @Test
+    public void addition_isCorrect() {
+        assertEquals(4, 2 + 2);
+    }
+}
\ No newline at end of file
diff --git a/build.gradle b/build.gradle
new file mode 100644
index 0000000..da807a2
--- /dev/null
+++ b/build.gradle
@@ -0,0 +1,24 @@
+// Top-level build file where you can add configuration options common to all sub-projects/modules.
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+    dependencies {
+        classpath "com.android.tools.build:gradle:4.0.0"
+
+        // NOTE: Do not place your application dependencies here; they belong
+        // in the individual module build.gradle files
+    }
+}
+
+allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+task clean(type: Delete) {
+    delete rootProject.buildDir
+}
\ No newline at end of file
diff --git a/gradle.properties b/gradle.properties
new file mode 100644
index 0000000..c52ac9b
--- /dev/null
+++ b/gradle.properties
@@ -0,0 +1,19 @@
+# Project-wide Gradle settings.
+# IDE (e.g. Android Studio) users:
+# Gradle settings configured through the IDE *will override*
+# any settings specified in this file.
+# For more details on how to configure your build environment visit
+# http://www.gradle.org/docs/current/userguide/build_environment.html
+# Specifies the JVM arguments used for the daemon process.
+# The setting is particularly useful for tweaking memory settings.
+org.gradle.jvmargs=-Xmx2048m
+# When configured, Gradle will run in incubating parallel mode.
+# This option should only be used with decoupled projects. More details, visit
+# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
+# org.gradle.parallel=true
+# AndroidX package structure to make it clearer which packages are bundled with the
+# Android operating system, and which are packaged with your app"s APK
+# https://developer.android.com/topic/libraries/support-library/androidx-rn
+android.useAndroidX=true
+# Automatically convert third-party libraries to use AndroidX
+android.enableJetifier=true
\ No newline at end of file
diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar
new file mode 100644
index 0000000..f6b961f
--- /dev/null
+++ b/gradle/wrapper/gradle-wrapper.jar
Binary files differ
diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 0000000..d402a46
--- /dev/null
+++ b/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,6 @@
+#Sat Jul 11 17:03:41 CST 2020
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-6.1.1-all.zip
diff --git a/gradlew b/gradlew
new file mode 100755
index 0000000..cccdd3d
--- /dev/null
+++ b/gradlew
@@ -0,0 +1,172 @@
+#!/usr/bin/env sh
+
+##############################################################################
+##
+##  Gradle start up script for UN*X
+##
+##############################################################################
+
+# Attempt to set APP_HOME
+# Resolve links: $0 may be a link
+PRG="$0"
+# Need this for relative symlinks.
+while [ -h "$PRG" ] ; do
+    ls=`ls -ld "$PRG"`
+    link=`expr "$ls" : '.*-> \(.*\)$'`
+    if expr "$link" : '/.*' > /dev/null; then
+        PRG="$link"
+    else
+        PRG=`dirname "$PRG"`"/$link"
+    fi
+done
+SAVED="`pwd`"
+cd "`dirname \"$PRG\"`/" >/dev/null
+APP_HOME="`pwd -P`"
+cd "$SAVED" >/dev/null
+
+APP_NAME="Gradle"
+APP_BASE_NAME=`basename "$0"`
+
+# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+DEFAULT_JVM_OPTS=""
+
+# Use the maximum available, or set MAX_FD != -1 to use that value.
+MAX_FD="maximum"
+
+warn () {
+    echo "$*"
+}
+
+die () {
+    echo
+    echo "$*"
+    echo
+    exit 1
+}
+
+# OS specific support (must be 'true' or 'false').
+cygwin=false
+msys=false
+darwin=false
+nonstop=false
+case "`uname`" in
+  CYGWIN* )
+    cygwin=true
+    ;;
+  Darwin* )
+    darwin=true
+    ;;
+  MINGW* )
+    msys=true
+    ;;
+  NONSTOP* )
+    nonstop=true
+    ;;
+esac
+
+CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
+
+# Determine the Java command to use to start the JVM.
+if [ -n "$JAVA_HOME" ] ; then
+    if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
+        # IBM's JDK on AIX uses strange locations for the executables
+        JAVACMD="$JAVA_HOME/jre/sh/java"
+    else
+        JAVACMD="$JAVA_HOME/bin/java"
+    fi
+    if [ ! -x "$JAVACMD" ] ; then
+        die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+    fi
+else
+    JAVACMD="java"
+    which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+fi
+
+# Increase the maximum file descriptors if we can.
+if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
+    MAX_FD_LIMIT=`ulimit -H -n`
+    if [ $? -eq 0 ] ; then
+        if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
+            MAX_FD="$MAX_FD_LIMIT"
+        fi
+        ulimit -n $MAX_FD
+        if [ $? -ne 0 ] ; then
+            warn "Could not set maximum file descriptor limit: $MAX_FD"
+        fi
+    else
+        warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
+    fi
+fi
+
+# For Darwin, add options to specify how the application appears in the dock
+if $darwin; then
+    GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
+fi
+
+# For Cygwin, switch paths to Windows format before running java
+if $cygwin ; then
+    APP_HOME=`cygpath --path --mixed "$APP_HOME"`
+    CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
+    JAVACMD=`cygpath --unix "$JAVACMD"`
+
+    # We build the pattern for arguments to be converted via cygpath
+    ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
+    SEP=""
+    for dir in $ROOTDIRSRAW ; do
+        ROOTDIRS="$ROOTDIRS$SEP$dir"
+        SEP="|"
+    done
+    OURCYGPATTERN="(^($ROOTDIRS))"
+    # Add a user-defined pattern to the cygpath arguments
+    if [ "$GRADLE_CYGPATTERN" != "" ] ; then
+        OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
+    fi
+    # Now convert the arguments - kludge to limit ourselves to /bin/sh
+    i=0
+    for arg in "$@" ; do
+        CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
+        CHECK2=`echo "$arg"|egrep -c "^-"`                                 ### Determine if an option
+
+        if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then                    ### Added a condition
+            eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
+        else
+            eval `echo args$i`="\"$arg\""
+        fi
+        i=$((i+1))
+    done
+    case $i in
+        (0) set -- ;;
+        (1) set -- "$args0" ;;
+        (2) set -- "$args0" "$args1" ;;
+        (3) set -- "$args0" "$args1" "$args2" ;;
+        (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
+        (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
+        (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
+        (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
+        (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
+        (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
+    esac
+fi
+
+# Escape application args
+save () {
+    for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
+    echo " "
+}
+APP_ARGS=$(save "$@")
+
+# Collect all arguments for the java command, following the shell quoting and substitution rules
+eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
+
+# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
+if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
+  cd "$(dirname "$0")"
+fi
+
+exec "$JAVACMD" "$@"
diff --git a/gradlew.bat b/gradlew.bat
new file mode 100644
index 0000000..e95643d
--- /dev/null
+++ b/gradlew.bat
@@ -0,0 +1,84 @@
+@if "%DEBUG%" == "" @echo off

+@rem ##########################################################################

+@rem

+@rem  Gradle startup script for Windows

+@rem

+@rem ##########################################################################

+

+@rem Set local scope for the variables with windows NT shell

+if "%OS%"=="Windows_NT" setlocal

+

+set DIRNAME=%~dp0

+if "%DIRNAME%" == "" set DIRNAME=.

+set APP_BASE_NAME=%~n0

+set APP_HOME=%DIRNAME%

+

+@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.

+set DEFAULT_JVM_OPTS=

+

+@rem Find java.exe

+if defined JAVA_HOME goto findJavaFromJavaHome

+

+set JAVA_EXE=java.exe

+%JAVA_EXE% -version >NUL 2>&1

+if "%ERRORLEVEL%" == "0" goto init

+

+echo.

+echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.

+echo.

+echo Please set the JAVA_HOME variable in your environment to match the

+echo location of your Java installation.

+

+goto fail

+

+:findJavaFromJavaHome

+set JAVA_HOME=%JAVA_HOME:"=%

+set JAVA_EXE=%JAVA_HOME%/bin/java.exe

+

+if exist "%JAVA_EXE%" goto init

+

+echo.

+echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%

+echo.

+echo Please set the JAVA_HOME variable in your environment to match the

+echo location of your Java installation.

+

+goto fail

+

+:init

+@rem Get command-line arguments, handling Windows variants

+

+if not "%OS%" == "Windows_NT" goto win9xME_args

+

+:win9xME_args

+@rem Slurp the command line arguments.

+set CMD_LINE_ARGS=

+set _SKIP=2

+

+:win9xME_args_slurp

+if "x%~1" == "x" goto execute

+

+set CMD_LINE_ARGS=%*

+

+:execute

+@rem Setup the command line

+

+set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar

+

+@rem Execute Gradle

+"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%

+

+:end

+@rem End local scope for the variables with windows NT shell

+if "%ERRORLEVEL%"=="0" goto mainEnd

+

+:fail

+rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of

+rem the _cmd.exe /c_ return code!

+if  not "" == "%GRADLE_EXIT_CONSOLE%" exit 1

+exit /b 1

+

+:mainEnd

+if "%OS%"=="Windows_NT" endlocal

+

+:omega

diff --git a/keystore.properties.sample b/keystore.properties.sample
new file mode 100644
index 0000000..af869b1
--- /dev/null
+++ b/keystore.properties.sample
@@ -0,0 +1,4 @@
+keyAlias=android
+keyPassword=android
+storeFile=testkey.jks
+storePassword=android
diff --git a/settings.gradle b/settings.gradle
new file mode 100644
index 0000000..26ec000
--- /dev/null
+++ b/settings.gradle
@@ -0,0 +1,2 @@
+include ':app'
+rootProject.name = "GamingMode"
\ No newline at end of file