2016-06-18 14:46:12 +02:00
/*************************************************************************/
/* export.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 14:16:55 +02:00
/* https://godotengine.org */
2016-06-18 14:46:12 +02:00
/*************************************************************************/
2021-01-01 20:13:46 +01:00
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
2016-06-18 14:46:12 +02:00
/* */
/* 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. */
/*************************************************************************/
2018-01-05 00:50:27 +01:00
2014-02-09 22:10:30 -03:00
# include "export.h"
2015-05-01 22:45:32 -03:00
2020-01-06 18:37:39 +00:00
# include "core/io/image_loader.h"
2018-09-11 18:13:45 +02:00
# include "core/io/marshalls.h"
# include "core/io/zip_io.h"
2019-08-09 13:45:30 +02:00
# include "core/os/dir_access.h"
2018-09-11 18:13:45 +02:00
# include "core/os/file_access.h"
# include "core/os/os.h"
# include "core/project_settings.h"
2021-01-31 13:34:42 +01:00
# include "core/safe_refcount.h"
2018-09-11 18:13:45 +02:00
# include "core/version.h"
2020-01-06 18:37:39 +00:00
# include "drivers/png/png_driver_common.h"
2017-08-27 21:07:15 +02:00
# include "editor/editor_export.h"
2019-12-24 15:17:23 +08:00
# include "editor/editor_log.h"
2017-08-27 21:07:15 +02:00
# include "editor/editor_node.h"
# include "editor/editor_settings.h"
2020-08-19 15:38:50 -07:00
# include "main/splash.gen.h"
2020-07-01 11:06:50 -04:00
# include "platform/android/export/gradle_export_util.h"
2017-08-27 21:07:15 +02:00
# include "platform/android/logo.gen.h"
2020-04-24 00:45:14 -07:00
# include "platform/android/plugin/godot_plugin_config.h"
2017-08-27 21:07:15 +02:00
# include "platform/android/run_icon.gen.h"
2014-02-09 22:10:30 -03:00
2017-08-27 21:07:15 +02:00
# include <string.h>
2014-02-09 22:10:30 -03:00
2017-03-23 20:14:12 -03:00
static const char * android_perms [ ] = {
" ACCESS_CHECKIN_PROPERTIES " ,
" ACCESS_COARSE_LOCATION " ,
" ACCESS_FINE_LOCATION " ,
" ACCESS_LOCATION_EXTRA_COMMANDS " ,
" ACCESS_MOCK_LOCATION " ,
" ACCESS_NETWORK_STATE " ,
" ACCESS_SURFACE_FLINGER " ,
" ACCESS_WIFI_STATE " ,
" ACCOUNT_MANAGER " ,
" ADD_VOICEMAIL " ,
" AUTHENTICATE_ACCOUNTS " ,
" BATTERY_STATS " ,
" BIND_ACCESSIBILITY_SERVICE " ,
" BIND_APPWIDGET " ,
" BIND_DEVICE_ADMIN " ,
" BIND_INPUT_METHOD " ,
" BIND_NFC_SERVICE " ,
" BIND_NOTIFICATION_LISTENER_SERVICE " ,
" BIND_PRINT_SERVICE " ,
" BIND_REMOTEVIEWS " ,
" BIND_TEXT_SERVICE " ,
" BIND_VPN_SERVICE " ,
" BIND_WALLPAPER " ,
" BLUETOOTH " ,
" BLUETOOTH_ADMIN " ,
" BLUETOOTH_PRIVILEGED " ,
" BRICK " ,
" BROADCAST_PACKAGE_REMOVED " ,
" BROADCAST_SMS " ,
" BROADCAST_STICKY " ,
" BROADCAST_WAP_PUSH " ,
" CALL_PHONE " ,
" CALL_PRIVILEGED " ,
" CAMERA " ,
" CAPTURE_AUDIO_OUTPUT " ,
" CAPTURE_SECURE_VIDEO_OUTPUT " ,
" CAPTURE_VIDEO_OUTPUT " ,
" CHANGE_COMPONENT_ENABLED_STATE " ,
" CHANGE_CONFIGURATION " ,
" CHANGE_NETWORK_STATE " ,
" CHANGE_WIFI_MULTICAST_STATE " ,
" CHANGE_WIFI_STATE " ,
" CLEAR_APP_CACHE " ,
" CLEAR_APP_USER_DATA " ,
" CONTROL_LOCATION_UPDATES " ,
" DELETE_CACHE_FILES " ,
" DELETE_PACKAGES " ,
" DEVICE_POWER " ,
" DIAGNOSTIC " ,
" DISABLE_KEYGUARD " ,
" DUMP " ,
" EXPAND_STATUS_BAR " ,
" FACTORY_TEST " ,
" FLASHLIGHT " ,
" FORCE_BACK " ,
" GET_ACCOUNTS " ,
" GET_PACKAGE_SIZE " ,
" GET_TASKS " ,
" GET_TOP_ACTIVITY_INFO " ,
" GLOBAL_SEARCH " ,
" HARDWARE_TEST " ,
" INJECT_EVENTS " ,
" INSTALL_LOCATION_PROVIDER " ,
" INSTALL_PACKAGES " ,
" INSTALL_SHORTCUT " ,
" INTERNAL_SYSTEM_WINDOW " ,
" INTERNET " ,
" KILL_BACKGROUND_PROCESSES " ,
" LOCATION_HARDWARE " ,
" MANAGE_ACCOUNTS " ,
" MANAGE_APP_TOKENS " ,
" MANAGE_DOCUMENTS " ,
" MASTER_CLEAR " ,
" MEDIA_CONTENT_CONTROL " ,
" MODIFY_AUDIO_SETTINGS " ,
" MODIFY_PHONE_STATE " ,
" MOUNT_FORMAT_FILESYSTEMS " ,
" MOUNT_UNMOUNT_FILESYSTEMS " ,
" NFC " ,
" PERSISTENT_ACTIVITY " ,
" PROCESS_OUTGOING_CALLS " ,
" READ_CALENDAR " ,
" READ_CALL_LOG " ,
" READ_CONTACTS " ,
" READ_EXTERNAL_STORAGE " ,
" READ_FRAME_BUFFER " ,
" READ_HISTORY_BOOKMARKS " ,
" READ_INPUT_STATE " ,
" READ_LOGS " ,
" READ_PHONE_STATE " ,
" READ_PROFILE " ,
" READ_SMS " ,
" READ_SOCIAL_STREAM " ,
" READ_SYNC_SETTINGS " ,
" READ_SYNC_STATS " ,
" READ_USER_DICTIONARY " ,
" REBOOT " ,
" RECEIVE_BOOT_COMPLETED " ,
" RECEIVE_MMS " ,
" RECEIVE_SMS " ,
" RECEIVE_WAP_PUSH " ,
" RECORD_AUDIO " ,
" REORDER_TASKS " ,
" RESTART_PACKAGES " ,
" SEND_RESPOND_VIA_MESSAGE " ,
" SEND_SMS " ,
" SET_ACTIVITY_WATCHER " ,
" SET_ALARM " ,
" SET_ALWAYS_FINISH " ,
" SET_ANIMATION_SCALE " ,
" SET_DEBUG_APP " ,
" SET_ORIENTATION " ,
" SET_POINTER_SPEED " ,
" SET_PREFERRED_APPLICATIONS " ,
" SET_PROCESS_LIMIT " ,
" SET_TIME " ,
" SET_TIME_ZONE " ,
" SET_WALLPAPER " ,
" SET_WALLPAPER_HINTS " ,
" SIGNAL_PERSISTENT_PROCESSES " ,
" STATUS_BAR " ,
" SUBSCRIBED_FEEDS_READ " ,
" SUBSCRIBED_FEEDS_WRITE " ,
" SYSTEM_ALERT_WINDOW " ,
" TRANSMIT_IR " ,
" UNINSTALL_SHORTCUT " ,
" UPDATE_DEVICE_STATS " ,
" USE_CREDENTIALS " ,
" USE_SIP " ,
" VIBRATE " ,
" WAKE_LOCK " ,
" WRITE_APN_SETTINGS " ,
" WRITE_CALENDAR " ,
" WRITE_CALL_LOG " ,
" WRITE_CONTACTS " ,
" WRITE_EXTERNAL_STORAGE " ,
" WRITE_GSERVICES " ,
" WRITE_HISTORY_BOOKMARKS " ,
" WRITE_PROFILE " ,
" WRITE_SECURE_SETTINGS " ,
" WRITE_SETTINGS " ,
" WRITE_SMS " ,
" WRITE_SOCIAL_STREAM " ,
" WRITE_SYNC_SETTINGS " ,
" WRITE_USER_DICTIONARY " ,
2021-05-04 16:00:45 +02:00
nullptr
2017-03-23 20:14:12 -03:00
} ;
2021-04-12 16:33:37 -07:00
static const char * SPLASH_IMAGE_EXPORT_PATH = " res/drawable-nodpi/splash.png " ;
static const char * LEGACY_BUILD_SPLASH_IMAGE_EXPORT_PATH = " res/drawable-nodpi-v4/splash.png " ;
static const char * SPLASH_BG_COLOR_PATH = " res/drawable-nodpi/splash_bg_color.png " ;
static const char * LEGACY_BUILD_SPLASH_BG_COLOR_PATH = " res/drawable-nodpi-v4/splash_bg_color.png " ;
2021-02-18 01:54:35 -08:00
static const char * SPLASH_CONFIG_PATH = " res://android/build/res/drawable/splash_drawable.xml " ;
const String SPLASH_CONFIG_XML_CONTENT = R " SPLASH(<?xml version= " 1.0 " encoding= " utf - 8 " ?>
< layer - list xmlns : android = " http://schemas.android.com/apk/res/android " >
< item android : drawable = " @drawable/splash_bg_color " / >
< item >
< bitmap
2021-04-12 16:33:37 -07:00
android : gravity = " center "
2021-02-18 01:54:35 -08:00
android : filter = " %s "
android : src = " @drawable/splash " / >
< / item >
< / layer - list >
) SPLASH " ;
2020-08-19 15:38:50 -07:00
2017-11-24 11:09:15 +07:00
struct LauncherIcon {
2018-09-27 13:11:41 +02:00
const char * export_path ;
2020-01-06 18:37:39 +00:00
int dimensions ;
2017-11-24 11:09:15 +07:00
} ;
2020-01-06 18:37:39 +00:00
static const int icon_densities_count = 6 ;
2020-01-17 13:06:28 +01:00
static const char * launcher_icon_option = " launcher_icons/main_192x192 " ;
static const char * launcher_adaptive_icon_foreground_option = " launcher_icons/adaptive_foreground_432x432 " ;
static const char * launcher_adaptive_icon_background_option = " launcher_icons/adaptive_background_432x432 " ;
2020-01-06 18:37:39 +00:00
static const LauncherIcon launcher_icons [ icon_densities_count ] = {
{ " res/mipmap-xxxhdpi-v4/icon.png " , 192 } ,
{ " res/mipmap-xxhdpi-v4/icon.png " , 144 } ,
{ " res/mipmap-xhdpi-v4/icon.png " , 96 } ,
{ " res/mipmap-hdpi-v4/icon.png " , 72 } ,
{ " res/mipmap-mdpi-v4/icon.png " , 48 } ,
{ " res/mipmap/icon.png " , 192 }
} ;
static const LauncherIcon launcher_adaptive_icon_foregrounds [ icon_densities_count ] = {
{ " res/mipmap-xxxhdpi-v4/icon_foreground.png " , 432 } ,
{ " res/mipmap-xxhdpi-v4/icon_foreground.png " , 324 } ,
{ " res/mipmap-xhdpi-v4/icon_foreground.png " , 216 } ,
{ " res/mipmap-hdpi-v4/icon_foreground.png " , 162 } ,
{ " res/mipmap-mdpi-v4/icon_foreground.png " , 108 } ,
{ " res/mipmap/icon_foreground.png " , 432 }
} ;
static const LauncherIcon launcher_adaptive_icon_backgrounds [ icon_densities_count ] = {
{ " res/mipmap-xxxhdpi-v4/icon_background.png " , 432 } ,
{ " res/mipmap-xxhdpi-v4/icon_background.png " , 324 } ,
{ " res/mipmap-xhdpi-v4/icon_background.png " , 216 } ,
{ " res/mipmap-hdpi-v4/icon_background.png " , 162 } ,
{ " res/mipmap-mdpi-v4/icon_background.png " , 108 } ,
{ " res/mipmap/icon_background.png " , 432 }
2017-11-24 11:09:15 +07:00
} ;
2020-12-23 01:30:29 -08:00
static const int EXPORT_FORMAT_APK = 0 ;
static const int EXPORT_FORMAT_AAB = 1 ;
2019-02-12 15:43:54 +01:00
class EditorExportPlatformAndroid : public EditorExportPlatform {
2019-03-19 14:35:57 -04:00
GDCLASS ( EditorExportPlatformAndroid , EditorExportPlatform ) ;
2017-03-23 20:14:12 -03:00
Ref < ImageTexture > logo ;
2017-05-25 20:57:13 +02:00
Ref < ImageTexture > run_icon ;
2017-03-23 20:14:12 -03:00
struct Device {
String id ;
String name ;
String description ;
2017-08-30 20:21:26 +02:00
int api_level ;
2017-03-23 20:14:12 -03:00
} ;
struct APKExportData {
zipFile apk ;
EditorProgress * ep ;
} ;
2021-01-14 15:05:42 +03:00
Vector < PluginConfigAndroid > plugins ;
2020-05-24 02:33:07 +02:00
String last_plugin_names ;
uint64_t last_custom_build_time = 0 ;
2021-01-31 13:34:42 +01:00
SafeFlag plugins_changed ;
2021-01-27 10:43:02 +01:00
Mutex plugins_lock ;
2017-03-23 20:14:12 -03:00
Vector < Device > devices ;
2021-01-31 13:34:42 +01:00
SafeFlag devices_changed ;
2021-01-27 10:43:02 +01:00
Mutex device_lock ;
2021-01-27 20:10:10 +01:00
Thread check_for_changes_thread ;
2021-01-31 13:34:42 +01:00
SafeFlag quit_request ;
2017-03-23 20:14:12 -03:00
2020-04-24 00:45:14 -07:00
static void _check_for_changes_poll_thread ( void * ud ) {
2019-02-12 15:43:54 +01:00
EditorExportPlatformAndroid * ea = ( EditorExportPlatformAndroid * ) ud ;
2017-03-23 20:14:12 -03:00
2021-01-31 13:34:42 +01:00
while ( ! ea - > quit_request . is_set ( ) ) {
2020-04-24 00:45:14 -07:00
// Check for plugins updates
{
// Nothing to do if we already know the plugins have changed.
2021-01-31 13:34:42 +01:00
if ( ! ea - > plugins_changed . is_set ( ) ) {
2021-01-14 15:05:42 +03:00
Vector < PluginConfigAndroid > loaded_plugins = get_plugins ( ) ;
2020-04-24 00:45:14 -07:00
2021-01-27 10:43:02 +01:00
ea - > plugins_lock . lock ( ) ;
2020-04-24 00:45:14 -07:00
if ( ea - > plugins . size ( ) ! = loaded_plugins . size ( ) ) {
2021-01-31 13:34:42 +01:00
ea - > plugins_changed . set ( ) ;
2020-04-24 00:45:14 -07:00
} else {
for ( int i = 0 ; i < ea - > plugins . size ( ) ; i + + ) {
if ( ea - > plugins [ i ] . name ! = loaded_plugins [ i ] . name ) {
2021-01-31 13:34:42 +01:00
ea - > plugins_changed . set ( ) ;
2020-04-24 00:45:14 -07:00
break ;
}
}
}
2017-03-23 20:14:12 -03:00
2021-01-31 13:34:42 +01:00
if ( ea - > plugins_changed . is_set ( ) ) {
2020-04-24 00:45:14 -07:00
ea - > plugins = loaded_plugins ;
}
2021-01-27 10:43:02 +01:00
ea - > plugins_lock . unlock ( ) ;
2020-04-24 00:45:14 -07:00
}
}
// Check for devices updates
2020-12-23 22:16:56 -08:00
String adb = get_adb_path ( ) ;
2017-03-23 20:14:12 -03:00
if ( FileAccess : : exists ( adb ) ) {
String devices ;
List < String > args ;
args . push_back ( " devices " ) ;
int ec ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & devices , & ec ) ;
2017-03-23 20:14:12 -03:00
Vector < String > ds = devices . split ( " \n " ) ;
Vector < String > ldevices ;
for ( int i = 1 ; i < ds . size ( ) ; i + + ) {
String d = ds [ i ] ;
2020-01-27 19:59:24 +01:00
int dpos = d . find ( " device " ) ;
2020-04-24 00:45:14 -07:00
if ( dpos = = - 1 ) {
2017-03-23 20:14:12 -03:00
continue ;
2020-04-24 00:45:14 -07:00
}
2017-03-23 20:14:12 -03:00
d = d . substr ( 0 , dpos ) . strip_edges ( ) ;
ldevices . push_back ( d ) ;
}
2021-01-27 10:43:02 +01:00
ea - > device_lock . lock ( ) ;
2017-03-23 20:14:12 -03:00
bool different = false ;
if ( ea - > devices . size ( ) ! = ldevices . size ( ) ) {
different = true ;
} else {
for ( int i = 0 ; i < ea - > devices . size ( ) ; i + + ) {
if ( ea - > devices [ i ] . id ! = ldevices [ i ] ) {
different = true ;
break ;
}
}
}
if ( different ) {
Vector < Device > ndevices ;
for ( int i = 0 ; i < ldevices . size ( ) ; i + + ) {
Device d ;
d . id = ldevices [ i ] ;
for ( int j = 0 ; j < ea - > devices . size ( ) ; j + + ) {
if ( ea - > devices [ j ] . id = = ldevices [ i ] ) {
d . description = ea - > devices [ j ] . description ;
d . name = ea - > devices [ j ] . name ;
2017-08-30 20:21:26 +02:00
d . api_level = ea - > devices [ j ] . api_level ;
2017-03-23 20:14:12 -03:00
}
}
if ( d . description = = " " ) {
//in the oven, request!
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( d . id ) ;
args . push_back ( " shell " ) ;
2018-03-20 13:53:33 +09:00
args . push_back ( " getprop " ) ;
2019-02-12 21:10:08 +01:00
int ec2 ;
2017-03-23 20:14:12 -03:00
String dp ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & dp , & ec2 ) ;
2017-03-23 20:14:12 -03:00
Vector < String > props = dp . split ( " \n " ) ;
String vendor ;
String device ;
2020-02-23 09:28:54 +01:00
d . description = " Device ID: " + d . id + " \n " ;
2017-08-30 20:21:26 +02:00
d . api_level = 0 ;
2017-03-23 20:14:12 -03:00
for ( int j = 0 ; j < props . size ( ) ; j + + ) {
2018-03-20 13:53:33 +09:00
// got information by `shell cat /system/build.prop` before and its format is "property=value"
// it's now changed to use `shell getporp` because of permission issue with Android 8.0 and above
// its format is "[property]: [value]" so changed it as like build.prop
2017-03-23 20:14:12 -03:00
String p = props [ j ] ;
2018-03-20 13:53:33 +09:00
p = p . replace ( " ]: " , " = " ) ;
p = p . replace ( " [ " , " " ) ;
p = p . replace ( " ] " , " " ) ;
2017-03-23 20:14:12 -03:00
if ( p . begins_with ( " ro.product.model= " ) ) {
device = p . get_slice ( " = " , 1 ) . strip_edges ( ) ;
} else if ( p . begins_with ( " ro.product.brand= " ) ) {
vendor = p . get_slice ( " = " , 1 ) . strip_edges ( ) . capitalize ( ) ;
} else if ( p . begins_with ( " ro.build.display.id= " ) ) {
d . description + = " Build: " + p . get_slice ( " = " , 1 ) . strip_edges ( ) + " \n " ;
} else if ( p . begins_with ( " ro.build.version.release= " ) ) {
2017-08-30 20:21:26 +02:00
d . description + = " Release: " + p . get_slice ( " = " , 1 ) . strip_edges ( ) + " \n " ;
} else if ( p . begins_with ( " ro.build.version.sdk= " ) ) {
d . api_level = p . get_slice ( " = " , 1 ) . to_int ( ) ;
2017-03-23 20:14:12 -03:00
} else if ( p . begins_with ( " ro.product.cpu.abi= " ) ) {
d . description + = " CPU: " + p . get_slice ( " = " , 1 ) . strip_edges ( ) + " \n " ;
} else if ( p . begins_with ( " ro.product.manufacturer= " ) ) {
d . description + = " Manufacturer: " + p . get_slice ( " = " , 1 ) . strip_edges ( ) + " \n " ;
} else if ( p . begins_with ( " ro.board.platform= " ) ) {
d . description + = " Chipset: " + p . get_slice ( " = " , 1 ) . strip_edges ( ) + " \n " ;
} else if ( p . begins_with ( " ro.opengles.version= " ) ) {
uint32_t opengl = p . get_slice ( " = " , 1 ) . to_int ( ) ;
d . description + = " OpenGL: " + itos ( opengl > > 16 ) + " . " + itos ( ( opengl > > 8 ) & 0xFF ) + " . " + itos ( ( opengl ) & 0xFF ) + " \n " ;
}
}
d . name = vendor + " " + device ;
2021-05-05 12:44:11 +02:00
if ( device = = String ( ) ) {
2021-05-04 14:28:27 +02:00
continue ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
}
ndevices . push_back ( d ) ;
}
ea - > devices = ndevices ;
2021-01-31 13:34:42 +01:00
ea - > devices_changed . set ( ) ;
2017-03-23 20:14:12 -03:00
}
2021-01-27 10:43:02 +01:00
ea - > device_lock . unlock ( ) ;
2017-03-23 20:14:12 -03:00
}
2017-12-17 11:50:11 +01:00
uint64_t sleep = OS : : get_singleton ( ) - > get_power_state ( ) = = OS : : POWERSTATE_ON_BATTERY ? 1000 : 100 ;
2017-03-23 20:14:12 -03:00
uint64_t wait = 3000000 ;
uint64_t time = OS : : get_singleton ( ) - > get_ticks_usec ( ) ;
while ( OS : : get_singleton ( ) - > get_ticks_usec ( ) - time < wait ) {
2017-12-17 11:50:11 +01:00
OS : : get_singleton ( ) - > delay_usec ( 1000 * sleep ) ;
2021-05-05 12:44:11 +02:00
if ( ea - > quit_request . is_set ( ) ) {
2017-03-23 20:14:12 -03:00
break ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
}
}
if ( EditorSettings : : get_singleton ( ) - > get ( " export/android/shutdown_adb_on_exit " ) ) {
2020-12-23 22:16:56 -08:00
String adb = get_adb_path ( ) ;
2017-03-23 20:14:12 -03:00
if ( ! FileAccess : : exists ( adb ) ) {
return ; //adb not configured
}
List < String > args ;
args . push_back ( " kill-server " ) ;
OS : : get_singleton ( ) - > execute ( adb , args , true ) ;
} ;
}
String get_project_name ( const String & p_name ) const {
String aname ;
if ( p_name ! = " " ) {
aname = p_name ;
} else {
2017-07-19 17:00:46 -03:00
aname = ProjectSettings : : get_singleton ( ) - > get ( " application/config/name " ) ;
2017-03-23 20:14:12 -03:00
}
if ( aname = = " " ) {
2017-11-19 21:18:01 +01:00
aname = VERSION_NAME ;
2017-03-23 20:14:12 -03:00
}
return aname ;
}
2018-10-27 12:54:39 +01:00
String get_package_name ( const String & p_package ) const {
2017-03-23 20:14:12 -03:00
String pname = p_package ;
2017-07-19 17:00:46 -03:00
String basename = ProjectSettings : : get_singleton ( ) - > get ( " application/config/name " ) ;
2017-03-23 20:14:12 -03:00
basename = basename . to_lower ( ) ;
String name ;
bool first = true ;
for ( int i = 0 ; i < basename . length ( ) ; i + + ) {
CharType c = basename [ i ] ;
if ( c > = ' 0 ' & & c < = ' 9 ' & & first ) {
continue ;
}
if ( ( c > = ' a ' & & c < = ' z ' ) | | ( c > = ' A ' & & c < = ' Z ' ) | | ( c > = ' 0 ' & & c < = ' 9 ' ) ) {
name + = String : : chr ( c ) ;
first = false ;
}
}
2021-05-05 12:44:11 +02:00
if ( name = = " " ) {
2017-03-23 20:14:12 -03:00
name = " noname " ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
pname = pname . replace ( " $genname " , name ) ;
2019-04-07 15:46:52 -03:00
2017-03-23 20:14:12 -03:00
return pname ;
}
2021-05-04 16:00:45 +02:00
bool is_package_name_valid ( const String & p_package , String * r_error = nullptr ) const {
2018-10-27 12:54:39 +01:00
String pname = p_package ;
if ( pname . length ( ) = = 0 ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = TTR ( " Package name is missing. " ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
int segments = 0 ;
bool first = true ;
for ( int i = 0 ; i < pname . length ( ) ; i + + ) {
CharType c = pname [ i ] ;
if ( first & & c = = ' . ' ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = TTR ( " Package segments must be of non-zero length. " ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
if ( c = = ' . ' ) {
segments + + ;
first = true ;
continue ;
}
if ( ! ( ( c > = ' a ' & & c < = ' z ' ) | | ( c > = ' A ' & & c < = ' Z ' ) | | ( c > = ' 0 ' & & c < = ' 9 ' ) | | c = = ' _ ' ) ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = vformat ( TTR ( " The character '%s' is not allowed in Android application package names. " ) , String : : chr ( c ) ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
if ( first & & ( c > = ' 0 ' & & c < = ' 9 ' ) ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = TTR ( " A digit cannot be the first character in a package segment. " ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
if ( first & & c = = ' _ ' ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = vformat ( TTR ( " The character '%s' cannot be the first character in a package segment. " ) , String : : chr ( c ) ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
first = false ;
}
if ( segments = = 0 ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = TTR ( " The package must have at least one '.' separator. " ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
if ( first ) {
if ( r_error ) {
2019-01-21 18:34:53 +01:00
* r_error = TTR ( " Package segments must be of non-zero length. " ) ;
2018-10-27 12:54:39 +01:00
}
return false ;
}
return true ;
}
2017-03-23 20:14:12 -03:00
static bool _should_compress_asset ( const String & p_path , const Vector < uint8_t > & p_data ) {
/*
* By not compressing files with little or not benefit in doing so ,
* a performance gain is expected attime . Moreover , if the APK is
* zip - aligned , assets stored as they are can be efficiently read by
* Android by memory - mapping them .
*/
// -- Unconditional uncompress to mimic AAPT plus some other
static const char * unconditional_compress_ext [ ] = {
// From https://github.com/android/platform_frameworks_base/blob/master/tools/aapt/Package.cpp
// These formats are already compressed, or don't compress well:
" .jpg " , " .jpeg " , " .png " , " .gif " ,
" .wav " , " .mp2 " , " .mp3 " , " .ogg " , " .aac " ,
" .mpg " , " .mpeg " , " .mid " , " .midi " , " .smf " , " .jet " ,
" .rtttl " , " .imy " , " .xmf " , " .mp4 " , " .m4a " ,
" .m4v " , " .3gp " , " .3gpp " , " .3g2 " , " .3gpp2 " ,
" .amr " , " .awb " , " .wma " , " .wmv " ,
// Godot-specific:
" .webp " , // Same reasoning as .png
" .cfb " , // Don't let small config files slow-down startup
" .scn " , // Binary scenes are usually already compressed
" .stex " , // Streamable textures are usually already compressed
// Trailer for easier processing
2021-05-04 16:00:45 +02:00
nullptr
2017-03-23 20:14:12 -03:00
} ;
for ( const char * * ext = unconditional_compress_ext ; * ext ; + + ext ) {
if ( p_path . to_lower ( ) . ends_with ( String ( * ext ) ) ) {
return false ;
}
}
// -- Compressed resource?
if ( p_data . size ( ) > = 4 & & p_data [ 0 ] = = ' R ' & & p_data [ 1 ] = = ' S ' & & p_data [ 2 ] = = ' C ' & & p_data [ 3 ] = = ' C ' ) {
// Already compressed
return false ;
}
// --- TODO: Decide on texture resources according to their image compression setting
return true ;
}
static zip_fileinfo get_zip_fileinfo ( ) {
OS : : Time time = OS : : get_singleton ( ) - > get_time ( ) ;
OS : : Date date = OS : : get_singleton ( ) - > get_date ( ) ;
zip_fileinfo zipfi ;
zipfi . tmz_date . tm_hour = time . hour ;
zipfi . tmz_date . tm_mday = date . day ;
zipfi . tmz_date . tm_min = time . min ;
2020-10-23 15:05:48 -04:00
zipfi . tmz_date . tm_mon = date . month - 1 ; // tm_mon is zero indexed
2017-03-23 20:14:12 -03:00
zipfi . tmz_date . tm_sec = time . sec ;
zipfi . tmz_date . tm_year = date . year ;
zipfi . dosDate = 0 ;
zipfi . external_fa = 0 ;
zipfi . internal_fa = 0 ;
return zipfi ;
}
2017-11-21 01:12:36 +07:00
static Vector < String > get_abis ( ) {
Vector < String > abis ;
abis . push_back ( " armeabi-v7a " ) ;
abis . push_back ( " arm64-v8a " ) ;
abis . push_back ( " x86 " ) ;
2019-01-16 11:16:00 +01:00
abis . push_back ( " x86_64 " ) ;
2017-08-18 21:17:35 +07:00
return abis ;
}
2017-03-23 20:14:12 -03:00
2020-04-24 00:45:14 -07:00
/// List the gdap files in the directory specified by the p_path parameter.
static Vector < String > list_gdap_files ( const String & p_path ) {
Vector < String > dir_files ;
DirAccessRef da = DirAccess : : open ( p_path ) ;
if ( da ) {
da - > list_dir_begin ( ) ;
while ( true ) {
String file = da - > get_next ( ) ;
if ( file = = " " ) {
break ;
}
if ( da - > current_is_dir ( ) | | da - > current_is_hidden ( ) ) {
continue ;
}
2021-01-14 15:05:42 +03:00
if ( file . ends_with ( PluginConfigAndroid : : PLUGIN_CONFIG_EXT ) ) {
2020-04-24 00:45:14 -07:00
dir_files . push_back ( file ) ;
}
}
da - > list_dir_end ( ) ;
}
return dir_files ;
}
2021-01-14 15:05:42 +03:00
static Vector < PluginConfigAndroid > get_plugins ( ) {
Vector < PluginConfigAndroid > loaded_plugins ;
2020-04-24 00:45:14 -07:00
String plugins_dir = ProjectSettings : : get_singleton ( ) - > get_resource_path ( ) . plus_file ( " android/plugins " ) ;
// Add the prebuilt plugins
loaded_plugins . append_array ( get_prebuilt_plugins ( plugins_dir ) ) ;
if ( DirAccess : : exists ( plugins_dir ) ) {
Vector < String > plugins_filenames = list_gdap_files ( plugins_dir ) ;
if ( ! plugins_filenames . empty ( ) ) {
Ref < ConfigFile > config_file = memnew ( ConfigFile ) ;
for ( int i = 0 ; i < plugins_filenames . size ( ) ; i + + ) {
2021-01-14 15:05:42 +03:00
PluginConfigAndroid config = load_plugin_config ( config_file , plugins_dir . plus_file ( plugins_filenames [ i ] ) ) ;
2020-04-24 00:45:14 -07:00
if ( config . valid_config ) {
loaded_plugins . push_back ( config ) ;
} else {
print_error ( " Invalid plugin config file " + plugins_filenames [ i ] ) ;
}
}
}
}
return loaded_plugins ;
}
2021-01-14 15:05:42 +03:00
static Vector < PluginConfigAndroid > get_enabled_plugins ( const Ref < EditorExportPreset > & p_presets ) {
Vector < PluginConfigAndroid > enabled_plugins ;
Vector < PluginConfigAndroid > all_plugins = get_plugins ( ) ;
2020-04-24 00:45:14 -07:00
for ( int i = 0 ; i < all_plugins . size ( ) ; i + + ) {
2021-01-14 15:05:42 +03:00
PluginConfigAndroid plugin = all_plugins [ i ] ;
2020-04-24 00:45:14 -07:00
bool enabled = p_presets - > get ( " plugins/ " + plugin . name ) ;
if ( enabled ) {
enabled_plugins . push_back ( plugin ) ;
}
}
return enabled_plugins ;
}
2017-11-21 01:12:36 +07:00
static Error store_in_apk ( APKExportData * ed , const String & p_path , const Vector < uint8_t > & p_data , int compression_method = Z_DEFLATED ) {
2017-03-23 20:14:12 -03:00
zip_fileinfo zipfi = get_zip_fileinfo ( ) ;
zipOpenNewFileInZip ( ed - > apk ,
2017-11-21 01:12:36 +07:00
p_path . utf8 ( ) . get_data ( ) ,
2017-03-23 20:14:12 -03:00
& zipfi ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-11-21 01:12:36 +07:00
compression_method ,
2017-03-23 20:14:12 -03:00
Z_DEFAULT_COMPRESSION ) ;
zipWriteInFileInZip ( ed - > apk , p_data . ptr ( ) , p_data . size ( ) ) ;
zipCloseFileInZip ( ed - > apk ) ;
2017-11-21 01:12:36 +07:00
return OK ;
}
static Error save_apk_so ( void * p_userdata , const SharedObject & p_so ) {
if ( ! p_so . path . get_file ( ) . begins_with ( " lib " ) ) {
String err = " Android .so file names must start with \" lib \" , but got: " + p_so . path ;
2019-01-21 18:34:53 +01:00
ERR_PRINTS ( err ) ;
2017-11-21 01:12:36 +07:00
return FAILED ;
}
APKExportData * ed = ( APKExportData * ) p_userdata ;
Vector < String > abis = get_abis ( ) ;
bool exported = false ;
for ( int i = 0 ; i < p_so . tags . size ( ) ; + + i ) {
// shared objects can be fat (compatible with multiple ABIs)
int abi_index = abis . find ( p_so . tags [ i ] ) ;
if ( abi_index ! = - 1 ) {
exported = true ;
String abi = abis [ abi_index ] ;
2019-06-16 13:31:57 +01:00
String dst_path = String ( " lib " ) . plus_file ( abi ) . plus_file ( p_so . path . get_file ( ) ) ;
2017-11-21 01:12:36 +07:00
Vector < uint8_t > array = FileAccess : : get_file_as_array ( p_so . path ) ;
Error store_err = store_in_apk ( ed , dst_path , array ) ;
2019-09-25 10:28:50 +02:00
ERR_FAIL_COND_V_MSG ( store_err , store_err , " Cannot store in apk file ' " + dst_path + " '. " ) ;
2017-11-21 01:12:36 +07:00
}
}
if ( ! exported ) {
String abis_string = String ( " " ) . join ( abis ) ;
String err = " Cannot determine ABI for library \" " + p_so . path + " \" . One of the supported ABIs must be used as a tag: " + abis_string ;
2019-01-21 18:34:53 +01:00
ERR_PRINTS ( err ) ;
2017-11-21 01:12:36 +07:00
return FAILED ;
}
return OK ;
}
static Error save_apk_file ( void * p_userdata , const String & p_path , const Vector < uint8_t > & p_data , int p_file , int p_total ) {
APKExportData * ed = ( APKExportData * ) p_userdata ;
String dst_path = p_path . replace_first ( " res:// " , " assets/ " ) ;
store_in_apk ( ed , dst_path , p_data , _should_compress_asset ( p_path , p_data ) ? Z_DEFLATED : 0 ) ;
2017-03-23 20:14:12 -03:00
return OK ;
}
2018-01-16 12:36:18 +07:00
static Error ignore_apk_file ( void * p_userdata , const String & p_path , const Vector < uint8_t > & p_data , int p_file , int p_total ) {
return OK ;
}
2021-04-16 06:42:44 -07:00
bool _has_storage_permission ( const Vector < String > & p_permissions ) {
return p_permissions . find ( " android.permission.READ_EXTERNAL_STORAGE " ) ! = - 1 | | p_permissions . find ( " android.permission.WRITE_EXTERNAL_STORAGE " ) ! = - 1 ;
}
2020-06-26 14:58:16 -04:00
void _get_permissions ( const Ref < EditorExportPreset > & p_preset , bool p_give_internet , Vector < String > & r_permissions ) {
const char * * aperms = android_perms ;
while ( * aperms ) {
bool enabled = p_preset - > get ( " permissions/ " + String ( * aperms ) . to_lower ( ) ) ;
if ( enabled ) {
r_permissions . push_back ( " android.permission. " + String ( * aperms ) ) ;
}
aperms + + ;
}
PoolStringArray user_perms = p_preset - > get ( " permissions/custom_permissions " ) ;
for ( int i = 0 ; i < user_perms . size ( ) ; i + + ) {
String user_perm = user_perms [ i ] . strip_edges ( ) ;
if ( ! user_perm . empty ( ) ) {
r_permissions . push_back ( user_perm ) ;
}
}
if ( p_give_internet ) {
if ( r_permissions . find ( " android.permission.INTERNET " ) = = - 1 ) {
r_permissions . push_back ( " android.permission.INTERNET " ) ;
}
}
2017-03-23 20:14:12 -03:00
2020-06-26 14:58:16 -04:00
int xr_mode_index = p_preset - > get ( " xr_features/xr_mode " ) ;
if ( xr_mode_index = = 1 /* XRMode.OVR */ ) {
int hand_tracking_index = p_preset - > get ( " xr_features/hand_tracking " ) ; // 0: none, 1: optional, 2: required
if ( hand_tracking_index > 0 ) {
if ( r_permissions . find ( " com.oculus.permission.HAND_TRACKING " ) = = - 1 ) {
r_permissions . push_back ( " com.oculus.permission.HAND_TRACKING " ) ;
}
}
}
}
2020-07-23 12:17:02 -04:00
void _write_tmp_manifest ( const Ref < EditorExportPreset > & p_preset , bool p_give_internet , bool p_debug ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Building temporary manifest.. " ) ;
2020-07-23 12:17:02 -04:00
String manifest_text =
" <?xml version= \" 1.0 \" encoding= \" utf-8 \" ?> \n "
" <manifest xmlns:android= \" http://schemas.android.com/apk/res/android \" \n "
" xmlns:tools= \" http://schemas.android.com/tools \" > \n " ;
manifest_text + = _get_screen_sizes_tag ( p_preset ) ;
manifest_text + = _get_gles_tag ( ) ;
Vector < String > perms ;
_get_permissions ( p_preset , p_give_internet , perms ) ;
for ( int i = 0 ; i < perms . size ( ) ; i + + ) {
manifest_text + = vformat ( " <uses-permission android:name= \" %s \" /> \n " , perms . get ( i ) ) ;
}
manifest_text + = _get_xr_features_tag ( p_preset ) ;
manifest_text + = _get_instrumentation_tag ( p_preset ) ;
2021-04-16 06:42:44 -07:00
manifest_text + = _get_application_tag ( p_preset , _has_storage_permission ( perms ) ) ;
2020-07-23 12:17:02 -04:00
manifest_text + = " </manifest> \n " ;
String manifest_path = vformat ( " res://android/build/src/%s/AndroidManifest.xml " , ( p_debug ? " debug " : " release " ) ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Storing manifest into " + manifest_path + " : " + " \n " + manifest_text ) ;
2020-07-23 12:17:02 -04:00
store_string_at_path ( manifest_path , manifest_text ) ;
}
2020-06-26 14:58:16 -04:00
void _fix_manifest ( const Ref < EditorExportPreset > & p_preset , Vector < uint8_t > & p_manifest , bool p_give_internet ) {
2017-09-02 22:32:31 +02:00
// Leaving the unused types commented because looking these constants up
// again later would be annoying
// const int CHUNK_AXML_FILE = 0x00080003;
// const int CHUNK_RESOURCEIDS = 0x00080180;
2017-03-23 20:14:12 -03:00
const int CHUNK_STRINGS = 0x001C0001 ;
2017-09-02 22:32:31 +02:00
// const int CHUNK_XML_END_NAMESPACE = 0x00100101;
2018-07-04 15:47:33 +02:00
const int CHUNK_XML_END_TAG = 0x00100103 ;
2017-09-02 22:32:31 +02:00
// const int CHUNK_XML_START_NAMESPACE = 0x00100100;
2017-03-23 20:14:12 -03:00
const int CHUNK_XML_START_TAG = 0x00100102 ;
2017-09-02 22:32:31 +02:00
// const int CHUNK_XML_TEXT = 0x00100104;
2017-03-23 20:14:12 -03:00
const int UTF8_FLAG = 0x00000100 ;
Vector < String > string_table ;
2017-09-02 22:32:31 +02:00
uint32_t ofs = 8 ;
2017-03-23 20:14:12 -03:00
2017-09-01 22:33:39 +02:00
uint32_t string_count = 0 ;
2018-09-27 12:07:59 +02:00
//uint32_t styles_count = 0;
2017-09-01 22:33:39 +02:00
uint32_t string_flags = 0 ;
uint32_t string_data_offset = 0 ;
2018-09-27 12:07:59 +02:00
//uint32_t styles_offset = 0;
2017-09-01 22:33:39 +02:00
uint32_t string_table_begins = 0 ;
uint32_t string_table_ends = 0 ;
2017-03-23 20:14:12 -03:00
Vector < uint8_t > stable_extra ;
String version_name = p_preset - > get ( " version/name " ) ;
int version_code = p_preset - > get ( " version/code " ) ;
String package_name = p_preset - > get ( " package/unique_name " ) ;
2020-11-13 11:48:21 -08:00
const int screen_orientation = _get_android_orientation_value ( _get_screen_orientation ( ) ) ;
2017-03-23 20:14:12 -03:00
2019-03-05 13:46:51 +01:00
bool min_gles3 = ProjectSettings : : get_singleton ( ) - > get ( " rendering/quality/driver/driver_name " ) = = " GLES3 " & &
! ProjectSettings : : get_singleton ( ) - > get ( " rendering/quality/driver/fallback_to_gles2 " ) ;
2017-03-23 20:14:12 -03:00
bool screen_support_small = p_preset - > get ( " screen/support_small " ) ;
bool screen_support_normal = p_preset - > get ( " screen/support_normal " ) ;
bool screen_support_large = p_preset - > get ( " screen/support_large " ) ;
bool screen_support_xlarge = p_preset - > get ( " screen/support_xlarge " ) ;
2020-01-09 17:10:04 -08:00
int xr_mode_index = p_preset - > get ( " xr_features/xr_mode " ) ;
2020-06-04 14:54:36 -07:00
bool focus_awareness = p_preset - > get ( " xr_features/focus_awareness " ) ;
2019-06-16 01:58:34 -07:00
2018-07-04 15:47:33 +02:00
Vector < String > perms ;
2020-06-26 14:58:16 -04:00
// Write permissions into the perms variable.
_get_permissions ( p_preset , p_give_internet , perms ) ;
2021-04-16 06:42:44 -07:00
bool has_storage_permission = _has_storage_permission ( perms ) ;
2018-07-04 15:47:33 +02:00
2017-03-23 20:14:12 -03:00
while ( ofs < ( uint32_t ) p_manifest . size ( ) ) {
uint32_t chunk = decode_uint32 ( & p_manifest [ ofs ] ) ;
uint32_t size = decode_uint32 ( & p_manifest [ ofs + 4 ] ) ;
switch ( chunk ) {
case CHUNK_STRINGS : {
int iofs = ofs + 8 ;
string_count = decode_uint32 ( & p_manifest [ iofs ] ) ;
2018-09-27 12:07:59 +02:00
//styles_count = decode_uint32(&p_manifest[iofs + 4]);
2017-03-23 20:14:12 -03:00
string_flags = decode_uint32 ( & p_manifest [ iofs + 8 ] ) ;
string_data_offset = decode_uint32 ( & p_manifest [ iofs + 12 ] ) ;
2018-09-27 12:07:59 +02:00
//styles_offset = decode_uint32(&p_manifest[iofs + 16]);
2017-03-23 20:14:12 -03:00
/*
printf ( " string count: %i \n " , string_count ) ;
printf ( " flags: %i \n " , string_flags ) ;
printf ( " sdata ofs: %i \n " , string_data_offset ) ;
printf ( " styles ofs: %i \n " , styles_offset ) ;
2018-09-27 12:07:59 +02:00
*/
2017-03-23 20:14:12 -03:00
uint32_t st_offset = iofs + 20 ;
string_table . resize ( string_count ) ;
uint32_t string_end = 0 ;
string_table_begins = st_offset ;
for ( uint32_t i = 0 ; i < string_count ; i + + ) {
uint32_t string_at = decode_uint32 ( & p_manifest [ st_offset + i * 4 ] ) ;
string_at + = st_offset + string_count * 4 ;
2019-08-09 06:49:33 +02:00
ERR_FAIL_COND_MSG ( string_flags & UTF8_FLAG , " Unimplemented, can't read UTF-8 string table. " ) ;
2017-03-23 20:14:12 -03:00
if ( string_flags & UTF8_FLAG ) {
} else {
uint32_t len = decode_uint16 ( & p_manifest [ string_at ] ) ;
Vector < CharType > ucstring ;
ucstring . resize ( len + 1 ) ;
for ( uint32_t j = 0 ; j < len ; j + + ) {
uint16_t c = decode_uint16 ( & p_manifest [ string_at + 2 + 2 * j ] ) ;
2018-07-25 03:11:03 +02:00
ucstring . write [ j ] = c ;
2017-03-23 20:14:12 -03:00
}
string_end = MAX ( string_at + 2 + 2 * len , string_end ) ;
2018-07-25 03:11:03 +02:00
ucstring . write [ len ] = 0 ;
string_table . write [ i ] = ucstring . ptr ( ) ;
2017-03-23 20:14:12 -03:00
}
}
for ( uint32_t i = string_end ; i < ( ofs + size ) ; i + + ) {
stable_extra . push_back ( p_manifest [ i ] ) ;
}
string_table_ends = ofs + size ;
} break ;
case CHUNK_XML_START_TAG : {
int iofs = ofs + 8 ;
uint32_t name = decode_uint32 ( & p_manifest [ iofs + 12 ] ) ;
String tname = string_table [ name ] ;
uint32_t attrcount = decode_uint32 ( & p_manifest [ iofs + 20 ] ) ;
iofs + = 28 ;
2020-06-27 16:40:22 -07:00
bool is_focus_aware_metadata = false ;
2017-09-02 22:32:31 +02:00
2017-03-23 20:14:12 -03:00
for ( uint32_t i = 0 ; i < attrcount ; i + + ) {
uint32_t attr_nspace = decode_uint32 ( & p_manifest [ iofs ] ) ;
uint32_t attr_name = decode_uint32 ( & p_manifest [ iofs + 4 ] ) ;
uint32_t attr_value = decode_uint32 ( & p_manifest [ iofs + 8 ] ) ;
uint32_t attr_resid = decode_uint32 ( & p_manifest [ iofs + 16 ] ) ;
2019-10-11 11:12:53 -07:00
const String value = ( attr_value ! = 0xFFFFFFFF ) ? string_table [ attr_value ] : " Res # " + itos ( attr_resid ) ;
2017-03-23 20:14:12 -03:00
String attrname = string_table [ attr_name ] ;
2019-10-11 11:12:53 -07:00
const String nspace = ( attr_nspace ! = 0xFFFFFFFF ) ? string_table [ attr_nspace ] : " " ;
2017-03-23 20:14:12 -03:00
//replace project information
if ( tname = = " manifest " & & attrname = = " package " ) {
2018-07-25 03:11:03 +02:00
string_table . write [ attr_value ] = get_package_name ( package_name ) ;
2017-03-23 20:14:12 -03:00
}
2018-08-24 09:35:07 +02:00
if ( tname = = " manifest " & & attrname = = " versionCode " ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( version_code , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
}
2018-08-24 09:35:07 +02:00
if ( tname = = " manifest " & & attrname = = " versionName " ) {
2017-03-23 20:14:12 -03:00
if ( attr_value = = 0xFFFFFFFF ) {
2019-06-11 14:49:34 +02:00
WARN_PRINT ( " Version name in a resource, should be plain text " ) ;
2021-05-05 12:44:11 +02:00
} else {
2018-07-25 03:11:03 +02:00
string_table . write [ attr_value ] = version_name ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
}
2021-04-16 06:42:44 -07:00
if ( tname = = " application " & & attrname = = " requestLegacyExternalStorage " ) {
encode_uint32 ( has_storage_permission ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
}
2019-09-02 17:31:51 -07:00
if ( tname = = " instrumentation " & & attrname = = " targetPackage " ) {
string_table . write [ attr_value ] = get_package_name ( package_name ) ;
}
2018-08-24 09:35:07 +02:00
if ( tname = = " activity " & & attrname = = " screenOrientation " ) {
2020-11-13 11:48:21 -08:00
encode_uint32 ( screen_orientation , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
}
if ( tname = = " supports-screens " ) {
if ( attrname = = " smallScreens " ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( screen_support_small ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
} else if ( attrname = = " normalScreens " ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( screen_support_normal ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
} else if ( attrname = = " largeScreens " ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( screen_support_large ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
} else if ( attrname = = " xlargeScreens " ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( screen_support_xlarge ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
2017-03-23 20:14:12 -03:00
}
}
2019-03-05 13:46:51 +01:00
if ( tname = = " uses-feature " & & attrname = = " glEsVersion " ) {
encode_uint32 ( min_gles3 ? 0x00030000 : 0x00020000 , & p_manifest . write [ iofs + 16 ] ) ;
}
2019-10-11 09:03:21 +02:00
// FIXME: `attr_value != 0xFFFFFFFF` below added as a stopgap measure for GH-32553,
// but the issue should be debugged further and properly addressed.
2019-10-11 11:12:53 -07:00
if ( tname = = " meta-data " & & attrname = = " name " & & value = = " xr_mode_metadata_name " ) {
2019-07-02 09:22:47 -07:00
// Update the meta-data 'android:name' attribute based on the selected XR mode.
if ( xr_mode_index = = 1 /* XRMode.OVR */ ) {
string_table . write [ attr_value ] = " com.samsung.android.vr.application.mode " ;
}
}
2019-10-11 11:12:53 -07:00
if ( tname = = " meta-data " & & attrname = = " value " & & value = = " xr_mode_metadata_value " ) {
2019-07-02 09:22:47 -07:00
// Update the meta-data 'android:value' attribute based on the selected XR mode.
2019-06-16 01:58:34 -07:00
if ( xr_mode_index = = 1 /* XRMode.OVR */ ) {
string_table . write [ attr_value ] = " vr_only " ;
}
}
2020-06-27 16:40:22 -07:00
if ( tname = = " meta-data " & & attrname = = " value " & & is_focus_aware_metadata ) {
2020-06-04 14:54:36 -07:00
// Update the focus awareness meta-data value
2020-06-27 16:40:22 -07:00
encode_uint32 ( xr_mode_index = = /* XRMode.OVR */ 1 & & focus_awareness ? 0xFFFFFFFF : 0 , & p_manifest . write [ iofs + 16 ] ) ;
2020-06-04 14:54:36 -07:00
}
2020-06-27 16:40:22 -07:00
is_focus_aware_metadata = tname = = " meta-data " & & attrname = = " name " & & value = = " com.oculus.vr.focusaware " ;
2017-03-23 20:14:12 -03:00
iofs + = 20 ;
}
} break ;
2018-07-04 15:47:33 +02:00
case CHUNK_XML_END_TAG : {
int iofs = ofs + 8 ;
uint32_t name = decode_uint32 ( & p_manifest [ iofs + 12 ] ) ;
String tname = string_table [ name ] ;
2020-01-09 17:10:04 -08:00
if ( tname = = " uses-feature " ) {
Vector < String > feature_names ;
Vector < bool > feature_required_list ;
Vector < int > feature_versions ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
if ( xr_mode_index = = 1 /* XRMode.OVR */ ) {
// Check for degrees of freedom
int dof_index = p_preset - > get ( " xr_features/degrees_of_freedom " ) ; // 0: none, 1: 3dof and 6dof, 2: 6dof
if ( dof_index > 0 ) {
feature_names . push_back ( " android.hardware.vr.headtracking " ) ;
feature_required_list . push_back ( dof_index = = 2 ) ;
feature_versions . push_back ( 1 ) ;
}
// Check for hand tracking
int hand_tracking_index = p_preset - > get ( " xr_features/hand_tracking " ) ; // 0: none, 1: optional, 2: required
if ( hand_tracking_index > 0 ) {
feature_names . push_back ( " oculus.software.handtracking " ) ;
feature_required_list . push_back ( hand_tracking_index = = 2 ) ;
feature_versions . push_back ( - 1 ) ; // no version attribute should be added.
}
2019-07-28 17:51:27 -07:00
}
2020-01-09 17:10:04 -08:00
if ( feature_names . size ( ) > 0 ) {
ofs + = 24 ; // skip over end tag
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
// save manifest ending so we can restore it
Vector < uint8_t > manifest_end ;
uint32_t manifest_cur_size = p_manifest . size ( ) ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
manifest_end . resize ( p_manifest . size ( ) - ofs ) ;
memcpy ( manifest_end . ptrw ( ) , & p_manifest [ ofs ] , manifest_end . size ( ) ) ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t attr_name_string = string_table . find ( " name " ) ;
ERR_FAIL_COND_MSG ( attr_name_string = = - 1 , " Template does not have 'name' attribute. " ) ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t ns_android_string = string_table . find ( " http://schemas.android.com/apk/res/android " ) ;
if ( ns_android_string = = - 1 ) {
string_table . push_back ( " http://schemas.android.com/apk/res/android " ) ;
ns_android_string = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t attr_uses_feature_string = string_table . find ( " uses-feature " ) ;
if ( attr_uses_feature_string = = - 1 ) {
string_table . push_back ( " uses-feature " ) ;
attr_uses_feature_string = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t attr_required_string = string_table . find ( " required " ) ;
if ( attr_required_string = = - 1 ) {
string_table . push_back ( " required " ) ;
attr_required_string = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
for ( int i = 0 ; i < feature_names . size ( ) ; i + + ) {
String feature_name = feature_names [ i ] ;
bool feature_required = feature_required_list [ i ] ;
int feature_version = feature_versions [ i ] ;
bool has_version_attribute = feature_version ! = - 1 ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
print_line ( " Adding feature " + feature_name ) ;
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t feature_string = string_table . find ( feature_name ) ;
if ( feature_string = = - 1 ) {
string_table . push_back ( feature_name ) ;
feature_string = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
String required_value_string = feature_required ? " true " : " false " ;
int32_t required_value = string_table . find ( required_value_string ) ;
if ( required_value = = - 1 ) {
string_table . push_back ( required_value_string ) ;
required_value = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
int32_t attr_version_string = - 1 ;
int32_t version_value = - 1 ;
int tag_size ;
int attr_count ;
if ( has_version_attribute ) {
attr_version_string = string_table . find ( " version " ) ;
if ( attr_version_string = = - 1 ) {
string_table . push_back ( " version " ) ;
attr_version_string = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
version_value = string_table . find ( itos ( feature_version ) ) ;
if ( version_value = = - 1 ) {
string_table . push_back ( itos ( feature_version ) ) ;
version_value = string_table . size ( ) - 1 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
tag_size = 96 ; // node and three attrs + end node
attr_count = 3 ;
} else {
tag_size = 76 ; // node and two attrs + end node
attr_count = 2 ;
}
manifest_cur_size + = tag_size + 24 ;
p_manifest . resize ( manifest_cur_size ) ;
// start tag
encode_uint16 ( 0x102 , & p_manifest . write [ ofs ] ) ; // type
encode_uint16 ( 16 , & p_manifest . write [ ofs + 2 ] ) ; // headersize
encode_uint32 ( tag_size , & p_manifest . write [ ofs + 4 ] ) ; // size
encode_uint32 ( 0 , & p_manifest . write [ ofs + 8 ] ) ; // lineno
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 12 ] ) ; // comment
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 16 ] ) ; // ns
encode_uint32 ( attr_uses_feature_string , & p_manifest . write [ ofs + 20 ] ) ; // name
encode_uint16 ( 20 , & p_manifest . write [ ofs + 24 ] ) ; // attr_start
encode_uint16 ( 20 , & p_manifest . write [ ofs + 26 ] ) ; // attr_size
encode_uint16 ( attr_count , & p_manifest . write [ ofs + 28 ] ) ; // num_attrs
encode_uint16 ( 0 , & p_manifest . write [ ofs + 30 ] ) ; // id_index
encode_uint16 ( 0 , & p_manifest . write [ ofs + 32 ] ) ; // class_index
encode_uint16 ( 0 , & p_manifest . write [ ofs + 34 ] ) ; // style_index
// android:name attribute
encode_uint32 ( ns_android_string , & p_manifest . write [ ofs + 36 ] ) ; // ns
encode_uint32 ( attr_name_string , & p_manifest . write [ ofs + 40 ] ) ; // 'name'
encode_uint32 ( feature_string , & p_manifest . write [ ofs + 44 ] ) ; // raw_value
encode_uint16 ( 8 , & p_manifest . write [ ofs + 48 ] ) ; // typedvalue_size
p_manifest . write [ ofs + 50 ] = 0 ; // typedvalue_always0
p_manifest . write [ ofs + 51 ] = 0x03 ; // typedvalue_type (string)
encode_uint32 ( feature_string , & p_manifest . write [ ofs + 52 ] ) ; // typedvalue reference
// android:required attribute
encode_uint32 ( ns_android_string , & p_manifest . write [ ofs + 56 ] ) ; // ns
encode_uint32 ( attr_required_string , & p_manifest . write [ ofs + 60 ] ) ; // 'name'
encode_uint32 ( required_value , & p_manifest . write [ ofs + 64 ] ) ; // raw_value
encode_uint16 ( 8 , & p_manifest . write [ ofs + 68 ] ) ; // typedvalue_size
p_manifest . write [ ofs + 70 ] = 0 ; // typedvalue_always0
p_manifest . write [ ofs + 71 ] = 0x03 ; // typedvalue_type (string)
encode_uint32 ( required_value , & p_manifest . write [ ofs + 72 ] ) ; // typedvalue reference
ofs + = 76 ;
if ( has_version_attribute ) {
// android:version attribute
encode_uint32 ( ns_android_string , & p_manifest . write [ ofs ] ) ; // ns
encode_uint32 ( attr_version_string , & p_manifest . write [ ofs + 4 ] ) ; // 'name'
encode_uint32 ( version_value , & p_manifest . write [ ofs + 8 ] ) ; // raw_value
encode_uint16 ( 8 , & p_manifest . write [ ofs + 12 ] ) ; // typedvalue_size
p_manifest . write [ ofs + 14 ] = 0 ; // typedvalue_always0
p_manifest . write [ ofs + 15 ] = 0x03 ; // typedvalue_type (string)
encode_uint32 ( version_value , & p_manifest . write [ ofs + 16 ] ) ; // typedvalue reference
ofs + = 20 ;
}
2019-07-28 17:51:27 -07:00
2020-01-09 17:10:04 -08:00
// end tag
encode_uint16 ( 0x103 , & p_manifest . write [ ofs ] ) ; // type
encode_uint16 ( 16 , & p_manifest . write [ ofs + 2 ] ) ; // headersize
encode_uint32 ( 24 , & p_manifest . write [ ofs + 4 ] ) ; // size
encode_uint32 ( 0 , & p_manifest . write [ ofs + 8 ] ) ; // lineno
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 12 ] ) ; // comment
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 16 ] ) ; // ns
encode_uint32 ( attr_uses_feature_string , & p_manifest . write [ ofs + 20 ] ) ; // name
ofs + = 24 ;
}
memcpy ( & p_manifest . write [ ofs ] , manifest_end . ptr ( ) , manifest_end . size ( ) ) ;
ofs - = 24 ; // go back over back end
2019-07-28 17:51:27 -07:00
}
}
2018-07-04 15:47:33 +02:00
if ( tname = = " manifest " ) {
// save manifest ending so we can restore it
Vector < uint8_t > manifest_end ;
uint32_t manifest_cur_size = p_manifest . size ( ) ;
manifest_end . resize ( p_manifest . size ( ) - ofs ) ;
memcpy ( manifest_end . ptrw ( ) , & p_manifest [ ofs ] , manifest_end . size ( ) ) ;
int32_t attr_name_string = string_table . find ( " name " ) ;
2019-08-09 06:49:33 +02:00
ERR_FAIL_COND_MSG ( attr_name_string = = - 1 , " Template does not have 'name' attribute. " ) ;
2018-07-04 15:47:33 +02:00
int32_t ns_android_string = string_table . find ( " android " ) ;
2019-08-09 06:49:33 +02:00
ERR_FAIL_COND_MSG ( ns_android_string = = - 1 , " Template does not have 'android' namespace. " ) ;
2018-07-04 15:47:33 +02:00
int32_t attr_uses_permission_string = string_table . find ( " uses-permission " ) ;
if ( attr_uses_permission_string = = - 1 ) {
string_table . push_back ( " uses-permission " ) ;
attr_uses_permission_string = string_table . size ( ) - 1 ;
}
for ( int i = 0 ; i < perms . size ( ) ; + + i ) {
print_line ( " Adding permission " + perms [ i ] ) ;
manifest_cur_size + = 56 + 24 ; // node + end node
p_manifest . resize ( manifest_cur_size ) ;
// Add permission to the string pool
int32_t perm_string = string_table . find ( perms [ i ] ) ;
if ( perm_string = = - 1 ) {
string_table . push_back ( perms [ i ] ) ;
perm_string = string_table . size ( ) - 1 ;
}
// start tag
2018-07-25 03:11:03 +02:00
encode_uint16 ( 0x102 , & p_manifest . write [ ofs ] ) ; // type
encode_uint16 ( 16 , & p_manifest . write [ ofs + 2 ] ) ; // headersize
encode_uint32 ( 56 , & p_manifest . write [ ofs + 4 ] ) ; // size
encode_uint32 ( 0 , & p_manifest . write [ ofs + 8 ] ) ; // lineno
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 12 ] ) ; // comment
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 16 ] ) ; // ns
encode_uint32 ( attr_uses_permission_string , & p_manifest . write [ ofs + 20 ] ) ; // name
encode_uint16 ( 20 , & p_manifest . write [ ofs + 24 ] ) ; // attr_start
encode_uint16 ( 20 , & p_manifest . write [ ofs + 26 ] ) ; // attr_size
encode_uint16 ( 1 , & p_manifest . write [ ofs + 28 ] ) ; // num_attrs
encode_uint16 ( 0 , & p_manifest . write [ ofs + 30 ] ) ; // id_index
encode_uint16 ( 0 , & p_manifest . write [ ofs + 32 ] ) ; // class_index
encode_uint16 ( 0 , & p_manifest . write [ ofs + 34 ] ) ; // style_index
2018-07-04 15:47:33 +02:00
// attribute
2018-07-25 03:11:03 +02:00
encode_uint32 ( ns_android_string , & p_manifest . write [ ofs + 36 ] ) ; // ns
encode_uint32 ( attr_name_string , & p_manifest . write [ ofs + 40 ] ) ; // 'name'
encode_uint32 ( perm_string , & p_manifest . write [ ofs + 44 ] ) ; // raw_value
encode_uint16 ( 8 , & p_manifest . write [ ofs + 48 ] ) ; // typedvalue_size
p_manifest . write [ ofs + 50 ] = 0 ; // typedvalue_always0
p_manifest . write [ ofs + 51 ] = 0x03 ; // typedvalue_type (string)
encode_uint32 ( perm_string , & p_manifest . write [ ofs + 52 ] ) ; // typedvalue reference
2018-07-04 15:47:33 +02:00
ofs + = 56 ;
// end tag
2018-07-25 03:11:03 +02:00
encode_uint16 ( 0x103 , & p_manifest . write [ ofs ] ) ; // type
encode_uint16 ( 16 , & p_manifest . write [ ofs + 2 ] ) ; // headersize
encode_uint32 ( 24 , & p_manifest . write [ ofs + 4 ] ) ; // size
encode_uint32 ( 0 , & p_manifest . write [ ofs + 8 ] ) ; // lineno
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 12 ] ) ; // comment
encode_uint32 ( - 1 , & p_manifest . write [ ofs + 16 ] ) ; // ns
encode_uint32 ( attr_uses_permission_string , & p_manifest . write [ ofs + 20 ] ) ; // name
2018-07-04 15:47:33 +02:00
ofs + = 24 ;
}
// copy footer back in
2018-07-25 03:11:03 +02:00
memcpy ( & p_manifest . write [ ofs ] , manifest_end . ptr ( ) , manifest_end . size ( ) ) ;
2018-07-04 15:47:33 +02:00
}
} break ;
2017-03-23 20:14:12 -03:00
}
ofs + = size ;
}
//create new andriodmanifest binary
Vector < uint8_t > ret ;
ret . resize ( string_table_begins + string_table . size ( ) * 4 ) ;
for ( uint32_t i = 0 ; i < string_table_begins ; i + + ) {
2018-07-25 03:11:03 +02:00
ret . write [ i ] = p_manifest [ i ] ;
2017-03-23 20:14:12 -03:00
}
ofs = 0 ;
for ( int i = 0 ; i < string_table . size ( ) ; i + + ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( ofs , & ret . write [ string_table_begins + i * 4 ] ) ;
2017-03-23 20:14:12 -03:00
ofs + = string_table [ i ] . length ( ) * 2 + 2 + 2 ;
}
2018-07-04 15:47:33 +02:00
2017-03-23 20:14:12 -03:00
ret . resize ( ret . size ( ) + ofs ) ;
2018-07-04 15:47:33 +02:00
string_data_offset = ret . size ( ) - ofs ;
2018-07-25 03:11:03 +02:00
uint8_t * chars = & ret . write [ string_data_offset ] ;
2017-03-23 20:14:12 -03:00
for ( int i = 0 ; i < string_table . size ( ) ; i + + ) {
String s = string_table [ i ] ;
encode_uint16 ( s . length ( ) , chars ) ;
chars + = 2 ;
2018-07-04 15:47:33 +02:00
for ( int j = 0 ; j < s . length ( ) ; j + + ) {
2017-03-23 20:14:12 -03:00
encode_uint16 ( s [ j ] , chars ) ;
chars + = 2 ;
}
encode_uint16 ( 0 , chars ) ;
chars + = 2 ;
}
for ( int i = 0 ; i < stable_extra . size ( ) ; i + + ) {
ret . push_back ( stable_extra [ i ] ) ;
}
2018-07-04 15:47:33 +02:00
//pad
2021-05-05 12:44:11 +02:00
while ( ret . size ( ) % 4 ) {
2017-03-23 20:14:12 -03:00
ret . push_back ( 0 ) ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
uint32_t new_stable_end = ret . size ( ) ;
uint32_t extra = ( p_manifest . size ( ) - string_table_ends ) ;
ret . resize ( new_stable_end + extra ) ;
2021-05-05 12:44:11 +02:00
for ( uint32_t i = 0 ; i < extra ; i + + ) {
2018-07-25 03:11:03 +02:00
ret . write [ new_stable_end + i ] = p_manifest [ string_table_ends + i ] ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
2021-05-05 12:44:11 +02:00
while ( ret . size ( ) % 4 ) {
2017-03-23 20:14:12 -03:00
ret . push_back ( 0 ) ;
2021-05-05 12:44:11 +02:00
}
2018-07-25 03:11:03 +02:00
encode_uint32 ( ret . size ( ) , & ret . write [ 4 ] ) ; //update new file size
2017-03-23 20:14:12 -03:00
2018-07-25 03:11:03 +02:00
encode_uint32 ( new_stable_end - 8 , & ret . write [ 12 ] ) ; //update new string table size
encode_uint32 ( string_table . size ( ) , & ret . write [ 16 ] ) ; //update new number of strings
encode_uint32 ( string_data_offset - 8 , & ret . write [ 28 ] ) ; //update new string data offset
2017-03-23 20:14:12 -03:00
p_manifest = ret ;
}
static String _parse_string ( const uint8_t * p_bytes , bool p_utf8 ) {
uint32_t offset = 0 ;
2020-03-20 16:49:50 +00:00
uint32_t len = 0 ;
2017-03-23 20:14:12 -03:00
if ( p_utf8 ) {
2020-03-20 16:49:50 +00:00
uint8_t byte = p_bytes [ offset ] ;
2021-05-05 12:44:11 +02:00
if ( byte & 0x80 ) {
2020-03-20 16:49:50 +00:00
offset + = 2 ;
2021-05-05 12:44:11 +02:00
} else {
2020-03-20 16:49:50 +00:00
offset + = 1 ;
2021-05-05 12:44:11 +02:00
}
2020-03-20 16:49:50 +00:00
byte = p_bytes [ offset ] ;
offset + + ;
if ( byte & 0x80 ) {
len = byte & 0x7F ;
len = ( len < < 8 ) + p_bytes [ offset ] ;
offset + + ;
} else {
len = byte ;
}
} else {
len = decode_uint16 ( & p_bytes [ offset ] ) ;
offset + = 2 ;
if ( len & 0x8000 ) {
len & = 0x7FFF ;
len = ( len < < 16 ) + decode_uint16 ( & p_bytes [ offset ] ) ;
offset + = 2 ;
}
2017-03-23 20:14:12 -03:00
}
if ( p_utf8 ) {
Vector < uint8_t > str8 ;
str8 . resize ( len + 1 ) ;
for ( uint32_t i = 0 ; i < len ; i + + ) {
2018-07-25 03:11:03 +02:00
str8 . write [ i ] = p_bytes [ offset + i ] ;
2017-03-23 20:14:12 -03:00
}
2018-07-25 03:11:03 +02:00
str8 . write [ len ] = 0 ;
2017-03-23 20:14:12 -03:00
String str ;
str . parse_utf8 ( ( const char * ) str8 . ptr ( ) ) ;
return str ;
} else {
String str ;
for ( uint32_t i = 0 ; i < len ; i + + ) {
CharType c = decode_uint16 ( & p_bytes [ offset + i * 2 ] ) ;
2021-05-05 12:44:11 +02:00
if ( c = = 0 ) {
2017-03-23 20:14:12 -03:00
break ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
str + = String : : chr ( c ) ;
}
return str ;
}
}
2020-06-26 14:58:16 -04:00
void _fix_resources ( const Ref < EditorExportPreset > & p_preset , Vector < uint8_t > & r_manifest ) {
2017-03-23 20:14:12 -03:00
const int UTF8_FLAG = 0x00000100 ;
2020-06-26 14:58:16 -04:00
uint32_t string_block_len = decode_uint32 ( & r_manifest [ 16 ] ) ;
uint32_t string_count = decode_uint32 ( & r_manifest [ 20 ] ) ;
uint32_t string_flags = decode_uint32 ( & r_manifest [ 28 ] ) ;
2017-03-23 20:14:12 -03:00
const uint32_t string_table_begins = 40 ;
Vector < String > string_table ;
2017-03-24 15:26:55 +09:00
String package_name = p_preset - > get ( " package/name " ) ;
2017-03-23 20:14:12 -03:00
for ( uint32_t i = 0 ; i < string_count ; i + + ) {
2020-06-26 14:58:16 -04:00
uint32_t offset = decode_uint32 ( & r_manifest [ string_table_begins + i * 4 ] ) ;
2017-03-23 20:14:12 -03:00
offset + = string_table_begins + string_count * 4 ;
2020-06-26 14:58:16 -04:00
String str = _parse_string ( & r_manifest [ offset ] , string_flags & UTF8_FLAG ) ;
2017-03-23 20:14:12 -03:00
if ( str . begins_with ( " godot-project-name " ) ) {
if ( str = = " godot-project-name " ) {
//project name
2017-03-24 15:26:55 +09:00
str = get_project_name ( package_name ) ;
2017-03-23 20:14:12 -03:00
} else {
String lang = str . substr ( str . find_last ( " - " ) + 1 , str . length ( ) ) . replace ( " - " , " _ " ) ;
2017-07-17 21:05:38 -03:00
String prop = " application/config/name_ " + lang ;
2017-10-05 15:34:34 -03:00
if ( ProjectSettings : : get_singleton ( ) - > has_setting ( prop ) ) {
2017-07-19 17:00:46 -03:00
str = ProjectSettings : : get_singleton ( ) - > get ( prop ) ;
2017-03-23 20:14:12 -03:00
} else {
2017-03-24 15:26:55 +09:00
str = get_project_name ( package_name ) ;
2017-03-23 20:14:12 -03:00
}
}
}
string_table . push_back ( str ) ;
}
//write a new string table, but use 16 bits
Vector < uint8_t > ret ;
ret . resize ( string_table_begins + string_table . size ( ) * 4 ) ;
for ( uint32_t i = 0 ; i < string_table_begins ; i + + ) {
2020-06-26 14:58:16 -04:00
ret . write [ i ] = r_manifest [ i ] ;
2017-03-23 20:14:12 -03:00
}
int ofs = 0 ;
for ( int i = 0 ; i < string_table . size ( ) ; i + + ) {
2018-07-25 03:11:03 +02:00
encode_uint32 ( ofs , & ret . write [ string_table_begins + i * 4 ] ) ;
2017-03-23 20:14:12 -03:00
ofs + = string_table [ i ] . length ( ) * 2 + 2 + 2 ;
}
ret . resize ( ret . size ( ) + ofs ) ;
2018-07-25 03:11:03 +02:00
uint8_t * chars = & ret . write [ ret . size ( ) - ofs ] ;
2017-03-23 20:14:12 -03:00
for ( int i = 0 ; i < string_table . size ( ) ; i + + ) {
String s = string_table [ i ] ;
encode_uint16 ( s . length ( ) , chars ) ;
chars + = 2 ;
for ( int j = 0 ; j < s . length ( ) ; j + + ) {
encode_uint16 ( s [ j ] , chars ) ;
chars + = 2 ;
}
encode_uint16 ( 0 , chars ) ;
chars + = 2 ;
}
//pad
2021-05-05 12:44:11 +02:00
while ( ret . size ( ) % 4 ) {
2017-03-23 20:14:12 -03:00
ret . push_back ( 0 ) ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
//change flags to not use utf8
2018-07-25 03:11:03 +02:00
encode_uint32 ( string_flags & ~ 0x100 , & ret . write [ 28 ] ) ;
2017-03-23 20:14:12 -03:00
//change length
2018-07-25 03:11:03 +02:00
encode_uint32 ( ret . size ( ) - 12 , & ret . write [ 16 ] ) ;
2017-03-23 20:14:12 -03:00
//append the rest...
int rest_from = 12 + string_block_len ;
int rest_to = ret . size ( ) ;
2020-06-26 14:58:16 -04:00
int rest_len = ( r_manifest . size ( ) - rest_from ) ;
ret . resize ( ret . size ( ) + ( r_manifest . size ( ) - rest_from ) ) ;
2017-03-23 20:14:12 -03:00
for ( int i = 0 ; i < rest_len ; i + + ) {
2020-06-26 14:58:16 -04:00
ret . write [ rest_to + i ] = r_manifest [ rest_from + i ] ;
2017-03-23 20:14:12 -03:00
}
//finally update the size
2018-07-25 03:11:03 +02:00
encode_uint32 ( ret . size ( ) , & ret . write [ 4 ] ) ;
2017-03-23 20:14:12 -03:00
2020-06-26 14:58:16 -04:00
r_manifest = ret ;
2017-03-23 20:14:12 -03:00
//printf("end\n");
}
2020-08-19 15:38:50 -07:00
void _load_image_data ( const Ref < Image > & p_splash_image , Vector < uint8_t > & p_data ) {
PoolVector < uint8_t > png_buffer ;
Error err = PNGDriverCommon : : image_to_png ( p_splash_image , png_buffer ) ;
if ( err = = OK ) {
p_data . resize ( png_buffer . size ( ) ) ;
memcpy ( p_data . ptrw ( ) , png_buffer . read ( ) . ptr ( ) , p_data . size ( ) ) ;
} else {
String err_str = String ( " Failed to convert splash image to png. " ) ;
WARN_PRINT ( err_str . utf8 ( ) . get_data ( ) ) ;
}
}
2020-07-22 15:52:26 -04:00
void _process_launcher_icons ( const String & p_file_name , const Ref < Image > & p_source_image , int dimension , Vector < uint8_t > & p_data ) {
Ref < Image > working_image = p_source_image ;
2020-01-06 18:37:39 +00:00
2020-07-22 15:52:26 -04:00
if ( p_source_image - > get_width ( ) ! = dimension | | p_source_image - > get_height ( ) ! = dimension ) {
working_image = p_source_image - > duplicate ( ) ;
working_image - > resize ( dimension , dimension , Image : : Interpolation : : INTERPOLATE_LANCZOS ) ;
}
PoolVector < uint8_t > png_buffer ;
Error err = PNGDriverCommon : : image_to_png ( working_image , png_buffer ) ;
if ( err = = OK ) {
p_data . resize ( png_buffer . size ( ) ) ;
memcpy ( p_data . ptrw ( ) , png_buffer . read ( ) . ptr ( ) , p_data . size ( ) ) ;
} else {
String err_str = String ( " Failed to convert resized icon ( " ) + p_file_name + " ) to png. " ;
WARN_PRINT ( err_str . utf8 ( ) . get_data ( ) ) ;
}
}
2021-02-18 01:54:35 -08:00
String load_splash_refs ( Ref < Image > & splash_image , Ref < Image > & splash_bg_color_image ) {
bool scale_splash = ProjectSettings : : get_singleton ( ) - > get ( " application/boot_splash/fullsize " ) ;
bool apply_filter = ProjectSettings : : get_singleton ( ) - > get ( " application/boot_splash/use_filter " ) ;
2020-08-19 15:38:50 -07:00
String project_splash_path = ProjectSettings : : get_singleton ( ) - > get ( " application/boot_splash/image " ) ;
if ( ! project_splash_path . empty ( ) ) {
splash_image . instance ( ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Loading splash image: " + project_splash_path ) ;
2020-08-19 15:38:50 -07:00
const Error err = ImageLoader : : load_image ( project_splash_path , splash_image ) ;
if ( err ) {
2021-02-16 16:33:15 -08:00
if ( OS : : get_singleton ( ) - > is_stdout_verbose ( ) ) {
print_error ( " - unable to load splash image from " + project_splash_path + " ( " + itos ( err ) + " ) " ) ;
}
2020-08-19 15:38:50 -07:00
splash_image . unref ( ) ;
}
}
if ( splash_image . is_null ( ) ) {
// Use the default
2021-02-16 16:33:15 -08:00
print_verbose ( " Using default splash image. " ) ;
2020-08-19 15:38:50 -07:00
splash_image = Ref < Image > ( memnew ( Image ( boot_splash_png ) ) ) ;
}
2021-04-12 16:33:37 -07:00
if ( scale_splash ) {
Size2 screen_size = Size2 ( ProjectSettings : : get_singleton ( ) - > get ( " display/window/size/width " ) , ProjectSettings : : get_singleton ( ) - > get ( " display/window/size/height " ) ) ;
int width , height ;
if ( screen_size . width > screen_size . height ) {
// scale horizontally
height = screen_size . height ;
width = splash_image - > get_width ( ) * screen_size . height / splash_image - > get_height ( ) ;
} else {
// scale vertically
width = screen_size . width ;
height = splash_image - > get_height ( ) * screen_size . width / splash_image - > get_width ( ) ;
}
splash_image - > resize ( width , height ) ;
}
2020-08-19 15:38:50 -07:00
// Setup the splash bg color
bool bg_color_valid ;
Color bg_color = ProjectSettings : : get_singleton ( ) - > get ( " application/boot_splash/bg_color " , & bg_color_valid ) ;
if ( ! bg_color_valid ) {
bg_color = boot_splash_bg_color ;
}
2021-02-16 16:33:15 -08:00
print_verbose ( " Creating splash background color image. " ) ;
2020-08-19 15:38:50 -07:00
splash_bg_color_image . instance ( ) ;
splash_bg_color_image - > create ( splash_image - > get_width ( ) , splash_image - > get_height ( ) , false , splash_image - > get_format ( ) ) ;
splash_bg_color_image - > fill ( bg_color ) ;
2021-02-18 01:54:35 -08:00
2021-04-12 16:33:37 -07:00
String processed_splash_config_xml = vformat ( SPLASH_CONFIG_XML_CONTENT , bool_to_string ( apply_filter ) ) ;
2021-02-18 01:54:35 -08:00
return processed_splash_config_xml ;
2020-08-19 15:38:50 -07:00
}
2020-07-22 15:52:26 -04:00
void load_icon_refs ( const Ref < EditorExportPreset > & p_preset , Ref < Image > & icon , Ref < Image > & foreground , Ref < Image > & background ) {
String project_icon_path = ProjectSettings : : get_singleton ( ) - > get ( " application/config/icon " ) ;
icon . instance ( ) ;
foreground . instance ( ) ;
background . instance ( ) ;
// Regular icon: user selection -> project icon -> default.
String path = static_cast < String > ( p_preset - > get ( launcher_icon_option ) ) . strip_edges ( ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Loading regular icon from " + path ) ;
2020-07-22 15:52:26 -04:00
if ( path . empty ( ) | | ImageLoader : : load_image ( path , icon ) ! = OK ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " - falling back to project icon: " + project_icon_path ) ;
2020-07-22 15:52:26 -04:00
ImageLoader : : load_image ( project_icon_path , icon ) ;
}
// Adaptive foreground: user selection -> regular icon (user selection -> project icon -> default).
path = static_cast < String > ( p_preset - > get ( launcher_adaptive_icon_foreground_option ) ) . strip_edges ( ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Loading adaptive foreground icon from " + path ) ;
2020-07-22 15:52:26 -04:00
if ( path . empty ( ) | | ImageLoader : : load_image ( path , foreground ) ! = OK ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " - falling back to using the regular icon " ) ;
2020-07-22 15:52:26 -04:00
foreground = icon ;
}
// Adaptive background: user selection -> default.
path = static_cast < String > ( p_preset - > get ( launcher_adaptive_icon_background_option ) ) . strip_edges ( ) ;
if ( ! path . empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Loading adaptive background icon from " + path ) ;
2020-07-22 15:52:26 -04:00
ImageLoader : : load_image ( path , background ) ;
}
}
void store_image ( const LauncherIcon launcher_icon , const Vector < uint8_t > & data ) {
2020-08-19 15:38:50 -07:00
store_image ( launcher_icon . export_path , data ) ;
}
void store_image ( const String & export_path , const Vector < uint8_t > & data ) {
String img_path = export_path . insert ( 0 , " res://android/build/ " ) ;
2020-07-22 15:52:26 -04:00
store_file_at_path ( img_path , data ) ;
}
2020-08-19 15:38:50 -07:00
void _copy_icons_to_gradle_project ( const Ref < EditorExportPreset > & p_preset ,
2021-02-18 01:54:35 -08:00
const String & processed_splash_config_xml ,
2020-08-19 15:38:50 -07:00
const Ref < Image > & splash_image ,
const Ref < Image > & splash_bg_color_image ,
const Ref < Image > & main_image ,
const Ref < Image > & foreground ,
const Ref < Image > & background ) {
2021-02-18 01:54:35 -08:00
// Store the splash configuration
if ( ! processed_splash_config_xml . empty ( ) ) {
print_verbose ( " Storing processed splash configuration: " + String ( " \n " ) + processed_splash_config_xml ) ;
store_string_at_path ( SPLASH_CONFIG_PATH , processed_splash_config_xml ) ;
}
2020-08-19 15:38:50 -07:00
// Store the splash image
if ( splash_image . is_valid ( ) & & ! splash_image - > empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Storing splash image in " + String ( SPLASH_IMAGE_EXPORT_PATH ) ) ;
2020-08-19 15:38:50 -07:00
Vector < uint8_t > data ;
_load_image_data ( splash_image , data ) ;
store_image ( SPLASH_IMAGE_EXPORT_PATH , data ) ;
}
// Store the splash bg color image
if ( splash_bg_color_image . is_valid ( ) & & ! splash_bg_color_image - > empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Storing splash background image in " + String ( SPLASH_BG_COLOR_PATH ) ) ;
2020-08-19 15:38:50 -07:00
Vector < uint8_t > data ;
_load_image_data ( splash_bg_color_image , data ) ;
store_image ( SPLASH_BG_COLOR_PATH , data ) ;
}
2020-07-22 15:52:26 -04:00
// Prepare images to be resized for the icons. If some image ends up being uninitialized,
// the default image from the export template will be used.
for ( int i = 0 ; i < icon_densities_count ; + + i ) {
if ( main_image . is_valid ( ) & & ! main_image - > empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Processing launcher icon for dimension " + itos ( launcher_icons [ i ] . dimensions ) + " into " + launcher_icons [ i ] . export_path ) ;
2020-07-22 15:52:26 -04:00
Vector < uint8_t > data ;
_process_launcher_icons ( launcher_icons [ i ] . export_path , main_image , launcher_icons [ i ] . dimensions , data ) ;
store_image ( launcher_icons [ i ] , data ) ;
2020-01-06 18:37:39 +00:00
}
2020-07-22 15:52:26 -04:00
if ( foreground . is_valid ( ) & & ! foreground - > empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Processing launcher adaptive icon foreground for dimension " + itos ( launcher_adaptive_icon_foregrounds [ i ] . dimensions ) + " into " + launcher_adaptive_icon_foregrounds [ i ] . export_path ) ;
2020-07-22 15:52:26 -04:00
Vector < uint8_t > data ;
_process_launcher_icons ( launcher_adaptive_icon_foregrounds [ i ] . export_path , foreground ,
launcher_adaptive_icon_foregrounds [ i ] . dimensions , data ) ;
store_image ( launcher_adaptive_icon_foregrounds [ i ] , data ) ;
}
if ( background . is_valid ( ) & & ! background - > empty ( ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Processing launcher adaptive icon background for dimension " + itos ( launcher_adaptive_icon_backgrounds [ i ] . dimensions ) + " into " + launcher_adaptive_icon_backgrounds [ i ] . export_path ) ;
2020-07-22 15:52:26 -04:00
Vector < uint8_t > data ;
_process_launcher_icons ( launcher_adaptive_icon_backgrounds [ i ] . export_path , background ,
launcher_adaptive_icon_backgrounds [ i ] . dimensions , data ) ;
store_image ( launcher_adaptive_icon_backgrounds [ i ] , data ) ;
2020-01-06 18:37:39 +00:00
}
}
}
2017-11-21 01:12:36 +07:00
static Vector < String > get_enabled_abis ( const Ref < EditorExportPreset > & p_preset ) {
Vector < String > abis = get_abis ( ) ;
Vector < String > enabled_abis ;
for ( int i = 0 ; i < abis . size ( ) ; + + i ) {
bool is_enabled = p_preset - > get ( " architectures/ " + abis [ i ] ) ;
if ( is_enabled ) {
enabled_abis . push_back ( abis [ i ] ) ;
}
}
return enabled_abis ;
}
2017-03-23 20:14:12 -03:00
public :
typedef Error ( * EditorExportSaveFunction ) ( void * p_userdata , const String & p_path , const Vector < uint8_t > & p_data , int p_file , int p_total ) ;
public :
virtual void get_preset_features ( const Ref < EditorExportPreset > & p_preset , List < String > * r_features ) {
2018-09-30 23:10:18 -03:00
String driver = ProjectSettings : : get_singleton ( ) - > get ( " rendering/quality/driver/driver_name " ) ;
2019-03-03 11:52:53 +01:00
if ( driver = = " GLES2 " ) {
2018-09-30 23:10:18 -03:00
r_features - > push_back ( " etc " ) ;
2019-03-03 11:52:53 +01:00
} else if ( driver = = " GLES3 " ) {
2018-09-30 23:10:18 -03:00
r_features - > push_back ( " etc2 " ) ;
2019-03-05 13:46:51 +01:00
if ( ProjectSettings : : get_singleton ( ) - > get ( " rendering/quality/driver/fallback_to_gles2 " ) ) {
r_features - > push_back ( " etc " ) ;
}
2018-09-30 23:10:18 -03:00
}
2017-11-21 01:12:36 +07:00
Vector < String > abis = get_enabled_abis ( p_preset ) ;
for ( int i = 0 ; i < abis . size ( ) ; + + i ) {
r_features - > push_back ( abis [ i ] ) ;
}
2017-03-23 20:14:12 -03:00
}
virtual void get_export_options ( List < ExportOption > * r_options ) {
2020-01-07 13:29:02 +01:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " custom_template/debug " , PROPERTY_HINT_GLOBAL_FILE , " *.apk " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " custom_template/release " , PROPERTY_HINT_GLOBAL_FILE , " *.apk " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " custom_template/use_custom_build " ) , false ) ) ;
2020-12-23 01:30:29 -08:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : INT , " custom_template/export_format " , PROPERTY_HINT_ENUM , " Export APK,Export AAB " ) , EXPORT_FORMAT_APK ) ) ;
2020-04-24 00:45:14 -07:00
2021-01-14 15:05:42 +03:00
Vector < PluginConfigAndroid > plugins_configs = get_plugins ( ) ;
2020-04-24 00:45:14 -07:00
for ( int i = 0 ; i < plugins_configs . size ( ) ; i + + ) {
print_verbose ( " Found Android plugin " + plugins_configs [ i ] . name ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " plugins/ " + plugins_configs [ i ] . name ) , false ) ) ;
}
2021-01-31 13:34:42 +01:00
plugins_changed . clear ( ) ;
2020-04-24 00:45:14 -07:00
2020-11-20 10:45:39 +01:00
Vector < String > abis = get_abis ( ) ;
for ( int i = 0 ; i < abis . size ( ) ; + + i ) {
String abi = abis [ i ] ;
bool is_default = ( abi = = " armeabi-v7a " | | abi = = " arm64-v8a " ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " architectures/ " + abi ) , is_default ) ) ;
}
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/debug " , PROPERTY_HINT_GLOBAL_FILE , " *.keystore,*.jks " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/debug_user " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/debug_password " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/release " , PROPERTY_HINT_GLOBAL_FILE , " *.keystore,*.jks " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/release_user " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " keystore/release_password " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " one_click_deploy/clear_previous_install " ) , false ) ) ;
2018-05-29 11:06:20 +07:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : INT , " version/code " , PROPERTY_HINT_RANGE , " 1,4096,1,or_greater " ) , 1 ) ) ;
2017-03-23 20:14:12 -03:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " version/name " ) , " 1.0 " ) ) ;
2019-04-07 15:46:52 -03:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " package/unique_name " , PROPERTY_HINT_PLACEHOLDER_TEXT , " ext.domain.name " ) , " org.godotengine.$genname " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " package/name " , PROPERTY_HINT_PLACEHOLDER_TEXT , " Game Name [default if blank] " ) , " " ) ) ;
2017-03-23 20:14:12 -03:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " package/signed " ) , true ) ) ;
2020-11-20 10:45:39 +01:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , launcher_icon_option , PROPERTY_HINT_FILE , " *.png " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , launcher_adaptive_icon_foreground_option , PROPERTY_HINT_FILE , " *.png " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , launcher_adaptive_icon_background_option , PROPERTY_HINT_FILE , " *.png " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " graphics/32_bits_framebuffer " ) , true ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " graphics/opengl_debug " ) , false ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : INT , " xr_features/xr_mode " , PROPERTY_HINT_ENUM , " Regular,Oculus Mobile VR " ) , 0 ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : INT , " xr_features/degrees_of_freedom " , PROPERTY_HINT_ENUM , " None,3DOF and 6DOF,6DOF " ) , 0 ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : INT , " xr_features/hand_tracking " , PROPERTY_HINT_ENUM , " None,Optional,Required " ) , 0 ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " xr_features/focus_awareness " ) , false ) ) ;
2017-03-23 20:14:12 -03:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " screen/immersive_mode " ) , true ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " screen/support_small " ) , true ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " screen/support_normal " ) , true ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " screen/support_large " ) , true ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " screen/support_xlarge " ) , true ) ) ;
2020-11-20 10:45:39 +01:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " command_line/extra_args " ) , " " ) ) ;
2017-03-23 20:14:12 -03:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " apk_expansion/enable " ) , false ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " apk_expansion/SALT " ) , " " ) ) ;
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : STRING , " apk_expansion/public_key " , PROPERTY_HINT_MULTILINE_TEXT ) , " " ) ) ;
2018-11-30 11:13:50 -02:00
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : POOL_STRING_ARRAY , " permissions/custom_permissions " ) , PoolStringArray ( ) ) ) ;
2017-03-23 20:14:12 -03:00
const char * * perms = android_perms ;
while ( * perms ) {
r_options - > push_back ( ExportOption ( PropertyInfo ( Variant : : BOOL , " permissions/ " + String ( * perms ) . to_lower ( ) ) , false ) ) ;
perms + + ;
}
}
virtual String get_name ( ) const {
return " Android " ;
}
2017-07-19 17:00:46 -03:00
virtual String get_os_name ( ) const {
return " Android " ;
}
2017-03-23 20:14:12 -03:00
virtual Ref < Texture > get_logo ( ) const {
return logo ;
}
2020-04-24 00:45:14 -07:00
virtual bool should_update_export_options ( ) {
2021-01-31 13:34:42 +01:00
bool export_options_changed = plugins_changed . is_set ( ) ;
2020-04-24 00:45:14 -07:00
if ( export_options_changed ) {
// don't clear unless we're reporting true, to avoid race
2021-01-31 13:34:42 +01:00
plugins_changed . clear ( ) ;
2020-04-24 00:45:14 -07:00
}
return export_options_changed ;
}
2019-10-11 13:47:28 +02:00
virtual bool poll_export ( ) {
2021-01-31 13:34:42 +01:00
bool dc = devices_changed . is_set ( ) ;
2018-07-13 13:30:50 +01:00
if ( dc ) {
// don't clear unless we're reporting true, to avoid race
2021-01-31 13:34:42 +01:00
devices_changed . clear ( ) ;
2018-07-13 13:30:50 +01:00
}
2017-03-23 20:14:12 -03:00
return dc ;
}
2019-10-11 13:47:28 +02:00
virtual int get_options_count ( ) const {
2021-01-27 10:43:02 +01:00
device_lock . lock ( ) ;
2017-03-23 20:14:12 -03:00
int dc = devices . size ( ) ;
2021-01-27 10:43:02 +01:00
device_lock . unlock ( ) ;
2017-03-23 20:14:12 -03:00
return dc ;
}
2019-10-11 13:47:28 +02:00
virtual String get_options_tooltip ( ) const {
return TTR ( " Select device from the list " ) ;
}
virtual String get_option_label ( int p_index ) const {
ERR_FAIL_INDEX_V ( p_index , devices . size ( ) , " " ) ;
2021-01-27 10:43:02 +01:00
device_lock . lock ( ) ;
2019-10-11 13:47:28 +02:00
String s = devices [ p_index ] . name ;
2021-01-27 10:43:02 +01:00
device_lock . unlock ( ) ;
2017-03-23 20:14:12 -03:00
return s ;
}
2019-10-11 13:47:28 +02:00
virtual String get_option_tooltip ( int p_index ) const {
ERR_FAIL_INDEX_V ( p_index , devices . size ( ) , " " ) ;
2021-01-27 10:43:02 +01:00
device_lock . lock ( ) ;
2019-10-11 13:47:28 +02:00
String s = devices [ p_index ] . description ;
if ( devices . size ( ) = = 1 ) {
// Tooltip will be:
// Name
// Description
s = devices [ p_index ] . name + " \n \n " + s ;
}
2021-01-27 10:43:02 +01:00
device_lock . unlock ( ) ;
2017-03-23 20:14:12 -03:00
return s ;
}
virtual Error run ( const Ref < EditorExportPreset > & p_preset , int p_device , int p_debug_flags ) {
ERR_FAIL_INDEX_V ( p_device , devices . size ( ) , ERR_INVALID_PARAMETER ) ;
2020-01-26 11:45:17 +08:00
String can_export_error ;
bool can_export_missing_templates ;
if ( ! can_export ( p_preset , can_export_error , can_export_missing_templates ) ) {
EditorNode : : add_io_error ( can_export_error ) ;
return ERR_UNCONFIGURED ;
}
2021-01-27 10:43:02 +01:00
device_lock . lock ( ) ;
2017-03-23 20:14:12 -03:00
EditorProgress ep ( " run " , " Running on " + devices [ p_device ] . name , 3 ) ;
2020-12-23 22:16:56 -08:00
String adb = get_adb_path ( ) ;
2017-03-23 20:14:12 -03:00
2019-08-09 13:45:30 +02:00
// Export_temp APK.
2020-02-05 22:21:47 +01:00
if ( ep . step ( " Exporting APK... " , 0 ) ) {
2021-01-27 10:43:02 +01:00
device_lock . unlock ( ) ;
2018-12-19 16:50:40 -02:00
return ERR_SKIP ;
}
2017-03-23 20:14:12 -03:00
2017-08-30 20:25:09 +02:00
const bool use_remote = ( p_debug_flags & DEBUG_FLAG_REMOTE_DEBUG ) | | ( p_debug_flags & DEBUG_FLAG_DUMB_CLIENT ) ;
2020-01-27 19:59:24 +01:00
const bool use_reverse = devices [ p_device ] . api_level > = 21 ;
2017-08-30 20:25:09 +02:00
2021-05-05 12:44:11 +02:00
if ( use_reverse ) {
2017-08-30 20:25:09 +02:00
p_debug_flags | = DEBUG_FLAG_REMOTE_DEBUG_LOCALHOST ;
2021-05-05 12:44:11 +02:00
}
2017-03-23 20:14:12 -03:00
2021-04-15 13:26:05 -07:00
String tmp_export_path = EditorSettings : : get_singleton ( ) - > get_cache_dir ( ) . plus_file ( " tmpexport. " + uitos ( OS : : get_singleton ( ) - > get_unix_time ( ) ) + " .apk " ) ;
2019-08-09 13:45:30 +02:00
# define CLEANUP_AND_RETURN(m_err) \
{ \
DirAccess : : remove_file_or_error ( tmp_export_path ) ; \
2021-01-27 10:43:02 +01:00
device_lock . unlock ( ) ; \
2019-08-09 13:45:30 +02:00
return m_err ; \
}
// Export to temporary APK before sending to device.
2020-12-23 01:30:29 -08:00
Error err = export_project_helper ( p_preset , true , tmp_export_path , EXPORT_FORMAT_APK , true , p_debug_flags ) ;
2019-08-09 13:45:30 +02:00
if ( err ! = OK ) {
CLEANUP_AND_RETURN ( err ) ;
2017-03-23 20:14:12 -03:00
}
List < String > args ;
int rv ;
2021-04-15 13:26:05 -07:00
String output ;
2017-03-23 20:14:12 -03:00
bool remove_prev = p_preset - > get ( " one_click_deploy/clear_previous_install " ) ;
String version_name = p_preset - > get ( " version/name " ) ;
String package_name = p_preset - > get ( " package/unique_name " ) ;
2014-02-09 22:10:30 -03:00
2017-03-23 20:14:12 -03:00
if ( remove_prev ) {
2018-12-19 16:50:40 -02:00
if ( ep . step ( " Uninstalling... " , 1 ) ) {
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2018-12-19 16:50:40 -02:00
}
2017-03-23 20:14:12 -03:00
print_line ( " Uninstalling previous version: " + devices [ p_device ] . name ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " uninstall " ) ;
args . push_back ( get_package_name ( package_name ) ) ;
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
err = OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-03-23 20:14:12 -03:00
}
2018-08-24 09:35:07 +02:00
print_line ( " Installing to device (please wait...): " + devices [ p_device ] . name ) ;
2020-02-05 22:21:47 +01:00
if ( ep . step ( " Installing to device, please wait... " , 2 ) ) {
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2018-12-19 16:50:40 -02:00
}
2017-03-23 20:14:12 -03:00
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " install " ) ;
args . push_back ( " -r " ) ;
2019-08-09 13:45:30 +02:00
args . push_back ( tmp_export_path ) ;
2017-03-23 20:14:12 -03:00
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
err = OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-03-23 20:14:12 -03:00
if ( err | | rv ! = 0 ) {
EditorNode : : add_io_error ( " Could not install to device. " ) ;
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_CANT_CREATE ) ;
2017-03-23 20:14:12 -03:00
}
2017-08-30 20:25:09 +02:00
if ( use_remote ) {
if ( use_reverse ) {
2019-09-20 01:56:09 +02:00
static const char * const msg = " --- Device API >= 21; debugging over USB --- " ;
EditorNode : : get_singleton ( ) - > get_log ( ) - > add_message ( msg , EditorLog : : MSG_TYPE_EDITOR ) ;
2017-08-30 20:25:09 +02:00
print_line ( String ( msg ) . to_upper ( ) ) ;
2017-03-23 20:14:12 -03:00
2017-08-30 20:25:09 +02:00
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " reverse " ) ;
args . push_back ( " --remove-all " ) ;
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-03-23 20:14:12 -03:00
2017-08-30 20:25:09 +02:00
if ( p_debug_flags & DEBUG_FLAG_REMOTE_DEBUG ) {
int dbg_port = EditorSettings : : get_singleton ( ) - > get ( " network/debug/remote_port " ) ;
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " reverse " ) ;
args . push_back ( " tcp: " + itos ( dbg_port ) ) ;
args . push_back ( " tcp: " + itos ( dbg_port ) ) ;
2017-03-23 20:14:12 -03:00
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-08-30 20:25:09 +02:00
print_line ( " Reverse result: " + itos ( rv ) ) ;
}
2017-03-23 20:14:12 -03:00
2017-08-30 20:25:09 +02:00
if ( p_debug_flags & DEBUG_FLAG_DUMB_CLIENT ) {
int fs_port = EditorSettings : : get_singleton ( ) - > get ( " filesystem/file_server/port " ) ;
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " reverse " ) ;
args . push_back ( " tcp: " + itos ( fs_port ) ) ;
args . push_back ( " tcp: " + itos ( fs_port ) ) ;
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
err = OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-08-30 20:25:09 +02:00
print_line ( " Reverse result2: " + itos ( rv ) ) ;
}
} else {
2020-01-27 19:59:24 +01:00
static const char * const msg = " --- Device API < 21; debugging over Wi-Fi --- " ;
2019-09-20 01:56:09 +02:00
EditorNode : : get_singleton ( ) - > get_log ( ) - > add_message ( msg , EditorLog : : MSG_TYPE_EDITOR ) ;
2017-08-30 20:25:09 +02:00
print_line ( String ( msg ) . to_upper ( ) ) ;
}
2017-03-23 20:14:12 -03:00
}
2020-02-05 22:21:47 +01:00
if ( ep . step ( " Running on device... " , 3 ) ) {
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2018-12-19 16:50:40 -02:00
}
2017-03-23 20:14:12 -03:00
args . clear ( ) ;
args . push_back ( " -s " ) ;
args . push_back ( devices [ p_device ] . id ) ;
args . push_back ( " shell " ) ;
args . push_back ( " am " ) ;
args . push_back ( " start " ) ;
2017-08-30 20:21:26 +02:00
if ( ( bool ) EditorSettings : : get_singleton ( ) - > get ( " export/android/force_system_user " ) & & devices [ p_device ] . api_level > = 17 ) { // Multi-user introduced in Android 17
2017-07-27 17:20:19 +02:00
args . push_back ( " --user " ) ;
args . push_back ( " 0 " ) ;
}
2017-03-23 20:14:12 -03:00
args . push_back ( " -a " ) ;
args . push_back ( " android.intent.action.MAIN " ) ;
args . push_back ( " -n " ) ;
2019-09-22 15:02:10 +03:00
args . push_back ( get_package_name ( package_name ) + " /com.godot.game.GodotApp " ) ;
2017-03-23 20:14:12 -03:00
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
err = OS : : get_singleton ( ) - > execute ( adb , args , true , nullptr , & output , & rv , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2017-03-23 20:14:12 -03:00
if ( err | | rv ! = 0 ) {
2017-07-27 17:20:19 +02:00
EditorNode : : add_io_error ( " Could not execute on device. " ) ;
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_CANT_CREATE ) ;
2017-03-23 20:14:12 -03:00
}
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( OK ) ;
# undef CLEANUP_AND_RETURN
2017-03-23 20:14:12 -03:00
}
2017-05-25 20:57:13 +02:00
virtual Ref < Texture > get_run_icon ( ) const {
return run_icon ;
}
2020-12-23 22:16:56 -08:00
static String get_adb_path ( ) {
String exe_ext = " " ;
if ( OS : : get_singleton ( ) - > get_name ( ) = = " Windows " ) {
exe_ext = " .exe " ;
}
String sdk_path = EditorSettings : : get_singleton ( ) - > get ( " export/android/android_sdk_path " ) ;
return sdk_path . plus_file ( " platform-tools/adb " + exe_ext ) ;
}
static String get_apksigner_path ( ) {
String exe_ext = " " ;
if ( OS : : get_singleton ( ) - > get_name ( ) = = " Windows " ) {
exe_ext = " .bat " ;
}
String apksigner_command_name = " apksigner " + exe_ext ;
String sdk_path = EditorSettings : : get_singleton ( ) - > get ( " export/android/android_sdk_path " ) ;
String apksigner_path = " " ;
Error errn ;
String build_tools_dir = sdk_path . plus_file ( " build-tools " ) ;
DirAccessRef da = DirAccess : : open ( build_tools_dir , & errn ) ;
if ( errn ! = OK ) {
print_error ( " Unable to open Android 'build-tools' directory. " ) ;
return apksigner_path ;
}
// There are additional versions directories we need to go through.
da - > list_dir_begin ( ) ;
String sub_dir = da - > get_next ( ) ;
while ( ! sub_dir . empty ( ) ) {
if ( ! sub_dir . begins_with ( " . " ) & & da - > current_is_dir ( ) ) {
// Check if the tool is here.
String tool_path = build_tools_dir . plus_file ( sub_dir ) . plus_file ( apksigner_command_name ) ;
if ( FileAccess : : exists ( tool_path ) ) {
apksigner_path = tool_path ;
break ;
}
}
sub_dir = da - > get_next ( ) ;
}
da - > list_dir_end ( ) ;
if ( apksigner_path . empty ( ) ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Unable to find the 'apksigner' tool. " ) ) ;
}
return apksigner_path ;
}
2017-03-23 20:14:12 -03:00
virtual bool can_export ( const Ref < EditorExportPreset > & p_preset , String & r_error , bool & r_missing_templates ) const {
2018-08-20 17:35:33 +02:00
String err ;
2020-01-07 13:29:02 +01:00
bool valid = false ;
2017-03-23 20:14:12 -03:00
2020-01-07 13:29:02 +01:00
// Look for export templates (first official, and if defined custom templates).
2019-04-07 15:46:52 -03:00
2020-01-07 13:29:02 +01:00
if ( ! bool ( p_preset - > get ( " custom_template/use_custom_build " ) ) ) {
2020-06-01 23:13:31 +02:00
String template_err ;
bool dvalid = false ;
bool rvalid = false ;
2021-02-26 12:28:19 -08:00
bool has_export_templates = false ;
2019-04-07 15:46:52 -03:00
2020-01-07 13:29:02 +01:00
if ( p_preset - > get ( " custom_template/debug " ) ! = " " ) {
dvalid = FileAccess : : exists ( p_preset - > get ( " custom_template/debug " ) ) ;
if ( ! dvalid ) {
2020-06-01 23:13:31 +02:00
template_err + = TTR ( " Custom debug template not found. " ) + " \n " ;
2019-04-07 15:46:52 -03:00
}
2020-06-01 23:13:31 +02:00
} else {
2021-02-26 12:28:19 -08:00
has_export_templates | = exists_export_template ( " android_debug.apk " , & template_err ) ;
2018-08-20 17:35:33 +02:00
}
2020-06-01 23:13:31 +02:00
2020-01-07 13:29:02 +01:00
if ( p_preset - > get ( " custom_template/release " ) ! = " " ) {
rvalid = FileAccess : : exists ( p_preset - > get ( " custom_template/release " ) ) ;
if ( ! rvalid ) {
2020-06-01 23:13:31 +02:00
template_err + = TTR ( " Custom release template not found. " ) + " \n " ;
2019-04-07 15:46:52 -03:00
}
2020-06-01 23:13:31 +02:00
} else {
2021-02-26 12:28:19 -08:00
has_export_templates | = exists_export_template ( " android_release.apk " , & template_err ) ;
2018-08-20 17:35:33 +02:00
}
2020-01-07 13:29:02 +01:00
2021-02-26 12:28:19 -08:00
r_missing_templates = ! has_export_templates ;
valid = dvalid | | rvalid | | has_export_templates ;
2020-06-01 23:13:31 +02:00
if ( ! valid ) {
err + = template_err ;
}
2020-01-07 13:29:02 +01:00
} else {
2021-02-26 12:28:19 -08:00
r_missing_templates = ! exists_export_template ( " android_source.zip " , & err ) ;
2018-08-20 17:35:33 +02:00
2021-02-26 12:28:19 -08:00
bool installed_android_build_template = FileAccess : : exists ( " res://android/build/build.gradle " ) ;
if ( ! installed_android_build_template ) {
2020-12-23 22:16:56 -08:00
err + = TTR ( " Android build template not installed in the project. Install it from the Project menu. " ) + " \n " ;
}
2021-02-26 12:28:19 -08:00
valid = installed_android_build_template & & ! r_missing_templates ;
2017-03-23 20:14:12 -03:00
}
2020-12-23 22:16:56 -08:00
// Validate the rest of the configuration.
2017-03-23 20:14:12 -03:00
2018-12-19 15:09:09 -02:00
String dk = p_preset - > get ( " keystore/debug " ) ;
2017-03-23 20:14:12 -03:00
if ( ! FileAccess : : exists ( dk ) ) {
2018-12-19 15:09:09 -02:00
dk = EditorSettings : : get_singleton ( ) - > get ( " export/android/debug_keystore " ) ;
if ( ! FileAccess : : exists ( dk ) ) {
valid = false ;
2019-01-21 18:34:53 +01:00
err + = TTR ( " Debug keystore not configured in the Editor Settings nor in the preset. " ) + " \n " ;
2018-12-19 15:09:09 -02:00
}
2017-03-23 20:14:12 -03:00
}
2020-06-04 13:01:15 +02:00
String rk = p_preset - > get ( " keystore/release " ) ;
if ( ! rk . empty ( ) & & ! FileAccess : : exists ( rk ) ) {
valid = false ;
err + = TTR ( " Release keystore incorrectly configured in the export preset. " ) + " \n " ;
}
2020-12-23 22:16:56 -08:00
String sdk_path = EditorSettings : : get_singleton ( ) - > get ( " export/android/android_sdk_path " ) ;
if ( sdk_path = = " " ) {
err + = TTR ( " A valid Android SDK path is required in Editor Settings. " ) + " \n " ;
valid = false ;
} else {
Error errn ;
// Check for the platform-tools directory.
DirAccessRef da = DirAccess : : open ( sdk_path . plus_file ( " platform-tools " ) , & errn ) ;
if ( errn ! = OK ) {
err + = TTR ( " Invalid Android SDK path in Editor Settings. " ) ;
err + = TTR ( " Missing 'platform-tools' directory! " ) ;
err + = " \n " ;
2019-04-07 15:46:52 -03:00
valid = false ;
}
2020-12-23 22:16:56 -08:00
// Validate that adb is available
String adb_path = get_adb_path ( ) ;
if ( ! FileAccess : : exists ( adb_path ) ) {
err + = TTR ( " Unable to find Android SDK platform-tools' adb command. " ) ;
err + = TTR ( " Please check in the Android SDK directory specified in Editor Settings. " ) ;
err + = " \n " ;
valid = false ;
}
2019-04-07 15:46:52 -03:00
2020-12-23 22:16:56 -08:00
// Check for the build-tools directory.
DirAccessRef build_tools_da = DirAccess : : open ( sdk_path . plus_file ( " build-tools " ) , & errn ) ;
if ( errn ! = OK ) {
err + = TTR ( " Invalid Android SDK path in Editor Settings. " ) ;
err + = TTR ( " Missing 'build-tools' directory! " ) ;
err + = " \n " ;
valid = false ;
}
// Validate that apksigner is available
String apksigner_path = get_apksigner_path ( ) ;
if ( ! FileAccess : : exists ( apksigner_path ) ) {
err + = TTR ( " Unable to find Android SDK build-tools' apksigner command. " ) ;
err + = TTR ( " Please check in the Android SDK directory specified in Editor Settings. " ) ;
err + = " \n " ;
2019-04-07 15:46:52 -03:00
valid = false ;
}
}
2017-03-23 20:14:12 -03:00
bool apk_expansion = p_preset - > get ( " apk_expansion/enable " ) ;
if ( apk_expansion ) {
String apk_expansion_pkey = p_preset - > get ( " apk_expansion/public_key " ) ;
if ( apk_expansion_pkey = = " " ) {
valid = false ;
2019-01-21 18:34:53 +01:00
err + = TTR ( " Invalid public key for APK expansion. " ) + " \n " ;
2017-03-23 20:14:12 -03:00
}
}
2018-10-27 12:54:39 +01:00
String pn = p_preset - > get ( " package/unique_name " ) ;
String pn_err ;
if ( ! is_package_name_valid ( get_package_name ( pn ) , & pn_err ) ) {
valid = false ;
2019-01-21 18:34:53 +01:00
err + = TTR ( " Invalid package name: " ) + " " + pn_err + " \n " ;
2018-10-27 12:54:39 +01:00
}
2019-02-26 18:43:37 -03:00
String etc_error = test_etc2 ( ) ;
if ( etc_error ! = String ( ) ) {
valid = false ;
err + = etc_error ;
}
2020-06-22 09:17:26 +02:00
// The GodotPaymentV3 module was converted to the external GodotGooglePlayBilling plugin in Godot 3.2.2,
2020-05-15 11:43:42 +02:00
// this check helps users to notice the change to ensure that they change their settings.
String modules = ProjectSettings : : get_singleton ( ) - > get ( " android/modules " ) ;
if ( modules . find ( " org/godotengine/godot/GodotPaymentV3 " ) ! = - 1 ) {
2020-06-22 09:17:26 +02:00
bool godot_google_play_billing_enabled = p_preset - > get ( " plugins/GodotGooglePlayBilling " ) ;
if ( ! godot_google_play_billing_enabled ) {
2020-05-15 11:43:42 +02:00
valid = false ;
err + = TTR ( " Invalid \" GodotPaymentV3 \" module included in the \" android/modules \" project setting (changed in Godot 3.2.2). \n "
2020-06-22 09:17:26 +02:00
" Replace it with the first-party \" GodotGooglePlayBilling \" plugin. \n "
" Note that the singleton was also renamed from \" GodotPayments \" to \" GodotGooglePlayBilling \" . " ) ;
2020-05-15 11:43:42 +02:00
err + = " \n " ;
}
}
2020-05-27 12:04:31 -07:00
// Ensure that `Use Custom Build` is enabled if a plugin is selected.
String enabled_plugins_names = get_plugins_names ( get_enabled_plugins ( p_preset ) ) ;
bool custom_build_enabled = p_preset - > get ( " custom_template/use_custom_build " ) ;
if ( ! enabled_plugins_names . empty ( ) & & ! custom_build_enabled ) {
valid = false ;
err + = TTR ( " \" Use Custom Build \" must be enabled to use the plugins. " ) ;
err + = " \n " ;
}
2020-06-04 14:54:36 -07:00
// Validate the Xr features are properly populated
int xr_mode_index = p_preset - > get ( " xr_features/xr_mode " ) ;
int degrees_of_freedom = p_preset - > get ( " xr_features/degrees_of_freedom " ) ;
int hand_tracking = p_preset - > get ( " xr_features/hand_tracking " ) ;
bool focus_awareness = p_preset - > get ( " xr_features/focus_awareness " ) ;
if ( xr_mode_index ! = /* XRMode.OVR*/ 1 ) {
if ( degrees_of_freedom > 0 ) {
valid = false ;
err + = TTR ( " \" Degrees Of Freedom \" is only valid when \" Xr Mode \" is \" Oculus Mobile VR \" . " ) ;
err + = " \n " ;
}
if ( hand_tracking > 0 ) {
valid = false ;
err + = TTR ( " \" Hand Tracking \" is only valid when \" Xr Mode \" is \" Oculus Mobile VR \" . " ) ;
err + = " \n " ;
}
if ( focus_awareness ) {
valid = false ;
err + = TTR ( " \" Focus Awareness \" is only valid when \" Xr Mode \" is \" Oculus Mobile VR \" . " ) ;
err + = " \n " ;
}
}
2020-12-23 01:30:29 -08:00
if ( int ( p_preset - > get ( " custom_template/export_format " ) ) = = EXPORT_FORMAT_AAB & &
2020-07-23 12:17:02 -04:00
! bool ( p_preset - > get ( " custom_template/use_custom_build " ) ) ) {
valid = false ;
err + = TTR ( " \" Export AAB \" is only valid when \" Use Custom Build \" is enabled. " ) ;
err + = " \n " ;
}
2017-03-23 20:14:12 -03:00
r_error = err ;
return valid ;
}
2018-10-29 18:18:49 -03:00
virtual List < String > get_binary_extensions ( const Ref < EditorExportPreset > & p_preset ) const {
List < String > list ;
list . push_back ( " apk " ) ;
2020-07-23 12:17:02 -04:00
list . push_back ( " aab " ) ;
2018-10-29 18:18:49 -03:00
return list ;
2017-03-23 20:14:12 -03:00
}
2019-04-07 15:46:52 -03:00
void _update_custom_build_project ( ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Updating custom build project.. " ) ;
2019-04-07 15:46:52 -03:00
DirAccessRef da = DirAccess : : open ( " res://android " ) ;
2019-09-25 10:28:50 +02:00
ERR_FAIL_COND_MSG ( ! da , " Cannot open directory 'res://android'. " ) ;
2021-05-04 14:20:36 +02:00
Map < String , List < String > > directory_paths ;
Map < String , List < String > > manifest_sections ;
Map < String , List < String > > gradle_sections ;
2019-04-07 15:46:52 -03:00
da - > list_dir_begin ( ) ;
String d = da - > get_next ( ) ;
while ( d ! = String ( ) ) {
if ( ! d . begins_with ( " . " ) & & d ! = " build " & & da - > current_is_dir ( ) ) { //a dir and not the build dir
//add directories found
DirAccessRef ds = DirAccess : : open ( String ( " res://android " ) . plus_file ( d ) ) ;
if ( ds ) {
ds - > list_dir_begin ( ) ;
String sd = ds - > get_next ( ) ;
while ( sd ! = String ( ) ) {
if ( ! sd . begins_with ( " . " ) & & ds - > current_is_dir ( ) ) {
String key = sd . to_upper ( ) ;
if ( ! directory_paths . has ( key ) ) {
directory_paths [ key ] = List < String > ( ) ;
}
String path = ProjectSettings : : get_singleton ( ) - > get_resource_path ( ) . plus_file ( " android " ) . plus_file ( d ) . plus_file ( sd ) ;
directory_paths [ key ] . push_back ( path ) ;
print_line ( " Add: " + sd + " : " + path ) ;
}
sd = ds - > get_next ( ) ;
}
ds - > list_dir_end ( ) ;
}
//parse manifest
{
FileAccessRef f = FileAccess : : open ( String ( " res://android " ) . plus_file ( d ) . plus_file ( " AndroidManifest.conf " ) , FileAccess : : READ ) ;
if ( f ) {
String section ;
while ( ! f - > eof_reached ( ) ) {
String l = f - > get_line ( ) ;
String k = l . strip_edges ( ) ;
if ( k . begins_with ( " [ " ) ) {
section = k . substr ( 1 , k . length ( ) - 2 ) . strip_edges ( ) . to_upper ( ) ;
print_line ( " Section: " + section ) ;
} else if ( k ! = String ( ) ) {
if ( ! manifest_sections . has ( section ) ) {
manifest_sections [ section ] = List < String > ( ) ;
}
manifest_sections [ section ] . push_back ( l ) ;
}
}
f - > close ( ) ;
}
}
//parse gradle
{
FileAccessRef f = FileAccess : : open ( String ( " res://android " ) . plus_file ( d ) . plus_file ( " gradle.conf " ) , FileAccess : : READ ) ;
if ( f ) {
String section ;
while ( ! f - > eof_reached ( ) ) {
String l = f - > get_line ( ) . strip_edges ( ) ;
String k = l . strip_edges ( ) ;
if ( k . begins_with ( " [ " ) ) {
section = k . substr ( 1 , k . length ( ) - 2 ) . strip_edges ( ) . to_upper ( ) ;
print_line ( " Section: " + section ) ;
} else if ( k ! = String ( ) ) {
if ( ! gradle_sections . has ( section ) ) {
gradle_sections [ section ] = List < String > ( ) ;
}
gradle_sections [ section ] . push_back ( l ) ;
}
}
}
}
}
d = da - > get_next ( ) ;
}
da - > list_dir_end ( ) ;
{ //fix gradle build
String new_file ;
{
FileAccessRef f = FileAccess : : open ( " res://android/build/build.gradle " , FileAccess : : READ ) ;
if ( f ) {
while ( ! f - > eof_reached ( ) ) {
String l = f - > get_line ( ) ;
2020-03-03 10:49:13 +01:00
bool append_line = false ;
2019-04-07 15:46:52 -03:00
if ( l . begins_with ( " //CHUNK_ " ) ) {
String text = l . replace_first ( " //CHUNK_ " , " " ) ;
int begin_pos = text . find ( " _BEGIN " ) ;
if ( begin_pos ! = - 1 ) {
text = text . substr ( 0 , begin_pos ) ;
text = text . to_upper ( ) ; //just in case
String end_marker = " //CHUNK_ " + text + " _END " ;
2019-03-26 18:51:13 +01:00
uint64_t pos = f - > get_position ( ) ;
2019-04-07 15:46:52 -03:00
bool found = false ;
while ( ! f - > eof_reached ( ) ) {
l = f - > get_line ( ) ;
if ( l . begins_with ( end_marker ) ) {
found = true ;
break ;
}
}
new_file + = " //CHUNK_ " + text + " _BEGIN \n " ;
if ( ! found ) {
ERR_PRINTS ( " No end marker found in build.gradle for chunk: " + text ) ;
f - > seek ( pos ) ;
} else {
//add chunk lines
if ( gradle_sections . has ( text ) ) {
for ( List < String > : : Element * E = gradle_sections [ text ] . front ( ) ; E ; E = E - > next ( ) ) {
new_file + = E - > get ( ) + " \n " ;
}
}
2020-03-03 10:49:13 +01:00
if ( f - > eof_reached ( ) ) {
new_file + = end_marker ;
} else {
new_file + = end_marker + " \n " ;
}
2019-04-07 15:46:52 -03:00
}
} else {
2020-03-03 10:49:13 +01:00
append_line = true ;
2019-04-07 15:46:52 -03:00
}
} else if ( l . begins_with ( " //DIR_ " ) ) {
String text = l . replace_first ( " //DIR_ " , " " ) ;
int begin_pos = text . find ( " _BEGIN " ) ;
if ( begin_pos ! = - 1 ) {
text = text . substr ( 0 , begin_pos ) ;
text = text . to_upper ( ) ; //just in case
String end_marker = " //DIR_ " + text + " _END " ;
2019-03-26 18:51:13 +01:00
uint64_t pos = f - > get_position ( ) ;
2019-04-07 15:46:52 -03:00
bool found = false ;
while ( ! f - > eof_reached ( ) ) {
l = f - > get_line ( ) ;
if ( l . begins_with ( end_marker ) ) {
found = true ;
break ;
}
}
new_file + = " //DIR_ " + text + " _BEGIN \n " ;
if ( ! found ) {
ERR_PRINTS ( " No end marker found in build.gradle for dir: " + text ) ;
f - > seek ( pos ) ;
} else {
//add chunk lines
if ( directory_paths . has ( text ) ) {
for ( List < String > : : Element * E = directory_paths [ text ] . front ( ) ; E ; E = E - > next ( ) ) {
new_file + = " ,' " + E - > get ( ) . replace ( " ' " , " \' " ) + " ' " ;
new_file + = " \n " ;
}
}
2020-03-03 10:49:13 +01:00
if ( f - > eof_reached ( ) ) {
new_file + = end_marker ;
} else {
new_file + = end_marker + " \n " ;
}
2019-04-07 15:46:52 -03:00
}
} else {
2020-03-03 10:49:13 +01:00
append_line = true ;
2019-04-07 15:46:52 -03:00
}
} else {
2020-03-03 10:49:13 +01:00
append_line = true ;
}
if ( append_line ) {
if ( f - > eof_reached ( ) ) {
new_file + = l ;
} else {
new_file + = l + " \n " ;
}
2019-04-07 15:46:52 -03:00
}
}
}
}
FileAccessRef f = FileAccess : : open ( " res://android/build/build.gradle " , FileAccess : : WRITE ) ;
f - > store_string ( new_file ) ;
f - > close ( ) ;
}
{ //fix manifest
String new_file ;
{
FileAccessRef f = FileAccess : : open ( " res://android/build/AndroidManifest.xml " , FileAccess : : READ ) ;
if ( f ) {
while ( ! f - > eof_reached ( ) ) {
String l = f - > get_line ( ) ;
2020-03-03 10:49:13 +01:00
bool append_line = false ;
2019-04-07 15:46:52 -03:00
if ( l . begins_with ( " <!--CHUNK_ " ) ) {
String text = l . replace_first ( " <!--CHUNK_ " , " " ) ;
int begin_pos = text . find ( " _BEGIN--> " ) ;
if ( begin_pos ! = - 1 ) {
text = text . substr ( 0 , begin_pos ) ;
text = text . to_upper ( ) ; //just in case
String end_marker = " <!--CHUNK_ " + text + " _END--> " ;
2019-03-26 18:51:13 +01:00
uint64_t pos = f - > get_position ( ) ;
2019-04-07 15:46:52 -03:00
bool found = false ;
while ( ! f - > eof_reached ( ) ) {
l = f - > get_line ( ) ;
if ( l . begins_with ( end_marker ) ) {
found = true ;
break ;
}
}
new_file + = " <!--CHUNK_ " + text + " _BEGIN--> \n " ;
if ( ! found ) {
2019-10-03 11:06:54 +02:00
ERR_PRINTS ( " No end marker found in AndroidManifest.xml for chunk: " + text ) ;
2019-04-07 15:46:52 -03:00
f - > seek ( pos ) ;
} else {
//add chunk lines
if ( manifest_sections . has ( text ) ) {
for ( List < String > : : Element * E = manifest_sections [ text ] . front ( ) ; E ; E = E - > next ( ) ) {
new_file + = E - > get ( ) + " \n " ;
}
}
2020-03-03 10:49:13 +01:00
if ( f - > eof_reached ( ) ) {
new_file + = end_marker ;
} else {
new_file + = end_marker + " \n " ;
}
2019-04-07 15:46:52 -03:00
}
} else {
2020-03-03 10:49:13 +01:00
append_line = true ;
2019-04-07 15:46:52 -03:00
}
} else if ( l . strip_edges ( ) . begins_with ( " <application " ) ) {
2020-01-06 18:37:39 +00:00
String last_tag = " android:icon= \" @mipmap/icon \" " ;
2019-04-07 15:46:52 -03:00
int last_tag_pos = l . find ( last_tag ) ;
if ( last_tag_pos = = - 1 ) {
2019-10-03 11:06:54 +02:00
ERR_PRINTS ( " Not adding application attributes as the expected tag was not found in '<application': " + last_tag ) ;
2020-03-03 10:49:13 +01:00
append_line = true ;
2019-04-07 15:46:52 -03:00
} else {
String base = l . substr ( 0 , last_tag_pos + last_tag . length ( ) ) ;
2019-04-27 12:22:47 -03:00
if ( manifest_sections . has ( " application_attribs " ) ) {
for ( List < String > : : Element * E = manifest_sections [ " application_attribs " ] . front ( ) ; E ; E = E - > next ( ) ) {
2019-04-07 15:46:52 -03:00
String to_add = E - > get ( ) . strip_edges ( ) ;
base + = " " + to_add + " " ;
}
}
base + = " > \n " ;
new_file + = base ;
}
} else {
2020-03-03 10:49:13 +01:00
append_line = true ;
}
if ( append_line ) {
new_file + = l ;
if ( ! f - > eof_reached ( ) ) {
new_file + = " \n " ;
}
2019-04-07 15:46:52 -03:00
}
}
}
}
FileAccessRef f = FileAccess : : open ( " res://android/build/AndroidManifest.xml " , FileAccess : : WRITE ) ;
f - > store_string ( new_file ) ;
f - > close ( ) ;
}
}
2021-01-14 15:05:42 +03:00
inline bool is_clean_build_required ( Vector < PluginConfigAndroid > enabled_plugins ) {
2020-05-24 02:33:07 +02:00
String plugin_names = get_plugins_names ( enabled_plugins ) ;
bool first_build = last_custom_build_time = = 0 ;
bool have_plugins_changed = false ;
if ( ! first_build ) {
have_plugins_changed = plugin_names ! = last_plugin_names ;
if ( ! have_plugins_changed ) {
for ( int i = 0 ; i < enabled_plugins . size ( ) ; i + + ) {
if ( enabled_plugins . get ( i ) . last_updated > last_custom_build_time ) {
have_plugins_changed = true ;
break ;
}
}
}
}
last_custom_build_time = OS : : get_singleton ( ) - > get_unix_time ( ) ;
last_plugin_names = plugin_names ;
return have_plugins_changed | | first_build ;
}
2020-07-23 12:17:02 -04:00
String get_apk_expansion_fullpath ( const Ref < EditorExportPreset > & p_preset , const String & p_path ) {
int version_code = p_preset - > get ( " version/code " ) ;
String package_name = p_preset - > get ( " package/unique_name " ) ;
String apk_file_name = " main. " + itos ( version_code ) + " . " + get_package_name ( package_name ) + " .obb " ;
String fullpath = p_path . get_base_dir ( ) . plus_file ( apk_file_name ) ;
return fullpath ;
}
Error save_apk_expansion_file ( const Ref < EditorExportPreset > & p_preset , const String & p_path ) {
String fullpath = get_apk_expansion_fullpath ( p_preset , p_path ) ;
Error err = save_pack ( p_preset , fullpath ) ;
return err ;
}
void get_command_line_flags ( const Ref < EditorExportPreset > & p_preset , const String & p_path , int p_flags , Vector < uint8_t > & r_command_line_flags ) {
2020-06-26 14:58:16 -04:00
String cmdline = p_preset - > get ( " command_line/extra_args " ) ;
Vector < String > command_line_strings = cmdline . strip_edges ( ) . split ( " " ) ;
for ( int i = 0 ; i < command_line_strings . size ( ) ; i + + ) {
if ( command_line_strings [ i ] . strip_edges ( ) . length ( ) = = 0 ) {
command_line_strings . remove ( i ) ;
i - - ;
}
}
gen_export_flags ( command_line_strings , p_flags ) ;
bool apk_expansion = p_preset - > get ( " apk_expansion/enable " ) ;
if ( apk_expansion ) {
2020-07-23 12:17:02 -04:00
String fullpath = get_apk_expansion_fullpath ( p_preset , p_path ) ;
2020-06-26 14:58:16 -04:00
String apk_expansion_public_key = p_preset - > get ( " apk_expansion/public_key " ) ;
command_line_strings . push_back ( " --use_apk_expansion " ) ;
command_line_strings . push_back ( " --apk_expansion_md5 " ) ;
command_line_strings . push_back ( FileAccess : : get_md5 ( fullpath ) ) ;
command_line_strings . push_back ( " --apk_expansion_key " ) ;
command_line_strings . push_back ( apk_expansion_public_key . strip_edges ( ) ) ;
}
int xr_mode_index = p_preset - > get ( " xr_features/xr_mode " ) ;
if ( xr_mode_index = = 1 ) {
command_line_strings . push_back ( " --xr_mode_ovr " ) ;
} else { // XRMode.REGULAR is the default.
command_line_strings . push_back ( " --xr_mode_regular " ) ;
}
bool use_32_bit_framebuffer = p_preset - > get ( " graphics/32_bits_framebuffer " ) ;
if ( use_32_bit_framebuffer ) {
command_line_strings . push_back ( " --use_depth_32 " ) ;
}
bool immersive = p_preset - > get ( " screen/immersive_mode " ) ;
if ( immersive ) {
command_line_strings . push_back ( " --use_immersive " ) ;
}
2020-11-20 10:45:39 +01:00
bool debug_opengl = p_preset - > get ( " graphics/opengl_debug " ) ;
2020-06-26 14:58:16 -04:00
if ( debug_opengl ) {
command_line_strings . push_back ( " --debug_opengl " ) ;
}
if ( command_line_strings . size ( ) ) {
r_command_line_flags . resize ( 4 ) ;
encode_uint32 ( command_line_strings . size ( ) , & r_command_line_flags . write [ 0 ] ) ;
for ( int i = 0 ; i < command_line_strings . size ( ) ; i + + ) {
print_line ( itos ( i ) + " param: " + command_line_strings [ i ] ) ;
CharString command_line_argument = command_line_strings [ i ] . utf8 ( ) ;
int base = r_command_line_flags . size ( ) ;
int length = command_line_argument . length ( ) ;
2021-05-05 12:44:11 +02:00
if ( length = = 0 ) {
2020-06-26 14:58:16 -04:00
continue ;
2021-05-05 12:44:11 +02:00
}
2020-06-26 14:58:16 -04:00
r_command_line_flags . resize ( base + 4 + length ) ;
encode_uint32 ( length , & r_command_line_flags . write [ base ] ) ;
2021-04-29 12:34:11 +02:00
memcpy ( & r_command_line_flags . write [ base + 4 ] , command_line_argument . ptr ( ) , length ) ;
2020-06-26 14:58:16 -04:00
}
}
}
2020-12-23 22:16:56 -08:00
Error sign_apk ( const Ref < EditorExportPreset > & p_preset , bool p_debug , const String & export_path , EditorProgress & ep ) {
2020-11-13 14:14:49 -08:00
int export_format = int ( p_preset - > get ( " custom_template/export_format " ) ) ;
2020-12-23 01:30:29 -08:00
String export_label = export_format = = EXPORT_FORMAT_AAB ? " AAB " : " APK " ;
2020-07-27 11:14:16 -04:00
String release_keystore = p_preset - > get ( " keystore/release " ) ;
String release_username = p_preset - > get ( " keystore/release_user " ) ;
String release_password = p_preset - > get ( " keystore/release_password " ) ;
2020-12-23 22:16:56 -08:00
String apksigner = get_apksigner_path ( ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Starting signing of the " + export_label + " binary using " + apksigner ) ;
2020-12-23 22:16:56 -08:00
if ( ! FileAccess : : exists ( apksigner ) ) {
EditorNode : : add_io_error ( " 'apksigner' could not be found. \n Please check the command is available in the Android SDK build-tools directory. \n The resulting " + export_label + " is unsigned. " ) ;
2020-07-27 11:14:16 -04:00
return OK ;
}
String keystore ;
String password ;
String user ;
if ( p_debug ) {
keystore = p_preset - > get ( " keystore/debug " ) ;
password = p_preset - > get ( " keystore/debug_password " ) ;
user = p_preset - > get ( " keystore/debug_user " ) ;
if ( keystore . empty ( ) ) {
keystore = EditorSettings : : get_singleton ( ) - > get ( " export/android/debug_keystore " ) ;
password = EditorSettings : : get_singleton ( ) - > get ( " export/android/debug_keystore_pass " ) ;
user = EditorSettings : : get_singleton ( ) - > get ( " export/android/debug_keystore_user " ) ;
}
2020-12-23 22:16:56 -08:00
if ( ep . step ( " Signing debug " + export_label + " ... " , 104 ) ) {
2020-07-27 11:14:16 -04:00
return ERR_SKIP ;
}
} else {
keystore = release_keystore ;
password = release_password ;
user = release_username ;
2020-12-23 22:16:56 -08:00
if ( ep . step ( " Signing release " + export_label + " ... " , 104 ) ) {
2020-07-27 11:14:16 -04:00
return ERR_SKIP ;
}
}
if ( ! FileAccess : : exists ( keystore ) ) {
EditorNode : : add_io_error ( " Could not find keystore, unable to export. " ) ;
return ERR_FILE_CANT_OPEN ;
}
2021-04-15 13:26:05 -07:00
String output ;
2020-07-27 11:14:16 -04:00
List < String > args ;
2020-12-23 22:16:56 -08:00
args . push_back ( " sign " ) ;
args . push_back ( " --verbose " ) ;
args . push_back ( " --ks " ) ;
2020-07-27 11:14:16 -04:00
args . push_back ( keystore ) ;
2020-12-23 22:16:56 -08:00
args . push_back ( " --ks-pass " ) ;
args . push_back ( " pass: " + password ) ;
args . push_back ( " --ks-key-alias " ) ;
2020-07-27 11:14:16 -04:00
args . push_back ( user ) ;
2020-12-23 22:16:56 -08:00
args . push_back ( export_path ) ;
2021-02-16 16:33:15 -08:00
if ( p_debug ) {
// We only print verbose logs for debug builds to avoid leaking release keystore credentials.
print_verbose ( " Signing debug binary using: " + String ( " \n " ) + apksigner + " " + join_list ( args , String ( " " ) ) ) ;
}
2020-07-27 11:14:16 -04:00
int retval ;
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( apksigner , args , true , nullptr , & output , & retval , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2020-07-27 11:14:16 -04:00
if ( retval ) {
2020-12-23 22:16:56 -08:00
EditorNode : : add_io_error ( " 'apksigner' returned with error # " + itos ( retval ) ) ;
2020-07-27 11:14:16 -04:00
return ERR_CANT_CREATE ;
}
2020-12-23 22:16:56 -08:00
if ( ep . step ( " Verifying " + export_label + " ... " , 105 ) ) {
2020-07-27 11:14:16 -04:00
return ERR_SKIP ;
}
args . clear ( ) ;
2020-12-23 22:16:56 -08:00
args . push_back ( " verify " ) ;
args . push_back ( " --verbose " ) ;
2020-11-13 14:14:49 -08:00
args . push_back ( export_path ) ;
2021-02-16 16:33:15 -08:00
if ( p_debug ) {
print_verbose ( " Verifying signed build using: " + String ( " \n " ) + apksigner + " " + join_list ( args , String ( " " ) ) ) ;
}
2020-07-27 11:14:16 -04:00
2021-04-15 13:26:05 -07:00
output . clear ( ) ;
2021-05-04 16:00:45 +02:00
OS : : get_singleton ( ) - > execute ( apksigner , args , true , nullptr , & output , & retval , true ) ;
2021-04-15 13:26:05 -07:00
print_verbose ( output ) ;
2020-07-27 11:14:16 -04:00
if ( retval ) {
2020-12-23 22:16:56 -08:00
EditorNode : : add_io_error ( " 'apksigner' verification of " + export_label + " failed. " ) ;
2020-07-27 11:14:16 -04:00
return ERR_CANT_CREATE ;
}
2021-02-16 16:33:15 -08:00
print_verbose ( " Successfully completed signing build. " ) ;
2020-07-27 11:14:16 -04:00
return OK ;
}
2020-11-13 14:14:49 -08:00
void _clear_assets_directory ( ) {
DirAccessRef da_res = DirAccess : : create ( DirAccess : : ACCESS_RESOURCES ) ;
if ( da_res - > dir_exists ( " res://android/build/assets " ) ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Clearing assets directory.. " ) ;
2020-11-13 14:14:49 -08:00
DirAccessRef da_assets = DirAccess : : open ( " res://android/build/assets " ) ;
da_assets - > erase_contents_recursive ( ) ;
da_res - > remove ( " res://android/build/assets " ) ;
}
}
2021-02-16 16:33:15 -08:00
String join_list ( List < String > parts , const String & separator ) const {
String ret ;
for ( int i = 0 ; i < parts . size ( ) ; + + i ) {
if ( i > 0 ) {
ret + = separator ;
}
ret + = parts [ i ] ;
}
return ret ;
}
2017-03-23 20:14:12 -03:00
virtual Error export_project ( const Ref < EditorExportPreset > & p_preset , bool p_debug , const String & p_path , int p_flags = 0 ) {
2020-12-23 01:30:29 -08:00
int export_format = int ( p_preset - > get ( " custom_template/export_format " ) ) ;
bool should_sign = p_preset - > get ( " package/signed " ) ;
return export_project_helper ( p_preset , p_debug , p_path , export_format , should_sign , p_flags ) ;
}
Error export_project_helper ( const Ref < EditorExportPreset > & p_preset , bool p_debug , const String & p_path , int export_format , bool should_sign , int p_flags ) {
2018-04-26 23:07:21 +02:00
ExportNotifier notifier ( * this , p_preset , p_debug , p_path , p_flags ) ;
2017-03-23 20:14:12 -03:00
String src_apk ;
2020-07-23 12:17:02 -04:00
Error err ;
2017-03-23 20:14:12 -03:00
2018-12-19 16:50:40 -02:00
EditorProgress ep ( " export " , " Exporting for Android " , 105 , true ) ;
2017-03-23 20:14:12 -03:00
2020-07-01 11:06:50 -04:00
bool use_custom_build = bool ( p_preset - > get ( " custom_template/use_custom_build " ) ) ;
2020-07-23 12:17:02 -04:00
bool p_give_internet = p_flags & ( DEBUG_FLAG_DUMB_CLIENT | DEBUG_FLAG_REMOTE_DEBUG ) ;
2020-07-23 12:17:02 -04:00
bool apk_expansion = p_preset - > get ( " apk_expansion/enable " ) ;
Vector < String > enabled_abis = get_enabled_abis ( p_preset ) ;
2017-03-23 20:14:12 -03:00
2021-02-16 16:33:15 -08:00
print_verbose ( " Exporting for Android... " ) ;
print_verbose ( " - debug build: " + bool_to_string ( p_debug ) ) ;
print_verbose ( " - export path: " + p_path ) ;
print_verbose ( " - export format: " + itos ( export_format ) ) ;
print_verbose ( " - sign build: " + bool_to_string ( should_sign ) ) ;
print_verbose ( " - custom build enabled: " + bool_to_string ( use_custom_build ) ) ;
print_verbose ( " - apk expansion enabled: " + bool_to_string ( apk_expansion ) ) ;
print_verbose ( " - enabled abis: " + String ( " , " ) . join ( enabled_abis ) ) ;
2021-02-21 21:34:27 -08:00
print_verbose ( " - export filter: " + itos ( p_preset - > get_export_filter ( ) ) ) ;
print_verbose ( " - include filter: " + p_preset - > get_include_filter ( ) ) ;
print_verbose ( " - exclude filter: " + p_preset - > get_exclude_filter ( ) ) ;
2021-02-16 16:33:15 -08:00
2020-08-19 15:38:50 -07:00
Ref < Image > splash_image ;
Ref < Image > splash_bg_color_image ;
2021-02-18 01:54:35 -08:00
String processed_splash_config_xml = load_splash_refs ( splash_image , splash_bg_color_image ) ;
2020-08-19 15:38:50 -07:00
2020-07-22 15:52:26 -04:00
Ref < Image > main_image ;
Ref < Image > foreground ;
Ref < Image > background ;
load_icon_refs ( p_preset , main_image , foreground , background ) ;
2020-07-23 12:17:02 -04:00
Vector < uint8_t > command_line_flags ;
// Write command line flags into the command_line_flags variable.
get_command_line_flags ( p_preset , p_path , p_flags , command_line_flags ) ;
2020-12-23 01:30:29 -08:00
if ( export_format = = EXPORT_FORMAT_AAB ) {
2020-07-23 12:17:02 -04:00
if ( ! p_path . ends_with ( " .aab " ) ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Invalid filename! Android App Bundle requires the *.aab extension. " ) ) ;
return ERR_UNCONFIGURED ;
}
if ( apk_expansion ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " APK Expansion not compatible with Android App Bundle. " ) ) ;
return ERR_UNCONFIGURED ;
}
}
2020-12-23 01:30:29 -08:00
if ( export_format = = EXPORT_FORMAT_APK & & ! p_path . ends_with ( " .apk " ) ) {
2020-07-23 12:17:02 -04:00
EditorNode : : get_singleton ( ) - > show_warning (
TTR ( " Invalid filename! Android APK requires the *.apk extension. " ) ) ;
return ERR_UNCONFIGURED ;
}
2020-12-23 01:30:29 -08:00
if ( export_format > EXPORT_FORMAT_AAB | | export_format < EXPORT_FORMAT_APK ) {
2020-07-23 12:17:02 -04:00
EditorNode : : add_io_error ( " Unsupported export format! \n " ) ;
return ERR_UNCONFIGURED ; //TODO: is this the right error?
}
2020-07-01 11:06:50 -04:00
if ( use_custom_build ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Starting custom build.. " ) ;
2020-07-23 12:17:02 -04:00
//test that installed build version is alright
2020-10-28 09:07:20 +01:00
{
2021-02-16 16:33:15 -08:00
print_verbose ( " Checking build version.. " ) ;
2020-10-28 09:07:20 +01:00
FileAccessRef f = FileAccess : : open ( " res://android/.build_version " , FileAccess : : READ ) ;
if ( ! f ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Trying to build from a custom built template, but no version info for it exists. Please reinstall from the 'Project' menu. " ) ) ;
return ERR_UNCONFIGURED ;
}
String version = f - > get_line ( ) . strip_edges ( ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " - build version: " + version ) ;
2020-10-28 09:07:20 +01:00
f - > close ( ) ;
if ( version ! = VERSION_FULL_CONFIG ) {
EditorNode : : get_singleton ( ) - > show_warning ( vformat ( TTR ( " Android build version mismatch: \n Template installed: %s \n Godot Version: %s \n Please reinstall Android build template from 'Project' menu. " ) , version , VERSION_FULL_CONFIG ) ) ;
return ERR_UNCONFIGURED ;
}
2020-07-23 12:17:02 -04:00
}
2020-12-23 22:16:56 -08:00
String sdk_path = EDITOR_GET ( " export/android/android_sdk_path " ) ;
2021-02-16 16:33:15 -08:00
ERR_FAIL_COND_V_MSG ( sdk_path . empty ( ) , ERR_UNCONFIGURED , " Android SDK path must be configured in Editor Settings at 'export/android/android_sdk_path'. " ) ;
print_verbose ( " Android sdk path: " + sdk_path ) ;
2020-07-01 11:06:50 -04:00
// TODO: should we use "package/name" or "application/config/name"?
String project_name = get_project_name ( p_preset - > get ( " package/name " ) ) ;
2020-07-23 12:17:02 -04:00
err = _create_project_name_strings_files ( p_preset , project_name ) ; //project name localization.
2020-07-01 11:06:50 -04:00
if ( err ! = OK ) {
EditorNode : : add_io_error ( " Unable to overwrite res://android/build/res/*.xml files with project name " ) ;
}
2020-07-23 12:17:02 -04:00
// Copies the project icon files into the appropriate Gradle project directory.
2021-02-18 01:54:35 -08:00
_copy_icons_to_gradle_project ( p_preset , processed_splash_config_xml , splash_image , splash_bg_color_image , main_image , foreground , background ) ;
2020-07-23 12:17:02 -04:00
// Write an AndroidManifest.xml file into the Gradle project directory.
_write_tmp_manifest ( p_preset , p_give_internet , p_debug ) ;
2019-04-07 15:46:52 -03:00
_update_custom_build_project ( ) ;
2020-07-23 12:17:02 -04:00
//stores all the project files inside the Gradle project directory. Also includes all ABIs
2020-11-13 14:14:49 -08:00
_clear_assets_directory ( ) ;
2020-07-23 12:17:02 -04:00
if ( ! apk_expansion ) {
2021-02-16 16:33:15 -08:00
print_verbose ( " Exporting project files.. " ) ;
2021-05-04 16:00:45 +02:00
err = export_project_files ( p_preset , rename_and_store_file_in_gradle_project , nullptr , ignore_so_file ) ;
2020-07-23 12:17:02 -04:00
if ( err ! = OK ) {
EditorNode : : add_io_error ( " Could not export project files to gradle project \n " ) ;
return err ;
}
} else {
2021-02-16 16:33:15 -08:00
print_verbose ( " Saving apk expansion file.. " ) ;
2020-07-23 12:17:02 -04:00
err = save_apk_expansion_file ( p_preset , p_path ) ;
if ( err ! = OK ) {
EditorNode : : add_io_error ( " Could not write expansion package file! " ) ;
return err ;
}
}
2021-02-16 16:33:15 -08:00
print_verbose ( " Storing command line flags.. " ) ;
2020-07-23 12:17:02 -04:00
store_file_at_path ( " res://android/build/assets/_cl_ " , command_line_flags ) ;
2019-04-07 15:46:52 -03:00
2021-02-16 16:33:15 -08:00
print_verbose ( " Updating ANDROID_HOME environment to " + sdk_path ) ;
2019-04-07 15:46:52 -03:00
OS : : get_singleton ( ) - > set_environment ( " ANDROID_HOME " , sdk_path ) ; //set and overwrite if required
String build_command ;
2020-07-23 12:17:02 -04:00
2019-04-07 15:46:52 -03:00
# ifdef WINDOWS_ENABLED
build_command = " gradlew.bat " ;
# else
build_command = " gradlew " ;
# endif
String build_path = ProjectSettings : : get_singleton ( ) - > get_resource_path ( ) . plus_file ( " android/build " ) ;
build_command = build_path . plus_file ( build_command ) ;
2019-12-12 19:52:57 -05:00
String package_name = get_package_name ( p_preset - > get ( " package/unique_name " ) ) ;
2020-07-23 12:17:02 -04:00
String version_code = itos ( p_preset - > get ( " version/code " ) ) ;
String version_name = p_preset - > get ( " version/name " ) ;
2020-07-23 12:17:02 -04:00
String enabled_abi_string = String ( " | " ) . join ( enabled_abis ) ;
2020-12-23 01:30:29 -08:00
String sign_flag = should_sign ? " true " : " false " ;
2020-11-21 14:09:33 -08:00
String zipalign_flag = " true " ;
2020-04-24 00:45:14 -07:00
2021-01-14 15:05:42 +03:00
Vector < PluginConfigAndroid > enabled_plugins = get_enabled_plugins ( p_preset ) ;
String local_plugins_binaries = get_plugins_binaries ( PluginConfigAndroid : : BINARY_TYPE_LOCAL , enabled_plugins ) ;
String remote_plugins_binaries = get_plugins_binaries ( PluginConfigAndroid : : BINARY_TYPE_REMOTE , enabled_plugins ) ;
2020-04-24 00:45:14 -07:00
String custom_maven_repos = get_plugins_custom_maven_repos ( enabled_plugins ) ;
2020-05-24 02:33:07 +02:00
bool clean_build_required = is_clean_build_required ( enabled_plugins ) ;
2019-12-12 19:52:57 -05:00
2019-04-07 15:46:52 -03:00
List < String > cmdline ;
2020-05-24 02:33:07 +02:00
if ( clean_build_required ) {
cmdline . push_back ( " clean " ) ;
}
2020-07-23 12:17:02 -04:00
String build_type = p_debug ? " Debug " : " Release " ;
2020-12-23 01:30:29 -08:00
if ( export_format = = EXPORT_FORMAT_AAB ) {
2020-07-23 12:17:02 -04:00
String bundle_build_command = vformat ( " bundle%s " , build_type ) ;
cmdline . push_back ( bundle_build_command ) ;
2020-12-23 01:30:29 -08:00
} else if ( export_format = = EXPORT_FORMAT_APK ) {
2020-07-23 12:17:02 -04:00
String apk_build_command = vformat ( " assemble%s " , build_type ) ;
cmdline . push_back ( apk_build_command ) ;
}
2021-02-16 16:33:15 -08:00
cmdline . push_back ( " -p " ) ; // argument to specify the start directory.
cmdline . push_back ( build_path ) ; // start directory.
2019-12-12 19:52:57 -05:00
cmdline . push_back ( " -Pexport_package_name= " + package_name ) ; // argument to specify the package name.
2020-07-23 12:17:02 -04:00
cmdline . push_back ( " -Pexport_version_code= " + version_code ) ; // argument to specify the version code.
cmdline . push_back ( " -Pexport_version_name= " + version_name ) ; // argument to specify the version name.
2020-07-23 12:17:02 -04:00
cmdline . push_back ( " -Pexport_enabled_abis= " + enabled_abi_string ) ; // argument to specify enabled ABIs.
2020-04-24 00:45:14 -07:00
cmdline . push_back ( " -Pplugins_local_binaries= " + local_plugins_binaries ) ; // argument to specify the list of plugins local dependencies.
cmdline . push_back ( " -Pplugins_remote_binaries= " + remote_plugins_binaries ) ; // argument to specify the list of plugins remote dependencies.
cmdline . push_back ( " -Pplugins_maven_repos= " + custom_maven_repos ) ; // argument to specify the list of custom maven repos for the plugins dependencies.
2020-11-21 14:09:33 -08:00
cmdline . push_back ( " -Pperform_zipalign= " + zipalign_flag ) ; // argument to specify whether the build should be zipaligned.
cmdline . push_back ( " -Pperform_signing= " + sign_flag ) ; // argument to specify whether the build should be signed.
2021-02-24 03:49:00 -08:00
cmdline . push_back ( " -Pgodot_editor_version= " + String ( VERSION_FULL_CONFIG ) ) ;
2021-02-16 16:33:15 -08:00
// NOTE: The release keystore is not included in the verbose logging
// to avoid accidentally leaking sensitive information when sharing verbose logs for troubleshooting.
// Any non-sensitive additions to the command line arguments must be done above this section.
// Sensitive additions must be done below the logging statement.
print_verbose ( " Build Android project using gradle command: " + String ( " \n " ) + build_command + " " + join_list ( cmdline , String ( " " ) ) ) ;
2020-12-23 01:30:29 -08:00
if ( should_sign & & ! p_debug ) {
2020-11-21 14:09:33 -08:00
// Pass the release keystore info as well
String release_keystore = p_preset - > get ( " keystore/release " ) ;
String release_username = p_preset - > get ( " keystore/release_user " ) ;
String release_password = p_preset - > get ( " keystore/release_password " ) ;
if ( ! FileAccess : : exists ( release_keystore ) ) {
EditorNode : : add_io_error ( " Could not find keystore, unable to export. " ) ;
return ERR_FILE_CANT_OPEN ;
}
cmdline . push_back ( " -Prelease_keystore_file= " + release_keystore ) ; // argument to specify the release keystore file.
cmdline . push_back ( " -Prelease_keystore_alias= " + release_username ) ; // argument to specify the release keystore alias.
cmdline . push_back ( " -Prelease_keystore_password= " + release_password ) ; // argument to specity the release keystore password.
}
2019-04-07 15:46:52 -03:00
int result = EditorNode : : get_singleton ( ) - > execute_and_show_output ( TTR ( " Building Android Project (gradle) " ) , build_command , cmdline ) ;
if ( result ! = 0 ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Building of Android project failed, check output for the error. \n Alternatively visit docs.godotengine.org for Android build documentation. " ) ) ;
return ERR_CANT_CREATE ;
}
2020-07-23 12:17:02 -04:00
List < String > copy_args ;
String copy_command ;
2020-12-23 01:30:29 -08:00
if ( export_format = = EXPORT_FORMAT_AAB ) {
2020-07-23 12:17:02 -04:00
copy_command = vformat ( " copyAndRename%sAab " , build_type ) ;
2020-12-23 01:30:29 -08:00
} else if ( export_format = = EXPORT_FORMAT_APK ) {
2020-07-23 12:17:02 -04:00
copy_command = vformat ( " copyAndRename%sApk " , build_type ) ;
2019-04-07 15:46:52 -03:00
}
2020-07-23 12:17:02 -04:00
copy_args . push_back ( copy_command ) ;
copy_args . push_back ( " -p " ) ; // argument to specify the start directory.
copy_args . push_back ( build_path ) ; // start directory.
String export_filename = p_path . get_file ( ) ;
String export_path = p_path . get_base_dir ( ) ;
2020-11-13 16:17:34 -08:00
if ( export_path . is_rel_path ( ) ) {
export_path = OS : : get_singleton ( ) - > get_resource_dir ( ) . plus_file ( export_path ) ;
}
export_path = ProjectSettings : : get_singleton ( ) - > globalize_path ( export_path ) . simplify_path ( ) ;
2020-07-23 12:17:02 -04:00
copy_args . push_back ( " -Pexport_path=file: " + export_path ) ;
copy_args . push_back ( " -Pexport_filename= " + export_filename ) ;
2021-02-16 16:33:15 -08:00
print_verbose ( " Copying Android binary using gradle command: " + String ( " \n " ) + build_command + " " + join_list ( copy_args , String ( " " ) ) ) ;
2020-07-23 12:17:02 -04:00
int copy_result = EditorNode : : get_singleton ( ) - > execute_and_show_output ( TTR ( " Moving output " ) , build_command , copy_args ) ;
if ( copy_result ! = 0 ) {
EditorNode : : get_singleton ( ) - > show_warning ( TTR ( " Unable to copy and rename export file, check gradle project directory for outputs. " ) ) ;
2019-04-07 15:46:52 -03:00
return ERR_CANT_CREATE ;
2017-03-23 20:14:12 -03:00
}
2019-04-07 15:46:52 -03:00
2021-02-16 16:33:15 -08:00
print_verbose ( " Successfully completed Android custom build. " ) ;
2020-07-23 12:17:02 -04:00
return OK ;
}
// This is the start of the Legacy build system
2021-02-16 16:33:15 -08:00
print_verbose ( " Starting legacy build system.. " ) ;
2021-05-05 12:44:11 +02:00
if ( p_debug ) {
2020-07-23 12:17:02 -04:00
src_apk = p_preset - > get ( " custom_template/debug " ) ;
2021-05-05 12:44:11 +02:00
} else {
2020-07-23 12:17:02 -04:00
src_apk = p_preset - > get ( " custom_template/release " ) ;
2021-05-05 12:44:11 +02:00
}
2019-04-07 15:46:52 -03:00
2020-07-23 12:17:02 -04:00
src_apk = src_apk . strip_edges ( ) ;
if ( src_apk = = " " ) {
if ( p_debug ) {
src_apk = find_export_template ( " android_debug.apk " ) ;
} else {
src_apk = find_export_template ( " android_release.apk " ) ;
}
2017-03-23 20:14:12 -03:00
if ( src_apk = = " " ) {
2020-07-23 12:17:02 -04:00
EditorNode : : add_io_error ( " Package not found: " + src_apk ) ;
return ERR_FILE_NOT_FOUND ;
2017-03-23 20:14:12 -03:00
}
}
2019-03-06 21:20:18 +09:00
if ( ! DirAccess : : exists ( p_path . get_base_dir ( ) ) ) {
2019-03-05 08:52:45 +01:00
return ERR_FILE_BAD_PATH ;
}
2021-05-04 16:00:45 +02:00
FileAccess * src_f = nullptr ;
2017-03-23 20:14:12 -03:00
zlib_filefunc_def io = zipio_create_io_from_file ( & src_f ) ;
2020-02-05 22:21:47 +01:00
if ( ep . step ( " Creating APK... " , 0 ) ) {
2018-12-19 16:50:40 -02:00
return ERR_SKIP ;
}
2017-03-23 20:14:12 -03:00
unzFile pkg = unzOpen2 ( src_apk . utf8 ( ) . get_data ( ) , & io ) ;
if ( ! pkg ) {
EditorNode : : add_io_error ( " Could not find template APK to export: \n " + src_apk ) ;
return ERR_FILE_NOT_FOUND ;
}
int ret = unzGoToFirstFile ( pkg ) ;
zlib_filefunc_def io2 = io ;
2021-05-04 16:00:45 +02:00
FileAccess * dst_f = nullptr ;
2017-03-23 20:14:12 -03:00
io2 . opaque = & dst_f ;
2019-08-09 13:45:30 +02:00
2021-04-15 13:26:05 -07:00
String tmp_unaligned_path = EditorSettings : : get_singleton ( ) - > get_cache_dir ( ) . plus_file ( " tmpexport-unaligned. " + uitos ( OS : : get_singleton ( ) - > get_unix_time ( ) ) + " .apk " ) ;
2019-08-09 13:45:30 +02:00
# define CLEANUP_AND_RETURN(m_err) \
{ \
DirAccess : : remove_file_or_error ( tmp_unaligned_path ) ; \
return m_err ; \
}
2021-05-04 16:00:45 +02:00
zipFile unaligned_apk = zipOpen2 ( tmp_unaligned_path . utf8 ( ) . get_data ( ) , APPEND_STATUS_CREATE , nullptr , & io2 ) ;
2017-03-23 20:14:12 -03:00
String cmdline = p_preset - > get ( " command_line/extra_args " ) ;
String version_name = p_preset - > get ( " version/name " ) ;
String package_name = p_preset - > get ( " package/unique_name " ) ;
String apk_expansion_pkey = p_preset - > get ( " apk_expansion/public_key " ) ;
2020-03-31 09:11:23 +02:00
Vector < String > invalid_abis ( enabled_abis ) ;
2017-03-23 20:14:12 -03:00
while ( ret = = UNZ_OK ) {
//get filename
unz_file_info info ;
char fname [ 16384 ] ;
2021-05-04 16:00:45 +02:00
ret = unzGetCurrentFileInfo ( pkg , & info , fname , 16384 , nullptr , 0 , nullptr , 0 ) ;
2017-03-23 20:14:12 -03:00
bool skip = false ;
String file = fname ;
Vector < uint8_t > data ;
data . resize ( info . uncompressed_size ) ;
//read
unzOpenCurrentFile ( pkg ) ;
2017-11-25 00:07:54 -03:00
unzReadCurrentFile ( pkg , data . ptrw ( ) , data . size ( ) ) ;
2017-03-23 20:14:12 -03:00
unzCloseCurrentFile ( pkg ) ;
//write
2020-07-23 12:17:02 -04:00
if ( file = = " AndroidManifest.xml " ) {
_fix_manifest ( p_preset , data , p_give_internet ) ;
}
if ( file = = " resources.arsc " ) {
_fix_resources ( p_preset , data ) ;
}
2020-08-19 15:38:50 -07:00
// Process the splash image
2021-04-12 16:33:37 -07:00
if ( ( file = = SPLASH_IMAGE_EXPORT_PATH | | file = = LEGACY_BUILD_SPLASH_IMAGE_EXPORT_PATH ) & & splash_image . is_valid ( ) & & ! splash_image - > empty ( ) ) {
2020-08-19 15:38:50 -07:00
_load_image_data ( splash_image , data ) ;
}
// Process the splash bg color image
2021-04-12 16:33:37 -07:00
if ( ( file = = SPLASH_BG_COLOR_PATH | | file = = LEGACY_BUILD_SPLASH_BG_COLOR_PATH ) & & splash_bg_color_image . is_valid ( ) & & ! splash_bg_color_image - > empty ( ) ) {
2020-08-19 15:38:50 -07:00
_load_image_data ( splash_bg_color_image , data ) ;
}
2020-07-23 12:17:02 -04:00
for ( int i = 0 ; i < icon_densities_count ; + + i ) {
if ( main_image . is_valid ( ) & & ! main_image - > empty ( ) ) {
if ( file = = launcher_icons [ i ] . export_path ) {
_process_launcher_icons ( file , main_image , launcher_icons [ i ] . dimensions , data ) ;
2020-07-22 15:52:26 -04:00
}
2020-07-23 12:17:02 -04:00
}
if ( foreground . is_valid ( ) & & ! foreground - > empty ( ) ) {
if ( file = = launcher_adaptive_icon_foregrounds [ i ] . export_path ) {
_process_launcher_icons ( file , foreground , launcher_adaptive_icon_foregrounds [ i ] . dimensions , data ) ;
2020-07-22 15:52:26 -04:00
}
2020-07-23 12:17:02 -04:00
}
if ( background . is_valid ( ) & & ! background - > empty ( ) ) {
if ( file = = launcher_adaptive_icon_backgrounds [ i ] . export_path ) {
_process_launcher_icons ( file , background , launcher_adaptive_icon_backgrounds [ i ] . dimensions , data ) ;
2020-07-22 15:52:26 -04:00
}
2017-03-23 20:14:12 -03:00
}
}
2017-11-21 01:12:36 +07:00
if ( file . ends_with ( " .so " ) ) {
bool enabled = false ;
for ( int i = 0 ; i < enabled_abis . size ( ) ; + + i ) {
if ( file . begins_with ( " lib/ " + enabled_abis [ i ] + " / " ) ) {
2020-03-31 09:11:23 +02:00
invalid_abis . erase ( enabled_abis [ i ] ) ;
2017-11-21 01:12:36 +07:00
enabled = true ;
break ;
}
}
if ( ! enabled ) {
skip = true ;
}
2017-07-25 12:28:31 +02:00
}
2020-12-23 01:30:29 -08:00
if ( file . begins_with ( " META-INF " ) & & should_sign ) {
2017-03-23 20:14:12 -03:00
skip = true ;
}
if ( ! skip ) {
2017-11-21 01:12:36 +07:00
print_line ( " ADDING: " + file ) ;
2017-03-23 20:14:12 -03:00
// Respect decision on compression made by AAPT for the export template
const bool uncompressed = info . compression_method = = 0 ;
zip_fileinfo zipfi = get_zip_fileinfo ( ) ;
zipOpenNewFileInZip ( unaligned_apk ,
file . utf8 ( ) . get_data ( ) ,
& zipfi ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
uncompressed ? 0 : Z_DEFLATED ,
Z_DEFAULT_COMPRESSION ) ;
zipWriteInFileInZip ( unaligned_apk , data . ptr ( ) , data . size ( ) ) ;
zipCloseFileInZip ( unaligned_apk ) ;
}
ret = unzGoToNextFile ( pkg ) ;
}
2020-03-31 09:11:23 +02:00
if ( ! invalid_abis . empty ( ) ) {
String unsupported_arch = String ( " , " ) . join ( invalid_abis ) ;
EditorNode : : add_io_error ( " Missing libraries in the export template for the selected architectures: " + unsupported_arch + " . \n " +
" Please build a template with all required libraries, or uncheck the missing architectures in the export preset. " ) ;
CLEANUP_AND_RETURN ( ERR_FILE_NOT_FOUND ) ;
}
2020-02-05 22:21:47 +01:00
if ( ep . step ( " Adding files... " , 1 ) ) {
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2018-12-19 16:50:40 -02:00
}
2020-07-23 12:17:02 -04:00
err = OK ;
2017-03-23 20:14:12 -03:00
if ( p_flags & DEBUG_FLAG_DUMB_CLIENT ) {
2018-01-16 12:36:18 +07:00
APKExportData ed ;
ed . ep = & ep ;
ed . apk = unaligned_apk ;
err = export_project_files ( p_preset , ignore_apk_file , & ed , save_apk_so ) ;
2020-07-23 12:17:02 -04:00
} else {
if ( apk_expansion ) {
err = save_apk_expansion_file ( p_preset , p_path ) ;
if ( err ! = OK ) {
EditorNode : : add_io_error ( " Could not write expansion package file! " ) ;
return err ;
}
} else {
APKExportData ed ;
ed . ep = & ep ;
ed . apk = unaligned_apk ;
err = export_project_files ( p_preset , save_apk_file , & ed , save_apk_so ) ;
}
2018-01-16 12:36:18 +07:00
}
2017-11-24 11:09:15 +07:00
2020-06-26 14:58:16 -04:00
if ( err ! = OK ) {
unzClose ( pkg ) ;
EditorNode : : add_io_error ( " Could not export project files " ) ;
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2019-06-16 01:58:34 -07:00
}
2020-06-26 14:58:16 -04:00
zip_fileinfo zipfi = get_zip_fileinfo ( ) ;
zipOpenNewFileInZip ( unaligned_apk ,
" assets/_cl_ " ,
& zipfi ,
2021-05-04 16:00:45 +02:00
nullptr ,
2020-06-26 14:58:16 -04:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2020-06-26 14:58:16 -04:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2020-06-26 14:58:16 -04:00
0 , // No compress (little size gain and potentially slower startup)
Z_DEFAULT_COMPRESSION ) ;
zipWriteInFileInZip ( unaligned_apk , command_line_flags . ptr ( ) , command_line_flags . size ( ) ) ;
zipCloseFileInZip ( unaligned_apk ) ;
2017-03-23 20:14:12 -03:00
2021-05-04 16:00:45 +02:00
zipClose ( unaligned_apk , nullptr ) ;
2017-03-23 20:14:12 -03:00
unzClose ( pkg ) ;
2019-08-09 13:45:30 +02:00
if ( err ! = OK ) {
CLEANUP_AND_RETURN ( err ) ;
2017-03-23 20:14:12 -03:00
}
2020-12-23 22:16:56 -08:00
// Let's zip-align (must be done before signing)
2017-03-23 20:14:12 -03:00
static const int ZIP_ALIGNMENT = 4 ;
2020-12-23 22:16:56 -08:00
// If we're not signing the apk, then the next step should be the last.
const int next_step = should_sign ? 103 : 105 ;
if ( ep . step ( " Aligning APK... " , next_step ) ) {
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( ERR_SKIP ) ;
2018-12-19 16:50:40 -02:00
}
2017-03-23 20:14:12 -03:00
2019-08-09 13:45:30 +02:00
unzFile tmp_unaligned = unzOpen2 ( tmp_unaligned_path . utf8 ( ) . get_data ( ) , & io ) ;
2017-03-23 20:14:12 -03:00
if ( ! tmp_unaligned ) {
2019-08-09 13:45:30 +02:00
EditorNode : : add_io_error ( " Could not unzip temporary unaligned APK. " ) ;
CLEANUP_AND_RETURN ( ERR_FILE_NOT_FOUND ) ;
2017-03-23 20:14:12 -03:00
}
ret = unzGoToFirstFile ( tmp_unaligned ) ;
io2 = io ;
2021-05-04 16:00:45 +02:00
dst_f = nullptr ;
2017-03-23 20:14:12 -03:00
io2 . opaque = & dst_f ;
2021-05-04 16:00:45 +02:00
zipFile final_apk = zipOpen2 ( p_path . utf8 ( ) . get_data ( ) , APPEND_STATUS_CREATE , nullptr , & io2 ) ;
2017-03-23 20:14:12 -03:00
// Take files from the unaligned APK and write them out to the aligned one
// in raw mode, i.e. not uncompressing and recompressing, aligning them as needed,
// following what is done in https://github.com/android/platform_build/blob/master/tools/zipalign/ZipAlign.cpp
int bias = 0 ;
while ( ret = = UNZ_OK ) {
unz_file_info info ;
memset ( & info , 0 , sizeof ( info ) ) ;
char fname [ 16384 ] ;
char extra [ 16384 ] ;
2021-05-04 16:00:45 +02:00
ret = unzGetCurrentFileInfo ( tmp_unaligned , & info , fname , 16384 , extra , 16384 - ZIP_ALIGNMENT , nullptr , 0 ) ;
2017-03-23 20:14:12 -03:00
String file = fname ;
Vector < uint8_t > data ;
data . resize ( info . compressed_size ) ;
// read
int method , level ;
unzOpenCurrentFile2 ( tmp_unaligned , & method , & level , 1 ) ; // raw read
long file_offset = unzGetCurrentFileZStreamPos64 ( tmp_unaligned ) ;
2017-11-25 00:07:54 -03:00
unzReadCurrentFile ( tmp_unaligned , data . ptrw ( ) , data . size ( ) ) ;
2017-03-23 20:14:12 -03:00
unzCloseCurrentFile ( tmp_unaligned ) ;
// align
int padding = 0 ;
if ( ! info . compression_method ) {
// Uncompressed file => Align
long new_offset = file_offset + bias ;
padding = ( ZIP_ALIGNMENT - ( new_offset % ZIP_ALIGNMENT ) ) % ZIP_ALIGNMENT ;
}
memset ( extra + info . size_file_extra , 0 , padding ) ;
2020-06-26 14:58:16 -04:00
zip_fileinfo fileinfo = get_zip_fileinfo ( ) ;
2017-03-23 20:14:12 -03:00
zipOpenNewFileInZip2 ( final_apk ,
file . utf8 ( ) . get_data ( ) ,
2020-06-26 14:58:16 -04:00
& fileinfo ,
2017-03-23 20:14:12 -03:00
extra ,
info . size_file_extra + padding ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
0 ,
2021-05-04 16:00:45 +02:00
nullptr ,
2017-03-23 20:14:12 -03:00
method ,
level ,
1 ) ; // raw write
zipWriteInFileInZip ( final_apk , data . ptr ( ) , data . size ( ) ) ;
zipCloseFileInZipRaw ( final_apk , info . uncompressed_size , info . crc ) ;
bias + = padding ;
ret = unzGoToNextFile ( tmp_unaligned ) ;
}
2021-05-04 16:00:45 +02:00
zipClose ( final_apk , nullptr ) ;
2017-03-23 20:14:12 -03:00
unzClose ( tmp_unaligned ) ;
2020-12-23 22:16:56 -08:00
if ( should_sign ) {
// Signing must be done last as any additional modifications to the
// file will invalidate the signature.
err = sign_apk ( p_preset , p_debug , p_path , ep ) ;
if ( err ! = OK ) {
CLEANUP_AND_RETURN ( err ) ;
}
}
2019-08-09 13:45:30 +02:00
CLEANUP_AND_RETURN ( OK ) ;
2017-03-23 20:14:12 -03:00
}
2017-07-19 17:00:46 -03:00
virtual void get_platform_features ( List < String > * r_features ) {
r_features - > push_back ( " mobile " ) ;
r_features - > push_back ( " Android " ) ;
}
2018-08-21 22:56:04 -04:00
virtual void resolve_platform_feature_priorities ( const Ref < EditorExportPreset > & p_preset , Set < String > & p_features ) {
}
2019-02-12 15:43:54 +01:00
EditorExportPlatformAndroid ( ) {
2017-05-17 07:36:47 -03:00
Ref < Image > img = memnew ( Image ( _android_logo ) ) ;
2017-05-25 20:57:13 +02:00
logo . instance ( ) ;
2017-03-23 20:14:12 -03:00
logo - > create_from_image ( img ) ;
2017-05-25 20:57:13 +02:00
img = Ref < Image > ( memnew ( Image ( _android_run_icon ) ) ) ;
run_icon . instance ( ) ;
run_icon - > create_from_image ( img ) ;
2021-01-31 13:34:42 +01:00
devices_changed . set ( ) ;
plugins_changed . set ( ) ;
2021-01-27 20:10:10 +01:00
check_for_changes_thread . start ( _check_for_changes_poll_thread , this ) ;
2017-03-23 20:14:12 -03:00
}
2017-04-10 19:41:27 +02:00
2019-02-12 15:43:54 +01:00
~ EditorExportPlatformAndroid ( ) {
2021-01-31 13:34:42 +01:00
quit_request . set ( ) ;
2021-01-27 20:10:10 +01:00
check_for_changes_thread . wait_to_finish ( ) ;
2017-04-10 19:41:27 +02:00
}
2017-03-23 20:14:12 -03:00
} ;
void register_android_exporter ( ) {
String exe_ext ;
if ( OS : : get_singleton ( ) - > get_name ( ) = = " Windows " ) {
exe_ext = " *.exe " ;
}
2020-12-23 22:16:56 -08:00
EDITOR_DEF ( " export/android/android_sdk_path " , " " ) ;
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : STRING , " export/android/android_sdk_path " , PROPERTY_HINT_GLOBAL_DIR ) ) ;
2017-03-23 20:14:12 -03:00
EDITOR_DEF ( " export/android/debug_keystore " , " " ) ;
2020-11-14 00:29:23 +01:00
EditorSettings : : get_singleton ( ) - > add_property_hint ( PropertyInfo ( Variant : : STRING , " export/android/debug_keystore " , PROPERTY_HINT_GLOBAL_FILE , " *.keystore,*.jks " ) ) ;
2017-03-23 20:14:12 -03:00
EDITOR_DEF ( " export/android/debug_keystore_user " , " androiddebugkey " ) ;
EDITOR_DEF ( " export/android/debug_keystore_pass " , " android " ) ;
2017-07-27 17:20:19 +02:00
EDITOR_DEF ( " export/android/force_system_user " , false ) ;
2017-03-23 20:14:12 -03:00
EDITOR_DEF ( " export/android/shutdown_adb_on_exit " , true ) ;
2019-02-12 15:43:54 +01:00
Ref < EditorExportPlatformAndroid > exporter = Ref < EditorExportPlatformAndroid > ( memnew ( EditorExportPlatformAndroid ) ) ;
2017-03-23 20:14:12 -03:00
EditorExport : : get_singleton ( ) - > add_export_platform ( exporter ) ;
2017-02-19 23:19:30 -03:00
}