Creating UIs Using Xcode Storyboards¶
For complex UIs you can use Xcode’s Interface Builder for designing and Multi-OS Engine for generating the Java* bindings in Android Studio*. This topic demonstrates how to use this approach for building a sample Quiz App.
1. Create a Stock Android Application¶
data:image/s3,"s3://crabby-images/42d2a/42d2a24f9d8d5899493aef2f4d16916784bc85d6" alt="../../../_images/1-1.png"
2. Create a New Multi-OS Engine Module¶
Right click on the Android project and select New > Multi-OS Engine Module to create a new iOS* module.
data:image/s3,"s3://crabby-images/67c64/67c64ec557abad316eed0f3d26779e46ca9fc18b" alt="../../../_images/2-1.png"
Select “Single View Application with Storyboard” in the “Multi-OS Engine” category.
data:image/s3,"s3://crabby-images/b2147/b21478bd6aef11852a3f0e4439b9deda5285b1da" alt="../../../_images/2-2.png"
Fill out the fields and select “Keep Xcode Project”.
data:image/s3,"s3://crabby-images/4fda1/4fda15b4d363527027fafc5a75b3ddc97b78e245" alt="../../../_images/2-3.png"
Change the module name and paths to use “ios”.
data:image/s3,"s3://crabby-images/aa96d/aa96d773d4d2f8a569f3f943bef005d00e29adb7" alt="../../../_images/2-4.png"
3. Delete Unneeded Files¶
We will create the view controller and the storyboard from scratch, so let’s delete the generated ones. Select the “ui” package and “MainUI.storyboard” file and delete them.
data:image/s3,"s3://crabby-images/f90d2/f90d2dbbb9976c81d311c75b653c9e9c2f653145" alt="../../../_images/3-1.png"
Open the “Main” class and remove the incorrect import of AppViewController.
data:image/s3,"s3://crabby-images/eca33/eca338eebfbcc5042333a0c6e8e47778c7424803" alt="../../../_images/3-2.png"
4. Edit the Xcode Project¶
Change your view to “Project” in Android Studio.
data:image/s3,"s3://crabby-images/e16ec/e16ec0b66299d7bf9c6017924d1da1fdbd7ba9ad" alt="../../../_images/4-1.png"
Right-click on the “ios” module and select “Multi-OS Engine Actions” > “Open Project in Xcode”.
data:image/s3,"s3://crabby-images/80117/8011740dab6bc58c4e6a0276dccc7843c764853c" alt="../../../_images/5-1.png"
Delete the remaining reference to the removed “MainUI.storyboard” file in Xcode.
data:image/s3,"s3://crabby-images/c4335/c43356cda2bfa127ccd4c73d3368f7f57bff77a1" alt="../../../_images/6-1.png"
5. Create the View Controller and Storyboard¶
Right-click on the “QuizApp” group and select “New File…”.
data:image/s3,"s3://crabby-images/c1f7b/c1f7b8ac2cdb2482aeac1c2ece7d4ad09f43d5df" alt="../../../_images/6-2.png"
Select “Cocoa Touch Class” in the “iOS” > “Source” category.
data:image/s3,"s3://crabby-images/d4a19/d4a199c135bd69e2b0014479d11fa31623aa1a09" alt="../../../_images/6-3.png"
Rename the class to “MyViewController” and make it a subclass of “UIViewController”.
data:image/s3,"s3://crabby-images/03e1c/03e1cc9e1e4caad7cddac1d371b780e182b252a8" alt="../../../_images/6-4.png"
Save in the Xcode target’s (QuizApp) directory. (This should be the default save location)
data:image/s3,"s3://crabby-images/02c4d/02c4d2dc6a23385eaa8980e201e0e4deed3eb463" alt="../../../_images/6-5.png"
Create another new file, but now a “Storyboard” from the “iOS” > “User Interface” category.
data:image/s3,"s3://crabby-images/6f1c9/6f1c9a2ad6ddb3b2636ef9d19b17a2c6b0bf7e38" alt="../../../_images/6-6.png"
Save it as “Storyboard” to the same location.
data:image/s3,"s3://crabby-images/aaaf2/aaaf27fa1c321fa3afbc985c3e1dc87c08fa17c5" alt="../../../_images/6-7.png"
6. Edit the Storyboard¶
Open up the Storyboard file for editing and drag a “View Controller” into the big empty editor area.
data:image/s3,"s3://crabby-images/6b313/6b3130f92bdf05395eb0a064de77eb0ee7439b0c" alt="../../../_images/7-1.png"
In the Attributes inspector check in the “Is Initial View Controller” checkbox.
data:image/s3,"s3://crabby-images/8b354/8b354bd457dcee04a70f85b53b11decc05e40973" alt="../../../_images/7-2.png"
In the Identity inspector change the view controller’s class to MyViewController.
data:image/s3,"s3://crabby-images/0493c/0493c02fa8fa6d8174d1bcf5e9b6017e75f5996e" alt="../../../_images/7-3.png"
7. Add UI Elements¶
Add two labels by searching for “label” and dragging them onto the view controller’s view.
data:image/s3,"s3://crabby-images/1d7ce/1d7ceed0074d9ac3d726968e23417614705d6469" alt="../../../_images/8-1.png"
Do the same for adding two buttons.
data:image/s3,"s3://crabby-images/0049a/0049addc2bdaccbdbd94f6a72650b04d9c43ad53" alt="../../../_images/8-2.png"
Edit the controls’ text by double-clicking on them and rearrange them by dragging them to their final location.
data:image/s3,"s3://crabby-images/cbd51/cbd5194565aac7744bbcf6ec652e2052ce33cc83" alt="../../../_images/8-3.png"
Select the view controller’s view and in the “Resolve Auto Layout Issues” menu select “Add Missing Constraints”.
data:image/s3,"s3://crabby-images/e13ae/e13aeb9faa9535acbfd9b0be0c7ee4ed8b2647f3" alt="../../../_images/8-4.png"
8. Binding the UI Elements to the Controller¶
Open up “MyViewController.h” in the Assistant Editor and ctrl + Drag the controls from the storyboard editor into the MyViewController’s source code.
data:image/s3,"s3://crabby-images/dfceb/dfcebf6c76e3fc2883a03b46ccf2b5c53fa5a87d" alt="../../../_images/9-1.png"
Add outlets for the labels, one as questionLabel, the other as answerLabel.
data:image/s3,"s3://crabby-images/af883/af8836e789f28c7ebd485df85d2ca44d3e60e82c" alt="../../../_images/9-2.png"
Add actions for the buttons, one as showQuestion, the other as showAnswer.
data:image/s3,"s3://crabby-images/37add/37add3cc259f0465ebb5327f92923d488e0f2f61" alt="../../../_images/9-3.png"
The result should look like this.
data:image/s3,"s3://crabby-images/9f7d7/9f7d76c22511c899b3949f2dd09ee9f641490376" alt="../../../_images/9-4.png"
9. Edit the Target¶
Select the “QuizApp” project in the Project navigator and “QuizApp” in the Targets listing. In the “General” tab set the Version, Build and Main Interface fields.
data:image/s3,"s3://crabby-images/eef0b/eef0bd238bcbef371e168bfe6e610260bb80c903" alt="../../../_images/10-1.png"
10. Editing the Java Code¶
Go back to Android Studio, select your MyViewController.h, right-click on it and select “Multi-OS Engine Actions” > “Synchronize to Java”.
data:image/s3,"s3://crabby-images/378af/378af0de545141aee6b978e1458d7721ed8dc902" alt="../../../_images/11-1.png"
It will ask for a package to generate the bindings into.
data:image/s3,"s3://crabby-images/d83db/d83db6414dc81207b312a9592d37d2410fd08af6" alt="../../../_images/11-2.png"
Add a new Java class called QuizDataSet.
public class QuizDataSet {
private ArrayList<String> questions = new ArrayList<>();
private ArrayList<String> answers = new ArrayList<>();
public QuizDataSet() {
questions.add("What is the State Capital of California?");
questions.add("What is the capital city of New York");
questions.add("Which state is the Niagara Falls located in?");
answers.add("Sacramento");
answers.add("New York City");
answers.add("New York");
}
public String getQuestion(int idx) {
return questions.get(idx);
}
public String getAnswer(int idx) {
return answers.get(idx);
}
public int getSize() {
return questions.size();
}
}
data:image/s3,"s3://crabby-images/1f740/1f740391881e2104ba1ba1ca64e86bb61683e4f7" alt="../../../_images/12-1.png"
Open up your MyViewController class for editing. You should see the two actions “showQuestion” and “showAnswer”.
data:image/s3,"s3://crabby-images/20186/20186d3a5933a71f2ed9f74a681a6fe3df5c1c33" alt="../../../_images/13-1.png"
Remove the @Generated annotation and the native keyword. Change the code to the following:
private QuizDataSet quiz = new QuizDataSet();
private int quizIdx = 0;
@Selector("showAnswer:")
public void showAnswer(UIButton sender) {
answerLabel().setText(quiz.getAnswer(quizIdx));
}
@Selector("showQuestion:")
public void showQuestion(UIButton sender) {
quizIdx = (quizIdx + 1) % quiz.getSize();
questionLabel().setText(quiz.getQuestion(quizIdx));
answerLabel().setText("???");
}
data:image/s3,"s3://crabby-images/c302f/c302f505e1b83c9db2a86c7da8cf682504b30bbf" alt="../../../_images/14-1.png"
Go to the top of the file and change the annotations as follows:
data:image/s3,"s3://crabby-images/e0ed2/e0ed2ac14b8f390ddea30aef9de0f0364dde401e" alt="../../../_images/14-2.png"