# Let's build a Calculator App (React Native)

Welcome back to the [Let's build Something Cool](https://favouritejome.hashnode.dev/series/lets-build-something-cool) series, where we build cool things, at least from my perspective 😊, but really it's a series where we go in-depth about the cool things I build, but then we get to build it together.

In this article, we'll build a Calculator application similar to Samsung's calculator app, and use [Expo](https://expo.dev/), a framework that allows us to get started with React Native applications quickly.

> There are a bunch of requirements we'll need before building in the Calculator application, but if you'll like to skip the requirements section cause you're familiar with React Native and Expo, [click here to do so](#heading-getting-started).

## Prerequisite

* Basic knowledge of React
    
* Basic knowledge of React Native
    

## Requirements

### Install Node

Node is pretty much essential, so visit [Nodejs.org](https://nodejs.org/en/) and download the Long Term Support (LTS) version of Nodejs.

Once you're done downloading and installing Node.js, run the command below to know if it's up and running on your machine.

```bash
node -v
```

### **Install Expo**

There are two tools we'll need to set up Expo, first is the [Expo CLI](https://docs.expo.dev/get-started/installation/#expo-cli) which serves the project, and second is the client app called [Expo GO](https://docs.expo.dev/get-started/installation/#expo-go-app-for-android-and-ios) which allows us to view the project on either an Andriod or IOS device.

**Install Expo CLI**

```bash
npm install -g expo-cli
```

**Install Expo Go**

* [Android Play Store](https://play.google.com/store/apps/details?id=host.exp.exponent)
    
* [iOS App Store](https://apps.apple.com/app/expo-go/id982107779)
    

> Further guides for the [Expo installation can be found here.](https://docs.expo.dev/get-started/installation/)

## Getting Started

Firstly we'll need to install the necessary dependencies to get building, and the first would be Expo.

```bash
npx create-expo-app calculator-app && cd calculator-app
```

Once the installation process is completed and we've changed our directory to `calculator-app`, let's start our app, making sure it works.

### Run the Project

```bash
npm start
```

`yarn start` would also work if you have yarn installed.

On the terminal, you'll see a QR code, which we'll need to scan with the [Expo Go app](https://play.google.com/store/apps/details?id=host.exp.exponent), which I guess you've downloaded from Play Store already.

> But if you've not got the Expo app yet, [download it here for Android](https://play.google.com/store/apps/details?id=host.exp.exponent) and here for [iOS](https://apps.apple.com/app/expo-go/id982107779) to follow along with the article.

To work with the Expo Go app and our project on our computers, we would need to be connected to the same network; this is very important.

The terminal interface should look like this when the `start` command is run:

![npm start](https://cdn.hashnode.com/res/hashnode/image/upload/v1673733081635/309e00d2-e30b-4de9-a295-4e27bd011a1a.png align="center")

Now, we need to open the Expo Go app on our mobile devices, scan the QR code, and we should see this interface on our phones.

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1674087239751/97c23339-e773-437c-86ac-c4d11bab48fc.jpeg align="center")

### Folder Structure

Let's go over what the folder structure entails:

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1673733844349/edc19858-73a1-411b-a886-5fe7e0b29364.png align="center")

> Note: As of the Expo version `47.0.12`, this is how the folder structure is

* /.expo - this folder got created when we started the application and contains devices.json and README.md files. Please go through the README file to know what this folder is about, but really we can ignore it.
    
* /assets - this is where we store any assets we may want to use in our application, like images, videos, fonts and icons
    
* /node\_modules - these are all the packages we have installed that allow Expo and React Native to run. Further packages installed with `npm install` get added here.
    
* .gitignore - this is a file to tell .git which files and directories we want to omit from source control (Git). node\_modules and .expo directories are omitted by default.
    
* App.js - the main entry point to our application. This is where you'll start adding code!
    
* app.json - another config file that contains expo metadata
    
* babel.config.js - babel config, used to add the expo preset
    
* package.json - this is where you list dependencies and add scripts
    
* package-lock.json - this file shows the map of all the dependencies installed
    

### Breakdown of the Calculator Layout

The calculator app UI can be broken down into four main components we'll need to build; first, an input box where the calculations would be done, referred to as the **Calculation Input** from the image below.

Then we have the **Preview Calculation** component, which shows a preview of the answer when a valid operation occurs; next, we have **Rows** of **Buttons**.

![Samsung Calculator app](https://cdn.hashnode.com/res/hashnode/image/upload/v1673902164713/cc7afd54-8a44-48d5-9261-63511f101360.jpeg align="center")

> *I'm sorry for the poor image quality, I'll look for better ways to edit images next time 😅*

### Coding Time

Let's open the `App.js` file, we should have the snippet of code below or similar in the file:

```javascript
import { StatusBar } from 'expo-status-bar';
import { StyleSheet, Text, View } from 'react-native';

export default function App() {
  return (
    <View style={styles.container}>
      <Text>Open up App.js to start working on your app!</Text>
      <StatusBar style="auto" />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
});
```

We'll need to create the Calculation Input first, so remove the `<Text>` component and replace it with an `<TextInput />` component.

```javascript
import { StatusBar } from 'expo-status-bar';
import { StyleSheet, Text, View } from 'react-native';

export default function App() {
  return (
    <View style={styles.container}>
      <TextInput value="Here's were we type our calculation!" />
      <StatusBar style="auto" />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
});
```

Here's what we should have on the Expo Go app on our devices.

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1673911341890/21c74f29-9f69-4c79-81c0-22a94b9b4cba.png align="center")

Notice the input is at the centre of the screen; let's fix that and add some colours.

```javascript
import { StatusBar } from "expo-status-bar";
import { StyleSheet, Text, TextInput, View } from "react-native";

export default function App() {
  return (
    <View style={styles.container}>
      <TextInput
        style={styles.input}
        value="Here's were we type our calculation!"
      />
      <StatusBar style='auto' />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: "#010101",
    paddingVertical: 30,
    paddingHorizontal: 20,
  },
  input: {
    marginTop: 40,
    borderWidth: 1,
    padding: 10,
    color: "#fff",
    fontSize: 36,
  },
});
```

Now the UI should look like this, which is much better.

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1673911915126/5991d2f6-d201-4c59-8751-ae4ce76d0cf4.png align="center")

Let's add state to the `<TextInput />` to keep track of the calculating number and operators entered.

```javascript
...
export default function App() {
  const [calValue, setCalValue] = useState("");
  return (
    <View style={styles.container}>
      <TextInput
        style={styles.input}
        value={calValue}
        onChangeText={setCalValue}
        cursorColor='#8ad8d1'
        autoFocus={true}
        textAlign='right'
        showSoftInputOnFocus={false}
      />
      <StatusBar style='auto' />
    </View>
  );
}
...
```

The Calculation Input is all set, but notice the `showSoftInputOnFocus={false}` prop on the `<TextInput />` component. The prop is added to make sure the phone keyboard doesn't show when the input is on focus cause we don't need the phone's keyboard; we'll create ours.

Let's move to the Preview Calculation Input, which follows the same pattern we did for the calculation input.

```javascript
import { StatusBar } from "expo-status-bar";
import { useState } from "react";
import { StyleSheet, Text, TextInput, View } from "react-native";

export default function App() {
  const [calValue, setCalValue] = useState("");
  const [previewValue, setPreviewValue] = useState("");

  return (
    <View style={styles.container}>
      <TextInput
        style={styles.input}
        value={calValue}
        onChangeText={setCalValue}
        cursorColor='#8ad8d1'
        autoFocus={true}
        textAlign='right'
        showSoftInputOnFocus={false}
      />
      <TextInput
        value={previewValue}
        onChangeText={setPreviewValue}
        cursorColor='#8ad8d1'
        textAlign='right'
        caretHidden={true}
        style={[styles.input, styles.prevInput]}
        showSoftInputOnFocus={false}
      />
      <StatusBar style='auto' />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: "#010101",
    paddingVertical: 30,
    paddingHorizontal: 20,
  },
  input: {
    marginTop: 40,
    borderWidth: 1,
    padding: 10,
    color: "#fff",
    fontSize: 36,
  },
  prevInput: {
    fontSize: 24,
    marginTop: 40,
    marginBottom: 30,
    color: "#616161",
  },
});
```

Next, let's move the backspace button; for this, we'll need to install a package `@expo/vector-icons` which comes with the icon we need. Run the command below:

```bash
npm install @expo/vector-icons
```

Once it's installed, we'll need to add the backspace icon to the code and its functionality.

```javascript
...
import { Pressable, StyleSheet, Text, TextInput, View } from "react-native";

export default function App() {
  const [calValue, setCalValue] = useState("");
  const [previewValue, setPreviewValue] = useState("");

  const handleBackSpace = () => {
    const remainValue = calValue.slice(0, calValue.length - 1);
    setCalValue(() => remainValue);
  };

  return (
    <View style={styles.container}>
      <TextInput
        style={styles.input} ...
      />
      <TextInput
        value={previewValue} ...
      />

      <View style={styles.backButton}>
        <Pressable
          onPress={() => handleBackSpace()}
          disabled={calValue ? false : true}
        >
          <Ionicons
            name='md-backspace-outline'
            size={24}
            color={calValue ? "green" : "#035903"}
          />
        </Pressable>
      </View>

      <View
        style={{
          height: 1,
          backgroundColor: "#575757",
          marginVertical: 20,
        }}
      />
      <StatusBar style='auto' />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {...},
  input: {...},
  prevInput: {...},
  backButton: {
    flexDirection: "row",
    justifyContent: "flex-end",
  },
});
```

Notice from the snippet above the `Pressable` component is used to indicate the icon would be clicked and has two props, `onPress` which listens to the pressing of the icon, which then calls the `handleBackSpace` function, and the second prop is `disabled` , which makes the icons not pressable when there's no value in the Calculation input.

The `handleBackSpace()` function removes the last value in the `calValue` state and then update the `calValue` state.

The self-closing `View` component is to indicate a line in the UI. Now the UI should look like this:

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1673918515288/77020d50-613a-4263-8966-ae7d887e874d.png align="center")

From the UI, we know we need a row of buttons, so let's create a component named `Row` but still, in this `App.js` file cause, it isn't a complicated component.

```javascript
...
export default function App() {
...
}

function Row({ children }) {
  return <View style={styles.row}>{children}</View>;
}

const styles = StyleSheet.create({
  ...
  row: {
    flex: 1,
    flexDirection: "row",
    justifyContent: "space-between",
  },
});
```

That's all for the `Row` component; it basically composes several button components, which we'll be working on next.

Let's create a new component named `Button.js` in a components directory from the root directory, that is:

```bash
|-components
| -- Button.js
App.js
```

```javascript
// Button.js

import { TouchableHighlight, Text, StyleSheet } from "react-native";
import React from "react";

export default function Button({ label, type, handlePress, icon }) {
  const getTextColor =
    type === "operator" ? "green" : type === "clear" ? "red" : "#fff";

  const getButtonColor =
    type === "equal"
      ? { backgroundColor: "green" }
      : { backgroundColor: "#171717" };

  return (
    <TouchableHighlight
      underlayColor={type === "equal" ? "#04ac04" : "#444444"}
      style={[styles.button, getButtonColor]}
      onPress={() => handlePress(label)}
    >
      {icon ? (
        icon
      ) : (
        <Text style={{ fontSize: 26, color: getTextColor, fontWeight: "bold" }}>
          {label}
        </Text>
      )}
    </TouchableHighlight>
  );
}

const styles = StyleSheet.create({
  button: {
    borderRadius: 50,
    backgroundColor: "#171717",
    width: 80,
    height: 80,
    alignItems: "center",
    justifyContent: "center",
  },
});
```

From the code above, we can see there's a lot going on. The Button component accepts four pros `{ label, type, handlePress, icon }` where

* `Label` is the text or number that the button should display
    
* `type` prop is to identify what the button does; is it going to be for an **operator** or **clear** (which removes the numbers and operator for the Calculation Input) or **number**.
    
* `handlePress` prop is a function that takes in the `label` as an argument and does the necessary operation with it, which we'll get into later on.
    
* `icon` prop is an icon component that will be displayed in the button instead of the `label` if available.
    

Now, let's add the button components to the \`App.js\` file and their various functionalities.

```javascript
import { StatusBar } from "expo-status-bar";
import { useState } from "react";
import { Pressable, StyleSheet, Text, TextInput, View } from "react-native";
import { Ionicons, FontAwesome5, FontAwesome } from "@expo/vector-icons";
import { MaterialCommunityIcons } from "@expo/vector-icons";
import Button from "./components/Button";

export default function App() {
  const [calValue, setCalValue] = useState("");
  const [previewValue, setPreviewValue] = useState("");
  const [isAnswer, setIsAnswer] = useState(false);

  const ansColor = {
    color: isAnswer ? "green" : "white",
  };

  const handleBackSpace = () => {
    const remainValue = calValue.slice(0, calValue.length - 1);
    setCalValue(() => remainValue);
  };

  const handlePress = (text) => {
    setIsAnswer(false);
    const corrText = text === "X" ? "*" : text === "+/-" ? "-" : text;
    setCalValue((prev) => prev + `${corrText}`);
  };

  const handleClear = () => {
    setCalValue("");
  };

  const handleEqual = () => {
    if (!previewValue) return;
    setCalValue(previewValue);
    setPreviewValue("");
    setIsAnswer(true);
  };

  return (
    <View style={styles.container}>
      <TextInput
        style={[styles.input, ansColor]}
        value={calValue}
        onChangeText={setCalValue}
        cursorColor='#8ad8d1'
        autoFocus={true}
        textAlign='right'
        showSoftInputOnFocus={false}
      />
      <TextInput
        value={previewValue}
        onChangeText={setPreviewValue}
        cursorColor='#8ad8d1'
        textAlign='right'
        caretHidden={true}
        showSoftInputOnFocus={false}
        style={[styles.input, styles.prevInput]}
      />

      <View style={styles.backButton}>
        <Pressable
          onPress={() => handleBackSpace()}
          disabled={calValue ? false : true}
        >
          <Ionicons
            name='md-backspace-outline'
            size={24}
            color={calValue ? "green" : "#035903"}
          />
        </Pressable>
      </View>

      <View
        style={{
          height: 2,
          backgroundColor: "#575757",
          marginVertical: 20,
        }}
      />

      <View style={styles.buttonContainer}>
        <Row>
          <Button handlePress={handleClear} label={"C"} type='clear' />
          <Button
            handlePress={handlePress}
            label={"()"}
            type='operator'
            icon={
              <MaterialCommunityIcons
                name='code-parentheses'
                size={30}
                color='green'
                style={{ fontWeight: "bold" }}
              />
            }
          />
          <Button handlePress={handlePress} label={"%"} type='operator' />
          <Button
            handlePress={handlePress}
            label={"/"}
            type='operator'
            icon={<FontAwesome5 name='divide' size={24} color='green' />}
          />
        </Row>
        <Row>
          <Button handlePress={handlePress} label={"7"} />
          <Button handlePress={handlePress} label={"8"} />
          <Button handlePress={handlePress} label={"9"} />
          <Button
            handlePress={handlePress}
            label={"X"}
            type='operator'
            icon={<FontAwesome5 name='times' size={24} color='green' />}
          />
        </Row>
        <Row>
          <Button handlePress={handlePress} label={"4"} />
          <Button handlePress={handlePress} label={"5"} />
          <Button handlePress={handlePress} label={"6"} />
          <Button
            handlePress={handlePress}
            label={"-"}
            type='operator'
            icon={<FontAwesome name='minus' size={24} color='green' />}
          />
        </Row>
        <Row>
          <Button handlePress={handlePress} label={"1"} />
          <Button handlePress={handlePress} label={"2"} />
          <Button handlePress={handlePress} label={"3"} />
          <Button
            handlePress={handlePress}
            label={"+"}
            type='operator'
            icon={<FontAwesome5 name='plus' size={24} color='green' />}
          />
        </Row>
        <Row>
          <Button handlePress={handlePress} label={"+/-"} />
          <Button handlePress={handlePress} label={"0"} />
          <Button handlePress={handlePress} label={"."} />
          <Button
            handlePress={handleEqual}
            label={"="}
            type='equal'
            icon={<FontAwesome5 name='equals' size={24} color='white' />}
          />
        </Row>
      </View>

      <StatusBar style='auto' />
    </View>
  );
}

function Row({ children }) {
  return <View style={styles.row}>{children}</View>;
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: "#010101",
    paddingVertical: 30,
    paddingHorizontal: 20,
  },
  input: {
    marginTop: 40,
    borderWidth: 1,
    padding: 10,
    color: "#fff",
    fontSize: 36,
  },
  prevInput: {
    fontSize: 24,
    marginTop: 40,
    marginBottom: 30,
    color: "#616161",
  },
  backButton: {
    flexDirection: "row",
    justifyContent: "flex-end",
  },
  row: {
    flex: 1,
    flexDirection: "row",
    justifyContent: "space-between",
  },
  buttonContainer: {
    flex: 4,
  },
});
```

From the snippet above, we can tell a lot is going on, so I've created an Expo snack for this point of the application so that we can see and play around with the application at this point, just in case you haven't been coding along 😊.

<iframe src="https://snack.expo.dev/embedded/@favouritejome/calculator-app-01?iframeId=zin9l29o0c&amp;preview=true&amp;platform=web&amp;theme=dark" height="100%" width="100%" style="display:block;overflow:hidden;background:#212121;border:1px solid var(--color-border);border-radius:4px;height:505px;width:100%"></iframe>

> To get the best experience please switch to the Andriod / iOS tab below (depending on your device), and scan the QR code to test on your mobile device with the Expo Go app.

The `Row` component composes of four `Button` components and each Button has a different or similar prop value. For example, the button for the equal operation, the prop `handlePress` uses the `handleEqual()` function, which sets the preview Calculation to the **Calculation Input.**

The `handlePress()` function takes a text, and if the text is `X` it returns `*` and if `+\-` it returns `-` to the `corrText` variable, which then added to the previous value of the `calValue` state, also, the `setIsAnswer(false)` function is made false.

Notice also a new state is added `const [isAnswer, setIsAnswer] = useState(false)` , and it's to keep track of when the arithmetic calculation is correct and determine what colour the Calculation Input should be, either green or white, in the `ansColor` variable, which is added to the Calculation Input styles.

> Just to be clear , Calculation Input is referred to the very first `<TextInput />` component, while Preview Input is the second `<TextInput />`.

Let's move to the preview input and make it work:

```javascript
...
import { useEffect, useState } from "react";
...

function valueHasOp(text) {
  if (text.length === 0) return false;
  if (
    text[text.length - 1] === "+" ||
    text[text.length - 1] === "-" ||
    text[text.length - 1] === "*" ||
    text[text.length - 1] === "/" ||
    text[text.length - 1] === "%" ||
    text[text.length - 1] === "(" ||
    text[text.length - 1] === ")"
  )
    return false;

  for (let i = 0; i <= text.length; i++) {
    if (
      text[i] === "+" ||
      text[i] === "-" ||
      text[i] === "*" ||
      text[i] === "/" ||
      text[i] === "%"
    ) {
      return true;
    }
  }
  return false;
}

export default function App() {
  const [calValue, setCalValue] = useState("");
  const [previewValue, setPreviewValue] = useState("");
  const [isAnswer, setIsAnswer] = useState(false);

  useEffect(() => {
    if (valueHasOp(calValue)) {
      let prevAns = eval(calValue);
      setPreviewValue(`${prevAns}`);
    } else {
      setPreviewValue(``);
    }
  }, [calValue]);

 ...

 return (
    ...
 )
}
```

<iframe src="https://snack.expo.dev/embedded/@favouritejome/calculator-app-02?iframeId=ti91x2koh5&amp;preview=true&amp;platform=web&amp;theme=dark" height="100%" width="100%" style="display:block;overflow:hidden;background:#212121;border:1px solid var(--color-border);border-radius:4px;height:505px;width:100%"></iframe>

From the snippet above, there's a `useEffect` hook, which checks for changes in the `calValue` state, and in its callback, there's a check with the `valueHasOp()` function, which checks if the string `calValue` has an operator, for example, if `calValue` is `2*2` the function returns `true`, if `calValue` is `2*` it returns `false` cause that isn't a valid operation, also if `calValue` is an empty string, it returns `false`.

If the `valueHasOp(calValue)` function returns `true` then this expression is executed ``Function(`return ${calValue}`)()`` which evaluate JavaScript expressions as strings. If you've seen the `eval()` method before, it is similar to the `Funtion()` method.

> If you'll like to learn more about the `Function` method, do read this guide: [Function - JavaScript | MDN (](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function)[mozilla.org](http://mozilla.org)[)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function) and also this article to know the difference between `eval()` and `Function()` : [eval() vs. Function() in JavaScript (](https://www.educative.io/answers/eval-vs-function-in-javascript)[educative.io](http://educative.io)[)](https://www.educative.io/answers/eval-vs-function-in-javascript)

The returned value from the ``Function(`return ${calValue}`)()`` method is set in the ``setPreviewValue(`${prevAns)`)`` but if the `valueHasOp(calValue)` function returns `false` then `setPreviewValue(``)` method is made an empty string for nothing should be previewed.

At this point, the calculator application's main functionality works, but then there is a bug with selecting a portion of the Calculation Input to add a new number. To demonstrate the bug, see the video below:

![](https://cdn.hashnode.com/res/hashnode/image/upload/v1674071876702/113fa48b-a285-4ca2-a449-143875522a6a.gif align="center")

For us to fix this bug, we'll need to know where the selection cursor is at every time on the Calculation Input (first TextInput); therefore, we need to listen to every change on the selection. Luckily, the React Native `<TextInput />` component allows us to keep track of this with some props.

```javascript
...

export default function App() {
    const [calValue, setCalValue] = useState("");
    const [previewValue, setPreviewValue] = useState("");
    const [isAnswer, setIsAnswer] = useState(false);
    const [cursorSel, setCursorSel] = useState({ end: 0, start: 0 });
    const [isCursorSel, setIsCursorSel] = useState(false);

    ...

    return (
        <View style={styles.container}>
          <TextInput
            style={[styles.input, ansColor]}
            value={calValue}
            onChangeText={setCalValue}
            selection={cursorSel}
            cursorColor='#8ad8d1'
            autoFocus={true}
            textAlign='right'
            onSelectionChange={(e) => {
              setIsCursorSel(true);
              setCursorSel(e.nativeEvent.selection);
            }}
            showSoftInputOnFocus={false}
          />
           <TextInput
             value={previewValue}...
           />

            ...
        </View>
    )
}
```

The above snippet shows only the changes; we'll need two new states. The first state `const [cursorSel, setCursorSel] = useState({ end: 0, start: 0 })` keeps track of the selection cursor end and start property, while the second state `const [isCursorSel, setIsCursorSel] = useState(false)` track if the selection cursor has been changed.

On the Calculation Input, two new props were added, `selection` which take in the `cursorSel` state and `onSelectionChange` an event handler. The `e.nativeEvent.selection` expression gets the current location of the selection cursor, which is an object like this `{start: 0, end: 0}`.

```javascript
onSelectionChange={(e) => {
   setIsCursorSel(true);
   setCursorSel(e.nativeEvent.selection);
}}
```

Even though we now keep track of when the selection cursor is moved with the `onSelectionChange` event handler, we haven't solved the bug. To solve the bug, every time a number is added to the Calculation Input, we need to know where the selection cursor is located and add the number to that spot. Therefore we'll need to change the functionality of the `handlePress` function.

```javascript
const handlePress = (text) => {
    setIsAnswer(false);
    const corrText = text === "X" ? "*" : text === "+/-" ? "-" : text;

    setCursorSel({ end: cursorSel.end + 1, start: cursorSel.start + 1 });
    setCalValue((prev) => {
      if (prev.length !== cursorSel.end && isCursorSel) {
        let leftOver = prev.slice(0, cursorSel.end);
        let rightOver = prev.slice(cursorSel.end, prev.length);

        return `${leftOver}` + `${corrText}` + `${rightOver}`;
      }
      return prev + `${corrText}`;
    });
  };
```

Looking at the above snippet, it might look scary, but really what going on is that we first set the current selection cursor position with the `setCursorSel` function and then in the `setCalValue` function, there is a check if the previous value length `prev` is not equal to the `cursorSel.end` and `isCursorSel` is `true`.

Let's say the `prev` value is `5*4` and the `cursorSel.end` is at `0` , which means the selection cursor is next to `5` .

The `leftOver` variable would be `"5*4".slice(0, 0)` which returns an empty string `""`

The `rightOver` variable would be `"5*4".slice(0,3)` which returns `5*4`

Now, if the `text` parameter is `"4"`, then `corrText` would be `"4"`

Put all together would be `"" + "4" + "5*4"` which evaluate to `"45*4"` .

> This is a hacky solutions I know, but it works 😅

Okay, so that is it; we've successfully solved the bug; if you'll like to play around with it on your device, use the Expo snack below along with the Expo Go app; it's not suitable for the web.

<iframe src="https://snack.expo.dev/embedded/@favouritejome/calculator-app3?iframeId=wt1gztt4p7&amp;preview=true&amp;platform=web&amp;theme=dark" height="100%" width="100%" style="display:block;overflow:hidden;background:#212121;border:1px solid var(--color-border);border-radius:4px;height:505px;width:100%"></iframe>

**GitHub repo:** [jomefavourite/calculator-native at blog branch (](https://github.com/jomefavourite/calculator-native/tree/blog)[github.com](http://github.com)[)](https://github.com/jomefavourite/calculator-native/tree/blog)

## Conclusion

If you've followed along with the article to this point, thank you very much, and if you can't tell, I'm still new to React Native 😊, and this is a project I built by myself without a guide just to get my hands dirty, and I can say React Native is really cool.

If you have suggestions on how the app could be improved, please leave a comment; also, I know there are other bugs. If you can fix them, do message on Twitter [Favourite Jome (@FavouriteJome1)](https://twitter.com/FavouriteJome1).

Thanks for reading once again, and see you in the next [Let's build Something Cool](https://favouritejome.hashnode.dev/series/lets-build-something-cool) series 🤩
