How to Build an Android NFC Tag Scanner App With React Native
Get the project source code below, and follow along with the lesson material.
Download Project Source CodeTo set up the project on your local machine, please follow the directions provided in the README.md
file. If you run into any issues with running the project source code, then feel free to reach out to the author in the course's Discord channel.
In this lesson, we will focus on the Android side. We will see the difference between Android and iOS and discuss how to provide a similar experience for our NFC game app.
The difference between iOS and Android#
Let's first run our previous work on a real Android device.
As you can see, I have pressed the START
button several times, but it looks like there's nothing happened.
What's wrong here? Actually, it's nothing wrong. It's because Android provides no built-in NFC scanning UI, so even though our app is ready to scan NFC tags, there's no visual cue for our users.
So, our first step in this lesson is to build a component to mimic the behavior for iOS NFC scanning UI.
Implement the AndroidPrompt component#
Let's call it AndroidPrompt
. This component will use React Native's built-in Modal
component to present content above the enclosing view.
Inside the Modal
, we place a View
and a Text
with the string Hello NFC
.
Then we set both transparent
and visible
props to be true for the Modal
component and provide some basic styling for our inner View
.
After the basic setup, let's add our AndroidPrompt
component into our App.js
.
Okay, now our AndroidPrompt
shows up on the screen.
Let's move on to style our content inside the Modal
.
The building blocks are:
a full
View
component withflex: 1
as its containera backdrop inside the container.
the actual prompt UI is rendered above the backdrop, and there are two components inside it, one hint text and one cancel button, just like on iOS.
Then, the next step is to apply our styles to our components.
Oops, it looks like there is something wrong with the backdrop. That's because we haven't applied the position: absolute
part into it. To write this kind of "cover-all" style, here's a little trick: we can use the StyleSheet.absoluteFill
. It's very convenient in such a case.
For now, the codes look like this:
xxxxxxxxxx
import React from 'react';
import {
View,
Text,
Modal,
StyleSheet,
Dimensions,
TouchableOpacity,
} from 'react-native';
​
function AndroidPrompt(props) {
return (
<Modal visible={true} transparent={true}>
<View style={styles.content}>
<View style={[styles.backdrop, StyleSheet.absoluteFill]} />
​
<View style={styles.prompt}>
<Text style={styles.hint}>Hello NFC</Text>
​
<TouchableOpacity style={styles.btn}>
<Text>CANCEL</Text>
</TouchableOpacity>
</View>
</View>
</Modal>
);
}
​
const styles = StyleSheet.create({
content: {
flex: 1,
},
backdrop: {
backgroundColor: 'rgba(0,0,0,0.3)',
},
prompt: {
position: 'absolute',
bottom: 0,
left: 20,
width: Dimensions.get('window').width - 2 * 20,
backgroundColor: 'white',
borderRadius: 8,
paddingVertical: 60,
paddingHorizontal: 20,
alignItems: 'center',
justifyContent: 'center',
},
hint: {
fontSize: 24,
marginBottom: 20,
},
btn: {
borderWidth: 1,
borderColor: '#ccc',
borderRadius: 8,
padding: 15,
},
});
​
export default AndroidPrompt;
Design the component interface#
The next step is to design the interface between our AndroidPrompt
and its parent.
Normally in React, we use props
passing as the primary interface between components. That's because of React's declarative nature. However, in our case, providing an imperative API set will be much easier.
The API we want to expose is something like React Native's built-in Alert
module. For Alert
, we simply call Alert.alert
, and it will appear. From the caller's perspective, we don't need to preserve a state like isAlertVisible
and pass it as props into Alert
. We'd like to do something similar for our AndroidPrompt
component here.
In order to accomplish this, we will use ref
.
First, wrap the original AndroidPrompt
component with React.forwardRef
. Once a function component is wrapped with forwardRef
, it will receive a second argument, which is the ref
object passing from its parent.
Then, we create our internal state, visible
, and hintText
. Since those states are within the current component, we need to provide a method to let the parent component access them.
We can use a useEffect
hook to accomplish this inside the hook. If we have a ref
object passed in, we can set the current
property of the ref
object and put our state mutation functions inside the current
object. So our parent component can use setVisible
and setHintText
.
And of course, we should pass our internal visible
state into the Modal
component and display our internal hintText
state. We also need to toggle the visible
state back to false
and clear the hintText
when the cancel
button is pressed.
Now, it's time to test them. Head back to our App.js
. We first call useRef
hook to obtain a ref
object and pass it into the AndroidPrompt
component.
Then, quickly create a test button. From the test button's onPress
handler, we can simply call current.setVisible(true)
.
As you can see, we can now toggle on our AndroidPrompt
component from the parent and toggle it off by pressing its own cancel button, without tracking the visible
state from the parent component.
This lesson preview is part of the The newline Guide to NFCs with React Native course and can be unlocked immediately with a \newline Pro subscription or a single-time purchase. Already have access to this course? Log in here.
Get unlimited access to The newline Guide to NFCs with React Native, plus 70+ \newline books, guides and courses with the \newline Pro subscription.
[00:00 - 00:21] Hello, in this lesson, we will focus on Android side. We will see the difference between Android and iOS, and discuss how to provide a similar experience for our NFC game app, the difference between iOS and Android.
[00:22 - 00:35] Let's first run our previous work on a real Android device. As you can see, I have pressed the start button several times, but it looks like there's nothing happen.
[00:36 - 00:39] What's wrong here? Actually, it's nothing wrong.
[00:40 - 00:52] It's because Android provides no built-in NFC scanning UI, so even our app is ready to scan NFC tags. There's no visual cue for our users.
[00:53 - 01:07] So our first step in this lesson is to build a component to mimic the behavior for iOS NFC scanning UI. Let's call it Android Prompt.
[01:08 - 01:51] This component will use real native built-in model component to present content about the enclosing view. Then we set both transparent and visible props to be true for the model component, and also provide some basic styling for our inner view.
[01:52 - 02:07] After the basic setup, let's add our Android Prompt component into our app.js. Okay, now our Android Prompt shows up on the screen.
[02:08 - 02:27] Let's move on to style our content inside the model. The built-in blocks are a full view component with flex1 as its container, and a backdrop inside the container.
[02:28 - 02:44] The actual prompt UI is rendered above the backdrop. And there are two components inside it.
[02:45 - 02:59] One hint tags and one cancel button, just like the iOS. Then the next step is to apply our styles to our components.
[03:00 - 03:16] Oops, it looks like there is something wrong with the backdrop. That's because we haven't applied the position absolute part into it.
[03:17 - 03:26] To write this kind of cover all style, here is a little trick. We can use the style sheet the absolute feel.
[03:27 - 03:36] It's very convenient in such a case. The next step is to design the interface between our Android Prompt and its parent.
[03:37 - 03:56] Normally in React, we use props passing as the primary interface between components. That's because of React's declarative nature, however, in our case, provide an imperative API set will be much easier.
[03:57 - 04:10] The API we want to expose is something like React Native's built-in alert module. For alert, we simply call alert.alert and it will appear.
[04:11 - 04:26] From the caller's perspective, we don't need to preserve a state like "is alert visible" and pass it as props into alert. We'd like to do something similar for our Android Prompt component here.
[04:27 - 04:38] In order to accomplish this, we will use ref. First, wrap the original Android Prompt component with React.forward ref.
[04:39 - 04:57] Once a function component is wrapped with 4 ref, it will receive a second argument, which is the ref object passing from its parent. Then we create our internal state, visible and hint text.
[04:58 - 05:20] Since those states are within the current component, we need to provide a method to let parent component to access them. We can use an use_effect_hook to accomplish this inside the hook.
[05:21 - 05:45] If we have a ref object passed in, we can set the current property of the ref object and put our state mutation function inside the current object. So the set visible and set hint text can be used by our parent component.
[05:46 - 06:13] And of course, we should pass our internal visible state into the modal component and display our internal hint text state. We also need to tackle the visible state back to false and clear the hint text when the cancel button is pressed.
[06:14 - 06:19] Now it's time to test them. Head back to our app.js.
[06:20 - 06:35] We first call use ref_hook to obtain a ref object and pass it into the Andrew prompt component. Then quickly create a test button.
[06:36 - 07:07] From the test buttons on press handler, we can simply call current.setvisible true. As you can see, we can now tackle our Andrew prompt component from parent and tackle it off by pressing its own cancel button without tracking the visible state from the parent component.
[07:08 - 07:17] Now it's the fun part. We'd like to add some animation into our Andrew prompt component.
[07:18 - 07:31] Import the animated module from ReNative. We also need an animated dot value initially set to 0.
[07:32 - 07:39] Then we create an underlined visible state. I will explain this in just a second.
[07:40 - 07:57] For now, just consider that we will use this underlined visible state instead of the previous visible state. Before we dive deeper, here's the thought process of creating animation.
[07:58 - 08:03] The first, what state triggered the animation? For us, that's the underlined visible state.
[08:04 - 08:19] The second, how to update the animated dot value object according to pyro state change. Third, how to update the style using the changing animated dot value object.
[08:20 - 08:35] To handle the first and second question, we create a new use effect hook. Depends on the underlined visible state and the anim value object.
[08:36 - 08:59] This hook function will be triggered when underlined visible value changed. And if this value is true, which means we are about to show our component, we first set the visible state to true and use animated dot timing to gradually change the anim value from 0 to 1.
[09:00 - 09:41] On the other hand, if the underlined visible is false, which means we are about to hide our component, we need to call animated dot timing first and then tackle the visible state to false in the completion handler, which is the callback passed into the start function. If we use the run order, the model will disappear instantly and no animation at all.
[09:42 - 09:58] That's why we need a second underlined visible state to signal the beginning of a transition. And the visible state is the one actually passed into our model component.
[09:59 - 10:18] After that, we change all the existing set visible into underlined set visible. Since its actual usage is to trigger the use effect hook, so we should only call set visible our effect hook.
[10:19 - 10:33] The next step is to update the style using the anim value. First, update the backdrop with the opacity.
[10:34 - 10:42] Test it and it looks pretty great. Then we can handle the animation for the prompt.
[10:43 - 10:59] The effect we want is a slide out effect. So we do a translate y transform and let the anim value to interpolate from 0 to 1.
[11:00 - 11:15] By the way, always remember only the animated view can have animation style in it, otherwise you will have some trouble. Test it again and it's awesome.
[11:16 - 11:35] The next step is to use our Android prompt component in our game. First, import it and set up the ref object.
[11:36 - 11:56] Then, when the NFC manager that registered tag event is called, we should set our Android prompt to be visible. By calling Android prompt ref.current.set visible true.
[11:57 - 12:16] And, when the user is playing, we can update the hint tags according to platforms. If it is Android, we will call our custom set hint tags function from ref.
[12:17 - 12:36] If it is iOS, we simply call set learn message iOS. Once the game is finished, we should set our Android prompt to be visible.
[12:37 - 13:11] Okay, it seems that we are ready to test it. Before testing, you will need to check the position of NFC antenna for your Android device, because they might be in different places according to different Android manufacturers.
[13:12 - 13:20] It looks pretty good. Though the app seems working correctly, but there is actually a potential bug.
[13:21 - 13:52] That is, once the user presses the cancel button in our Android prompt, we won 't actually stop the NFC from scanning, because our game component is not aware of it. To fix this issue, we first add the uncancel pressed into our Android prompt component, and call it when the cancel button is pressed.
[13:53 - 14:19] And then, go back to our game component, pass the uncancel pressed prompt into Android prompt component, and call unregistered tag event inside it. Besides the system scanning UI, another difference between Android and iOS.
[14:20 - 14:33] That is, NFC can be disabled in Android. So for Android app, before the game starts, we will need to confirm that the NFC is enabled.
[14:34 - 14:46] Let's do it now. First, create an enabled state, and in our check NFC function, we call NFC manager that is enabled.
[14:47 - 15:00] This API will resolve to a boolean value to indicate whether the NFC is enabled or not. And use this state in our render logic.
[15:01 - 15:24] If the NFC is supported but not enabled, we show a hint to our users, and provide a button for the user to jump to NFC system setting. By calling NFC manager that go to NFC setting, and please be aware, this API only works on Android.
[15:25 - 15:36] You might wonder, what about iOS? Actually, our NFC that is enabled will always return true since there is no NFC switch for iOS.
[15:37 - 15:51] So, we can confirm that the runtime execution flow will never reach here. Finally, we also provide a button to let the user recheck the NFC enabled state .
[15:52 - 16:01] It's about time to test our modifications. We first disable NFC, and then launch our app.
[16:02 - 16:11] As you can see, our UI hint at users about their NFC state. Then hit "Go to settings" to enable NFC.
[16:12 - 16:19] Then back to our app again, recheck the state, and now the user can enter the game. Cool.
[16:20 - 16:29] Now we have a simple but fully functional NFC tech counter game app for both Android and iOS.