expo-location
Version:
Allows reading geolocation information from the device. Your app can poll for the current location or subscribe to location update events.
203 lines (183 loc) • 7.26 kB
text/typescript
import { generateImageAsync } from '@expo/image-utils';
import {
AndroidConfig,
ConfigPlugin,
IOSConfig,
createRunOncePlugin,
withInfoPlist,
withDangerousMod,
withAndroidManifest,
} from 'expo/config-plugins';
import { writeFileSync, unlinkSync, existsSync, mkdirSync } from 'fs';
import { resolve } from 'path';
const pkg = require('expo-location/package.json');
const LOCATION_USAGE = 'Allow $(PRODUCT_NAME) to access your location';
type DPIString = 'mdpi' | 'hdpi' | 'xhdpi' | 'xxhdpi' | 'xxxhdpi';
type dpiMap = Record<DPIString, { folderName: string; scale: number }>;
export const ANDROID_RES_PATH = 'android/app/src/main/res/';
export const dpiValues: dpiMap = {
mdpi: { folderName: 'mipmap-mdpi', scale: 1 },
hdpi: { folderName: 'mipmap-hdpi', scale: 1.5 },
xhdpi: { folderName: 'mipmap-xhdpi', scale: 2 },
xxhdpi: { folderName: 'mipmap-xxhdpi', scale: 3 },
xxxhdpi: { folderName: 'mipmap-xxxhdpi', scale: 4 },
};
const {
addMetaDataItemToMainApplication,
getMainApplicationOrThrow,
removeMetaDataItemFromMainApplication,
} = AndroidConfig.Manifest;
const BASELINE_PIXEL_SIZE = 24;
const ERROR_MSG_PREFIX = 'An error occurred while configuring Android location. ';
export const FOREGROUND_SERVICE_ICON = 'location_foreground_service_icon';
export const FOREGROUND_SERVICE_ICON_RESOURCE = `@drawable/${FOREGROUND_SERVICE_ICON}`;
export const META_DATA_FOREGROUND_SERVICE_ICON = 'expo.modules.location.foreground_service_icon';
const withBackgroundLocation: ConfigPlugin = (config) => {
return withInfoPlist(config, (config) => {
if (!Array.isArray(config.modResults.UIBackgroundModes)) {
config.modResults.UIBackgroundModes = [];
}
if (!config.modResults.UIBackgroundModes.includes('location')) {
config.modResults.UIBackgroundModes.push('location');
}
return config;
});
};
export const withForegroundServiceIcon: ConfigPlugin<{ icon: string | null }> = (
config,
{ icon }
) => {
// Update icon assets
const configWithIconAssets = withDangerousMod(config, [
'android',
async (config) => {
await setForegroundServiceIconAsync(config.modRequest.projectRoot, icon);
return config;
},
]);
// Update assets Android manifest
return withAndroidManifest(configWithIconAssets, (config) => {
const manifest = config.modResults;
const mainApplication = getMainApplicationOrThrow(manifest);
if (icon) {
addMetaDataItemToMainApplication(
mainApplication,
META_DATA_FOREGROUND_SERVICE_ICON,
FOREGROUND_SERVICE_ICON_RESOURCE,
'resource'
);
} else {
removeMetaDataItemFromMainApplication(mainApplication, META_DATA_FOREGROUND_SERVICE_ICON);
}
config.modResults = manifest;
return config;
});
};
/**
* Applies foreground service icon configuration for expo-location
*/
export async function setForegroundServiceIconAsync(projectRoot: string, icon: string | null) {
if (icon) {
await writeForegroundServiceIconImageFilesAsync(icon, projectRoot);
} else {
removeForegroundServiceIconImageFiles(projectRoot);
}
}
async function writeForegroundServiceIconImageFilesAsync(icon: string, projectRoot: string) {
await Promise.all(
Object.values(dpiValues).map(async ({ folderName, scale }) => {
const drawableFolderName = folderName.replace('mipmap', 'drawable');
const dpiFolderPath = resolve(projectRoot, ANDROID_RES_PATH, drawableFolderName);
if (!existsSync(dpiFolderPath)) {
mkdirSync(dpiFolderPath, { recursive: true });
}
const iconSizePx = BASELINE_PIXEL_SIZE * scale;
try {
const resizedIcon = (
await generateImageAsync(
{ projectRoot, cacheType: 'android-location' },
{
src: icon,
width: iconSizePx,
height: iconSizePx,
resizeMode: 'cover',
backgroundColor: 'transparent',
}
)
).source;
writeFileSync(resolve(dpiFolderPath, FOREGROUND_SERVICE_ICON + '.png'), resizedIcon);
} catch (e) {
throw new Error(
ERROR_MSG_PREFIX + 'Encountered an issue resizing Android foreground service icon: ' + e
);
}
})
);
}
function removeForegroundServiceIconImageFiles(projectRoot: string) {
Object.values(dpiValues).forEach(async ({ folderName }) => {
const drawableFolderName = folderName.replace('mipmap', 'drawable');
const dpiFolderPath = resolve(projectRoot, ANDROID_RES_PATH, drawableFolderName);
const iconFile = resolve(dpiFolderPath, FOREGROUND_SERVICE_ICON + '.png');
if (existsSync(iconFile)) {
unlinkSync(iconFile);
}
});
}
const withLocation: ConfigPlugin<
{
locationAlwaysAndWhenInUsePermission?: string | false;
locationAlwaysPermission?: string | false;
locationWhenInUsePermission?: string | false;
isIosBackgroundLocationEnabled?: boolean;
isAndroidBackgroundLocationEnabled?: boolean;
isAndroidForegroundServiceEnabled?: boolean;
androidForegroundServiceIcon?: string;
} | void
> = (
config,
{
locationAlwaysAndWhenInUsePermission,
locationAlwaysPermission,
locationWhenInUsePermission,
isIosBackgroundLocationEnabled,
isAndroidBackgroundLocationEnabled,
isAndroidForegroundServiceEnabled,
androidForegroundServiceIcon,
} = {}
) => {
if (isIosBackgroundLocationEnabled) {
config = withBackgroundLocation(config);
}
config = withForegroundServiceIcon(config, { icon: androidForegroundServiceIcon ?? null });
IOSConfig.Permissions.createPermissionsPlugin({
NSLocationAlwaysAndWhenInUseUsageDescription: LOCATION_USAGE,
NSLocationAlwaysUsageDescription: LOCATION_USAGE,
NSLocationWhenInUseUsageDescription: LOCATION_USAGE,
})(config, {
NSLocationAlwaysAndWhenInUseUsageDescription: locationAlwaysAndWhenInUsePermission,
NSLocationAlwaysUsageDescription: locationAlwaysPermission,
NSLocationWhenInUseUsageDescription: locationWhenInUsePermission,
});
// If the user has not specified a value for isAndroidForegroundServiceEnabled,
// we default to the value of isAndroidBackgroundLocationEnabled because we want
// to enable foreground by default if background location is enabled.
const enableAndroidForegroundService =
typeof isAndroidForegroundServiceEnabled === 'undefined'
? isAndroidBackgroundLocationEnabled
: isAndroidForegroundServiceEnabled;
return AndroidConfig.Permissions.withPermissions(
config,
[
// Note: these are already added in the library AndroidManifest.xml and so
// are not required here, we may want to remove them in the future.
'android.permission.ACCESS_COARSE_LOCATION',
'android.permission.ACCESS_FINE_LOCATION',
// These permissions are optional, and not listed in the library AndroidManifest.xml
isAndroidBackgroundLocationEnabled && 'android.permission.ACCESS_BACKGROUND_LOCATION',
enableAndroidForegroundService && 'android.permission.FOREGROUND_SERVICE',
enableAndroidForegroundService && 'android.permission.FOREGROUND_SERVICE_LOCATION',
].filter(Boolean) as string[]
);
};
export default createRunOncePlugin(withLocation, pkg.name, pkg.version);