ManageAccountActivity.java

  1package eu.siacs.conversations.ui;
  2
  3import android.Manifest;
  4import android.content.ActivityNotFoundException;
  5import android.content.ComponentName;
  6import android.content.Intent;
  7import android.content.pm.PackageManager;
  8import android.os.Build;
  9import android.os.Bundle;
 10import android.security.KeyChain;
 11import android.security.KeyChainAliasCallback;
 12import android.util.Pair;
 13import android.view.ContextMenu;
 14import android.view.ContextMenu.ContextMenuInfo;
 15import android.view.Menu;
 16import android.view.MenuItem;
 17import android.view.View;
 18import android.widget.AdapterView.AdapterContextMenuInfo;
 19import android.widget.ListView;
 20import android.widget.Toast;
 21
 22import androidx.annotation.NonNull;
 23import androidx.appcompat.app.ActionBar;
 24import androidx.appcompat.app.AlertDialog;
 25
 26import org.openintents.openpgp.util.OpenPgpApi;
 27
 28import java.util.ArrayList;
 29import java.util.List;
 30import java.util.concurrent.atomic.AtomicBoolean;
 31
 32import eu.siacs.conversations.Config;
 33import eu.siacs.conversations.R;
 34import eu.siacs.conversations.entities.Account;
 35import eu.siacs.conversations.entities.Contact;
 36import eu.siacs.conversations.services.XmppConnectionService;
 37import eu.siacs.conversations.services.XmppConnectionService.OnAccountUpdate;
 38import eu.siacs.conversations.ui.adapter.AccountAdapter;
 39import eu.siacs.conversations.ui.util.MenuDoubleTabUtil;
 40import eu.siacs.conversations.utils.Resolver;
 41import eu.siacs.conversations.xmpp.Jid;
 42import eu.siacs.conversations.xmpp.XmppConnection;
 43
 44import static eu.siacs.conversations.utils.PermissionUtils.allGranted;
 45import static eu.siacs.conversations.utils.PermissionUtils.writeGranted;
 46
 47public class ManageAccountActivity extends XmppActivity implements OnAccountUpdate, KeyChainAliasCallback, XmppConnectionService.OnAccountCreated, AccountAdapter.OnTglAccountState {
 48
 49    private final String STATE_SELECTED_ACCOUNT = "selected_account";
 50
 51    private static final int REQUEST_IMPORT_BACKUP = 0x63fb;
 52    private static final int REQUEST_MICROPHONE = 0x63fb1;
 53
 54    protected Account selectedAccount = null;
 55    protected Jid selectedAccountJid = null;
 56
 57    protected final List<Account> accountList = new ArrayList<>();
 58    protected ListView accountListView;
 59    protected AccountAdapter mAccountAdapter;
 60    protected AtomicBoolean mInvokedAddAccount = new AtomicBoolean(false);
 61    protected Intent mMicIntent = null;
 62
 63    protected Pair<Integer, Intent> mPostponedActivityResult = null;
 64
 65    @Override
 66    public void onAccountUpdate() {
 67        refreshUi();
 68    }
 69
 70    @Override
 71    protected void refreshUiReal() {
 72        synchronized (this.accountList) {
 73            accountList.clear();
 74            accountList.addAll(xmppConnectionService.getAccounts());
 75        }
 76        ActionBar actionBar = getSupportActionBar();
 77        if (actionBar != null) {
 78            actionBar.setHomeButtonEnabled(this.accountList.size() > 0);
 79            actionBar.setDisplayHomeAsUpEnabled(this.accountList.size() > 0);
 80        }
 81        invalidateOptionsMenu();
 82        mAccountAdapter.notifyDataSetChanged();
 83
 84        findViewById(R.id.phone_accounts).setVisibility(View.GONE);
 85        findViewById(R.id.phone_accounts).setOnClickListener((View v) -> {
 86            mMicIntent = new Intent();
 87            mMicIntent.setComponent(new ComponentName("com.android.server.telecom",
 88                "com.android.server.telecom.settings.EnableAccountPreferenceActivity"));
 89            requestMicPermission();
 90        });
 91        findViewById(R.id.phone_accounts_settings).setOnClickListener((View v) -> {
 92            mMicIntent = new Intent(android.telecom.TelecomManager.ACTION_CHANGE_PHONE_ACCOUNTS);
 93            requestMicPermission();
 94        });
 95
 96        if (Build.VERSION.SDK_INT < 23) return;
 97        if (Build.VERSION.SDK_INT >= 33) {
 98            if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_TELECOM) && !getPackageManager().hasSystemFeature(PackageManager.FEATURE_CONNECTION_SERVICE)) return;
 99        } else {
100            if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_CONNECTION_SERVICE)) return;
101        }
102
103        outer:
104        for (Account account : xmppConnectionService.getAccounts()) {
105            for (Contact contact : account.getRoster().getContacts()) {
106                if (contact.getPresences().anyIdentity("gateway", "pstn")) {
107                    findViewById(R.id.phone_accounts).setVisibility(View.VISIBLE);
108                    break outer;
109                }
110            }
111        }
112    }
113
114    @Override
115    protected void onCreate(Bundle savedInstanceState) {
116
117        super.onCreate(savedInstanceState);
118
119        setContentView(R.layout.activity_manage_accounts);
120        Activities.setStatusAndNavigationBarColors(this, findViewById(android.R.id.content));
121        setSupportActionBar(findViewById(R.id.toolbar));
122        configureActionBar(getSupportActionBar());
123        if (savedInstanceState != null) {
124            String jid = savedInstanceState.getString(STATE_SELECTED_ACCOUNT);
125            if (jid != null) {
126                try {
127                    this.selectedAccountJid = Jid.ofEscaped(jid);
128                } catch (IllegalArgumentException e) {
129                    this.selectedAccountJid = null;
130                }
131            }
132        }
133
134        accountListView = findViewById(R.id.account_list);
135        this.mAccountAdapter = new AccountAdapter(this, accountList);
136        accountListView.setAdapter(this.mAccountAdapter);
137        accountListView.setOnItemClickListener((arg0, view, position, arg3) -> switchToAccount(accountList.get(position)));
138        registerForContextMenu(accountListView);
139    }
140
141    @Override
142    public void onSaveInstanceState(final Bundle savedInstanceState) {
143        if (selectedAccount != null) {
144            savedInstanceState.putString(STATE_SELECTED_ACCOUNT, selectedAccount.getJid().asBareJid().toEscapedString());
145        }
146        super.onSaveInstanceState(savedInstanceState);
147    }
148
149    @Override
150    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) {
151        super.onCreateContextMenu(menu, v, menuInfo);
152        ManageAccountActivity.this.getMenuInflater().inflate(
153                R.menu.manageaccounts_context, menu);
154        AdapterContextMenuInfo acmi = (AdapterContextMenuInfo) menuInfo;
155        this.selectedAccount = accountList.get(acmi.position);
156        if (this.selectedAccount.isEnabled()) {
157            menu.findItem(R.id.mgmt_account_enable).setVisible(false);
158            menu.findItem(R.id.mgmt_account_announce_pgp).setVisible(Config.supportOpenPgp());
159        } else {
160            menu.findItem(R.id.mgmt_account_disable).setVisible(false);
161            menu.findItem(R.id.mgmt_account_announce_pgp).setVisible(false);
162            menu.findItem(R.id.mgmt_account_publish_avatar).setVisible(false);
163        }
164        menu.setHeaderTitle(this.selectedAccount.getJid().asBareJid().toEscapedString());
165    }
166
167    @Override
168    protected void onBackendConnected() {
169        if (selectedAccountJid != null) {
170            this.selectedAccount = xmppConnectionService.findAccountByJid(selectedAccountJid);
171        }
172        refreshUiReal();
173        if (this.mPostponedActivityResult != null) {
174            this.onActivityResult(mPostponedActivityResult.first, RESULT_OK, mPostponedActivityResult.second);
175        }
176        if (Config.X509_VERIFICATION && this.accountList.size() == 0) {
177            if (mInvokedAddAccount.compareAndSet(false, true)) {
178                addAccountFromKey();
179            }
180        }
181    }
182
183    @Override
184    public boolean onCreateOptionsMenu(Menu menu) {
185        getMenuInflater().inflate(R.menu.manageaccounts, menu);
186        MenuItem enableAll = menu.findItem(R.id.action_enable_all);
187        MenuItem addAccount = menu.findItem(R.id.action_add_account);
188        MenuItem addAccountWithCertificate = menu.findItem(R.id.action_add_account_with_cert);
189
190        if (Config.X509_VERIFICATION) {
191            addAccount.setVisible(false);
192            addAccountWithCertificate.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
193        }
194
195        if (!accountsLeftToEnable()) {
196            enableAll.setVisible(false);
197        }
198        MenuItem disableAll = menu.findItem(R.id.action_disable_all);
199        if (!accountsLeftToDisable()) {
200            disableAll.setVisible(false);
201        }
202        return true;
203    }
204
205    @Override
206    public boolean onContextItemSelected(MenuItem item) {
207        switch (item.getItemId()) {
208            case R.id.mgmt_account_publish_avatar:
209                publishAvatar(selectedAccount);
210                return true;
211            case R.id.mgmt_account_disable:
212                disableAccount(selectedAccount);
213                return true;
214            case R.id.mgmt_account_enable:
215                enableAccount(selectedAccount);
216                return true;
217            case R.id.mgmt_account_delete:
218                deleteAccount(selectedAccount);
219                return true;
220            case R.id.mgmt_account_announce_pgp:
221                publishOpenPGPPublicKey(selectedAccount);
222                return true;
223            default:
224                return super.onContextItemSelected(item);
225        }
226    }
227
228    @Override
229    public boolean onOptionsItemSelected(MenuItem item) {
230        if (MenuDoubleTabUtil.shouldIgnoreTap()) {
231            return false;
232        }
233        switch (item.getItemId()) {
234            case R.id.action_add_account:
235                startActivity(new Intent(this, EditAccountActivity.class));
236                break;
237            case R.id.action_import_backup:
238                if (hasStoragePermission(REQUEST_IMPORT_BACKUP)) {
239                    startActivity(new Intent(this, ImportBackupActivity.class));
240                }
241                break;
242            case R.id.action_disable_all:
243                disableAllAccounts();
244                break;
245            case R.id.action_enable_all:
246                enableAllAccounts();
247                break;
248            case R.id.action_add_account_with_cert:
249                addAccountFromKey();
250                break;
251            default:
252                break;
253        }
254        return super.onOptionsItemSelected(item);
255    }
256
257    private void requestMicPermission() {
258        final String[] permissions;
259        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
260            permissions = new String[]{Manifest.permission.RECORD_AUDIO, Manifest.permission.BLUETOOTH_CONNECT};
261        } else {
262            permissions = new String[]{Manifest.permission.RECORD_AUDIO};
263        }
264        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
265            if (checkSelfPermission(Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED && shouldShowRequestPermissionRationale(Manifest.permission.RECORD_AUDIO)) {
266                final AlertDialog.Builder builder = new AlertDialog.Builder(this);
267                builder.setTitle("Dialler Integration");
268                builder.setMessage("You will be asked to grant microphone permission, which is needed for the dialler integration to function.");
269                builder.setPositiveButton("I Understand", (dialog, which) -> {
270                    requestPermissions(permissions, REQUEST_MICROPHONE);
271                });
272                builder.setCancelable(true);
273                final AlertDialog dialog = builder.create();
274                dialog.setCanceledOnTouchOutside(true);
275                dialog.show();
276            } else {
277                requestPermissions(new String[]{Manifest.permission.RECORD_AUDIO}, REQUEST_MICROPHONE);
278            }
279        }
280    }
281
282    @Override
283    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
284        if (grantResults.length > 0) {
285            if (allGranted(grantResults)) {
286                switch (requestCode) {
287                    case REQUEST_MICROPHONE:
288                        try {
289                            startActivity(mMicIntent);
290                        } catch (final android.content.ActivityNotFoundException e) {
291                            Toast.makeText(this, "Your OS has blocked dialler integration", Toast.LENGTH_SHORT).show();
292                        }
293                        mMicIntent = null;
294                        return;
295                    case REQUEST_IMPORT_BACKUP:
296                        startActivity(new Intent(this, ImportBackupActivity.class));
297                        break;
298                }
299            } else {
300                if (requestCode == REQUEST_MICROPHONE) {
301                    Toast.makeText(this, "Microphone access was denied", Toast.LENGTH_SHORT).show();
302                } else {
303                    Toast.makeText(this, R.string.no_storage_permission, Toast.LENGTH_SHORT).show();
304                }
305            }
306        }
307        if (writeGranted(grantResults, permissions)) {
308            if (xmppConnectionService != null) {
309                xmppConnectionService.restartFileObserver();
310            }
311        }
312    }
313
314    @Override
315    public boolean onNavigateUp() {
316        if (xmppConnectionService.getConversations().size() == 0) {
317            Intent contactsIntent = new Intent(this,
318                    StartConversationActivity.class);
319            contactsIntent.setFlags(
320                    // if activity exists in stack, pop the stack and go back to it
321                    Intent.FLAG_ACTIVITY_CLEAR_TOP |
322                            // otherwise, make a new task for it
323                            Intent.FLAG_ACTIVITY_NEW_TASK |
324                            // don't use the new activity animation; finish
325                            // animation runs instead
326                            Intent.FLAG_ACTIVITY_NO_ANIMATION);
327            startActivity(contactsIntent);
328            finish();
329            return true;
330        } else {
331            return super.onNavigateUp();
332        }
333    }
334
335    @Override
336    public void onClickTglAccountState(Account account, boolean enable) {
337        if (enable) {
338            enableAccount(account);
339        } else {
340            disableAccount(account);
341        }
342    }
343
344    private void addAccountFromKey() {
345        try {
346            KeyChain.choosePrivateKeyAlias(this, this, null, null, null, -1, null);
347        } catch (ActivityNotFoundException e) {
348            Toast.makeText(this, R.string.device_does_not_support_certificates, Toast.LENGTH_LONG).show();
349        }
350    }
351
352    private void publishAvatar(Account account) {
353        Intent intent = new Intent(getApplicationContext(),
354                PublishProfilePictureActivity.class);
355        intent.putExtra(EXTRA_ACCOUNT, account.getJid().asBareJid().toEscapedString());
356        startActivity(intent);
357    }
358
359    private void disableAllAccounts() {
360        List<Account> list = new ArrayList<>();
361        synchronized (this.accountList) {
362            for (Account account : this.accountList) {
363                if (account.isEnabled()) {
364                    list.add(account);
365                }
366            }
367        }
368        for (Account account : list) {
369            disableAccount(account);
370        }
371    }
372
373    private boolean accountsLeftToDisable() {
374        synchronized (this.accountList) {
375            for (Account account : this.accountList) {
376                if (account.isEnabled()) {
377                    return true;
378                }
379            }
380            return false;
381        }
382    }
383
384    private boolean accountsLeftToEnable() {
385        synchronized (this.accountList) {
386            for (Account account : this.accountList) {
387                if (!account.isEnabled()) {
388                    return true;
389                }
390            }
391            return false;
392        }
393    }
394
395    private void enableAllAccounts() {
396        List<Account> list = new ArrayList<>();
397        synchronized (this.accountList) {
398            for (Account account : this.accountList) {
399                if (!account.isEnabled()) {
400                    list.add(account);
401                }
402            }
403        }
404        for (Account account : list) {
405            enableAccount(account);
406        }
407    }
408
409    private void disableAccount(Account account) {
410        Resolver.clearCache();
411        account.setOption(Account.OPTION_DISABLED, true);
412        if (!xmppConnectionService.updateAccount(account)) {
413            Toast.makeText(this, R.string.unable_to_update_account, Toast.LENGTH_SHORT).show();
414        }
415    }
416
417    private void enableAccount(Account account) {
418        account.setOption(Account.OPTION_DISABLED, false);
419        final XmppConnection connection = account.getXmppConnection();
420        if (connection != null) {
421            connection.resetEverything();
422        }
423        if (!xmppConnectionService.updateAccount(account)) {
424            Toast.makeText(this, R.string.unable_to_update_account, Toast.LENGTH_SHORT).show();
425        }
426    }
427
428    private void publishOpenPGPPublicKey(Account account) {
429        if (ManageAccountActivity.this.hasPgp()) {
430            announcePgp(selectedAccount, null, null, onOpenPGPKeyPublished);
431        } else {
432            this.showInstallPgpDialog();
433        }
434    }
435
436    @Override
437    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
438        super.onActivityResult(requestCode, resultCode, data);
439        if (resultCode == RESULT_OK) {
440            if (xmppConnectionServiceBound) {
441                if (requestCode == REQUEST_CHOOSE_PGP_ID) {
442                    if (data.getExtras().containsKey(OpenPgpApi.EXTRA_SIGN_KEY_ID)) {
443                        selectedAccount.setPgpSignId(data.getExtras().getLong(OpenPgpApi.EXTRA_SIGN_KEY_ID));
444                        announcePgp(selectedAccount, null, null, onOpenPGPKeyPublished);
445                    } else {
446                        choosePgpSignId(selectedAccount);
447                    }
448                } else if (requestCode == REQUEST_ANNOUNCE_PGP) {
449                    announcePgp(selectedAccount, null, data, onOpenPGPKeyPublished);
450                }
451                this.mPostponedActivityResult = null;
452            } else {
453                this.mPostponedActivityResult = new Pair<>(requestCode, data);
454            }
455        }
456    }
457
458    @Override
459    public void alias(final String alias) {
460        if (alias != null) {
461            xmppConnectionService.createAccountFromKey(alias, this);
462        }
463    }
464
465    @Override
466    public void onAccountCreated(final Account account) {
467        final Intent intent = new Intent(this, EditAccountActivity.class);
468        intent.putExtra("jid", account.getJid().asBareJid().toString());
469        intent.putExtra("init", true);
470        startActivity(intent);
471    }
472
473    @Override
474    public void informUser(final int r) {
475        runOnUiThread(() -> Toast.makeText(ManageAccountActivity.this, r, Toast.LENGTH_LONG).show());
476    }
477}