Androidデバイスのメインのメールアドレスを取得する方法


412

Androidのメインのメールアドレス(またはメールアドレスのリスト)を取得するにはどうすればよいですか?

OS 2.0以降では複数の電子メールアドレスがサポートされますが、2.0未満では、デバイスごとに1つの電子メールアドレスしか持てないというのが私の理解です。


連絡先のメールアドレスを取得することについて話していますか?
オースティンマホニー2010

1
いいえ、デバイスのプライマリ電子メールアドレス。
Brandon O'Rourke

Androidデバイスに関連付けられている電子メールアドレスは1つ以上ありますか?それが私が望んでいることです。
Brandon O'Rourke

2
@ BrandonO'Rourke Androidマーケットに関連付けられているものとして「デバイスのメインのメールアドレス」を意味しますか?Androidマーケットに関連付けられているGmail IDと他のメールに違いがあるためです。この質問を
ご覧ください。stackoverflow.com/

回答:


749

以下に示すように、これを行うにはいくつかの方法があります。

わかりやすい警告として、アカウント、プロファイル、および連絡先データを処理するときは、注意して前もってユーザーに伝えてください。ユーザーのメールアドレスやその他の個人情報を悪用すると、悪いことが発生する可能性があります。

方法A:AccountManagerを使用する(APIレベル5以上)

AccountManager.getAccountsまたはAccountManager.getAccountsByTypeを使用して、デバイス上のすべてのアカウント名のリストを取得できます。さいわい、特定の種類のアカウント(などcom.google)では、アカウント名は電子メールアドレスです。以下のスニペットの例。

Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
Account[] accounts = AccountManager.get(context).getAccounts();
for (Account account : accounts) {
    if (emailPattern.matcher(account.name).matches()) {
        String possibleEmail = account.name;
        ...
    }
}

これにはGET_ACCOUNTS許可が必要です。

<uses-permission android:name="android.permission.GET_ACCOUNTS" />

使用の詳細については、SDKのContact Managerサンプルコードを参照AccountManagerしてください

方法B:ContactsContract.Profileを使用する(APIレベル14以上)

Android 4.0(Ice Cream Sandwich)以降では、ユーザーのプロファイルにアクセスしてユーザーのメールアドレスを取得できます。ユーザープロファイルへのアクセスは、2つのアクセス許可(詳細は以下を参照)を必要とするため、少し重いですが、電子メールアドレスはかなり機密性の高いデータであるため、これが入場料になります。

以下は、CursorLoader電子メールアドレスを含むプロファイルデータ行を取得するためにを使用する完全な例です。

public class ExampleActivity extends Activity implements LoaderManager.LoaderCallbacks<Cursor> {
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getLoaderManager().initLoader(0, null, this);
    }

    @Override
    public Loader<Cursor> onCreateLoader(int id, Bundle arguments) {
        return new CursorLoader(this,
                // Retrieve data rows for the device user's 'profile' contact.
                Uri.withAppendedPath(
                        ContactsContract.Profile.CONTENT_URI,
                        ContactsContract.Contacts.Data.CONTENT_DIRECTORY),
                ProfileQuery.PROJECTION,

                // Select only email addresses.
                ContactsContract.Contacts.Data.MIMETYPE + " = ?",
                new String[]{ContactsContract.CommonDataKinds.Email.CONTENT_ITEM_TYPE},

                // Show primary email addresses first. Note that there won't be
                // a primary email address if the user hasn't specified one.
                ContactsContract.Contacts.Data.IS_PRIMARY + " DESC");
    }

    @Override
    public void onLoadFinished(Loader<Cursor> cursorLoader, Cursor cursor) {
        List<String> emails = new ArrayList<String>();
        cursor.moveToFirst();
        while (!cursor.isAfterLast()) {
            emails.add(cursor.getString(ProfileQuery.ADDRESS));
            // Potentially filter on ProfileQuery.IS_PRIMARY
            cursor.moveToNext();
        }

        ...
    }

    @Override
    public void onLoaderReset(Loader<Cursor> cursorLoader) {
    }

    private interface ProfileQuery {
        String[] PROJECTION = {
                ContactsContract.CommonDataKinds.Email.ADDRESS,
                ContactsContract.CommonDataKinds.Email.IS_PRIMARY,
        };

        int ADDRESS = 0;
        int IS_PRIMARY = 1;
    }
}

これには、READ_PROFILEREAD_CONTACTS権限の両方が必要です。

<uses-permission android:name="android.permission.READ_PROFILE" />
<uses-permission android:name="android.permission.READ_CONTACTS" />

3
私のコードを使用して同様の質問があります。私の電話に関連付けられているすべてのGmail IDを取得できますが、メインのIDが必要です。電話と同期するためにメールIDを追加すると、0番目の位置のcom.google idを取得している場合はスタックに入るような解決策が1つ見つかりました。 。これが私のコードですAccount [] accounts = AccountManager.get(this).getAccountsByType( "com.google"); String myEmailid = accounts [0] .toString(); Log.d( "欲しいメールアドレス"、myEmailid); 私はそれが正しい方法ではないことを知っています。
PiyushMishra

59
プロファイル方法にはひどく欠陥があります(私の意見では)。私の電子メールを必要とする/必要とするアプリは、私のすべての連絡先を読みたいと思うアプリと比べて大したことではありませんが、両方に同じ権限が必要になるように作成しました。そのため、ユーザーとしては、自分の電子メールを読むアプリと500以上の連絡先を読むアプリの違いがわかりません。連絡先を乱用するアプリの数が増えているため、これは非常に現実的で実際的な問題です。
トム

3
@Muzikantそれは決して公式声明ではありませんが、変更される可能性がかなり低いものです。そうは言っても、ユーザーの電子メールアドレスにアクセスする「正しい」方法は方法Bです。これはより「公式」であり、いくつかのヘビー級のアクセス許可の背後にあるという事実は、そのようなデータに取り組むべき機密性を示しているはずです。
Roman Nurik

15
これについて@Tomに同意します。電話のすべての連絡先のデータについて、ユーザーの名と姓だけの許可を求めるのはばかげています。
tasomaniac 2013年

3
方法Bは、Android 4.4ですべてのサンプルコードをコピーして動作しません。cursor.isAfterLast()常にtrueを返します。何か案が?
cprcrack 14

55

これは他の人に役立つかもしれません:

AccountPickerを使用して、グローバル権限なしでユーザーの電子メールアドレスを取得し、ユーザーがプロセスを認識して承認またはキャンセルできるようにします。


1
これは非常に役立つ回答です。プライマリメールは通常Googleアカウントを意味するので、これが推奨されるオプションであると思います。これは、Google Play
Alex.F

@ Alex.Fこれはmarshmellow以降のAndroidバージョンで機能しますか?
Eswar 2018年

27

ICSで導入されたAndroidのAccountPickerを使用します。

Intent googlePicker = AccountPicker.newChooseAccountIntent(null, null, new String[]{GoogleAuthUtil.GOOGLE_ACCOUNT_TYPE}, true, null, null, null, null);
startActivityForResult(googlePicker, REQUEST_CODE);

そして結果を待ちます:

protected void onActivityResult(final int requestCode, final int resultCode,
                                final Intent data) {
    if (requestCode == REQUEST_CODE && resultCode == RESULT_OK) {
        String accountName = data.getStringExtra(AccountManager.KEY_ACCOUNT_NAME);
    }
}

2
Playサービスを使用する必要があり、場合によってはユーザーがアカウントを選択する必要があることを示すダイアログが表示されることに注意してください。
Androidデベロッパー

AccountManager.newChooseAccountIntent()を使用しても同じ処理が行われ、play-servicesライブラリは必要ありません。
デニス

これにより、最新のAndroidバージョンのコンテキストでアカウント認証のためのポップアップがユーザーにスローされますか?はいの場合、アカウントが1つしかないデバイスでそれをバイパスするにはどうすればよいですか?
Eswar 2018年

14
public String getUsername() {
    AccountManager manager = AccountManager.get(this);
    Account[] accounts = manager.getAccountsByType("com.google");
    List<String> possibleEmails = new LinkedList<String>();

    for (Account account : accounts) {
        // TODO: Check possibleEmail against an email regex or treat
        // account.name as an email address only for certain account.type values.
        possibleEmails.add(account.name);
    }

    if (!possibleEmails.isEmpty() && possibleEmails.get(0) != null) {
        String email = possibleEmails.get(0);
        String[] parts = email.split("@");

        if (parts.length > 1)
            return parts[0];
    }
    return null;
}


2
ご注意これは必要ですandroid.permission.GET_ACCOUNTS「危険な」権限として定義され、(ランタイムリクエストが必要です):developer.android.com/reference/android/...
SagiLow

@SagiLowどのようにこれを処理しましたか?私はユーザーに別の許可を求めたくありません、彼に彼のメールアドレスを入力するのを怠惰にするだけです:)
codebased

1
@codebasedしませんでした...私の知る限りでは不可能です。
SagiLow 2017年

2
manager.getAccountsByType( "com.google"); それ以降のバージョンのAndroidでは動作しません。
パウダー366

8

ユーザーが許可なしで自分のメールアドレスを選択できるようにするAndroid APIがあります。ご覧くださいhttps : //developers.google.com/identity/smartlock-passwords/android/retrieve-hints

HintRequest hintRequest = new HintRequest.Builder()
        .setHintPickerConfig(new CredentialPickerConfig.Builder()
                .setShowCancelButton(true)
                .build())
        .setEmailAddressIdentifierSupported(true)
        .setAccountTypes(IdentityProviders.GOOGLE)
        .build();

PendingIntent intent = mCredentialsClient.getHintPickerIntent(hintRequest);
try {
    startIntentSenderForResult(intent.getIntentSender(), RC_HINT, null, 0, 0, 0);
} catch (IntentSender.SendIntentException e) {
    Log.e(TAG, "Could not start hint picker Intent", e);
}

これにより、ユーザーがメールアドレスを選択できるピッカーが表示されます。結果は以下で配信されますonActivityResult()


私は私が見つけた、私はこれを使用することができただけで作業の答えに感謝を確認することができます
csga5000

唯一の実用的なソリューション、感謝の言葉
Zulqarnain

7

悲しいことに受け入れられた答えが機能していません。

遅れましたが、コンテンツURIがプロバイダーによって変更されない限り、内部Androidメールアプリケーションのソリューションは次のとおりです。

Uri EMAIL_ACCOUNTS_DATABASE_CONTENT_URI = 
              Uri.parse("content://com.android.email.provider/account");

public ArrayList<String> GET_EMAIL_ADDRESSES ()
{
    ArrayList<String> names = new ArrayList<String>();
    ContentResolver cr      = m_context.getContentResolver();
    Cursor cursor           = cr.query(EMAIL_ACCOUNTS_DATABASE_CONTENT_URI ,null, 
                             null, null, null);

    if (cursor == null) {
        Log.e("TEST", "Cannot access email accounts database");
        return null;
    }

    if (cursor.getCount() <= 0) {
        Log.e("TEST", "No accounts");
        return null;
    }

    while (cursor.moveToNext()) {
        names.add(cursor.getString(cursor.getColumnIndex("emailAddress")));
        Log.i("TEST", cursor.getString(cursor.getColumnIndex("emailAddress")));
    }
    return names;
}


2

この方法を使用します。

 public String getUserEmail() {
    AccountManager manager = AccountManager.get(App.getInstance());
    Account[] accounts = manager.getAccountsByType("com.google");
    List<String> possibleEmails = new LinkedList<>();
    for (Account account : accounts) {
        possibleEmails.add(account.name);
    }
    if (!possibleEmails.isEmpty() && possibleEmails.get(0) != null) {
        return possibleEmails.get(0);
    }
    return "";
}

これにはGET_ACCOUNTS許可が必要です。

<uses-permission android:name="android.permission.GET_ACCOUNTS" />

次に:

editTextEmailAddress.setText(getUserEmail());

これは、現在のアプリに関連付けられているアカウントのみを返すようです-テストで「なし」になります
csga5000

manager.getAccountsByType( "com.google")は、以降のバージョンのAndroidでは機能しません。そして、App.getInstance()は何からのものですか?
パウダー366


0

AndroidがGET_ACCOUNTS最近ロックダウンされたため、一部の回答がうまくいきませんでした。私はこれをAndroid 7.0で動作させましたが、ユーザーが権限ダイアログに耐えなければならないことに注意してください。

AndroidManifest.xml

<uses-permission android:name="android.permission.GET_ACCOUNTS"/>

MainActivity.java

package com.example.patrick.app2;
import android.content.pm.PackageManager;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.accounts.AccountManager;
import android.accounts.Account;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.*;

public class MainActivity extends AppCompatActivity {

    final static int requestcode = 4; //arbitrary constant less than 2^16

    private static String getEmailId(Context context) {
        AccountManager accountManager = AccountManager.get(context);
        Account[] accounts = accountManager.getAccountsByType("com.google");
        Account account;
        if (accounts.length > 0) {
            account = accounts[0];
        } else {
            return "length is zero";
        }
        return account.name;
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            case requestcode:
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                    String emailAddr = getEmailId(getApplicationContext());
                    ShowMessage(emailAddr);

                } else {
                    ShowMessage("Permission Denied");
                }
        }
    }

    public void ShowMessage(String email)
    {
        AlertDialog alertDialog = new AlertDialog.Builder(MainActivity.this).create();
        alertDialog.setTitle("Alert");
        alertDialog.setMessage(email);
        alertDialog.setButton(AlertDialog.BUTTON_NEUTRAL, "OK",
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                    }
                });
        alertDialog.show();
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        Context context = getApplicationContext();

        if ( ContextCompat.checkSelfPermission( context, android.Manifest.permission.GET_ACCOUNTS )
                != PackageManager.PERMISSION_GRANTED )
        {
            ActivityCompat.requestPermissions( this, new String[]
                            {  android.Manifest.permission.GET_ACCOUNTS  },requestcode );
        }
        else
        {
            String possibleEmail = getEmailId(getApplicationContext());
            ShowMessage(possibleEmail);
        }
    }
}

これは他の回答と違いはなく、それらは私にとってうまくいかないようです-Androidの新しいバージョンでは、これはアプリに関連付けられたアカウントのみを返し、許可を必要としなくなったためです。
csga5000 2018

0

MarshMallowオペレーティングシステムでの作業

    btn_click=(Button) findViewById(R.id.btn_click);

    btn_click.setOnClickListener(new View.OnClickListener() {

        @Override
        public void onClick(View arg0)
        {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
            {
                int permissionCheck = ContextCompat.checkSelfPermission(PermissionActivity.this,
                        android.Manifest.permission.CAMERA);
                if (permissionCheck == PackageManager.PERMISSION_GRANTED)
                {
                    //showing dialog to select image
                    String possibleEmail=null;

                     Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
                     Account[] accounts = AccountManager.get(PermissionActivity.this).getAccounts();
                     for (Account account : accounts) {
                         if (emailPattern.matcher(account.name).matches()) {
                             possibleEmail = account.name;
                             Log.e("keshav","possibleEmail"+possibleEmail);
                         }
                     }

                    Log.e("keshav","possibleEmail gjhh->"+possibleEmail);
                    Log.e("permission", "granted Marshmallow O/S");

                } else {                        ActivityCompat.requestPermissions(PermissionActivity.this,
                            new String[]{android.Manifest.permission.READ_EXTERNAL_STORAGE,
                                    android.Manifest.permission.READ_PHONE_STATE,
                                    Manifest.permission.GET_ACCOUNTS,
                                    android.Manifest.permission.CAMERA}, 1);
                }
            } else {
// Lower then Marshmallow

                    String possibleEmail=null;

                     Pattern emailPattern = Patterns.EMAIL_ADDRESS; // API level 8+
                     Account[] accounts = AccountManager.get(PermissionActivity.this).getAccounts();
                     for (Account account : accounts) {
                         if (emailPattern.matcher(account.name).matches()) {
                             possibleEmail = account.name;
                             Log.e("keshav","possibleEmail"+possibleEmail);
                     }

                    Log.e("keshav","possibleEmail gjhh->"+possibleEmail);


            }
        }
    });

<uses-permission android:name = "android.permission.GET_ACCOUNTS" />
Keshav Gera

1
アカウントの取得の応答が機能しません(0アカウントを返す)-ボタンのコールバックでコードを呼び出しても違いはないことを確認できます。
csga5000 2018

このコードは機能していますが、最近私は時間の問題があるので、私たちの側を確認してください
Keshav Gera

0

マニフェストにこの1行を追加します(許可用

<uses-permission android:name="android.permission.GET_ACCOUNTS" />

次に、このコードをアクティビティに貼り付けます

private ArrayList<String> getPrimaryMailId() {
    ArrayList<String> accountsList = new ArrayList<String>();
    try {
        Account[] accounts = AccountManager.get(this).getAccountsByType("com.google");
        for (Account account : accounts) {
            accountsList.add(account.name);
            Log.e("GetPrimaryMailId ", account.name);
        }
    } catch (Exception e) {
        Log.e("GetPrimaryMailId", " Exception : " + e);
    }
    return accountsList;
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.