react-native-qrcode-svg not working on eas build

I just created a new react-native expo app.

I installed a few dependencies:

{
  "name": "qrcode",
  "version": "1.0.0",
  "scripts": {
    "start": "expo start --dev-client",
    "android": "expo run:android",
    "ios": "expo run:ios",
    "web": "expo start --web"
  },
  "dependencies": {
    "expo": "46.0.0",
    "expo-dev-client": "~1.3.1",
    "expo-permissions": "~13.2.0",
    "expo-splash-screen": "~0.16.2",
    "expo-status-bar": "~1.4.2",
    "react": "18.0.0",
    "react-native": "0.69.6",
    "react-native-qrcode-svg": "^6.2.0",
    "react-native-svg": "^13.2.0",
    "expo-barcode-scanner": "~11.4.0"
  },
  "devDependencies": {
    "@babel/core": "^7.12.9"
  },
  "private": true
}

I am running on eas build instead of using the expo go app.
I am trying to use the react-native-qrcode-svg package but it doesn’t work

I always keep getting the error:

TypeError: _qrcode.default.create is not a function. (In '_qrcode.default.create(value, {
      errorCorrectionLevel: errorCorrectionLevel
    })', '_qrcode.default.create' is undefined)

I am extremely surprised of the lack of solution or direction to solve this error on the internet. I have never come across the anything like this tbh lol.

What have I tried?
Degrading my expo version from 47 to 46, equally degrading dependencies to fit the expo versions.

Please what could be the problem or a possible solution?

Hi @alex261816, did you try the using the workaround mention in a similar issue opened in the react-native-qrcode-svg library’s GitHub repo: app crashed on device after (expo) eas build -p but run at expo start · Issue #167 · awesomejerry/react-native-qrcode-svg · GitHub?

I’d suggest installing react-native-svg using npx expo install react-native-svg. This installs the compatible version of the package with the appropriate Expo SDK used in your project.

For SDK 46, the compatible version is "12.3.0".

The reason it might not have worked with SDK 47 or with react-native-svg version 13 is because of the details mentioned in this PR: react-native-svg 13 support by GunnarAK · Pull Request #168 · awesomejerry/react-native-qrcode-svg · GitHub. Seems like the package needs to upgrade to version 13. I’d suggest following this thread and asking the question in the library since this is a third party library.

1 Like

Hi @alex261816

Please post the react-native-qrcode-svg code you’re using.

This worked fine for me:

$ yarn create expo-app qrcode-svg
$ cd qrcode-svg
$ npx expo install react-native-svg react-native-qrcode-svg

Dependencies:

  "dependencies": {
    "expo": "~47.0.12",
    "expo-status-bar": "~1.4.2",
    "expo-updates": "~0.15.6",
    "react": "18.1.0",
    "react-native": "0.70.5",
    "react-native-qrcode-svg": "^6.2.0",
    "react-native-svg": "13.4.0"
  },
  "devDependencies": {
    "@babel/core": "^7.12.9"
  },

Code:

import { StatusBar } from "expo-status-bar";
import { StyleSheet, Text, View } from "react-native";
import QRCode from "react-native-qrcode-svg";

// Simple usage, defaults for all but the value
function SimpleQR() {
  return <QRCode value="http://awesome.link.qr" />;
}

// 30px logo from base64 string with transparent background
function Base64LogoQR() {
  let base64Logo = "";

  return (
    <QRCode
      value="Just some string value"
      logo={{ uri: base64Logo }}
      logoSize={30}
      logoBackgroundColor="transparent"
    />
  );
}

// 20% (default) sized logo from local file string with white logo backdrop
function LogoFromFileQR() {
  let logoFromFile = require("./assets/logo.png");

  return <QRCode value="Just some string value" logo={logoFromFile} />;
}

export default function App() {
  return (
    <View style={styles.container}>
      <SimpleQR />
      <Base64LogoQR />
      <LogoFromFileQR />
      <StatusBar style="auto" />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: "#fff",
    alignItems: "center",
    justifyContent: "space-around",
  },
});

I built this with eas build -p android --profile preview and installed it on a phone. See the screenshot below:

{
  "name": "qrcode",
  "version": "1.0.0",
  "scripts": {
    "start": "expo start --dev-client",
    "android": "expo run:android",
    "ios": "expo run:ios",
    "web": "expo start --web"
  },
  "dependencies": {
    "expo": "~47.0.12",
    "expo-status-bar": "~1.4.2",
    "react": "18.1.0",
    "react-native": "0.70.5",
    "react-native-qrcode-svg": "^6.2.0",
    "react-native-svg": "13.4.0",
    "expo-splash-screen": "~0.17.5",
    "expo-dev-client": "~2.0.1",
    "expo-barcode-scanner": "~12.1.0"
  },
  "devDependencies": {
    "@babel/core": "^7.12.9"
  },
  "private": true
}

Hello. This is my dependencies. Although the initial error has been fixed now thanks to your help, I installed the expo-barcode-scanner and I keep getting the error:

WARN  The native view manager required by name (ExpoBarCodeScannerView) from NativeViewManagerAdapter isn't exported by expo-modules-core. Views of this type may not render correctly. Exported view managers: [].
 ERROR  TypeError: undefined is not an object (evaluating '_ExpoBarCodeScannerModule.default.BarCodeType')
 ERROR  Invariant Violation: Failed to call into JavaScript module method AppRegistry.runApplication(). Module has not been registered as callable. Registered callable JavaScript modules (n = 11): Systrace, JSTimers, HeapCapture, SamplingProfiler, RCTLog, RCTDeviceEventEmitter, RCTNativeAppEventEmitter, GlobalPerformanceLogger, JSDevSupportModule, HMRClient, RCTEventEmitter.
        A frequent cause of the error is that the application entry file path is incorrect. This can also happen when the JS bundle 
is corrupt or there is an early initialization error when loading React Native.
 ERROR  Invariant Violation: Failed to call into JavaScript module method AppRegistry.runApplication(). Module has not been registered as callable. Registered callable JavaScript modules (n = 11): Systrace, JSTimers, HeapCapture, SamplingProfiler, RCTLog, RCTDeviceEventEmitter, RCTNativeAppEventEmitter, GlobalPerformanceLogger, JSDevSupportModule, HMRClient, RCTEventEmitter.
        A frequent cause of the error is that the application entry file path is incorrect. This can also happen when the JS bundle 
is corrupt or there is an early initialization error when loading React Native

What could be the cause?

EDIT: So I commented out the barcode scanner import and where it’s being used in my application, only to reload the app and I’m getting the exact same error as before. The initial error:

TypeError: _qrcode.default.create is not a function. (In '_qrcode.default.create(value, {  
      errorCorrectionLevel: errorCorrectionLevel
    })', '_qrcode.default.create' is undefined

I don’t know what else to do now. Please are you on eas build? I am not using expo go app.

My dependencies version is still exactly the same

Yes, I have tried this. Still didn’t work. Mind you I deleted my entire project and created a new one. It uses expo 47 now and "react-native-svg": "^13.4.0"

This is my package.json now:

{
  "name": "qrcode",
  "version": "1.0.0",
  "scripts": {
    "start": "expo start --dev-client",
    "android": "expo run:android",
    "ios": "expo run:ios",
    "web": "expo start --web"
  },
  "dependencies": {
    "expo": "~47.0.12",
    "expo-status-bar": "~1.4.2",
    "react": "18.1.0",
    "react-native": "0.70.5",
    "react-native-qrcode-svg": "^6.2.0",
    "react-native-svg": "13.4.0",
    "expo-splash-screen": "~0.17.5",
    "expo-dev-client": "~2.0.1",
    "expo-barcode-scanner": "~12.1.0"
  },
  "devDependencies": {
    "@babel/core": "^7.12.9"
  },
  "private": true
}

Hello??? Can’t anyone help?

And also some BarCodeScanner code.

This works for me:

  "dependencies": {
    "expo": "~47.0.12",
    "expo-barcode-scanner": "~12.1.0",
    "expo-status-bar": "~1.4.2",
    "expo-updates": "~0.15.6",
    "react": "18.1.0",
    "react-native": "0.70.5",
    "react-native-qrcode-svg": "^6.2.0",
    "react-native-svg": "13.4.0"
  },
  "devDependencies": {
    "@babel/core": "^7.12.9"
  },
import { useState, useEffect } from "react";
import { StatusBar } from "expo-status-bar";
import { Button, StyleSheet, Text, View } from "react-native";
import QRCode from "react-native-qrcode-svg";
import { BarCodeScanner } from "expo-barcode-scanner";

// Simple usage, defaults for all but the value
function SimpleQR() {
  return (
    <View style={{ borderWidth: 8, borderColor: "white", boderStyle: "solid" }}>
      <QRCode value="http://awesome.link.qr" />
    </View>
  );
}

// 30px logo from base64 string with transparent background
function Base64LogoQR() {
  let base64Logo =
    "";

  return (
    <View style={{ borderWidth: 8, borderColor: "white", boderStyle: "solid" }}>
      <QRCode
        value="Just some string value"
        logo={{ uri: base64Logo }}
        logoSize={30}
        logoBackgroundColor="transparent"
      />
    </View>
  );
}

// 20% (default) sized logo from local file string with white logo backdrop
function LogoFromFileQR() {
  let logoFromFile = require("./assets/logo.png");

  return (
    <View style={{ borderWidth: 8, borderColor: "white", boderStyle: "solid" }}>
      <QRCode value="Just some string value" logo={logoFromFile} />
    </View>
  );
}

export default function App() {
  const [hasPermission, setHasPermission] = useState(null);
  const [scanned, setScanned] = useState(false);

  useEffect(() => {
    const getBarCodeScannerPermissions = async () => {
      const { status } = await BarCodeScanner.requestPermissionsAsync();
      setHasPermission(status === "granted");
    };

    getBarCodeScannerPermissions();
  }, []);

  const handleBarCodeScanned = ({ type, data }) => {
    setScanned(true);
    alert(`Bar code with type ${type} and data ${data} has been scanned!`);
  };

  if (hasPermission === null) {
    return <Text>Requesting for camera permission</Text>;
  }
  if (hasPermission === false) {
    return <Text>No access to camera</Text>;
  }

  return (
    <View style={styles.container}>
      <BarCodeScanner
        onBarCodeScanned={scanned ? undefined : handleBarCodeScanned}
        style={StyleSheet.absoluteFillObject}
      />
      <SimpleQR />
      <Base64LogoQR />
      <LogoFromFileQR />
      {scanned && (
        <Button title={"Tap to Scan Again"} onPress={() => setScanned(false)} />
      )}
      <StatusBar style="auto" />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: "#fff",
    alignItems: "center",
    justifyContent: "space-around",
  },
});

I built with eas build -p android --profile preview and installed it on my Android phone.

import { StyleSheet, Text, View } from 'react-native'
import React, { useState } from 'react'
import SvgQRCode from 'react-native-qrcode-svg'
import * as Svg from 'react-native-svg';
import BarcodeScanner from './Modals.js/BarcodeScanner';

export default function App() {
    const [showScanner, setShowScanner] = useState(false)
    const handleBarCodeScanned = ({ type, data }) => {
        setScanned(true);
        alert(`Bar code with type ${type} and data ${data} has been scanned!`);
    };
    const closeScanner = () => {
      setShowScanner(false)
    }
  return (
    <View>
      <Text>Scan</Text>
      <SvgQRCode value='num=1000&id=1' />
      <BarcodeScanner showScanner={showScanner} closeScanner={closeScanner} handleBarCodeScanned={handleBarCodeScanned} />
    </View>
  )
}

const styles = StyleSheet.create({})

I build with eas build -p android --profile development

I have used QRcode before but it doesn’t work. I saw someone else use SvgQRCode and it worked but either still gives me the same error

As per the documentation this must be QRCode. Not SvgQRCode.

EDIT: Sorry, I wasn’t thinking straight. Because of the way it is being exported in the react-native-qrcode-svg code, it doesn’t matter what name you give it here.

As you can see from my code it works for me.

What is this?

It’s a modal containing the react native modal.

import { Modal, StyleSheet, Text, View } from 'react-native'
import React, { useState,useEffect } from 'react'
// import { BarCodeScanner } from 'expo-barcode-scanner';

export default function BarcodeScanner({ handleBarCodeScanned,closeScanner,showScanner }) {
    const [hasPermission, setHasPermission] = useState(null);
    const [scanned, setScanned] = useState(false);

    useEffect(() => {
        const getBarCodeScannerPermissions = async () => {
          const { status } = await BarCodeScanner.requestPermissionsAsync();
          setHasPermission(status === 'granted');
        };
        getBarCodeScannerPermissions();
    }, []);
  return (
    <Modal 
        transparent={false}
        visible={showScanner}
        onRequestClose={closeScanner}
        statusBarTranslucent={true}
    >
        <View>
          {/* <BarCodeScanner
            onBarCodeScanned={scanned ? undefined : handleBarCodeScanned}
            // style={StyleSheet.absoluteFillObject}
          /> */}
        </View>
    </Modal>
  )
}

const styles = StyleSheet.create({
    container:{
        flex:1
    }
})

I have changed the import from SvgQRCode to QRCode but still the same error. It’s so confusing how yours works but mine doesn’t but we have the same exact project.

How do you start your expo app?

Is it like mine using:
expo start --dev-client

or something else?

EDIT: Sorry, I wasn’t thinking straight about the SvgQRCode in my previous message. It doesn’t matter what name you give it when you import as long as you use that same name when calling it, as you are doing.

I’ve just installed expo-dev-client and imported it at the top of App.js:

import "expo-dev-client";
import { useState, useEffect } from "react";
import { StatusBar } from "expo-status-bar";
import { Button, StyleSheet, Text, View } from "react-native";
import QRCode from "react-native-qrcode-svg";
import { BarCodeScanner } from "expo-barcode-scanner";

// [...]

I then created a development build with:

eas build -p android --profile development

I installed that on my Android device and started the dev server:

npx expo start

then loaded the app in the dev build on my phone. It still works.

I then killed the dev server and started it like this:

npx expo start --dev-client

I killed and restarted the dev build on my phone and it still works.

Can you create a new app, install the necessary dependencies with npx expo install ..., replace the contents of App.js with the code that I posted above and see if that works?

Although it should not make a difference, I used yarn to create the app:

yarn create expo-app

Wow.
Works now.

I have a feeling it’s because I always ran expo prebuild before I then built the app to eas.

What about a situation where by I need to run expo prebuild? For instance to use onesignal or some other package?

What should I do then?

Please Try running expo prebuild and then build the app to eas and see if it still works on your end

If you are using config plugins and everything works fine for you then you do not need to run expo prebuild because it is automatically run by the build servers during the build process.

I generally avoid running expo prebuild unless I am trying to debug a config plugin or something like that. Also if you’re getting errors while building an app you might want to troubleshoot by building on your own machine, in which case you might use expo prebuild to generate the native code to build in Xcode or Android Studio. Or if you want to make changes to the native code, but then you can’t blindly run expo prebuild again because it will probably overwrite your changes to the native code.

If you run expo prebuild it basically switches you to the bare workflow instead of the managed workflow.

In your case it seems that you are not making changes to the native code or anything like that, so you should just switch back to the managed workflow and then avoid running expo prebuild or expo run:android / expo run:io after that.

See expo.fyi/prebuild-cleanup
The main thing is to delete the android and ios directories.

If for some reason you want to keep running expo prebuild then it seems that something is not quite right with the native code in your case, so you should run npx expo prebuild --clean to regenerate the native code from scratch. But I think it would be better to switch back to the managed workflow in your case.

EDIT: OneSignal has a config plugin, so you don’t need to prebuild. The config plugin makes the changes that you would otherwise have to make manually. If you are trying to integrate a library that does not have a config plugin, but needs you to make manual changes to the native code, then you might need to switch to the bare workflow, but it would be better to figure out how to write a config plugin for that library and stick to the managed workflow.

Yeah. I have tried using config plugins before and they are not consistent in terms of a successful application.

Example is trying to change targetsdkversion to 31 in accordance to google’s new policies for android 11. I have added config plugins for them but they do still do not work. I have to manually change them. And I use notifee to display notifications and I have to remove something in the android folder in order to allow notifee to display onesignal notifications.

I assume as the apps progresses, the need for the android/ios folder to be edited would become compulsory
So does this mean I can’t use the qrcode package if I run expo prebuild?

Have you tried running it and building to see if it works on your end?

At least part of this is probably because you ran expo prebuild.

This should work reliably, but it’s possible that something wasn’t regenerated properly if you previously ran expo prebuild.

I would need more details to figure out what’s going on there. It’s possible that “dangerous” config plugins might conflict with each other, though.

Not necessarily and I would resist making changes to the native code, because it makes upgrading to the next Expo SDK or version of React Native harder. Using config plugins is a way to avoid making manual changes to the native code.

No it should work fine with or without prebuild. react-native-qrcode-svg is written in pure JavaScript, but it depends on react-native-svg which includes native code. But even react-native-svg should have no problems with expo prebuild. If something has gone wrong with the native projects (and you still want to use expo prebuild) then you should make sure you know what changes you made manually and then run npx expo prebuild --clean before redoing those manual changes.

I suspect some combination of running expo prebuild multiple times with different Expo SDK versions and maybe combined with manual changes to the native code caused the issues you were having.

See also: Prebuild - Expo Documentation

I haven’t tried prebuilding on my machine before building, but as I said, the build server runs prebuild for you anyway if you have not done so. So for every app, either you have the native code already (e.g. because it’s a plain React Native app, or you have run npx expo prebuild to generate the native code) or the native code will be generated for you on the build server.

See step 7 of the remote part of the Android build process:

If the native code already exists (like in your case) that step will be skipped. In the case of the test app that I built, that step will execute and it would be the same as if I had run npx expo prebuild before building.