App Framework is a starting point for shortcut developers to create complex applications in Shortcuts. Study and learn from this documentation and the heavily-commented shortcut; you’ll be writing your own shortcut applications in no time!
- Why Shortcut Applications?
- Download App Framework
- Application Settings
- Preferences
- Localization
- Assets
- Application Loop
- vCard Menus
- Pseudo-Global Variables
- Pseudo-Local Variables
- Pseudo-Classes
- Pseudo-Functions
- Special Comments
- Miscellaneous Tips and Tricks
- Frequently Asked Questions
- Localization
- License
Shortcuts are capable of doing far more than simple actions to calculate driving directions, setting reminders, or converting files from one format to another. Complicated shortcuts are possible, but there hasn’t been an easy way to create them short of building your own framework.
Shortcuts do not feature traditional programming concepts like functions, while loops, and local variables. And, most shortcuts are designed to run from the top of its list of actions to the bottom. As a result, many seasoned developers have scoffed at Shortcuts as a toy or gimmick.
I’ll be the first to tell you, shortcuts are no gimmick, and you can actually create application experiences that actually go beyond what’s currently possible with apps developed with Xcode.
App Framework addresses these issues with design and programming methodologies for creating shortcuts. With App Framework, you have a starter shortcut that implements the following application features:
- Application Settings
- Preferences
- Localization
- Assets
- Application Loop
- vCard Menus
- Pseudo-Global Variables
- Pseudo-Local Variables
- Pseudo-Classes
- Pseudo-Functions
- Special Comments
With App Framework, you will be able to create complex applications in Shortcuts using a fully touch-oriented and visual programming interface.
The latest version of App Framework is available from RoutineHub.co:
Localization files and this documentation for App Framework are available on GitHub:
You’re welcome to send me a pull request for new localizations and corrections to the documentation.
Application settings and defaults are stored in a dictionary called App
. Here you’ll find things such as:
App Name
Shortcut Name
RoutineHub ID
Current Version
Build Number
Default Preferences
Default Language
Device-Specific Preferences
The App
variable is an example of a pseudo-global variable and is available throughout your entire application lifecycle.
Preferences are stored as a JSON file in the Shortcuts directory of iCloud Drive in a folder named App Framework
(the value of the App.App Name
variable):
iCloud Drive/Shortcuts/App Framework/config.json
Default Preferences, found in the App
global, are loaded the first time the shortcut is launched by the user. In this shortcut, we have six preferences:
- Copy Locally: When copying text, copy to the local clipboard or the user’s global clipboard on iCloud (which is available for devices supporting Handoff).
- Debug: Displays alerts when moving through the application. Useful for understanding what pseudo-function is being called at the exact moment.
- Language: The language that will be used to generate the
Localized Strings
global variable. - Number: A number that the user can modify in the
Program.step2
function. - Text: A string that the user can modify in the
Program.step2
function. - Version: This corresponds to the build number of the application. It’s updated whenever a new version of the shortcut is added and run by the user.
Preferences are available throughout your program’s execution via the Preferences
global variable.
As you build your own application, you’ll add and remove items to the Default Preferences
variable in the App
global. Remember to add them to your preferences menu in the pseudo-function Application.settings
.
When you make changes to the
Preferences
variable, remember to call Set Variable onPreferences
. Otherwise, your changes will not be saved back to thePreferences
. Follow this up with the Save File action to save your changes back to iCloud Drive.
You may want to have separate preferences for your shortcut when it is run on the user’s iPad or iPhone. If desired, set the value of the App.Device-Specific Preferences
boolean to true
.
This will use the Device Name of the iOS device as the unique name for the preferences file.
Note: If the user has multiple iOS devices with the same name, those devices will share the same preferences file.
The preference Debug
turns on additional debugging alerts when running the App Framework shortcut. It will stop execution at key moments during program execution, including the beginning and end of the Application Loop. You’ll be able to view the function that will be called next, the state of the Current Number
and Current Text
global variables, and more.
Use IF/EQUALS Debug True blocks when developing your application so you can inspect what is happening during complex portions of your code. You can also place a specialty formatted comment immediately below the IF block if you intend to remove this code prior to releasing your shortcut to the public.
Also consider using my shortcut Inspector, which lets you view and modify objects such as Dictionaries, Lists, Text, Numbers, and Booleans at runtime.
App Framework is ready to be translated into your preferred language. It comes with a default English localization and a test Emoji localization. Copy the default English localization and replace its strings with your translated version. Add your localized dictionary to the All Localizations
global variable.
{
"English": {
"Hello": "Hello",
…
},
"Emoji": {
"Hello": "👋🏻",
…
},
"Français": {
"Hello": "Bonjour",
…
},
…
}
Image assets for menus are stored as Base64 strings within the Assets
global variable. In general, menu icons should be resized to between 82 pixels and 132 pixels wide. Note that when displayed as a vCard menu, your image will be cropped to a circle. Be mindful of image elements that go edge-to-edge, as they may be cropped.
I used the Iconify shortcut from heyitzspencer to create the menu icons for App Framework.
The Application Loop is where your application lives most of the time. Once initialization is complete, the shortcut enters a repeat loop and stays there until the user presses the Cancel button or otherwise invokes a command which runs the Exit Shortcut action.
At the start and end of the repeat loop, there is a comment with the following block of text:
✳️✳️✳️✳️◻️✳️✳️✳️
✳️◻️◻️✳️◻️✳️◻️✳️
✳️✳️✳️✳️◻️✳️✳️✳️
✳️◻️✳️◻️◻️✳️◻️◻️
✳️◻️◻️✳️◻️✳️◻️◻️
This comment tells you that you are starting the primary application loop.
You may want to add this comment block to other long and complicated repeat loops.
The following diagram displays the flow of operations within the App Framework shortcut. Study this flowchart carefully so you understand how the loop functions.
Here’s what’s happening during each iteration of the application loop:
- At the beginning of the loop, the variable
Function Name
is evaluated in a series of IF statements. - It looks within the appropriate pseudo-class via an IF/CONTAINS comparison and all class pseudo-functions via an IF/EQUALS comparison for a matching string. When found, it executes the code contained in the IF statement block. If there’s no match, it goes to the next IF/CONTAINS block for a match.
- At the end of the matching function, the
Return
object must be properly set up. - Program execution goes to the end of the repeat loop, where the
Return
object is evaluated and global variables are reset. - Program execution returns to the top of the loop, where the process repeats itself.
Remember, at the end of the function, you must configure the
Return
object, which will be evaluated at the very end of the repeat loop.
The Return
object contains information about the next function, global variables to assign, and other things your shortcut needs to operate. It is a dictionary that can hold whatever data you wish.
App Framework does not evaluate for errors in the
Return
object. Consider it an exercise to add anError Code
andError Number
fields to theReturn
dictionary. So, if an error occurred during the shortcut’s execution, you can use this information to raise an alert to the user.
Within this version of App Framework, the primary values that the developer must specify in the Return
object are:
- handled: a boolean value. Set true if the function successfully ran. False otherwise. App Framework returns to the
Program.start
function if an error occurred. - next: a string that is the next function to call.
Function Name
will be assigned the value ofnext
at the end of the repeat loop. - args: a dictionary or singular element that will be assigned to the
Arguments
global variable. This can then be used by the next function. - selection: an array of elements. Selection is actually not used by App Framework, but it’s here to show how you can hold a list of items that the next function needs to operate on (i.e. a list of images, numbers, or strings).
In App Framework, the shortcut checks the value of the Shortcut Input
pseudo-global variable. If the shortcut was launched without input, it tells the Application.menus
function to go to Application.welcome
via the Arguments.next
variable. If input was provided to the shortcut, Application.menus
will go to Program.start
after setting up the menus.
The easiest way to evaluate if input was provided to the app is to coerce the input to a text string, surround it with #
characters and do a text comparison.
If the value is ##
, it means the app was launched without parameters. This could mean the user tapped on it from the Shortcuts home screen, launched it from the Notification Widget screen, or invoked via a Siri voice command.
If you use Cronios, your app can be launched with a Cronios dictionary object that contains a boolean variable named
Cronios
set to true. That way you know that your shortcut is bring run automatically and in the background.
If the text comparison returns false, your shortcut was launched with parameters. You may then choose to run a different function when the Application Loop begins.
Standard menus generated by the Choose from List or Choose from Menu actions are limited in what they can display. A trick many developers have been using in their shortcuts is supplying the Choose from List action with a list filled with Contact cards, also known as vCards.
As you can see below, vCards can hold a lot of information; fields not visible to the user can still be retrieved using the Get Details of Contacts action.
Each vCard is a block of text that looks like this. The only required field is the N
or ORG
field. If you don’t have an image for your menu, don’t include the PHOTO;ENCODING=b:
field.
BEGIN:VCARD
VERSION:3.0
N:{{Menu Name}};
TITLE:{{Menu Type}};
ORG:{{Menu Description}};
NOTE:{{Additional Field}}
PHOTO;ENCODING=b:{{Base64 Image}}
END:VCARD
- N: The menu title.
- TITLE: App Framework uses this field to differentiate between types of Menu Items: Actions, Separators, and Data.
- ORG: The menu description or subtitle. If
N
is blank andORG
is filled out, it will be displayed in both the title and subtitle sections. - NOTE: Not used in App Framework, you can store additional information in this field.
- PHOTO;ENCODING=b: A Base64 version of your menu icon. Keep the icons between 82 and 132 pixels in width/height. Image elements along the edges of the box may be cropped, since vCards icons are displayed in a cropped circle.
When transformed into a menu, a vCard looks like this:
To create a vCard menu, follow these steps:
- Create an empty list object.
- Use Set Variable to give your list object a name.
- Add your vCard text items to the list using the Add to Variable action.
- Run the Combine Text action on the list with
Spaces
as the separator when you have added all of the items. - Place a Set Name action and give the menu a name. You must add
.vcf
to the end of the name. - Add a Get Variable action.
- Tap Magic Variable.
- Set Name action that you created in Step 4.
- Tap on the variable in the Get Variable action and make sure that the variable is being interpreted as a Contact. Your menu will not display properly if you forget this step.
- Add the Choose from List action immediately after the Get Variable action.
Alternatively, you can add a Get Contacts From Input action in place of Get Variable in Step 8.
Shortcuts do not distinguish between global and local variables. All variables are considered global when writing shortcuts.
However, not all of the variables that you define will be accessible by every function, even though they will appear in the list of variables you can choose from in actions like Get Variable. This is because we are defining certain variables within a pseudo-class or pseudo-function. During program execution of one function, variables defined in another function may not be available; or if they are, they may not hold the right information.
For variables that you will be using everywhere in your shortcut, App Framework considers pseudo-global variables to have the following attributes:
- No Prefix: No periods in the variable name.
- Title Case: Every word in a global variable is capitalized.
Here are some examples of global variables used in App Framework:
App
All Localizations
Localized Strings
Current Number
Current Text
Function Name
Return
Arguments
Selection
Local variables are variables that are used solely within a specific pseudo-function. For instance, in the Program.step2
function, we create a vCard menu. We don’t use this menu variable anywhere else, so it makes sense to distinguish it from variables in other functions.
To accomplish this, we prefix the variable name with the name of the function that it belongs to. The other named components of a local variable are in lowercase or lower camel case. For example:
Program.step2.menu
Program.step2.menuChoice
Program.step2.menuType
Program.step2.newText
The local variables in the Application.settings
function are named as follows:
Application.settings.menu
Application.settings.language
Application.settings.args
As you can see in the screenshot below, it’s now easy to distinguish which variables belong to which function just by looking at their names.
Pseudo-classes combine related pseudo-functions under the same prefix. When evaluating the Function Name
variable, the shortcut can skip unrelated functions belong to other classes.
In App Framework, there are only two classes:
Program
Application
When you develop your own applications, you may have more pseudo-classes. LaunchCuts, for instance, features the following classes:
Folder
Shortcuts
Keywords
Cache
Application
Preferences
Settings
The start of a class is preceded by a comment with the following text block:
◻️🔵🔵◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
◻️🔵🔵◻️🔵🔵🔵
When you see this comment, you know we are defining a new pseudo-class. It’s good practice to add another comment explaining what the class does immediately following the IF/CONTAINS statement matching the class name with the global variable Function Name
.
We also separate pseudo-classes with columned comments. Choose an emoji that best represents what your class does when creating the column comments.
Pseudo-functions are contained within a pseudo-class. Their names are prefixed by the name of the class:
Program.start
Program.step2
Program.step3
Program.step4
Program.actionMenuHandler
You use functions within a class to organize your code in a logical manner.
The start of the function is a comment with the text block:
⬛️⬛️⬛️◻️⬛️◻️◻️⬛️
⬛️◻️◻️◻️⬛️⬛️◻️⬛️
⬛️⬛️⬛️◻️⬛️◻️⬛️⬛️
⬛️◻️◻️◻️⬛️◻️◻️⬛️
⬛️◻️◻️◻️⬛️◻️◻️⬛️
This helps you identify the upcoming block of code as belonging to a function.
Be sure to add a comment explaining what the function does immediately following the IF/EQUALS statement matching the function name with the global variable Function Name
.
App Framework has the following functions defined:
- Program.start: This function doesn’t do too much, although you can use it to do some additional initialization. You could, for instance, evaluate the
Original Input
global variable and display something different depending on whether the shortcut was launch with input or not. This function is called when the user taps Start Program fromApplication.welcome
or when the shortcut is run with input. - Program.step2: The primary function in App Framework. This displays the menu containing the
Current Number
andCurrent Text
preference variables. It also displays the Actions menu, and takes user input. - Program.step3: This is a stub function. It isn’t called anywhere by the application, but you can add a menu item and update the
Return
object to run it. - Program.step4: Another stub function. This one calls
Program.step5
, but that function doesn’t exist (so it should raise an error at the end of the Application Loop. - Program.actionMenuHandler: This function handles choosing an Action menu from
Program.step2
: Copy to Clipboard, View Application Flow, License, Settings, and Exit. - Application.menus: This function generates the Action menu used in
Program.step2
. - Application.welcome: This function is called when the App Framework shortcut is run with no input.
- Application.settings: This function is called when the Settings menu item is chosen from
Program.step2
orApplication.welcome
. It handles updating theCopy Locally
,Debug
, andLanguage
preferences. It also has a stub for checking for updates to the shortcut. Right now, it informs the user of the current version and build and redirects them to the RoutineHub.co page for App Framework. Your shortcut might call UpdateKit or another shortcut updater library.
Throughout this document, there have been mention of using specially formatted comments to distinguish different sections of your code. Here is a list of special comments that are used in App Framework and other apps:
This denotes the start of a pseudo-class.
◻️🔵🔵◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
🔵◻️◻️◻️🔵◻️◻️
◻️🔵🔵◻️🔵🔵🔵
This framework also separates classes with three sets of columned comments for each class. When scrolling through long sections of actions, this makes finding your code easier.
🔍🔍🔍 🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍
🔍🔍🔍 🔍🔍🔍
🔍🔍🔍
🔍🔍🔍
🔍🔍🔍
🔍🔍🔍
🔍🔍🔍
⚙️⚙️⚙️
⚙️⚙️⚙️
⚙️⚙️⚙️
⚙️⚙️⚙️
⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️ ⚙️⚙️⚙️
⚙️⚙️⚙️ ⚙️⚙️⚙️ ⚙️⚙️⚙️
This denotes code that should be removed prior to release of the shortcut.
‼️‼️‼️ ‼️‼️‼️ ‼️‼️‼️
‼️‼️‼️ ‼️‼️‼️ ‼️‼️‼️
‼️‼️‼️ ‼️‼️‼️ ‼️‼️‼️
‼️‼️‼️ ‼️‼️‼️ ‼️‼️‼️
‼️‼️‼️ ‼️‼️‼️ ‼️‼️‼️
Not used in App Framework, this comment denotes a call to another shortcut. It’s also used when calling your shortcut recursively.
◻️◻️◻️◻️🔵◻️◻️
◻️◻️◻️◻️🔵🔵◻️
🔵🔵🔵🔵🔵🔵🔵
◻️◻️◻️◻️🔵🔵◻️
◻️◻️◻️◻️🔵◻️◻️
This denotes the start of a pseudo-function.
⬛️⬛️⬛️◻️⬛️◻️◻️⬛️
⬛️◻️◻️◻️⬛️⬛️◻️⬛️
⬛️⬛️⬛️◻️⬛️◻️⬛️⬛️
⬛️◻️◻️◻️⬛️◻️◻️⬛️
⬛️◻️◻️◻️⬛️◻️◻️⬛️
This marks the beginning and end of a complex, multi-action repeat loop (like the Application Loop).
✳️✳️✳️✳️◻️✳️✳️✳️
✳️◻️◻️✳️◻️✳️◻️✳️
✳️✳️✳️✳️◻️✳️✳️✳️
✳️◻️✳️◻️◻️✳️◻️◻️
✳️◻️◻️✳️◻️✳️◻️◻️
Here are some other tips and tricks that you’ll find useful when developing shortcut applications.
There are plenty of shortcuts that back up your shortcuts to iCloud Drive. Choose one of them and use it regularly. When doing development, make copies of your shortcut applications and suffix the name with:
App Framework Backup YYYYMMDD_hhmm
For instance:
App Framework Backup 20190110_2032
App Framework Backup 20190109_0930
App Framework Backup 20190108_1050
Change the color of the shortcut icon so you don’t get confused with the current version you’re working on.
Make sure you have iCloud Sync turned on for your Shortcuts in Settings.
During the course of developing a complicated shortcut, Shortcuts may crash and display the following alert to you. This is alert looks very scary, and it’s not always obvious which button to press:
Corrupt Database
The Shortcuts database cannot be read because it is corrupt.
You can email support with your corrupted database or reset your custom shortcuts. Your shortcuts will be lost, but if you use iCloud Sync they will be restored.
- Email Support
- Reset Shortcuts
- Exit
Your mileage may vary, but I have found success in tapping Reset Shortcuts. This sometimes makes the Shortcuts app unusable (it launches and immediately crashes). When this happens, I turn restart the iOS device off and on by following these steps:
- Press the Volume Up button.
- Press the Volume Down button.
- Press and hold the Sleep/Wake button.
- Slide to power off.
- Wait a few moments after the screen goes completely black.
- Press and hold the Sleep/Wake button until the Apple logo appears
After the device has rebooted, I open back up to the Shortcuts app, which may be completely empty.
Do not panic.
I repeat. Do not panic.
After a minute, all of my shortcuts return from iCloud. The order of the shortcuts, however, may be completely hosed (so much for the Sync Shortcut Order preference in iOS Settings).
It may be time to panic if your shortcuts never return (make sure you are first connected to the internet because iCloud sync cannot work while offline). In this event, you will have to restore from a backup that you made earlier to iCloud Drive.
You can run JavaScript in your shortcuts using the following steps:
- Add a Text action containing your script. Consider using an app like the excellent Scriptable iOS application to debug your script prior to pasting it into Shortcuts.
const dict = { "name": "John Doe", "address": "123 Main Street" };
document.write( JSON.stringify( dict ) );
- Add a Text action that references your script in (1).
<!DOCTYPE html>
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-16">
</head>
<script>
{{Script Variable from Step 1}}
</script>
</head>
<body>
</body>
</html>
- Add a Base64 Encode action. Set Line Breaks to None.
- Add the URL action with the following text:
data:text/html;base64,{{Base64 Variable From Step 3}}
. - Add a Get Contents of Web Page action.
- Add a Get Text From Input action.
- Add a Replace Text action to get rid of the occasional trailing space in the output. Search string is
^\s+|\s+$|\s+(?=\s)/g
and the replace string is$1
with the Regular Expression slider activate. - Insert a Get Dictionary From Input action. What is returned is the contents of the
document.write( JSON.stringify( dict ) );
command in Step 1.
You can change the output format, but your JavaScript should end with a document.write
command. For instance, if all you want to return is a 0 or 1, just end the script with document.write(1)
or document.write(0)
. Make sure that the return string doesn’t have trailing spaces (see Step 7).
When your shortcut gets very large, it’s just a matter of time before the Shortcuts application becomes unstable and crashes. Here are symptoms you can look out for before it crashes:
- Scrolling becomes slow and jittery.
- Fields that have editable text go white and can’t be tapped on.
- Action blocks are shown visually overlapping each other.
- Tapping on an action to add it to the shortcut’s bottom takes a long time.
When you see any of the above, consider closing your shortcuts, waiting a minute, and force-quitting the Shortcuts app. This will free up any memory used up during your development period and make it available again for the Shortcuts app. Scrolling will become smoother and the other visual defects will be resolved (until it starts getting slow again).
Shortcut applications can get quite long, and it’s a pain to scroll through hundreds or thousands of lines of code at a time.
One tip is to put the most important code at the top or bottom of your shortcut. You can jump to the top of the shortcut by tapping on your device’s status bar.
I hope future versions of Shortcuts improves navigation in complicated shortcuts. It would be great to jump more quickly between sections of code through search or via something akin to Kodex’s minimap.
By tapping on an action in the Actions sidebar, the action will be added to the bottom of your shortcut. Remember to delete the action if you aren’t planning to use it.
I like to use the Comment action since it does nothing when run.
The faster your swipe, the faster Shortcuts will scroll through your code. By separating your classes with comments, you can see which section of code you’re in as you pass the columned comments.
You can also take a look at the scrollbar position of the current code block you’re in. If the app crashes, or you have to exit and re-open your shortcut, you can swipe scroll until you reach that remembered position in the scrollbar.
Be sure to enable Overwrite If File Exists when saving the Preferences
global back to disk. If you don’t, Shortcuts will create another preferences file but won’t change the one that your shortcut relies on.
App Framework is on GitHub and you’re welcome to send me a pull request for new localizations and corrections to this documentation.
It’s a little more difficult to get the actual shortcut into git, but there are some intriguing developments to watch for in the future.
Sign up for a GitHub account. Now that free accounts can create unlimited private repositories, there’s no excuse to version track your code.
Why put all this code into one shortcut? It’s too complex and takes too long to scroll through all of the code. Why not compartmentalize the functionality into smaller sets of shortcuts?
If you plan on distributing your shortcuts to the public, I think it’s better to place all of your code into one shortcut rather than having the user download (and maintain) several dependency shortcuts.
My hope is that the Shortcuts team at Apple addresses this in the future. Allowing shortcuts to be a bundle or container for multiple shortcuts would go a long way towards creating small re-usable shortcuts that can be used across multiple shortcut applications.
Of course, and I am not saying shortcut applications are. At the same time, can you efficiently and effectively program in Xcode while laying in bed or strolling in the park? You can’t lug your laptop everywhere you go like you can carry your iPhone. Can you make an app in 30 seconds in Xcode to select a group of photos from the user’s Photos Library and resize them for uploading to a website?
App Framework provides tools and design methodologies for creating complex applications just in Shortcuts. I wrote upwards of 30% of LaunchCuts, Cronios, and Inspector when away from my desk.
The visual programming style of Shortcuts, coupled with the touch UI of iOS enables on-the-go mobile programming. Rather than trying to shoehorn decades of desktop programming tools into a phone or tablet interface, why not try to create new development tools and interfaces optimized for mobile and touch? And, where appropriate, port the best tools from desktop programming to the mobile environment.
I’d rather have a beefed-up version of Shortcuts than a port of Xcode to iOS.
App Framework is available in English, but the application is fully ready to be localized. I have developed an application, Localization Helper, that will assist you in localizing App Framework into your language.
When the localization file is complete, either submit a pull request on my GitHub page.
Copyright © 2018-2019 Adam Tow • tow.com • @atow
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.