summaryrefslogtreecommitdiffhomepage
path: root/app/src/main/java/com/wireguard/android/BaseConfigActivity.java
blob: 1bd5c70d5083b904a356e4513578eb6f482dfaa3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
package com.wireguard.android;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;

import com.wireguard.android.backends.VpnService;
import com.wireguard.config.Config;

/**
 * Base class for activities that need to remember the current configuration and wait for a service.
 */

abstract class BaseConfigActivity extends Activity {
    protected static final String KEY_CURRENT_CONFIG = "currentConfig";
    protected static final String KEY_IS_EDITING = "isEditing";

    private Config currentConfig;
    private String initialConfig;
    private boolean isEditing;
    private boolean wasEditing;

    protected Config getCurrentConfig() {
        return currentConfig;
    }

    protected boolean isEditing() {
        return isEditing;
    }

    @Override
    protected void onCreate(final Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // Restore the saved configuration if there is one; otherwise grab it from the intent.
        if (savedInstanceState != null) {
            initialConfig = savedInstanceState.getString(KEY_CURRENT_CONFIG);
            wasEditing = savedInstanceState.getBoolean(KEY_IS_EDITING, false);
        } else {
            final Intent intent = getIntent();
            initialConfig = intent.getStringExtra(KEY_CURRENT_CONFIG);
            wasEditing = intent.getBooleanExtra(KEY_IS_EDITING, false);
        }
        // Trigger starting the service as early as possible
        if (VpnService.getInstance() != null)
            onServiceAvailable();
        else
            bindService(new Intent(this, VpnService.class), new ServiceConnectionCallbacks(),
                    Context.BIND_AUTO_CREATE);
    }

    protected abstract void onCurrentConfigChanged(Config oldCconfig, Config newConfig);

    protected abstract void onEditingStateChanged(boolean isEditing);

    @Override
    public void onSaveInstanceState(final Bundle outState) {
        super.onSaveInstanceState(outState);
        if (currentConfig != null)
            outState.putString(KEY_CURRENT_CONFIG, currentConfig.getName());
        outState.putBoolean(KEY_IS_EDITING, isEditing);
    }

    protected void onServiceAvailable() {
        // Make sure the subclass activity is initialized before setting its config.
        if (initialConfig != null && currentConfig == null)
            setCurrentConfig(VpnService.getInstance().get(initialConfig));
        setIsEditing(wasEditing);
    }

    public void setCurrentConfig(final Config config) {
        if (currentConfig == config)
            return;
        final Config oldConfig = currentConfig;
        currentConfig = config;
        onCurrentConfigChanged(oldConfig, config);
    }

    public void setIsEditing(final boolean isEditing) {
        if (this.isEditing == isEditing)
            return;
        this.isEditing = isEditing;
        onEditingStateChanged(isEditing);
    }

    private class ServiceConnectionCallbacks implements ServiceConnection {
        @Override
        public void onServiceConnected(final ComponentName component, final IBinder binder) {
            // We don't actually need a binding, only notification that the service is started.
            unbindService(this);
            onServiceAvailable();
        }

        @Override
        public void onServiceDisconnected(final ComponentName component) {
            // This can never happen; the service runs in the same thread as the activity.
            throw new IllegalStateException();
        }
    }
}