mirror of https://github.com/lukechilds/docs.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
608 lines
20 KiB
608 lines
20 KiB
6 years ago
|
---
|
||
5 years ago
|
|
||
6 years ago
|
description: Learn about the Android SDK
|
||
5 years ago
|
|
||
6 years ago
|
---
|
||
5 years ago
|
|
||
5 years ago
|
# Android DApps
|
||
5 years ago
|
|
||
6 years ago
|
{:.no_toc}
|
||
|
|
||
|
This tutorial is written for readers who are new to either or both Blockstack
|
||
|
and Android to create a decentralized application. It contains the following
|
||
|
content:
|
||
|
|
||
5 years ago
|
TOC
|
||
|
{:toc}
|
||
6 years ago
|
|
||
5 years ago
|
This tutorial was extensively tested using Android Studio 3.6 on a Dell XPS 13
|
||
|
running Ubuntu 19. If your environment is different, you may encounter
|
||
6 years ago
|
slight or even major discrepancies when performing the procedures in this
|
||
5 years ago
|
tutorial. Please [join the Blockstack discord server](https://chat.blockstack.org) and post questions or comments to
|
||
6 years ago
|
the `#support` channel.
|
||
|
|
||
|
Finally, this tutorial is written for all levels from the beginner to the most
|
||
|
experienced. For best results, beginners should follow the guide as written. It
|
||
|
is expected that the fast or furiously brilliant will skip ahead and improvise
|
||
|
on this material at will. Fair journey one and all.
|
||
|
|
||
|
If you prefer, you can skip working through the tutorial all together. Instead,
|
||
|
you can [download the final project code](images/helloandroid.zip) and import it
|
||
|
into Android Studio to review it.
|
||
|
|
||
|
## Understand the sample application flow
|
||
|
|
||
|
When complete, the sample application is a simple `hello-world` display. It is
|
||
|
intended for user on an Android phone.
|
||
|
|
||
|
![](images/final-app.png)
|
||
|
|
||
5 years ago
|
Only users with an existing blockstack id can run your
|
||
6 years ago
|
final sample application. When complete, users interact with the sample
|
||
|
application by doing the following:
|
||
|
|
||
|
![](images/app-flow.png)
|
||
|
|
||
|
## Set up your environment
|
||
|
|
||
|
This sample application has two code bases, a Blockstack `hello-blockstack`
|
||
5 years ago
|
web application and a `hello-andriod` Android application. Before you start
|
||
6 years ago
|
developing the sample, there are a few elements you need in your environment.
|
||
|
|
||
|
### Install Android Studio
|
||
|
|
||
|
If you are an experienced Android developer and already have an Android
|
||
|
development environment on your workstation, you can use that and skip this
|
||
|
step. However, you will need to adjust the remaining instructions for your
|
||
|
environment.
|
||
|
|
||
|
Follow the installation instructions to download and [Android Studio
|
||
5 years ago
|
3.6](https://developer.android.com/studio/install) for your operating system.
|
||
6 years ago
|
Depending on your network connection, this can take between 15-30 minutes.
|
||
|
|
||
|
![](images/studio-download.png)
|
||
|
|
||
5 years ago
|
### Do you have Node.js?
|
||
6 years ago
|
|
||
5 years ago
|
Node.js v10 or higher is recommended the minimum supported version is Node.js v8. Before you begin, verify you have the correct version of Node.js and its tools installed.
|
||
6 years ago
|
|
||
|
```bash
|
||
5 years ago
|
$ node -v
|
||
|
v12.10.0
|
||
|
$ which npm npx
|
||
6 years ago
|
/usr/local/bin/npm
|
||
5 years ago
|
/usr/local/bin/npx
|
||
6 years ago
|
```
|
||
|
|
||
5 years ago
|
If you don't have these installed, take a moment to install or upgrade as needed.
|
||
6 years ago
|
|
||
5 years ago
|
## Build the Blockstack hello-world web app
|
||
6 years ago
|
|
||
5 years ago
|
In this section, you build a Blockstack `hello-world` web application that acts as the companion site to your Android application.
|
||
6 years ago
|
|
||
|
### Generate and launch your hello-blockstack application
|
||
|
|
||
5 years ago
|
{% include scaffolding.md %}
|
||
|
|
||
6 years ago
|
In this section, you build an initial React.js application called
|
||
|
`hello-blockstack`.
|
||
|
|
||
|
1. Create a `hello-blockstack` directory.
|
||
|
|
||
5 years ago
|
```bash
|
||
|
mkdir hello-blockstack
|
||
|
```
|
||
6 years ago
|
|
||
|
2. Change into your new directory.
|
||
|
|
||
5 years ago
|
```bash
|
||
|
cd hello-blockstack
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
3. Use the Blockstack application generator to create your initial `hello-blockstack` application.
|
||
6 years ago
|
|
||
5 years ago
|
```bash
|
||
|
$ npx generator-blockstack --plain
|
||
|
create package.json
|
||
|
create .gitignore
|
||
|
create webpack.config.js
|
||
|
create netlify.toml
|
||
|
create firebase.json
|
||
|
...
|
||
|
npm WARN ajv-errors@1.0.1 requires a peer of ajv@>=5.0.0 but none is installed. You must install peer dependencies yourself.
|
||
|
|
||
|
added 840 packages from 582 contributors and audited 843 packages in 18.84s
|
||
|
found 0 vulnerabilities
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
Depending on your environment you may have some warnings with the installation. Optionally, you can fix these before continuing to the next section.
|
||
6 years ago
|
|
||
5 years ago
|
4. Depending on your environment, respond to the prompts to populate the initial app.
|
||
|
You might see a prompt similar to the following:
|
||
6 years ago
|
|
||
5 years ago
|
```bash
|
||
|
[fsevents] Success:
|
||
|
"/Users/theuser/repos/hello-blockstack/node_modules/fsevents/lib/binding/Release/node-v59-darwin-x64/fse.node"
|
||
|
is installed via remote npm notice created a lockfile as package-lock.json.
|
||
|
You should commit this file. added 1060 packages in 26.901s
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
5. After the initial setup you can now run the initial application.
|
||
6 years ago
|
|
||
5 years ago
|
```bash
|
||
|
$ npm run start
|
||
6 years ago
|
|
||
5 years ago
|
> hello-blockstack-webpack@0.0.0 start /home/user/hello-blockstack
|
||
6 years ago
|
> webpack-dev-server
|
||
|
|
||
5 years ago
|
ℹ 「wds」: Project is running at http://localhost:9000/
|
||
|
ℹ 「wds」: webpack output is served from /
|
||
|
ℹ 「wds」: Content not from webpack is served from /home/user/hello-blockstack/dist
|
||
|
ℹ 「wdm」: Hash: f5d88efe9c9194f66ddd
|
||
|
Version: webpack 4.43.0
|
||
|
Time: 2733ms
|
||
|
Built at: 05/19/2020 10:44:50 AM
|
||
|
Asset Size Chunks Chunk Names
|
||
|
main.js 5.39 MiB main [emitted] main
|
||
|
....
|
||
|
[./src/index.js] 1.8 KiB {main} [built]
|
||
|
+ 610 hidden modules
|
||
|
ℹ 「wdm」: Compiled successfully.
|
||
|
```
|
||
6 years ago
|
|
||
|
The system opens a browser displaying your running application.
|
||
|
|
||
|
![](images/blockstack-signin.png)
|
||
|
|
||
|
At this point, the browser is running a Blockstack server on your local host.
|
||
|
This is for testing your applications only.
|
||
|
|
||
5 years ago
|
6. Choose **Get started**
|
||
6 years ago
|
|
||
|
The system displays a prompt allowing you to create a new Blockstack ID or restore an existing one.
|
||
|
|
||
5 years ago
|
![](images/create-restore.png)
|
||
6 years ago
|
|
||
|
7. Follow the prompts appropriate to your situation.
|
||
|
|
||
5 years ago
|
At this point you have only a single application
|
||
|
on your test server. So, you should see this single application, with your
|
||
|
own `blockstack.id`, once you are signed in:
|
||
6 years ago
|
|
||
5 years ago
|
![](images/running-app.png)
|
||
6 years ago
|
|
||
|
## Create the hello-android project
|
||
|
|
||
5 years ago
|
In this section, you'll create an Android application in Android Studio. You'll
|
||
6 years ago
|
run the application in the emulator to test it.
|
||
|
|
||
|
### Create a simple project
|
||
|
|
||
|
In this section, you create an inital project. You'll validate the application's
|
||
|
iniatial state by creating an emulator to run it in. Open Android Studio and do the following:
|
||
|
|
||
|
1. Open Android Studio and choose **Start a new Andriod Studio project**.
|
||
|
|
||
|
If studio is already started, choose **File > New > New Project**.
|
||
|
|
||
5 years ago
|
2. Choose a project template: select **Empty Activity** and press **Next**
|
||
|
|
||
|
3. Enter these fields in the **Create Android Project** page.
|
||
|
|
||
|
<table class="uk-table">
|
||
|
<tr>
|
||
|
<th>Name</th>
|
||
|
<td><code>Hello Android</code></td>
|
||
|
</tr>
|
||
|
<tr>
|
||
|
<th>Package name</th>
|
||
|
<td><code>blockstack.id.<i>USERNAME</i>.hello</code></td>
|
||
|
</tr>
|
||
|
<tr>
|
||
|
<th>Project location</th>
|
||
|
<td><code>/home/<i>USERNAME</i>/AndroidStudioProjects/helloandroid</code></td>
|
||
|
</tr>
|
||
|
<tr>
|
||
|
<th>Language</th>
|
||
|
<td>Select (Kotlin)</td>
|
||
|
</tr>
|
||
|
<tr>
|
||
|
<th>Minimum SDK</th>
|
||
|
<td>Select (API 21: Android 5.0 (Lollipop))</td>
|
||
|
</tr>
|
||
|
</table>
|
||
|
|
||
|
![](images/configure-activity.png)
|
||
|
|
||
|
4. Press **Finish**.
|
||
6 years ago
|
|
||
|
Android studio builds your initial project. This can take a bit the first time you do it.
|
||
|
|
||
|
### Run the app in an emulator
|
||
|
|
||
|
In this section, you run the appliation and create an emulator when prompted.
|
||
|
|
||
|
1. Once the project is imported into studio, click the `app` module in the **Project** window.
|
||
|
|
||
5 years ago
|
2. Then, select **Run > Run** (or click the green arrow in the toolbar).
|
||
6 years ago
|
|
||
|
Studio prompts you to **Select Deployment Target**.
|
||
|
|
||
|
3. Choose **Create New Virtual Device** and press **OK**.
|
||
|
|
||
5 years ago
|
Studio prompts you to **Select Hardware**.
|
||
6 years ago
|
|
||
5 years ago
|
4. Choose a Phone device configuration like "Pixel".
|
||
6 years ago
|
|
||
5 years ago
|
![](images/select-hdw.png)
|
||
6 years ago
|
|
||
5 years ago
|
Studio prompts you for a system image.
|
||
6 years ago
|
|
||
5 years ago
|
5. Choose **Q** which is API level 29 and press **Next**.
|
||
6 years ago
|
|
||
5 years ago
|
![](images/q-api.png)
|
||
6 years ago
|
|
||
5 years ago
|
Studio asks you to verify your new emulator configuration.
|
||
6 years ago
|
|
||
|
6. Press **Finish**.
|
||
|
|
||
5 years ago
|
The emulation takes a moment to build. Then, studio launches the emulation and opens your application.
|
||
6 years ago
|
|
||
5 years ago
|
![](images/hello-andriod-1.png)
|
||
6 years ago
|
|
||
|
### Configure your application with the Blockstack SDK
|
||
|
|
||
|
Now that you have created your initial project and verified it running in an emulator, you are ready to begin configuring the application for use with Blockstack.
|
||
|
|
||
|
1. In studio, open the `AndroidManifest.xml` file.
|
||
|
|
||
5 years ago
|
2. Add an `<intent-filter>` with the custom handler for Blockstack. Replace the host ("flamboyant-darwin-d11c17.netlify.app") with your own url once you have hosted your web app publicly.
|
||
6 years ago
|
|
||
5 years ago
|
```XML
|
||
|
<intent-filter>
|
||
|
<action android:name="android.intent.action.VIEW" />
|
||
|
<category android:name="android.intent.category.DEFAULT" />
|
||
|
<category android:name="android.intent.category.BROWSABLE" />
|
||
|
<data android:scheme="https" android:host="flamboyant-darwin-d11c17.netlify.app" />
|
||
|
</intent-filter>
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
3. Open the Project's `build.gradle` file ("Project: Hello Android").
|
||
|
4. Add the Jitpack repository `maven { url 'https://jitpack.io' }` to the `repositories` section. for all projects
|
||
6 years ago
|
|
||
5 years ago
|
When you finish, that section looks like this:
|
||
6 years ago
|
|
||
5 years ago
|
```JS
|
||
|
allprojects {
|
||
|
repositories {
|
||
|
google()
|
||
|
jcenter()
|
||
|
maven { url 'https://jitpack.io' }
|
||
|
}
|
||
|
}
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
5. Open the Module `build.gradle` file ("Module: app").
|
||
|
6. Fix a problem with duplicate meta data in the Android SDK: add `packagingOptions`
|
||
6 years ago
|
|
||
|
```JS
|
||
|
android {
|
||
|
...
|
||
5 years ago
|
packagingOptions {
|
||
|
exclude 'META-INF/*'
|
||
|
}
|
||
6 years ago
|
}
|
||
5 years ago
|
```
|
||
6 years ago
|
|
||
|
7. Below this, add the Blockstack Android SDK dependency to your project's `dependencies` list:
|
||
|
|
||
5 years ago
|
When you are done you should see:
|
||
6 years ago
|
|
||
5 years ago
|
```JS
|
||
6 years ago
|
dependencies {
|
||
5 years ago
|
implementation fileTree(dir: 'libs', include: ['*.jar'])
|
||
6 years ago
|
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
|
||
5 years ago
|
implementation 'androidx.appcompat:appcompat:1.1.0'
|
||
|
implementation 'androidx.core:core-ktx:1.2.0'
|
||
|
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
|
||
6 years ago
|
testImplementation 'junit:junit:4.12'
|
||
5 years ago
|
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
|
||
|
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
|
||
|
|
||
|
implementation 'com.github.blockstack:blockstack-android:7940940'
|
||
|
implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.2.0'
|
||
6 years ago
|
}
|
||
|
|
||
5 years ago
|
```
|
||
6 years ago
|
|
||
5 years ago
|
**NOTE**: Ignore the warning on the `junit` dependencies.
|
||
6 years ago
|
|
||
|
8. Sync your project.
|
||
|
|
||
5 years ago
|
![](images/sync-project.png)
|
||
6 years ago
|
|
||
5 years ago
|
Be sure to check the sync completed successfully.
|
||
6 years ago
|
|
||
5 years ago
|
![](images/sync-success.png)
|
||
6 years ago
|
|
||
5 years ago
|
9. Run your app in the emulator.
|
||
6 years ago
|
|
||
5 years ago
|
You've made a lot of changes, make sure the emulator is still running
|
||
|
correctly.
|
||
6 years ago
|
|
||
|
### Add a simple interface
|
||
|
|
||
|
1. Open the `app/res/layout/activity_main.xml` file.
|
||
|
|
||
|
The `activity_main.xml` file defines the graphical elements. Some elements are required before you can functionality to your `MainActivity.kt` code.
|
||
|
|
||
5 years ago
|
2. Replace the entire content of the file with the following code:
|
||
6 years ago
|
|
||
5 years ago
|
The new interface includes a `BlockstackSignInButton` which is provided by
|
||
|
the SDK. This SDK includes a themed "Get Started" button
|
||
|
(`BlockstackSignInButton`). You use this button here with the
|
||
|
`org.blockstack.android.sdk.ui.BlockstackSignInButton` class.
|
||
6 years ago
|
|
||
|
```XML
|
||
|
<?xml version="1.0" encoding="utf-8"?>
|
||
5 years ago
|
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||
|
xmlns:tools="http://schemas.android.com/tools"
|
||
|
android:layout_width="match_parent"
|
||
|
android:layout_height="match_parent"
|
||
|
android:layout_margin="16dp"
|
||
|
tools:context=".MainActivity">
|
||
|
|
||
|
<org.blockstack.android.sdk.ui.BlockstackSignInButton
|
||
|
android:id="@+id/signInButton"
|
||
|
android:layout_width="match_parent"
|
||
|
android:layout_height="wrap_content"
|
||
|
app:layout_constraintTop_toTopOf="parent"
|
||
|
app:layout_constraintEnd_toEndOf="parent"
|
||
|
app:layout_constraintStart_toStartOf="parent"
|
||
|
app:layout_constraintBottom_toTopOf="@id/userDataTextView"/>
|
||
|
|
||
|
<TextView
|
||
|
android:id="@+id/userDataTextView"
|
||
|
android:layout_width="wrap_content"
|
||
|
android:layout_height="wrap_content"
|
||
|
android:text="Hello World!"
|
||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||
|
app:layout_constraintLeft_toLeftOf="parent"
|
||
|
app:layout_constraintRight_toRightOf="parent"
|
||
|
app:layout_constraintTop_toBottomOf="@id/signInButton" />
|
||
|
|
||
|
</androidx.constraintlayout.widget.ConstraintLayout>
|
||
6 years ago
|
```
|
||
|
|
||
|
This codes adds a button and some text to your application.
|
||
|
|
||
5 years ago
|
3. Choose **Run > Apply changes**.
|
||
6 years ago
|
|
||
5 years ago
|
4. Choose **Run > Run app** in the emulator.
|
||
6 years ago
|
|
||
|
The emulator now contains a new interface with a button:
|
||
|
|
||
|
![](images/new-interface.png)
|
||
|
|
||
|
### Add session & authentication code
|
||
|
|
||
|
1. Open the `MainActivity.kt` file.
|
||
5 years ago
|
2. Add some additional imports to the top, replace the `android.os.Bundle` import.
|
||
6 years ago
|
|
||
5 years ago
|
When you are done, your imports should appear as follows:
|
||
6 years ago
|
|
||
5 years ago
|
```kotlin
|
||
6 years ago
|
import android.content.Intent
|
||
|
import android.os.Bundle
|
||
5 years ago
|
import androidx.appcompat.app.AppCompatActivity
|
||
|
import androidx.lifecycle.lifecycleScope
|
||
6 years ago
|
import kotlinx.android.synthetic.main.activity_main.*
|
||
5 years ago
|
import kotlinx.coroutines.Dispatchers
|
||
|
import kotlinx.coroutines.launch
|
||
6 years ago
|
import org.blockstack.android.sdk.BlockstackSession
|
||
5 years ago
|
import org.blockstack.android.sdk.BlockstackSignIn
|
||
|
import org.blockstack.android.sdk.SessionStore
|
||
|
import org.blockstack.android.sdk.getBlockstackSharedPreferences
|
||
|
import org.blockstack.android.sdk.model.UserData
|
||
|
import org.blockstack.android.sdk.model.toBlockstackConfig
|
||
|
import org.blockstack.android.sdk.ui.SignInProvider
|
||
|
import org.blockstack.android.sdk.ui.showBlockstackConnect
|
||
|
```
|
||
|
|
||
|
3. Add one variable for the Blockstack sign-in flow and one for the Blockstack session that deals with all other function. This needs to be added before `onCreate`.
|
||
6 years ago
|
|
||
|
```kotlin
|
||
|
class MainActivity : AppCompatActivity() {
|
||
|
|
||
5 years ago
|
private lateinit var blockstackSession: BlockstackSession
|
||
|
private lateinit var blockstackSignIn: BlockstackSignIn
|
||
6 years ago
|
|
||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||
|
super.onCreate(savedInstanceState)
|
||
|
setContentView(R.layout.activity_main)
|
||
|
}
|
||
|
}
|
||
|
```
|
||
|
|
||
5 years ago
|
4. Replace the existing the `onCreate` function with the following, use your own domain if you are already hosting your web app somewhere:
|
||
6 years ago
|
|
||
5 years ago
|
```kotlin
|
||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||
|
super.onCreate(savedInstanceState)
|
||
|
setContentView(R.layout.activity_main)
|
||
|
|
||
|
val appConfig =
|
||
|
"https://flamboyant-darwin-d11c17.netlify.app".toBlockstackConfig()
|
||
|
val sessionStore = SessionStore(getBlockstackSharedPreferences())
|
||
|
blockstackSession = BlockstackSession(sessionStore, appConfig)
|
||
|
blockstackSignIn = BlockstackSignIn(sessionStore, appConfig)
|
||
|
BlockstackSignIn.shouldLaunchInCustomTabs = false
|
||
|
|
||
|
signInButton.setOnClickListener {
|
||
|
showBlockstackConnect()
|
||
|
}
|
||
6 years ago
|
|
||
5 years ago
|
if (intent?.action == Intent.ACTION_VIEW) {
|
||
|
// handle the redirect from sign in
|
||
|
userDataTextView.text = "Signing in now ..."
|
||
|
lifecycleScope.launch(Dispatchers.IO) {
|
||
|
handleAuthResponse(intent)
|
||
6 years ago
|
}
|
||
5 years ago
|
}
|
||
|
}
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
This new `onCreate` does several things:
|
||
6 years ago
|
|
||
5 years ago
|
- Supply authentication information for connecting to your Blockstack app: `appDomain` (for `scopes`, `redirectURI`, `manifestURI` the default values are used)
|
||
|
- Add a listener for the button click.
|
||
|
- Handles the redirect intent from the sign-in flow.
|
||
6 years ago
|
|
||
5 years ago
|
Notice that the application in this example is a URI you have not set up.
|
||
|
Registering and application name takes time, so in time's interest you'll
|
||
|
use an existing app (https://flamboyant-darwin-d11c17.netlify.app) that is identical to the `hello-world` you created
|
||
|
earlier. For a production version, you'll need to replace `appDomain`,
|
||
|
`redirectURI`, `manifestURI` and `scopes` with values appropriate for your
|
||
|
app.
|
||
6 years ago
|
|
||
|
5. Add a private function to reflect when a user successfully signs in.
|
||
|
|
||
5 years ago
|
```kotlin
|
||
|
private fun onSignIn(userData: UserData) {
|
||
|
userDataTextView.text = "Signed in as ${userData.decentralizedID}"
|
||
|
signInButton.isEnabled = false
|
||
|
}
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
6. Create a handler for the authentication response.
|
||
6 years ago
|
|
||
5 years ago
|
```kotlin
|
||
|
private fun handleAuthResponse(intent: Intent) {
|
||
|
val authResponse = intent.data?.getQueryParameter("authResponse")
|
||
|
if (authResponse != null) {
|
||
|
val userData = blockstackSession.handlePendingSignIn(authResponse)
|
||
|
if (userData.hasValue) {
|
||
|
// The user is now signed in!
|
||
|
runOnUiThread {
|
||
|
onSignIn(userData.value!!)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
```
|
||
6 years ago
|
|
||
5 years ago
|
7. Add `SignInProvider` interface for better user onboarding. Let the activity implement the interface and implement the member function `getBlockstackSignIn`
|
||
6 years ago
|
|
||
5 years ago
|
```kotlin
|
||
|
class MainActivity : AppCompatActivity(), SignInProvider {
|
||
|
...
|
||
|
|
||
|
override fun provideBlockstackSignIn(): BlockstackSignIn {
|
||
|
return blockstackSignIn
|
||
|
}
|
||
|
}
|
||
|
```
|
||
|
|
||
|
8. Verify your final `MainActivity.kt` code looks like this:
|
||
|
|
||
|
```kotlin
|
||
|
package blockstack.id.user.hello
|
||
6 years ago
|
|
||
5 years ago
|
import android.content.Intent
|
||
|
import android.os.Bundle
|
||
|
import androidx.appcompat.app.AppCompatActivity
|
||
|
import androidx.lifecycle.lifecycleScope
|
||
|
import kotlinx.android.synthetic.main.activity_main.\*
|
||
|
import kotlinx.coroutines.Dispatchers
|
||
|
import kotlinx.coroutines.launch
|
||
|
import org.blockstack.android.sdk.BlockstackSession
|
||
|
import org.blockstack.android.sdk.BlockstackSignIn
|
||
|
import org.blockstack.android.sdk.SessionStore
|
||
|
import org.blockstack.android.sdk.getBlockstackSharedPreferences
|
||
|
import org.blockstack.android.sdk.model.UserData
|
||
|
import org.blockstack.android.sdk.model.toBlockstackConfig
|
||
|
import org.blockstack.android.sdk.ui.SignInProvider
|
||
|
import org.blockstack.android.sdk.ui.showBlockstackConnect
|
||
6 years ago
|
|
||
5 years ago
|
class MainActivity : AppCompatActivity(), SignInProvider {
|
||
|
|
||
|
private lateinit var blockstackSession: BlockstackSession
|
||
|
private lateinit var blockstackSignIn: BlockstackSignIn
|
||
|
|
||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||
|
super.onCreate(savedInstanceState)
|
||
|
setContentView(R.layout.activity_main)
|
||
|
|
||
|
val appConfig =
|
||
|
"https://flamboyant-darwin-d11c17.netlify.app".toBlockstackConfig()
|
||
|
val sessionStore = SessionStore(getBlockstackSharedPreferences())
|
||
|
blockstackSession = BlockstackSession(sessionStore, appConfig)
|
||
|
blockstackSignIn = BlockstackSignIn(sessionStore, appConfig)
|
||
|
BlockstackSignIn.shouldLaunchInCustomTabs = false
|
||
|
|
||
|
signInButton.setOnClickListener {
|
||
|
showBlockstackConnect()
|
||
|
}
|
||
|
|
||
|
if (intent?.action == Intent.ACTION_VIEW) {
|
||
|
// handle the redirect from sign in
|
||
|
userDataTextView.text = "Signing in now ..."
|
||
|
lifecycleScope.launch(Dispatchers.IO) {
|
||
|
handleAuthResponse(intent)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private fun onSignIn(userData: UserData) {
|
||
|
userDataTextView.text = "Signed in as ${userData.decentralizedID}"
|
||
|
signInButton.isEnabled = false
|
||
|
}
|
||
|
|
||
|
private suspend fun handleAuthResponse(intent: Intent) {
|
||
|
val authResponse = intent.data?.getQueryParameter("authResponse")
|
||
|
if (authResponse != null) {
|
||
|
val userData = blockstackSession.handlePendingSignIn(authResponse)
|
||
|
if (userData.hasValue) {
|
||
|
// The user is now signed in!
|
||
|
runOnUiThread {
|
||
|
onSignIn(userData.value!!)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
override fun provideBlockstackSignIn(): BlockstackSignIn {
|
||
|
return blockstackSignIn
|
||
|
}
|
||
5 years ago
|
|
||
5 years ago
|
}
|
||
|
```
|
||
6 years ago
|
|
||
|
### Run the final app in the emulator
|
||
|
|
||
|
1. Choose **Run > Apply changes**.
|
||
|
2. Choose **Run > Run app** in the emulator.
|
||
5 years ago
|
3. When you see the application open, choose **Get Started**.
|
||
|
4. A small information is presented about the sign-in flow
|
||
6 years ago
|
|
||
5 years ago
|
![](images/connect-ui.png)
|
||
6 years ago
|
|
||
5 years ago
|
5. Select **Get Started**
|
||
5 years ago
|
The system might prompt you how to select a browser.
|
||
6 years ago
|
|
||
5 years ago
|
6. Work through the Blockstack prompts to login.
|
||
|
7. Blockstack redirects you to a web site. Open it with your Android app: Select **Hello Android** and **Always**
|
||
6 years ago
|
|
||
5 years ago
|
![](images/connect-response.png)
|
||
6 years ago
|
|
||
5 years ago
|
8. The screen after the sign-in flow shows the owner address of the username that was entered during sign-in.
|
||
6 years ago
|
|
||
5 years ago
|
![](images/final-app.png)
|
||
6 years ago
|
|
||
|
## Where to go next
|
||
|
|
||
5 years ago
|
Congratulations, you've completed your Android app using the Blockstack Android SDK.
|
||
6 years ago
|
|
||
|
Learn more about Blockstack by [trying another tutorial](https://blockstack.org/tutorials).
|