Откройте другое приложение из своего собственного (намерение)


136

Я знаю, как обновлять свои собственные программы, и я знаю, как открывать программы, используя предопределенный URI (например, для sms или электронной почты)

Мне нужно знать, как создать намерение для открытия MyTracks или любого другого приложения, которое я не знаю, какие намерения они слушают.

Я получил эту информацию из DDMS, но мне не удалось превратить это в намерение, которое я могу использовать. Это происходит при открытии MyTracks вручную.

Спасибо за вашу помощь

05-06 11:22:24.945: INFO/ActivityManager(76): Starting activity: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.google.android.maps.mytracks/com.google.android.apps.mytracks.MyTracks bnds=[243,338][317,417] }
05-06 11:22:25.005: INFO/ActivityManager(76): Start proc com.google.android.maps.mytracks for activity com.google.android.maps.mytracks/com.google.android.apps.mytracks.MyTracks: pid=1176 uid=10063 gids={3003, 1015}
05-06 11:22:26.995: INFO/ActivityManager(76): Displayed activity com.google.android.maps.mytracks/com.google.android.apps.mytracks.MyTracks: 1996 ms (total 1996 ms)

Ответы:


141

Во-первых, понятие «приложение» в Android несколько расширено.

Приложение - технически процесс - может иметь несколько действий, служб, поставщиков контента и / или слушателей вещания. Если хотя бы один из них запущен, приложение запущено и работает (процесс).

Итак, вам нужно определить, как вы хотите «запустить приложение».

Хорошо ... вот что вы можете попробовать:

  1. Создайте намерение с помощью action=MAINиcategory=LAUNCHER
  2. Получить PackageManagerиз текущего контекста, используяcontext.getPackageManager
  3. packageManager.queryIntentActivity(<intent>, 0)где имеется намерение category=LAUNCHER, action=MAINилиpackageManager.resolveActivity(<intent>, 0) получить первое действие с помощью main / launcher
  4. Получите то, что ActivityInfoвам интересно
  5. От ActivityInfoполучитьpackageName иname
  6. И, наконец, создать еще один замысел с с category=LAUNCHER, action=MAIN,componentName = new ComponentName(packageName, name) иsetFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
  7. В заключение, context.startActivity(newIntent)

Что, если бы я хотел открыть три отдельных приложения?
Si8

7
Ответ сообщества вики ниже лучше, если вы знаете имя пакета stackoverflow.com/a/7596063/379115
Мартин Белчер - AtWrk

Как можно передавать данные между вызывающим приложением и запущенным приложением? Я нашел Intent.putExtra (), но не знаю, как получить дополнительные данные в запущенном приложении.
Bram

onCreate=> Bundle extras = getIntent().getExtras()=> и if(extras != null) { extras.getString("blah") }т. д.
Гаурав Вайш

2
getPackageManager().getLaunchIntentForPackage()уже делает все это за вас github.com/android/platform_frameworks_base/blob/master/core/…
jrub 06

233

Я так работаю,

/** Open another app.
 * @param context current Context, like Activity, App, or Service
 * @param packageName the full package name of the app to open
 * @return true if likely successful, false if unsuccessful
 */
public static boolean openApp(Context context, String packageName) {
    PackageManager manager = context.getPackageManager();
    try {
        Intent i = manager.getLaunchIntentForPackage(packageName);
        if (i == null) {
            return false;
            //throw new ActivityNotFoundException();
        }
        i.addCategory(Intent.CATEGORY_LAUNCHER);
        context.startActivity(i);
        return true;
    } catch (ActivityNotFoundException e) {
        return false;
    }
}

Пример использования:

openApp(this, "com.google.android.maps.mytracks");

Надеюсь, это кому-то поможет.


5
потому что getLaunchIntentForPackage («имя пакета приложения») может вызвать исключение.
xtr

Это отличный ответ, просто перехватите исключение и сделайте то, что вам нужно, уведомите пользователя и т. Д.
IT-Dan

5
Не быть настойчивым, но нет причин выделять новое намерение в первой строке, если вы используете результат getLaunchIntentForPackageвызова.
Крис Лейси

2
getLaunchIntentForPackage()уже добавляет категорию, см. источник: github.com/android/platform_frameworks_base/blob/master/core/…
jrub

Единственная проблема для меня - это Contextобъект, где / как его получить / создать? И какие пакеты импортировать. Новое в Java для разработки под Android.
Алексей Концевич

96
    Intent intent = new Intent(Intent.ACTION_MAIN);
    intent.setComponent(ComponentName.unflattenFromString("com.google.android.maps.mytracks/com.google.android.apps.mytracks.MyTracks"));
    intent.addCategory(Intent.CATEGORY_LAUNCHER);
    startActivity(intent);

РЕДАКТИРОВАТЬ :

как предлагается в комментариях, добавьте одну строку перед startActivity(intent);

intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

18
но для этого необходимо знать название активности
njzk2

Я попробовал это и получил ошибку с рекомендацией использовать флаг FLAG_ACTIVITY_NEW_TASK. Я добавил эту строку перед startActivity, и она сработала: intent.setFlags (Intent.FLAG_ACTIVITY_NEW_TASK);
david-hoze

1
@ njzk2 найти имя пакета для чего-либо в Google Play легко; это прямо в URL. Например: play.google.com/store/apps/…
Да-Джин

@iceybobby: да, имя пакета найти легко, но как узнать имя класса активности, которую нужно запустить?
phreakhead

@phreakhead Вы правы. Я думаю, что использовал решение в этом ответе: stackoverflow.com/a/8944286/1224186, поэтому имя действия не нужно, и поэтому я думаю, что мой ответ на njzk2 здесь бесполезен.
Da-Jin

39

Если у вас уже есть имя пакета, который вы хотите активировать, вы можете использовать следующий код, который является немного более общим:

PackageManager pm = context.getPackageManager();
Intent appStartIntent = pm.getLaunchIntentForPackage(appPackageName);
if (null != appStartIntent)
{
    context.startActivity(appStartIntent);
}

Я обнаружил, что это лучше работает в тех случаях, когда основное действие не было обнаружено обычным методом запуска ОСНОВНОГО действия.


Лучший ответ для меня :-) полезно для меня. Спасибо
Джалпеш Хахи

Это идеально подходит для меня. Большое спасибо Музикант :)
Alex

13

Это код моей базы решения на основе решения MasterGaurav:

private void  launchComponent(String packageName, String name){
    Intent launch_intent = new Intent("android.intent.action.MAIN");
    launch_intent.addCategory("android.intent.category.LAUNCHER");
    launch_intent.setComponent(new ComponentName(packageName, name));
    launch_intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

    activity.startActivity(launch_intent);
}

public void startApplication(String application_name){
    try{
        Intent intent = new Intent("android.intent.action.MAIN");
        intent.addCategory("android.intent.category.LAUNCHER");

        intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
        List<ResolveInfo> resolveinfo_list = activity.getPackageManager().queryIntentActivities(intent, 0);

        for(ResolveInfo info:resolveinfo_list){
            if(info.activityInfo.packageName.equalsIgnoreCase(application_name)){
                launchComponent(info.activityInfo.packageName, info.activityInfo.name);
                break;
            }
        }
    }
    catch (ActivityNotFoundException e) {
        Toast.makeText(activity.getApplicationContext(), "There was a problem loading the application: "+application_name,Toast.LENGTH_SHORT).show();
    }
}

10

Используя решение от inversus, я расширил сниппет функцией, которая будет вызываться, когда нужное приложение в данный момент не установлено. Это работает так: Запускать приложение по имени пакета. Если не нашел, откройте Android Market - Google play для этого пакета .

public void startApplication(String packageName)
{
    try
    {
        Intent intent = new Intent("android.intent.action.MAIN");
        intent.addCategory("android.intent.category.LAUNCHER");

        intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
        List<ResolveInfo> resolveInfoList = getPackageManager().queryIntentActivities(intent, 0);

        for(ResolveInfo info : resolveInfoList)
            if(info.activityInfo.packageName.equalsIgnoreCase(packageName))
            {
                launchComponent(info.activityInfo.packageName, info.activityInfo.name);
                return;
            }

        // No match, so application is not installed
        showInMarket(packageName);
    }
    catch (Exception e) 
    {
        showInMarket(packageName);
    }
}

private void launchComponent(String packageName, String name)
{
    Intent intent = new Intent("android.intent.action.MAIN");
    intent.addCategory("android.intent.category.LAUNCHER");
    intent.setComponent(new ComponentName(packageName, name));
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

    startActivity(intent);
}

private void showInMarket(String packageName)
{
    Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + packageName));
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);
}

И используется он так:

startApplication("org.teepee.bazant");

6

Использовать это :

    PackageManager pm = getPackageManager();
    Intent intent = pm.getLaunchIntentForPackage("com.package.name");
    startActivity(intent);

Есть идеи? откройте test.apk вместо установленного в магазине устройства внутри существующего приложения. подсказывает при нажатии на кнопку обеда test.apk apps, которая была сохранена в существующем проекте. заранее спасибо.
Селим Раза

6

Откройте приложение, если оно существует, или откройте приложение Play Store для его установки:

private void open() {
    openApplication(getActivity(), "com.app.package.here");
}

public void openApplication(Context context, String packageN) {
    Intent i = context.getPackageManager().getLaunchIntentForPackage(packageN);
    if (i != null) {
        i.addCategory(Intent.CATEGORY_LAUNCHER);
        context.startActivity(i);
    } else {
        try {
            context.startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + packageN)));
        }
        catch (android.content.ActivityNotFoundException anfe) {
            context.startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("http://play.google.com/store/apps/details?id=" + packageN)));
        }
    }
}

должно быть if (i! = null)
vallllll

4

Чтобы запустить другое действие приложения из моего действия приложения. У меня все работает нормально.

Приведенный ниже код будет работать, если другое приложение уже установлено на вашем телефоне, в противном случае невозможно перенаправить из одного приложения в другое, поэтому убедитесь, что ваше приложение запущено или нет.

Intent intent = new Intent();
intent.setClassName("com.xyz.myapplication", "com.xyz.myapplication.SplashScreenActivity");
startActivity(intent);

Пожалуйста, не пишите несколько ответов, которые почти идентичны одному и тому же вопросу. Воспользуйтесь ссылкой «изменить» под ответом и измените оригинал.
AdrianHHH

3

// Это работает на Android Lollipop 5.0.2

public static boolean launchApp(Context context, String packageName) {

    final PackageManager manager = context.getPackageManager();
    final Intent appLauncherIntent = new Intent(Intent.ACTION_MAIN);
    appLauncherIntent.addCategory(Intent.CATEGORY_LAUNCHER);

    List<ResolveInfo> resolveInfos = manager.queryIntentActivities(appLauncherIntent, 0);
    if ((null != resolveInfos) && (!resolveInfos.isEmpty())) {
        for (ResolveInfo rInfo : resolveInfos) {
            String className = rInfo.activityInfo.name.trim();
            String targetPackageName = rInfo.activityInfo.packageName.trim();
            Log.d("AppsLauncher", "Class Name = " + className + " Target Package Name = " + targetPackageName + " Package Name = " + packageName);
            if (packageName.trim().equals(targetPackageName)) {
                Intent intent = new Intent();
                intent.setClassName(targetPackageName, className);
                intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                context.startActivity(intent);
                Log.d("AppsLauncher", "Launching Package '" + packageName + "' with Activity '" + className + "'");
                return true;
            }
        }
    }
    return false;
}

2

Поскольку приложениям не разрешено изменять многие настройки телефона, вы можете открыть действие настроек так же, как другое приложение.

Посмотрите на свой вывод LogCat после того, как вы изменили настройки вручную:

INFO/ActivityManager(1306): Starting activity: Intent { act=android.intent.action.MAIN cmp=com.android.settings/.DevelopmentSettings } from pid 1924

Затем используйте это, чтобы отобразить страницу настроек из вашего приложения:

String SettingsPage = "com.android.settings/.DevelopmentSettings";

try
{
Intent intent = new Intent(Intent.ACTION_MAIN);             
intent.setComponent(ComponentName.unflattenFromString(SettingsPage));             
intent.addCategory(Intent.CATEGORY_LAUNCHER );             
startActivity(intent); 
}
catch (ActivityNotFoundException e)
{
 log it
}

2

Для уровня API 3+ не более одной строчки кода:

Intent intent = context.getPackageManager().getLaunchIntentForPackage("name.of.package");

Возвращает намерение запуска CATEGORY_INFO (приложения без активности запуска, например, обои часто используют это для предоставления некоторой информации о приложении) и, если его не находят, возвращает CATEGORY_LAUNCH пакета, если он существует.


2

Если вы пытаетесь запустить СЛУЖБУ, а не действие, это сработало для меня:

Intent intent = new Intent();
intent.setClassName("com.example.otherapplication", "com.example.otherapplication.ServiceName");
context.startService(intent);

Если вы используете метод intent.setComponent (...), как упоминалось в других ответах, вы можете получить предупреждение «Неявные намерения с startService небезопасны».


2

В качестве альтернативы вы также можете открыть намерение из своего приложения в другом приложении с помощью:

Intent intent = new Intent(Intent.ACTION_VIEW, uri);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

где uriссылка на другое приложение


2

Используйте следующее:

String packagename = "com.example.app";
startActivity(getPackageManager().getLaunchIntentForPackage(packagename));

2

Запустить приложение из другого приложения на Android

  Intent launchIntent = getActivity.getPackageManager().getLaunchIntentForPackage("com.ionicframework.uengage");
        startActivity(launchIntent);

2

Если вы хотите открыть другое приложение, но оно не установлено, вы можете отправить его в Google App Store для загрузки.

  1. Сначала создайте метод openOtherApp, например

    public static boolean openOtherApp(Context context, String packageName) {
        PackageManager manager = context.getPackageManager();
         try {
            Intent intent = manager.getLaunchIntentForPackage(packageName);
            if (intent == null) {
                //the app is not installed
                try {
                    intent = new Intent(Intent.ACTION_VIEW);
                    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    intent.setData(Uri.parse("market://details?id=" + packageName));
                    context.startActivity(intent);
                } catch (ActivityNotFoundException e) {
                    //throw new ActivityNotFoundException();
                    return false;
                }
    
             }
             intent.addCategory(Intent.CATEGORY_LAUNCHER);
             context.startActivity(intent);
             return true;
        } catch (ActivityNotFoundException e) {
            return false;
        }
    
    }
    

2.- Использование

openOtherApp(getApplicationContext(), "com.packageappname");

0

Попробуйте этот код, надеюсь, это вам поможет. Если этот пакет доступен, это откроет приложение или откроет игровой магазин для загрузок

    String  packageN = "aman4india.com.pincodedirectory";

            Intent i = getPackageManager().getLaunchIntentForPackage(packageN);
            if (i != null) {
                i.addCategory(Intent.CATEGORY_LAUNCHER);
                startActivity(i);
            } else {
                try {
                    startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + packageN)));
                }
                catch (android.content.ActivityNotFoundException anfe) {
                    startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("http://play.google.com/store/apps/details?id=" + packageN)));
                }
            }


-3
Intent intent = new Intent(Intent.ACTION_MAIN);
        intent.setComponent(new ComponentName("package_name","package_name.class_name"));
        intent.putExtra("grace", "Hi");
        startActivity(intent);
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.