How to Create AI-Generated Images on iOS in React Native Using Stable Diffusion
2,871
Have you ever wondered if it's possible to generate images using Stable Diffusion natively on your iPhone or iPad while taking advantage of Core ML in an Expo and React Native app?
Well, I was too!
Building The Module
After doing some research on Google and reading numerous articles, I came across a Swift Package called Stable Diffusion that was released in December last year by Apple.
Good to know:
Stable Diffusion
Swift Package allows us to run converted Core ML Stable Diffusion models natively on Apple Silicon, 100% offline and free without relying on an external API! 🎉
Unfortunately, there is a good lack of documentation on how to use this package. So, I had to reverse engineer the swift code myself and try to figure out what functions to call and with what arguments.
But, that was only the first step, as my initial goal was to be able to run Stable Diffusion models in a React Native app which makes things even more complicated...
Luckily, the Expo team has done an amazing job at releasing Expo Modules API. It's an abstraction layer on top of JSI and other low-level primitives that React Native is built upon which allows us to write native modules easier than ever. I highly recommend having a look at their API if you are considering creating a native module for React Native yourself!
🧠 If are interested in how I managed to build the expo-stable-diffusion module using the
Expo Modules API
, let me know on Twitter and might write an article on this topic in the future! 😊
While trying to navigate these challenging waters, I have documented some parts of my early struggles in one of my recent tweets. 😁
How to Download and Convert Stable Diffusion Models?
Before diving into how to use the expo-stable-diffusion
module, we first need a converted Core ML Stable Diffusion model which we'll use to generate images.
Currently, there are 3 options on how to obtain a converted model:
1. Convert Your Own Model
You can convert your own model by following Apple's official guide. This is the Python snippet that I use to convert my model which you can also use:
_11python -m python_coreml_stable_diffusion.torch2coreml \_11 --model-version stabilityai/stable-diffusion-2-1-base \_11 --convert-unet \_11 --convert-text-encoder \_11 --convert-vae-decoder \_11 --convert-safety-checker \_11 --chunk-unet \_11 --attention-implementation SPLIT_EINSUM_V2 \_11 --compute-unit ALL \_11 --bundle-resources-for-swift-cli \_11 -o models/stable-diffusion-2-1/split_einsum_v2/compiled
💡 Make sure to have a look at Apple's official guide mentioned above to fully understand what these arguments mean as your use case may be different than mine!
2. Download a Converted Model
You can download an already converted model from Apple's official Hugging Face repo. Hugging Face doesn't allow downloading individual folders, so you would have to use git-lfs
in order to download the whole repo which can take some time if you have a slow internet.
However, you can use the following Python script which will allow you to download a specific folder instead:
_10from huggingface_hub import snapshot_download_10from pathlib import Path_10_10repo_id = "apple/coreml-stable-diffusion-2-1-base"_10variant = "split_einsum/compiled"_10_10model_path = Path("./models") / (repo_id.split("/")[-1] + "_" + variant.replace("/", "_"))_10snapshot_download(repo_id, allow_patterns=f"{variant}/*", local_dir=model_path, local_dir_use_symlinks=False)_10print(f"Model downloaded at {model_path}")
3. Download My Model
Using the Python script that I mentioned in the 1st step, I managed to convert my own model that is highly optimized for the current iOS 16 devices. You can download the model from my Hugging Face repo.
✅ You can download the compressed model in the
.zip
file format directly from my Hugging Face repo using this link!
Optional: Sending The Model To Your iPhone/iPad
In case you want to send your model to your iPhone/iPad from your Mac for testing purposes, you can use the following idb command which makes testing a lot easier!
_10idb file push --udid <DEVICE UDID> --bundle-id <APP BUNDLE ID> <Model Path>/models/stable-diffusion-2-1/split_einsum_v2/compiled/Resources/* /Documents/.
How to Use expo-stable-diffusion
Module?
Okay, now that we got our model, let's dive into how to install and use the expo-stable-diffusion
module!
1. Install
First things first, we have to install the expo-stable-diffusion
module in our Expo project:
_10npx expo install expo-stable-diffusion
2. Configure Expo Project
❗️ This package is not included in the Expo Go. You will have to use a Development Build or build it locally using Xcode!
2.1. Change iOS Deployment Target
One of the most important things to do in order for your project to build is to change the iOS Deployment Target to 16.2
because the Stable Diffusion
Swift Package is not supported on lower versions of iOS!
To change the Deployment Target, first, we need to install the expo-build-properties
plugin:
_10npx expo install expo-build-properties
Now we need to configure it by adding the following code to our app.json
file:
_14{_14 "expo": {_14 "plugins": [_14 [_14 "expo-build-properties",_14 {_14 "ios": {_14 "deploymentTarget": "16.2"_14 }_14 }_14 ]_14 ]_14 }_14}
2.2. Add the Increased Memory Limit
Capability
Apple also recommends adding the Increased Memory Limit capability to our iOS project to prevent the app from running out of memory.
Luckily this is quite easy to do in an Expo Managed app. Just add the following code to our app.json
file:
_10{_10 "expo": {_10 "ios": {_10 "entitlements": {_10 "com.apple.developer.kernel.increased-memory-limit": true_10 }_10 }_10 }_10}
2.3. Build Your iOS App
The last step is to build your iOS app by running the following 2 commands:
_10npx expo prebuild --clean --platform ios_10npx expo run:ios
3. Generate Images
Now that we have our module installed successfully and our Expo project configured properly, we can try to generate a simple image using the example below:
_25import * as FileSystem from 'expo-file-system';_25import * as ExpoStableDiffusion from 'expo-stable-diffusion';_25_25const MODEL_PATH = FileSystem.documentDirectory + 'Model/stable-diffusion-2-1';_25const SAVE_PATH = FileSystem.documentDirectory + 'image.jpeg';_25_25Alert.alert(`Loading Model: ${MODEL_PATH}`);_25_25await ExpoStableDiffusion.loadModel(MODEL_PATH);_25_25Alert.alert('Model Loaded, Generating Images!');_25_25const subscription = ExpoStableDiffusion.addStepListener(({ step }) => {_25 console.log(`Current Step: ${step}`);_25});_25_25await ExpoStableDiffusion.generateImage({_25 prompt: 'a cat coding at night',_25 stepCount: 25,_25 savePath: SAVE_PATH,_25});_25_25Alert.alert(`Image Generated: ${SAVE_PATH}`);_25_25subscription.remove();
Let's go through these 2 functions and explain what they do:
-
First, we load the model by calling the
ExpoStableDiffusion.loadModel(...)
function. In this case,modelPath
is the path to the directory holding all model and tokenization resources. I recommend storing the model in the app's document directory for easier access. -
Second, we subscribe to the current image generation step to measure progress by calling the
addStepListener(...)
listener. Don't forget to callsubscription.remove()
once the image is generated in order to prevent memory leaks! -
Finally, we call the
ExpoStableDiffusion.generateImage(...)
function to start the image generation process. Once it's finished, an image will be saved in thesavePath
location.
💡 If you are saving the image in a custom directory, make sure the directory exists. You can create a directory by calling the
FileSystem.makeDirectoryAsync(fileUri, options)
function fromexpo-file-system
.
❗️ The model load time and image generation duration take some time, especially on devices with lower RAM than 6GB! Find more information in Q6 in the FAQ section of the ml-stable-diffusion repo.
Good to know: You can also run
expo-stable-diffusion
directly in your simulator by providing the path to the model on your computer to themodelPath
argument. However from my testings, it doesn't use Apple's Neural Engine cores, so the model load and image generation duration will most likely be different!
Optional: Make Folders Appear in "Files" App on iOS
If you want to make the document directory of your app available so the user can see it in the "Files" app, all you need to do is add a few properties to your info.plist
file.
In our case, we can easily add the following code to our app.json
file:
_11{_11 "expo": {_11 "ios": {_11 "infoPlist": {_11 "UIFileSharingEnabled": true,_11 "LSSupportsOpeningDocumentsInPlace": true,_11 "UISupportsDocumentBrowser": true_11 }_11 }_11 }_11}
💡 Don't forget to build your iOS app again in order for the changes to take effect!
Running Stable Diffusion on Lower-End Devices
The current models can only run on higher-end devices such as iPhone 13 Pro, iPhone 14 Pro, iPad Pro and iPad Air which have more RAM than other lower-end devices. This is due to the model being quite heavy...
However, this year at WWDC, Apple updated the ml-stable-diffusion repo. It's now possible to convert and run Quantized Stable Diffusion Models on devices with lower RAM such as iPhone 12 Mini!
Using quantized models, based on Apple's official benchmarks, it seems that it takes around 20 seconds to generate an image on iPhone 12 Mini, which is quite impressive! However, quantized models can only run on devices with iOS 17 and macOS 14 which is currently in public beta.
I haven't run these models myself as I don't want to update my iPhone to iOS 17 yet. Also, we are only a month away from Apple finally releasing them publicly!
How You Can Help!
It will be very helpful if more developers will contribute to expo-stable-diffusion
module in any shape or form!
If you happen to be knowledgeable in this field, feel free to start a discussion on GitHub or shoot me a tweet directly with your suggestions.
In case you encounter any bugs with expo-stable-diffusion
, please open an issue on GitHub with the problem you are having.
And for the real adventurous developers 🗡️ who would like to get their hands dirty by writing some code, you can contribute by opening a pull request on GitHub. Your help will be greatly appreciated!
🌟 I will really appreciate if you could star the expo-stable-diffusion module on GitHub!
In case you have more specific questions about the expo-stable-diffusion
module, feel free to email me at: me@andreizgirvaci.com.
Take care and see you next time! 😊