[1/2] OmniGears: Events settings improvements

Change-Id: I9798bb1244fe1bd34f48373e4bbf18b83c7a63d2
diff --git a/res/layout/app_grid_item.xml b/res/layout/app_grid_item.xml
new file mode 100644
index 0000000..0b1644c
--- /dev/null
+++ b/res/layout/app_grid_item.xml
@@ -0,0 +1,30 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--  Copyright (C) 2016 The OmniROM Project
+
+  This program is free software: you can redistribute it and/or modify
+  it under the terms of the GNU General Public License as published by
+  the Free Software Foundation, either version 2 of the License, or
+  (at your option) any later version.
+
+  This program is distributed in the hope that it will be useful,
+  but WITHOUT ANY WARRANTY; without even the implied warranty of
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+  GNU General Public License for more details.
+
+  You should have received a copy of the GNU General Public License
+  along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ -->
+<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="match_parent">
+
+    <ImageView
+        android:id="@+id/appIcon"
+        android:layout_width="@android:dimen/app_icon_size"
+        android:layout_height="@android:dimen/app_icon_size"
+        android:layout_alignParentLeft="true"
+        android:layout_alignParentStart="true"
+        android:layout_alignParentTop="true"
+        android:background="?android:attr/selectableItemBackground"
+        android:scaleType="centerInside" />
+</RelativeLayout>
diff --git a/res/layout/app_grid_view.xml b/res/layout/app_grid_view.xml
new file mode 100644
index 0000000..38c575f
--- /dev/null
+++ b/res/layout/app_grid_view.xml
@@ -0,0 +1,29 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--  Copyright (C) 2016 The OmniROM Project
+
+  This program is free software: you can redistribute it and/or modify
+  it under the terms of the GNU General Public License as published by
+  the Free Software Foundation, either version 2 of the License, or
+  (at your option) any later version.
+
+  This program is distributed in the hope that it will be useful,
+  but WITHOUT ANY WARRANTY; without even the implied warranty of
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+  GNU General Public License for more details.
+
+  You should have received a copy of the GNU General Public License
+  along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ -->
+<GridView xmlns:android="http://schemas.android.com/apk/res/android"
+    android:columnWidth="@android:dimen/app_icon_size"
+    android:gravity="center"
+    android:horizontalSpacing="16dp"
+    android:id="@+id/app_grid_view"
+    android:layout_height="match_parent"
+    android:layout_width="match_parent"
+    android:listSelector="@android:color/transparent"
+    android:numColumns="auto_fit"
+    android:padding="16dp"
+    android:stretchMode="columnWidth"
+    android:verticalSpacing="16dp" />
+
diff --git a/res/layout/app_select_item.xml b/res/layout/app_select_item.xml
new file mode 100644
index 0000000..92a9b9a
--- /dev/null
+++ b/res/layout/app_select_item.xml
@@ -0,0 +1,57 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--  Copyright (C) 2016 The OmniROM Project
+
+  This program is free software: you can redistribute it and/or modify
+  it under the terms of the GNU General Public License as published by
+  the Free Software Foundation, either version 2 of the License, or
+  (at your option) any later version.
+
+  This program is distributed in the hope that it will be useful,
+  but WITHOUT ANY WARRANTY; without even the implied warranty of
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+  GNU General Public License for more details.
+
+  You should have received a copy of the GNU General Public License
+  along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ -->
+
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:minHeight="64dp"
+    android:paddingStart="6dp"
+    android:gravity="center_vertical"
+    android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
+    android:orientation="horizontal" >
+
+    <ImageView
+        android:id="@+id/app_icon"
+        android:layout_width="@android:dimen/app_icon_size"
+        android:layout_height="@android:dimen/app_icon_size"
+        android:layout_marginEnd="8dip"
+        android:layout_gravity="center_vertical"
+        android:scaleType="centerInside"
+        android:contentDescription="@null" />
+
+    <TextView
+        android:id="@+id/app_name"
+        android:layout_width="0dip"
+        android:layout_height="wrap_content"
+        android:layout_weight="1"
+        android:layout_gravity="center_vertical"
+        android:singleLine="true"
+        android:ellipsize="end"
+        android:focusable="false"
+        android:textAppearance="?android:attr/textAppearanceListItem"
+        android:textAlignment="viewStart" />
+
+    <CheckBox android:id="@android:id/checkbox"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_marginStart="8dip"
+        android:layout_gravity="center_vertical"
+        android:clickable="false"
+        android:focusable="false" />
+
+</LinearLayout>
+
diff --git a/res/layout/preference_app_list.xml b/res/layout/preference_app_list.xml
new file mode 100644
index 0000000..97177b9
--- /dev/null
+++ b/res/layout/preference_app_list.xml
@@ -0,0 +1,32 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--  Copyright (C) 2016 The OmniROM Project
+
+  This program is free software: you can redistribute it and/or modify
+  it under the terms of the GNU General Public License as published by
+  the Free Software Foundation, either version 2 of the License, or
+  (at your option) any later version.
+
+  This program is distributed in the hope that it will be useful,
+  but WITHOUT ANY WARRANTY; without even the implied warranty of
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+  GNU General Public License for more details.
+
+  You should have received a copy of the GNU General Public License
+  along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ -->
+
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    xmlns:tools="http://schemas.android.com/tools"
+    android:layout_width="match_parent"
+    android:layout_height="match_parent"
+    android:orientation="vertical"
+    android:paddingStart="@dimen/alert_dialog_padding_material"
+    android:paddingEnd="@dimen/alert_dialog_padding_material">
+
+    <ListView
+        android:id="@+id/app_list"
+        android:layout_width="match_parent"
+        android:layout_height="wrap_content"
+        android:divider="@null" />
+</LinearLayout>
+
diff --git a/res/layout/preference_dialog_applist.xml b/res/layout/preference_dialog_applist.xml
deleted file mode 100644
index 55ab014..0000000
--- a/res/layout/preference_dialog_applist.xml
+++ /dev/null
@@ -1,29 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!-- Copyright (C) 2016 The OmniROM 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.
--->
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
-        android:layout_width="match_parent"
-        android:layout_height="match_parent"
-        android:orientation="vertical"
-        android:gravity="center_horizontal"
-        android:paddingStart="24dp"
-        android:paddingEnd="24dp" >
-
-     <ListView android:id="@+id/applist"
-               android:layout_width="match_parent"
-               android:layout_height="match_parent"
-               android:divider="@null" />
-
-</LinearLayout>
diff --git a/res/layout/preference_selected_apps_view.xml b/res/layout/preference_selected_apps_view.xml
new file mode 100644
index 0000000..aa7c499
--- /dev/null
+++ b/res/layout/preference_selected_apps_view.xml
@@ -0,0 +1,60 @@
+<?xml version="1.0" encoding="utf-8"?><!--  Copyright (C) 2016 The OmniROM Project
+
+  This program is free software: you can redistribute it and/or modify
+  it under the terms of the GNU General Public License as published by
+  the Free Software Foundation, either version 2 of the License, or
+  (at your option) any later version.
+
+  This program is distributed in the hope that it will be useful,
+  but WITHOUT ANY WARRANTY; without even the implied warranty of
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+  GNU General Public License for more details.
+
+  You should have received a copy of the GNU General Public License
+  along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ -->
+
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:background="@android:drawable/list_selector_background"
+    android:gravity="center_vertical"
+    android:minHeight="?android:attr/listPreferredItemHeight"
+    android:paddingEnd="?android:attr/scrollbarSize">
+
+    <RelativeLayout
+        android:layout_width="0dp"
+        android:layout_height="wrap_content"
+        android:layout_marginBottom="6dip"
+        android:layout_marginEnd="6dip"
+        android:layout_marginStart="15dip"
+        android:layout_marginTop="6dip"
+        android:layout_weight="1">
+
+        <HorizontalScrollView
+            android:id="@+id/appsScrollView"
+            android:layout_width="wrap_content"
+            android:layout_height="wrap_content"
+            android:layout_gravity="center"
+            android:layout_marginBottom="8dip"
+            android:layout_marginTop="8dip"
+            android:scrollbars="none" >
+
+            <LinearLayout
+                android:id="@+id/selected_apps"
+                android:layout_width="wrap_content"
+                android:layout_height="wrap_content"
+                android:orientation="horizontal" />
+        </HorizontalScrollView>
+
+    </RelativeLayout>
+
+    <!-- Preference should place its actual preference widget here. -->
+    <LinearLayout
+        android:id="@android:id/widget_frame"
+        android:layout_width="wrap_content"
+        android:layout_height="match_parent"
+        android:gravity="center_vertical"
+        android:orientation="vertical" />
+
+</LinearLayout>
diff --git a/res/values/custom_strings.xml b/res/values/custom_strings.xml
index 2ea865a..e591423 100644
--- a/res/values/custom_strings.xml
+++ b/res/values/custom_strings.xml
@@ -698,13 +698,18 @@
 
     <string name="event_service_settings_title">Events</string>
     <string name="category_media_player_title">Media player</string>
-    <string name="bt_a2dp_connect_app_title">Bluetooth A2DP connect</string>
-    <string name="headset_connect_app_title">Wired headset connect</string>
+    <string name="bt_a2dp_connect_app_list_title">Bluetooth A2DP connect</string>
+    <string name="headset_connect_app_list_title">Wired headset connect</string>
     <string name="event_service_enabled_title">Enable event service</string>
     <string name="event_service_running">Service is running</string>
     <string name="event_service_stopped">Service is not running</string>
-    <string name="category_media_player_info_title">Select an app to be started for this event</string>
+    <string name="category_media_player_info_title">App selection</string>
     <string name="media_player_autostart_title">Send media play event after start</string>
+    <string name="media_player_music_active_title">Do not start event on playback</string>
+    <string name="autorun_single_app_title">Autorun single app</string>
+    <string name="autorun_single_app_summary">Dont show app selection dialog</string>
+    <string name="bt_a2dp_connect_app_list_summary">App(s) to be listed when an A2DP event is triggered, such as connecting a Bluetooth headset</string>
+    <string name="headset_connect_app_list_summary">App(s) to be listed when a wired headset is connected</string>
 
     <!-- Quick Pulldown-->
     <string name="quick_pulldown_title">Quick pulldown</string>
diff --git a/res/xml/event_service_settings.xml b/res/xml/event_service_settings.xml
index 4b1bf2c..cff6d4d 100644
--- a/res/xml/event_service_settings.xml
+++ b/res/xml/event_service_settings.xml
@@ -1,4 +1,5 @@
 <?xml version="1.0" encoding="utf-8"?>
+
 <!--  Copyright (C) 2018 The OmniROM Project
 
   This program is free software: you can redistribute it and/or modify
@@ -18,39 +19,71 @@
 <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"
     android:key="event_service_settings"
     android:title="@string/event_service_settings_title">
+
+    <SwitchPreference
+        android:key="event_service_enabled"
+        android:persistent="false"
+        android:title="@string/event_service_enabled_title" />
+
+    <PreferenceCategory
+        android:key="category_media_player"
+        android:title="@string/category_media_player_title">
+
+        <Preference
+            android:dependency="event_service_enabled"
+            android:icon="@drawable/ic_info_outline_24dp"
+            android:key="category_media_player_info"
+            android:persistent="false"
+            android:summary="@string/category_media_player_info_title" />
+
+        <org.omnirom.omnigears.preference.AppMultiSelectListPreference
+            android:dependency="event_service_enabled"
+            android:icon="@drawable/ic_settings_bluetooth"
+            android:key="bt_a2dp_connect_app_list"
+            android:persistent="false"
+            android:summary="@string/bt_a2dp_connect_app_list_summary"
+            android:title="@string/bt_a2dp_connect_app_list_title" />
+
+        <org.omnirom.omnigears.preference.ScrollAppsViewPreference
+            android:dependency="event_service_enabled"
+            android:key="a2dp_app_list"
+            android:selectable="false"
+            android:persistent="false" />
+
+        <org.omnirom.omnigears.preference.AppMultiSelectListPreference
+            android:dependency="event_service_enabled"
+            android:icon="@drawable/ic_headset_24dp"
+            android:key="headset_connect_app_list"
+            android:persistent="false"
+            android:summary="@string/headset_connect_app_list_summary"
+            android:title="@string/headset_connect_app_list_title" />
+
+        <org.omnirom.omnigears.preference.ScrollAppsViewPreference
+            android:dependency="event_service_enabled"
+            android:key="headset_app_list"
+            android:selectable="false"
+            android:persistent="false" />
+
         <SwitchPreference
-            android:key="event_service_enabled"
-            android:title="@string/event_service_enabled_title"
-            android:persistent="false"/>
+            android:defaultValue="true"
+            android:dependency="event_service_enabled"
+            android:key="autorun_single_app"
+            android:persistent="false"
+            android:summary="@string/autorun_single_app_summary"
+            android:title="@string/autorun_single_app_title" />
 
-        <PreferenceCategory
-            android:key="category_media_player"
-            android:title="@string/category_media_player_title">
+        <SwitchPreference
+            android:defaultValue="true"
+            android:dependency="event_service_enabled"
+            android:key="media_player_music_active"
+            android:persistent="false"
+            android:title="@string/media_player_music_active_title" />
 
-            <Preference
-                android:key="category_media_player_info"
-                android:summary="@string/category_media_player_info_title"
-                android:persistent="false"
-                android:icon="@drawable/ic_info_outline_24dp"
-                android:dependency="event_service_enabled" />
+        <SwitchPreference
+            android:dependency="event_service_enabled"
+            android:key="media_player_autostart"
+            android:persistent="false"
+            android:title="@string/media_player_autostart_title" />
 
-            <org.omnirom.omnigears.preference.AppSelectListPreference
-                android:key="bt_a2dp_connect_app"
-                android:title="@string/bt_a2dp_connect_app_title"
-                android:persistent="false"
-                android:dependency="event_service_enabled" />
-
-            <org.omnirom.omnigears.preference.AppSelectListPreference
-                android:key="headset_connect_app"
-                android:title="@string/headset_connect_app_title"
-                android:persistent="false"
-                android:dependency="event_service_enabled" />
-
-            <SwitchPreference
-                android:key="media_player_autostart"
-                android:title="@string/media_player_autostart_title"
-                android:persistent="false"
-                android:dependency="event_service_enabled" />
-
-        </PreferenceCategory>
+    </PreferenceCategory>
 </PreferenceScreen>
diff --git a/src/org/omnirom/omnigears/preference/AppMultiSelectListPreference.java b/src/org/omnirom/omnigears/preference/AppMultiSelectListPreference.java
new file mode 100644
index 0000000..62749cd
--- /dev/null
+++ b/src/org/omnirom/omnigears/preference/AppMultiSelectListPreference.java
@@ -0,0 +1,246 @@
+/*
+ *  Copyright (C) 2016 The OmniROM Project
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+package org.omnirom.omnigears.preference;
+
+import android.content.ComponentName;
+import android.content.Context;
+import android.content.Intent;
+import android.content.pm.PackageManager;
+import android.content.pm.ResolveInfo;
+import android.graphics.drawable.Drawable;
+import android.util.AttributeSet;
+import android.util.Log;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.AdapterView;
+import android.widget.ArrayAdapter;
+import android.widget.CheckBox;
+import android.widget.ImageView;
+import android.widget.ListView;
+import android.widget.TextView;
+
+import com.android.settings.R;
+import com.android.settingslib.CustomDialogPreference;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+
+public class AppMultiSelectListPreference extends CustomDialogPreference {
+    private static final String TAG = "AppMultiSelectList";
+    private static final boolean DEBUG = false;
+
+    private final List<PackageItem> mPackageInfoList = new ArrayList<PackageItem>();
+    private AppListAdapter mAdapter;
+    private Set<String> mValues = new HashSet<String>();
+    private PackageManager mPm;
+
+    public AppMultiSelectListPreference(Context context) {
+        this(context, null);
+    }
+
+    public AppMultiSelectListPreference(Context context, AttributeSet attrs) {
+        super(context, attrs);
+        setDialogLayoutResource(R.layout.preference_app_list);
+
+        mPm = context.getPackageManager();
+
+        final Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);
+        mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);
+        List<ResolveInfo> installedAppsInfo = getContext().getPackageManager().queryIntentActivities(
+                mainIntent, 0);
+
+        for (ResolveInfo info : installedAppsInfo) {
+            ComponentName componentName = new ComponentName(
+                    info.activityInfo.applicationInfo.packageName,
+                    info.activityInfo.name);
+
+            try {
+                final PackageItem item = new PackageItem(
+                        info.activityInfo.loadLabel(mPm), 0, componentName);
+                mPackageInfoList.add(item);
+            } catch (Exception e) {
+                if (DEBUG) Log.e(TAG, "Load installed apps", e);
+            }
+        }
+        Collections.sort(mPackageInfoList);
+
+        setPositiveButtonText(R.string.action_save);
+        setNegativeButtonText(android.R.string.cancel);
+    }
+
+    public void setValues(Set<String> values) {
+        mValues.clear();
+        mValues.addAll(values);
+    }
+
+    public Set<String> getValues() {
+        return mValues;
+    }
+
+    @Override
+    protected void onBindDialogView(View view) {
+        super.onBindDialogView(view);
+
+        mAdapter = new AppListAdapter(getContext());
+        final ListView listView = (ListView) view.findViewById(R.id.app_list);
+        listView.setAdapter(mAdapter);
+        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
+            @Override
+            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
+                final AppViewHolder holder = (AppViewHolder) view.getTag();
+                final boolean isChecked = !holder.checkBox.isChecked();
+
+                holder.checkBox.setChecked(isChecked);
+                PackageItem info = mAdapter.getItem(position);
+
+                if (isChecked) {
+                    mValues.add(info.mValue);
+                } else {
+                    mValues.remove(info.mValue);
+                }
+            }
+        });
+    }
+
+    @Override
+    protected void onDialogClosed(boolean positiveResult) {
+        super.onDialogClosed(positiveResult);
+        if (positiveResult) {
+            callChangeListener(mValues.size() > 0 ? mValues : null);
+        }
+    }
+
+    public class PackageItem implements Comparable<PackageItem> {
+        public final CharSequence mTitle;
+        public final int mAppIconResourceId;
+        public final ComponentName mComponentName;
+        public final String mValue;
+
+        PackageItem(CharSequence title, int iconResourceId, ComponentName componentName) {
+            mTitle = title;
+            mAppIconResourceId = iconResourceId;
+            mComponentName = componentName;
+            mValue = componentName.flattenToString();
+        }
+
+        PackageItem(CharSequence title, int iconResourceId, String value) {
+            mTitle = title;
+            mAppIconResourceId = iconResourceId;
+            mComponentName = null;
+            mValue = value;
+        }
+
+        @Override
+        public int compareTo(PackageItem another) {
+            return mTitle.toString().toUpperCase().compareTo(another.mTitle.toString().toUpperCase());
+        }
+
+        @Override
+        public int hashCode() {
+            return mValue.hashCode();
+        }
+
+        @Override
+        public boolean equals(Object another) {
+            if (another == null || !(another instanceof PackageItem)) {
+                return false;
+            }
+            return mValue.equals(((PackageItem) another).mValue);
+        }
+    }
+
+    public class AppListAdapter extends ArrayAdapter<PackageItem> {
+        private final LayoutInflater mInflater;
+
+        public AppListAdapter(Context context) {
+            super(context, 0);
+            mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
+            addAll(mPackageInfoList);
+        }
+
+        @Override
+        public View getView(int position, View convertView, ViewGroup parent) {
+            AppViewHolder holder = AppViewHolder.createOrRecycle(mInflater, convertView);
+            convertView = holder.rootView;
+            PackageItem info = getItem(position);
+            holder.appName.setText(info.mTitle);
+            if (info.mAppIconResourceId != 0) {
+                holder.appIcon.setImageResource(info.mAppIconResourceId);
+            } else {
+                Drawable d = resolveAppIcon(info);
+                holder.appIcon.setImageDrawable(d);
+            }
+            holder.checkBox.setChecked(mValues.contains(info.mValue));
+            return convertView;
+        }
+
+        @Override
+        public PackageItem getItem(int position) {
+            return mPackageInfoList.get(position);
+        }
+    }
+
+    public static class AppViewHolder {
+        public View rootView;
+        public TextView appName;
+        public ImageView appIcon;
+        public CheckBox checkBox;
+
+        public static AppViewHolder createOrRecycle(LayoutInflater inflater, View convertView) {
+            if (convertView == null) {
+                convertView = inflater.inflate(R.layout.app_select_item, null);
+
+                // Creates a ViewHolder and store references to the two children views
+                // we want to bind data to.
+                AppViewHolder holder = new AppViewHolder();
+                holder.rootView = convertView;
+                holder.appName = (TextView) convertView.findViewById(R.id.app_name);
+                holder.appIcon = (ImageView) convertView.findViewById(R.id.app_icon);
+                holder.checkBox = (CheckBox) convertView.findViewById(android.R.id.checkbox);
+                convertView.setTag(holder);
+                return holder;
+            } else {
+                // Get the ViewHolder back to get fast access to the TextView
+                // and the ImageView.
+                return (AppViewHolder) convertView.getTag();
+            }
+        }
+    }
+
+    private Drawable getDefaultActivityIcon() {
+        return getContext().getResources().getDrawable(android.R.drawable.sym_def_app_icon);
+    }
+
+    private Drawable resolveAppIcon(PackageItem item) {
+        Drawable appIcon = null;
+        try {
+            appIcon = mPm.getActivityIcon(item.mComponentName);
+        } catch (PackageManager.NameNotFoundException e) {
+            if (DEBUG) Log.e(TAG, "resolveAppIcon", e);
+        }
+        if (appIcon == null) {
+            appIcon = getDefaultActivityIcon();
+        }
+        return appIcon;
+    }
+}
+
diff --git a/src/org/omnirom/omnigears/preference/AppSelectListPreference.java b/src/org/omnirom/omnigears/preference/AppSelectListPreference.java
index 7fdd73e..2110725 100644
--- a/src/org/omnirom/omnigears/preference/AppSelectListPreference.java
+++ b/src/org/omnirom/omnigears/preference/AppSelectListPreference.java
@@ -217,7 +217,7 @@
 
     private void init() {
         mPm = getContext().getPackageManager();
-        setDialogLayoutResource(R.layout.preference_dialog_applist);
+        setDialogLayoutResource(R.layout.preference_app_list);
         setLayoutResource(R.layout.preference_app_select);
         setNegativeButtonText(android.R.string.cancel);
         setPositiveButtonText(null);
@@ -242,7 +242,7 @@
     protected void onBindDialogView(View view) {
         super.onBindDialogView(view);
 
-        final ListView list = (ListView) view.findViewById(R.id.applist);
+        final ListView list = (ListView) view.findViewById(R.id.app_list);
         list.setAdapter(mAdapter);
         list.setOnItemClickListener(new OnItemClickListener() {
             @Override
diff --git a/src/org/omnirom/omnigears/preference/ScrollAppsViewPreference.java b/src/org/omnirom/omnigears/preference/ScrollAppsViewPreference.java
new file mode 100644
index 0000000..4d42ecc
--- /dev/null
+++ b/src/org/omnirom/omnigears/preference/ScrollAppsViewPreference.java
@@ -0,0 +1,92 @@
+/*
+ * Copyright (C) 2017 The OmniROM Project
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+package org.omnirom.omnigears.preference;
+
+import android.content.ComponentName;
+import android.content.Context;
+import android.content.pm.PackageManager;
+import android.graphics.drawable.Drawable;
+import android.support.v7.preference.Preference;
+import android.support.v7.preference.PreferenceViewHolder;
+import android.util.AttributeSet;
+import android.util.Log;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.ImageView;
+import android.widget.LinearLayout;
+
+import com.android.settings.R;
+
+import java.util.HashSet;
+import java.util.Set;
+
+public class ScrollAppsViewPreference extends Preference {
+    private static final String TAG = "ScrollAppsPreference";
+
+    private Context mContext;
+    private Set<String> mValues = new HashSet<String>();
+    private PackageManager mPm;
+    private LayoutInflater mInflater;
+
+    public ScrollAppsViewPreference(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+        super(context, attrs, defStyleAttr, defStyleRes);
+        initPreference(context);
+    }
+
+    public ScrollAppsViewPreference(Context context, AttributeSet attrs, int defStyleAttr) {
+        super(context, attrs, defStyleAttr);
+        initPreference(context);
+    }
+
+    public ScrollAppsViewPreference(Context context, AttributeSet attrs) {
+        super(context, attrs);
+        initPreference(context);
+    }
+
+    public void setValues(Set<String> values) {
+        mValues.clear();
+        mValues.addAll(values);
+    }
+
+    @Override
+    public void onBindViewHolder(PreferenceViewHolder holder) {
+        super.onBindViewHolder(holder);
+        LinearLayout linearLayout = (LinearLayout) holder.findViewById(R.id.selected_apps);
+        if (linearLayout.getChildCount() > 0) linearLayout.removeAllViews();
+
+        for (String value : mValues) {
+            try {
+                View v = mInflater.inflate(R.layout.app_grid_item, null);
+                ComponentName componentName = ComponentName.unflattenFromString(value);
+                Drawable icon = mPm.getActivityIcon(componentName);
+                ((ImageView) v.findViewById(R.id.appIcon)).setImageDrawable(icon);
+                v.setPadding(10, 5, 10, 5);
+                linearLayout.addView(v);
+            } catch (PackageManager.NameNotFoundException e) {
+                Log.e(TAG, "Set app icon", e);
+            }
+        }
+    }
+
+    private void initPreference(Context context) {
+        mContext = context;
+        setLayoutResource(R.layout.preference_selected_apps_view);
+        mPm = context.getPackageManager();
+        mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
+    }
+}
diff --git a/src/org/omnirom/omnigears/service/EventService.java b/src/org/omnirom/omnigears/service/EventService.java
index 2756e15..8bcb452 100644
--- a/src/org/omnirom/omnigears/service/EventService.java
+++ b/src/org/omnirom/omnigears/service/EventService.java
@@ -14,132 +14,135 @@
  * You should have received a copy of the GNU General Public License
  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
  *
-*/
+ */
 package org.omnirom.omnigears.service;
 
-import android.app.ActivityManagerNative;
 import android.app.Service;
-import android.bluetooth.BluetoothAdapter;
 import android.bluetooth.BluetoothA2dp;
+import android.bluetooth.BluetoothAdapter;
 import android.bluetooth.BluetoothDevice;
 import android.bluetooth.BluetoothProfile;
 import android.content.BroadcastReceiver;
-import android.content.ComponentName;
 import android.content.Context;
 import android.content.Intent;
 import android.content.IntentFilter;
 import android.content.SharedPreferences;
-import android.media.IAudioService;
 import android.media.AudioManager;
-import android.media.session.MediaSessionLegacyHelper;
-import android.os.Handler;
+import android.media.AudioSystem;
 import android.os.Binder;
 import android.os.IBinder;
 import android.os.PowerManager;
 import android.os.UserHandle;
-import android.os.ServiceManager;
-import android.os.SystemClock;
-import android.text.TextUtils;
 import android.util.Log;
-import android.view.KeyEvent;
 
-import org.omnirom.omnigears.preference.AppSelectListPreference;
+import org.omnirom.omnigears.ui.MultiAppSelectorActivity;
+import org.omnirom.omnigears.utils.SetStringPackUtils;
 
-import java.util.ArrayList;
-import java.util.List;
+import java.util.Set;
+
 
 public class EventService extends Service {
     private static final String TAG = "OmniEventService";
-    private static final boolean DEBUG = true;
+    private static final boolean DEBUG = false;
     private PowerManager.WakeLock mWakeLock;
     private static boolean mIsRunning;
-    private Handler mHandler = new Handler();
-    private boolean mWiredHeadsetConnected;
-    private boolean mA2DPConnected;
+    private static boolean mWiredHeadsetConnected;
+    private static boolean mA2DPConnected;
 
     private BroadcastReceiver mStateListener = new BroadcastReceiver() {
         @Override
         public void onReceive(Context context, Intent intent) {
             String action = intent.getAction();
             mWakeLock.acquire();
+
             try {
                 if (DEBUG) Log.d(TAG, "onReceive " + action);
-                if (BluetoothAdapter.ACTION_STATE_CHANGED.equals(action)) {
-                    if(intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1) == BluetoothAdapter.STATE_OFF){
-                        mA2DPConnected = false;
-                    }
-                }
-                if (BluetoothDevice.ACTION_ACL_CONNECTED.equals(action)) {
-                }
-                if (BluetoothDevice.ACTION_ACL_DISCONNECTED.equals(action)) {
-                }
-                if (BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED.equals(action)) {
-                    int state = intent.getIntExtra(BluetoothProfile.EXTRA_STATE,
-                            BluetoothProfile.STATE_CONNECTED);
-                    if (state == BluetoothProfile.STATE_CONNECTED && !mA2DPConnected) {
-                        mA2DPConnected = true;
-                        if (DEBUG) Log.d(TAG, "BluetoothProfile.STATE_CONNECTED = true" );
-                        String app = getPrefs(context).getString(EventServiceSettings.EVENT_A2DP_CONNECT, null);
-                        if (!TextUtils.isEmpty(app) && !app.equals(AppSelectListPreference.DISABLED_ENTRY)) {
-                            if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG app = " + app);
-                            try {
-                                context.startActivityAsUser(createIntent(app), UserHandle.CURRENT);
-                                if (getPrefs(context).getBoolean(EventServiceSettings.EVENT_MEDIA_PLAYER_START, false)) {
-                                    mHandler.postDelayed(new Runnable() {
-                                        @Override
-                                        public void run() {
-                                            dispatchMediaKeyToAudioService(KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE);
-                                        }
-                                    }, 1000);
-                                }
-                            } catch (Exception e) {
-                                Log.e(TAG, "BluetoothProfile.STATE_CONNECTED", e);
-                            }
+
+                boolean disableIfMusicActive = getPrefs(context).getBoolean(EventServiceSettings.EVENT_MUSIC_ACTIVE, true);
+
+                switch (action) {
+                    case BluetoothAdapter.ACTION_STATE_CHANGED:
+                        if (intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1) == BluetoothAdapter.STATE_OFF) {
+                            mA2DPConnected = false;
                         }
-                    } else {
-                        mA2DPConnected = false;
-                        if (DEBUG) Log.d(TAG, "BluetoothProfile.STATE_CONNECTED = false" );
-                    }
-                }
-                if (AudioManager.ACTION_HEADSET_PLUG.equals(action)) {
-                    boolean useHeadset = intent.getIntExtra("state", 0) == 1;
-                    if (useHeadset && !mWiredHeadsetConnected) {
-                        mWiredHeadsetConnected = true;
-                        if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG = true" );
-                        String app = getPrefs(context).getString(EventServiceSettings.EVENT_WIRED_HEADSET_CONNECT, null);
-                        if (!TextUtils.isEmpty(app) && !app.equals(AppSelectListPreference.DISABLED_ENTRY)) {
-                            if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG app = " + app);
-                            try {
-                                context.startActivityAsUser(createIntent(app), UserHandle.CURRENT);
-                                if (getPrefs(context).getBoolean(EventServiceSettings.EVENT_MEDIA_PLAYER_START, false)) {
-                                    mHandler.postDelayed(new Runnable() {
-                                        @Override
-                                        public void run() {
-                                            dispatchMediaKeyToAudioService(KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE);
-                                        }
-                                    }, 1000);
+                        break;
+                    case BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED:
+                        int state = intent.getIntExtra(BluetoothProfile.EXTRA_STATE,
+                                BluetoothProfile.STATE_CONNECTED);
+                        if (state == BluetoothProfile.STATE_CONNECTED && !mA2DPConnected) {
+                            mA2DPConnected = true;
+                            if (DEBUG) Log.d(TAG, "BluetoothProfile.STATE_CONNECTED = true");
+
+                            if (!(disableIfMusicActive && isMusicActive())) {
+                                Set<String> apps = getPrefs(context).getStringSet(EventServiceSettings.EVENT_A2DP_CONNECT, null);
+                                if (apps != null) {
+                                    openMultiAppSelector(apps, context);
                                 }
-                            } catch (Exception e) {
-                                Log.e(TAG, "AudioManager.ACTION_HEADSET_PLUG", e);
                             }
+                        } else {
+                            mA2DPConnected = false;
+                            if (DEBUG) Log.d(TAG, "BluetoothProfile.STATE_CONNECTED = false");
                         }
-                    } else {
-                        mWiredHeadsetConnected = false;
-                        if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG = false" );
-                    }
+                        break;
+                    case AudioManager.ACTION_HEADSET_PLUG:
+                        boolean useHeadset = intent.getIntExtra("state", 0) == 1;
+                        if (useHeadset && !mWiredHeadsetConnected) {
+                            mWiredHeadsetConnected = true;
+                            if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG = true");
+
+                            if (!(disableIfMusicActive && isMusicActive())) {
+                                Set<String> apps = getPrefs(context).getStringSet(EventServiceSettings.EVENT_WIRED_HEADSET_CONNECT, null);
+                                if (apps != null) {
+                                    openMultiAppSelector(apps, context);
+                                }
+                            }
+                        } else {
+                            mWiredHeadsetConnected = false;
+                            if (DEBUG) Log.d(TAG, "AudioManager.ACTION_HEADSET_PLUG = false");
+                        }
+                        break;
                 }
-                
+
             } finally {
                 mWakeLock.release();
             }
         }
     };
 
+    private boolean isMusicActive() {
+        if (AudioSystem.isStreamActive(AudioSystem.STREAM_MUSIC, 0)) {
+            // local / wired / BT playback active
+            if (DEBUG) Log.d(TAG, "isMusicActive(): local");
+            return true;
+        }
+        if (AudioSystem.isStreamActiveRemotely(AudioSystem.STREAM_MUSIC, 0)) {
+            // remote submix playback active
+            if (DEBUG) Log.d(TAG, "isMusicActive(): remote submix");
+            return true;
+        }
+        if (DEBUG) Log.d(TAG, "isMusicActive(): no");
+        return false;
+    }
+
+    private void openMultiAppSelector(Set<String> apps, Context context) {
+        Intent intent = new Intent(context, MultiAppSelectorActivity.class);
+        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
+                | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED
+                | Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
+        intent.putExtra(MultiAppSelectorActivity.APPS, SetStringPackUtils.packSet(apps));
+        intent.putExtra(MultiAppSelectorActivity.AUTORUN_SINGLE,
+                getPrefs(context).getBoolean(EventServiceSettings.EVENT_AUTORUN_SINGLE, true));
+        intent.putExtra(MultiAppSelectorActivity.MEDIA_PLAYER_START,
+                getPrefs(context).getBoolean(EventServiceSettings.EVENT_MEDIA_PLAYER_START, false));
+        startActivityAsUser(intent, UserHandle.CURRENT);
+    }
+
     public class LocalBinder extends Binder {
         public EventService getService() {
             return EventService.this;
         }
     }
+
     private final LocalBinder mBinder = new LocalBinder();
 
     @Override
@@ -192,31 +195,4 @@
     private SharedPreferences getPrefs(Context context) {
         return context.getSharedPreferences(EventServiceSettings.EVENTS_PREFERENCES_NAME, Context.MODE_PRIVATE);
     }
-
-    private Intent createIntent(String value) {
-        ComponentName componentName = ComponentName.unflattenFromString(value);
-        Intent intent = new Intent(Intent.ACTION_MAIN);
-        intent.addCategory(Intent.CATEGORY_LAUNCHER);
-        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
-                | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
-        intent.setComponent(componentName);
-        return intent;
-    }
-
-    private void dispatchMediaKeyToAudioService(int keycode) {
-        if (ActivityManagerNative.isSystemReady()) {
-            IAudioService audioService = IAudioService.Stub
-                    .asInterface(ServiceManager.checkService(Context.AUDIO_SERVICE));
-            if (audioService != null) {
-                if (DEBUG) Log.d(TAG, "dispatchMediaKeyToAudioService " + keycode);
-
-                KeyEvent event = new KeyEvent(SystemClock.uptimeMillis(),
-                        SystemClock.uptimeMillis(), KeyEvent.ACTION_DOWN,
-                        keycode, 0);
-                MediaSessionLegacyHelper.getHelper(this).sendMediaButtonEvent(event, true);
-                event = KeyEvent.changeAction(event, KeyEvent.ACTION_UP);
-                MediaSessionLegacyHelper.getHelper(this).sendMediaButtonEvent(event, true);
-            }
-        }
-    }
 }
diff --git a/src/org/omnirom/omnigears/service/EventServiceSettings.java b/src/org/omnirom/omnigears/service/EventServiceSettings.java
index d29b9ce..e146278 100644
--- a/src/org/omnirom/omnigears/service/EventServiceSettings.java
+++ b/src/org/omnirom/omnigears/service/EventServiceSettings.java
@@ -14,7 +14,7 @@
  * You should have received a copy of the GNU General Public License
  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
  *
-*/
+ */
 
 package org.omnirom.omnigears.service;
 
@@ -23,43 +23,50 @@
 import android.content.SharedPreferences;
 import android.os.Bundle;
 import android.os.Handler;
-import android.support.v7.preference.ListPreference;
-import android.support.v7.preference.Preference;
-import android.support.v7.preference.PreferenceCategory;
-import android.support.v7.preference.PreferenceScreen;
-import android.support.v7.preference.Preference.OnPreferenceChangeListener;
-import android.support.v14.preference.SwitchPreference;
-import android.provider.Settings;
 import android.provider.SearchIndexableResource;
-import android.util.Log;
+import android.support.v14.preference.SwitchPreference;
+import android.support.v7.preference.Preference;
+import android.support.v7.preference.Preference.OnPreferenceChangeListener;
 
-import com.android.settings.R;
-import com.android.settings.SettingsActivity;
-import com.android.settings.SettingsPreferenceFragment;
-import com.android.settings.Utils;
-import com.android.internal.logging.MetricsLogger;
 import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
+import com.android.settings.R;
+import com.android.settings.SettingsPreferenceFragment;
 import com.android.settings.search.BaseSearchIndexProvider;
 import com.android.settings.search.Indexable;
 
-import org.omnirom.omnigears.preference.AppSelectListPreference;
+import org.omnirom.omnigears.preference.AppMultiSelectListPreference;
+import org.omnirom.omnigears.preference.ScrollAppsViewPreference;
 
-import java.util.List;
 import java.util.ArrayList;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
 
 public class EventServiceSettings extends SettingsPreferenceFragment implements OnPreferenceChangeListener, Indexable {
-    private static final String TAG = "EventServiceSettings";
     public static final String EVENTS_PREFERENCES_NAME = "event_service";
 
-    public static final String EVENT_A2DP_CONNECT = "bt_a2dp_connect_app";
-    public static final String EVENT_WIRED_HEADSET_CONNECT = "headset_connect_app";
+    public static final String EVENT_A2DP_CONNECT = "bt_a2dp_connect_app_list";
+    public static final String EVENT_WIRED_HEADSET_CONNECT = "headset_connect_app_list";
     public static final String EVENT_SERVICE_ENABLED = "event_service_enabled";
     public static final String EVENT_MEDIA_PLAYER_START = "media_player_autostart";
+    public static final String EVENT_MUSIC_ACTIVE = "media_player_music_active";
+    public static final String EVENT_AUTORUN_SINGLE = "autorun_single_app";
+    public static final String A2DP_APP_LIST = "a2dp_app_list";
+    public static final String HEADSET_APP_LIST = "headset_app_list";
 
-    private AppSelectListPreference mA2DPappSelect;
-    private AppSelectListPreference mWiredHeadsetAppSelect;
+    // -- For backward compatibility
+    public static final String OLD_EVENT_A2DP_CONNECT = "bt_a2dp_connect_app";
+    public static final String OLD_EVENT_WIRED_HEADSET_CONNECT = "headset_connect_app";
+    // -- End backward compatibility
+
+    private AppMultiSelectListPreference mA2DPappSelect;
+    private AppMultiSelectListPreference mWiredHeadsetAppSelect;
+    private ScrollAppsViewPreference mA2DPApps;
+    private ScrollAppsViewPreference mHeadsetApps;
     private SwitchPreference mEnable;
     private SwitchPreference mAutoStart;
+    private SwitchPreference mMusicActive;
+    private SwitchPreference mAutorun;
     private Handler mHandler = new Handler();
     private String mServiceRunning;
     private String mServiceStopped;
@@ -78,6 +85,17 @@
         super.onCreate(savedInstanceState);
         addPreferencesFromResource(R.xml.event_service_settings);
 
+        // -- For backward compatibility
+        String old_value = getPrefs().getString(OLD_EVENT_A2DP_CONNECT, null);
+        if (old_value != null) {
+            fixOldPreference(OLD_EVENT_A2DP_CONNECT, EVENT_A2DP_CONNECT, old_value);
+        }
+        old_value = getPrefs().getString(OLD_EVENT_WIRED_HEADSET_CONNECT, null);
+        if (old_value != null) {
+            fixOldPreference(OLD_EVENT_WIRED_HEADSET_CONNECT, EVENT_WIRED_HEADSET_CONNECT, old_value);
+        }
+        // -- End backward compatibility
+
         mEnable = (SwitchPreference) findPreference(EVENT_SERVICE_ENABLED);
         mEnable.setChecked(getPrefs().getBoolean(EventServiceSettings.EVENT_SERVICE_ENABLED, false));
         mEnable.setOnPreferenceChangeListener(this);
@@ -89,29 +107,73 @@
         mAutoStart.setChecked(getPrefs().getBoolean(EventServiceSettings.EVENT_MEDIA_PLAYER_START, false));
         mAutoStart.setOnPreferenceChangeListener(this);
 
-        mA2DPappSelect = (AppSelectListPreference) findPreference(EVENT_A2DP_CONNECT);
-        mEnable.setChecked(getPrefs().getBoolean(EventServiceSettings.EVENT_SERVICE_ENABLED, false));
-        String value = getPrefs().getString(EVENT_A2DP_CONNECT, null);
-        mA2DPappSelect.setValue(value);
+        mMusicActive = (SwitchPreference) findPreference(EVENT_MUSIC_ACTIVE);
+        mMusicActive.setChecked(getPrefs().getBoolean(EventServiceSettings.EVENT_MUSIC_ACTIVE, false));
+        mMusicActive.setOnPreferenceChangeListener(this);
+
+        mAutorun = (SwitchPreference) findPreference(EVENT_AUTORUN_SINGLE);
+        mAutorun.setChecked(getPrefs().getBoolean(EventServiceSettings.EVENT_AUTORUN_SINGLE, true));
+        mAutorun.setOnPreferenceChangeListener(this);
+
+        mA2DPappSelect = (AppMultiSelectListPreference) findPreference(EVENT_A2DP_CONNECT);
+        Set<String> value = getPrefs().getStringSet(EVENT_A2DP_CONNECT, null);
+        if (value != null) mA2DPappSelect.setValues(value);
         mA2DPappSelect.setOnPreferenceChangeListener(this);
 
-        mWiredHeadsetAppSelect = (AppSelectListPreference) findPreference(EVENT_WIRED_HEADSET_CONNECT);
-        value = getPrefs().getString(EVENT_WIRED_HEADSET_CONNECT, null);
-        mWiredHeadsetAppSelect.setValue(value);
+        mA2DPApps = (ScrollAppsViewPreference) findPreference(A2DP_APP_LIST);
+        if (value == null) {
+            mA2DPApps.setVisible(false);
+        } else {
+            mA2DPApps.setVisible(true);
+            mA2DPApps.setValues(value);
+        }
+
+        mWiredHeadsetAppSelect = (AppMultiSelectListPreference) findPreference(EVENT_WIRED_HEADSET_CONNECT);
+        value = getPrefs().getStringSet(EVENT_WIRED_HEADSET_CONNECT, null);
+        if (value != null) mWiredHeadsetAppSelect.setValues(value);
         mWiredHeadsetAppSelect.setOnPreferenceChangeListener(this);
+
+        mHeadsetApps = (ScrollAppsViewPreference) findPreference(HEADSET_APP_LIST);
+        if (value == null) {
+            mHeadsetApps.setVisible(false);
+        } else {
+            mHeadsetApps.setValues(value);
+            mHeadsetApps.setVisible(true);
+        }
+    }
+
+    private void fixOldPreference(String old_event, String new_event, String value) {
+        Set<String> mValues = new HashSet<String>();
+        mValues.add(value);
+        getPrefs().edit().putStringSet(new_event, mValues).commit();
+        getPrefs().edit().putString(old_event, null).commit();
     }
 
     @Override
     public boolean onPreferenceChange(Preference preference, Object newValue) {
         if (preference == mA2DPappSelect) {
-            String value = (String) newValue;
-            boolean appDisabled = value.equals(AppSelectListPreference.DISABLED_ENTRY);
-            getPrefs().edit().putString(EVENT_A2DP_CONNECT, appDisabled ? null : value).commit();
+            Set<String> value = (Set<String>) newValue;
+
+            getPrefs().edit().putStringSet(EVENT_A2DP_CONNECT, value).commit();
+
+            mA2DPApps.setVisible(false);
+            if (value != null) {
+                mA2DPApps.setValues(value);
+                mA2DPApps.setVisible(true);
+            }
+
             return true;
         } else if (preference == mWiredHeadsetAppSelect) {
-            String value = (String) newValue;
-            boolean appDisabled = value.equals(AppSelectListPreference.DISABLED_ENTRY);
-            getPrefs().edit().putString(EVENT_WIRED_HEADSET_CONNECT, appDisabled ? null : value).commit();
+            Set<String> value = (Set<String>) newValue;
+
+            getPrefs().edit().putStringSet(EVENT_WIRED_HEADSET_CONNECT, value).commit();
+
+            mHeadsetApps.setVisible(false);
+            if (value != null) {
+                mHeadsetApps.setValues(value);
+                mHeadsetApps.setVisible(true);
+            }
+
             return true;
         } else if (preference == mEnable) {
             boolean value = ((Boolean) newValue).booleanValue();
@@ -135,6 +197,14 @@
             boolean value = ((Boolean) newValue).booleanValue();
             getPrefs().edit().putBoolean(EVENT_MEDIA_PLAYER_START, value).commit();
             return true;
+        } else if (preference == mMusicActive) {
+            boolean value = ((Boolean) newValue).booleanValue();
+            getPrefs().edit().putBoolean(EVENT_MUSIC_ACTIVE, value).commit();
+            return true;
+        } else if (preference == mAutorun) {
+            boolean value = ((Boolean) newValue).booleanValue();
+            getPrefs().edit().putBoolean(EVENT_AUTORUN_SINGLE, value).commit();
+            return true;
         }
         return false;
     }
@@ -147,7 +217,7 @@
             new BaseSearchIndexProvider() {
                 @Override
                 public List<SearchIndexableResource> getXmlResourcesToIndex(Context context,
-                        boolean enabled) {
+                                                                            boolean enabled) {
                     ArrayList<SearchIndexableResource> result =
                             new ArrayList<SearchIndexableResource>();
 
@@ -163,5 +233,5 @@
                     ArrayList<String> result = new ArrayList<String>();
                     return result;
                 }
-    };
+            };
 }
diff --git a/src/org/omnirom/omnigears/ui/AppGridAdapter.java b/src/org/omnirom/omnigears/ui/AppGridAdapter.java
new file mode 100644
index 0000000..4dddb4e
--- /dev/null
+++ b/src/org/omnirom/omnigears/ui/AppGridAdapter.java
@@ -0,0 +1,89 @@
+/*
+ * Copyright (C) 2017 The OmniROM Project
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+package org.omnirom.omnigears.ui;
+
+import android.content.ComponentName;
+import android.content.Context;
+import android.content.pm.PackageManager;
+import android.graphics.drawable.Drawable;
+import android.util.Log;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.BaseAdapter;
+import android.widget.ImageView;
+
+import org.omnirom.omnigears.R;
+
+public class AppGridAdapter extends BaseAdapter {
+    private static final String TAG = "AppGridAdapter";
+
+    private LayoutInflater layoutinflater;
+    private Object[] appList;
+    private PackageManager mPm;
+
+    public AppGridAdapter(Context context, Object[] customizedListView) {
+        mPm = context.getPackageManager();
+        layoutinflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
+        appList = customizedListView;
+    }
+
+    @Override
+    public int getCount() {
+        return appList.length;
+    }
+
+    @Override
+    public Object getItem(int position) {
+        return appList[position];
+    }
+
+    @Override
+    public long getItemId(int position) {
+        return (long) position;
+    }
+
+    @Override
+    public View getView(int position, View convertView, ViewGroup parent) {
+
+        ViewHolder listViewHolder;
+
+        if (convertView == null) {
+            listViewHolder = new ViewHolder();
+            convertView = layoutinflater.inflate(R.layout.app_grid_item, parent, false);
+            listViewHolder.imageInListView = (ImageView) convertView.findViewById(R.id.appIcon);
+            convertView.setTag(listViewHolder);
+        } else {
+            listViewHolder = (ViewHolder) convertView.getTag();
+        }
+
+        try {
+            ComponentName componentName = ComponentName.unflattenFromString((String) appList[position]);
+            Drawable icon = mPm.getActivityIcon(componentName);
+            listViewHolder.imageInListView.setImageDrawable(icon);
+        } catch (PackageManager.NameNotFoundException e) {
+            Log.e(TAG, "Set app icon", e);
+        }
+
+        return convertView;
+    }
+
+    static class ViewHolder {
+        ImageView imageInListView;
+    }
+}
diff --git a/src/org/omnirom/omnigears/ui/MultiAppSelectorActivity.java b/src/org/omnirom/omnigears/ui/MultiAppSelectorActivity.java
new file mode 100644
index 0000000..1e5b9c6
--- /dev/null
+++ b/src/org/omnirom/omnigears/ui/MultiAppSelectorActivity.java
@@ -0,0 +1,120 @@
+package org.omnirom.omnigears.ui;
+
+
+import android.app.Activity;
+import android.app.ActivityManagerNative;
+import android.content.ComponentName;
+import android.content.Context;
+import android.content.Intent;
+import android.content.res.Configuration;
+import android.media.IAudioService;
+import android.media.session.MediaSessionLegacyHelper;
+import android.os.Bundle;
+import android.os.Handler;
+import android.os.ServiceManager;
+import android.os.SystemClock;
+import android.os.UserHandle;
+import android.util.Log;
+import android.view.KeyEvent;
+import android.view.View;
+import android.view.Window;
+import android.widget.AdapterView;
+import android.widget.GridView;
+
+import org.omnirom.omnigears.R;
+import org.omnirom.omnigears.utils.SetStringPackUtils;
+
+public class MultiAppSelectorActivity extends Activity {
+    private static final String TAG = "MultiAppSelector";
+    private static final boolean DEBUG = false;
+
+    public static final String APPS = "apps";
+    public static final String MEDIA_PLAYER_START = "media_player_start";
+    public static final String AUTORUN_SINGLE = "autorun_single";
+
+    private Handler mHandler = new Handler();
+    private Boolean mediaStart = false;
+    private Boolean autoRun = false;
+    private Object[] appList = null;
+    private GridView gridview;
+
+
+    @Override
+    public void onCreate(Bundle savedInstanceState) {
+        super.onCreate(savedInstanceState);
+        requestWindowFeature(Window.FEATURE_NO_TITLE);
+        setContentView(R.layout.app_grid_view);
+        Intent intent = getIntent();
+
+        mediaStart = intent.getBooleanExtra(MEDIA_PLAYER_START, false);
+        autoRun = intent.getBooleanExtra(AUTORUN_SINGLE, false);
+        appList = SetStringPackUtils.unpackString(intent.getStringExtra(APPS)).toArray();
+
+        gridview = (GridView) findViewById(R.id.app_grid_view);
+        gridview.setAdapter(new AppGridAdapter(MultiAppSelectorActivity.this, appList));
+        gridview.setOnItemClickListener(new AdapterView.OnItemClickListener() {
+            @Override
+            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
+                openApp((String) appList[position]);
+            }
+        });
+
+        if (autoRun && appList.length == 1) { // If there is only one app open it
+            openApp((String) appList[0]);
+        }
+    }
+
+    private void openApp(String app_uri) {
+        try {
+            startActivityAsUser(createIntent(app_uri), UserHandle.CURRENT);
+            if (mediaStart) {
+                mHandler.postDelayed(new Runnable() {
+                    @Override
+                    public void run() {
+                        dispatchMediaKeyToAudioService(KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE);
+                    }
+                }, 1000);
+            }
+        } catch (Exception e) {
+            Log.e(TAG, "MultiAppSelector.EVENT_MEDIA_PLAYER_START", e);
+        }
+    }
+
+    private Intent createIntent(String value) {
+        ComponentName componentName = ComponentName.unflattenFromString(value);
+        Intent intent = new Intent(Intent.ACTION_MAIN);
+        intent.addCategory(Intent.CATEGORY_LAUNCHER);
+        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
+                | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
+        intent.setComponent(componentName);
+        return intent;
+    }
+
+    private void dispatchMediaKeyToAudioService(int keycode) {
+        if (ActivityManagerNative.isSystemReady()) {
+            IAudioService audioService = IAudioService.Stub
+                    .asInterface(ServiceManager.checkService(Context.AUDIO_SERVICE));
+            if (audioService != null) {
+                if (DEBUG) Log.d(TAG, "dispatchMediaKeyToAudioService " + keycode);
+
+                KeyEvent event = new KeyEvent(SystemClock.uptimeMillis(),
+                        SystemClock.uptimeMillis(), KeyEvent.ACTION_DOWN,
+                        keycode, 0);
+                MediaSessionLegacyHelper.getHelper(this).sendMediaButtonEvent(event, true);
+                event = KeyEvent.changeAction(event, KeyEvent.ACTION_UP);
+                MediaSessionLegacyHelper.getHelper(this).sendMediaButtonEvent(event, true);
+            }
+        }
+    }
+
+    @Override
+    public void onConfigurationChanged(Configuration newConfig) {
+        super.onConfigurationChanged(newConfig);
+    }
+
+    @Override
+    protected void onStop() {
+        super.onStop();
+        finish();
+    }
+}
diff --git a/src/org/omnirom/omnigears/utils/SetStringPackUtils.java b/src/org/omnirom/omnigears/utils/SetStringPackUtils.java
new file mode 100644
index 0000000..758c97e
--- /dev/null
+++ b/src/org/omnirom/omnigears/utils/SetStringPackUtils.java
@@ -0,0 +1,34 @@
+/*
+ * Copyright (C) 2017 The OmniROM Project
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+package org.omnirom.omnigears.utils;
+
+import android.text.TextUtils;
+
+import java.util.Arrays;
+import java.util.HashSet;
+import java.util.Set;
+
+public class SetStringPackUtils {
+    public static Set<String> unpackString(String packed) {
+        return new HashSet<String>(Arrays.asList(packed.split("\\|")));
+    }
+
+    public static String packSet(Set<String> values) {
+        return TextUtils.join("|", values);
+    }
+}