Использование типов сборки в Gradle для запуска того же приложения, которое использует ContentProvider на одном устройстве
Я настроил Gradle для добавления суффикса имени пакета в мое приложение отладки, чтобы у меня могла быть версия выпуска, которую я использую, и версия отладки на одном телефоне. Я ссылался на это: http://tools.android.com/tech-docs/new-build-system/user-guide
Мой файл build.gradle выглядит так:
...
android
{
...
buildTypes
{
debug
{
packageNameSuffix ".debug"
versionNameSuffix " debug"
}
}
}
Все работает нормально, пока я не начну использовать ContentProvider в своем приложении. Я получил:
Failure [INSTALL_FAILED_CONFLICTING_PROVIDER]
Я понимаю, что это происходит потому, что два приложения (выпуск и отладка) регистрируют одинаковые полномочия ContentProvider.
Я вижу одну возможность решить это. Если я правильно понимаю, вы сможете указать разные файлы для использования при сборке. Тогда я смогу поместить разные права доступа в разные файлы ресурсов (и из Манифеста установить полномочия в качестве строкового ресурса) и сказать Gradle использовать разные ресурсы для отладочной сборки. Это возможно? Если да, то любые намеки на то, как этого добиться, были бы потрясающими!
Или, возможно, можно напрямую модифицировать Manifest с помощью Gradle? Любое другое решение о том, как запустить одно приложение с ContentProvider на одном устройстве, всегда приветствуется.
14 ответов
Ни один из существующих ответов не удовлетворил меня, однако Свобода была близка. Так вот как я это делаю. Прежде всего, сейчас я работаю с:
- Android Studio Beta 0.8.2
- Gradle плагин 0.12. +
- Gradle 1.12
Моя цель - бежать Debug
версия вместе с Release
версия на том же устройстве, используя тот же ContentProvider
,
В build.gradle вашего приложения установите суффикс для сборки Debug:
buildTypes {
debug {
applicationIdSuffix ".debug"
}
}
В наборе файлов AndroidManifest.xml android:authorities
собственность вашего ContentProvider
:
<provider
android:name="com.example.app.YourProvider"
android:authorities="${applicationId}.provider"
android:enabled="true"
android:exported="false" >
</provider>
В вашем кодовом наборе AUTHORITY
свойство, которое можно использовать везде, где необходимо в вашей реализации:
public static final String AUTHORITY = BuildConfig.APPLICATION_ID + ".provider";
Совет: прежде чем это было BuildConfig.PACKAGE_NAME
Это оно! Это будет работать как шарм. Продолжайте читать, если вы используете SyncAdapter!
Обновление для SyncAdapter (14.11.2014)
Еще раз начну с моей текущей настройки:
- Android Studio Beta 0.9.2
- Gradle плагин 0.14.1
- Gradle 2.1
По сути, если вам нужно настроить некоторые значения для разных сборок, вы можете сделать это из файла build.gradle:
- используйте buildConfigField для доступа к нему из
BuildConfig.java
учебный класс - используйте resValue для доступа к нему из ресурсов, например, @string/ your_value
В качестве альтернативы для ресурсов, вы можете создать отдельные каталоги buildType или flavour и переопределить XML или значения внутри них. Однако я не собираюсь использовать его в примере ниже.
пример
В файле build.gradle добавьте следующее:
defaultConfig {
resValue "string", "your_authorities", applicationId + '.provider'
resValue "string", "account_type", "your.syncadapter.type"
buildConfigField "String", "ACCOUNT_TYPE", '"your.syncadapter.type"'
}
buildTypes {
debug {
applicationIdSuffix ".debug"
resValue "string", "your_authorities", defaultConfig.applicationId + '.debug.provider'
resValue "string", "account_type", "your.syncadapter.type.debug"
buildConfigField "String", "ACCOUNT_TYPE", '"your.syncadapter.type.debug"'
}
}
Вы увидите результаты в классе BuildConfig.java
public static final String ACCOUNT_TYPE = "your.syncadapter.type.debug";
и в сборке / генерируется /res/ генерируется /debug/values /gen.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<!-- Automatically generated file. DO NOT MODIFY -->
<!-- Values from default config. -->
<item name="account_type" type="string">your.syncadapter.type.debug</item>
<item name="authorities" type="string">com.example.app.provider</item>
</resources>
В вашем authenticator.xml используйте ресурс, указанный в файле build.gradle
<?xml version="1.0" encoding="utf-8"?>
<account-authenticator xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="@string/account_type"
android:icon="@drawable/ic_launcher"
android:smallIcon="@drawable/ic_launcher"
android:label="@string/app_name"
/>
В вашем syncadapter.xml снова используйте тот же ресурс и @string/ полномочия тоже
<?xml version="1.0" encoding="utf-8"?>
<sync-adapter xmlns:android="http://schemas.android.com/apk/res/android"
android:contentAuthority="@string/authorities"
android:accountType="@string/account_type"
android:userVisible="true"
android:supportsUploading="false"
android:allowParallelSyncs="false"
android:isAlwaysSyncable="true"
/>
Совет: автозаполнение (Ctrl+ Пробел) не работает для этих сгенерированных ресурсов, поэтому вы должны ввести их вручную
Совет по новой системе сборки Android: переименование полномочий ContentProvider
Думаю, вы все слышали о новой системе сборки на базе Android Gradle. Давайте будем честными, эта новая система сборки - огромный шаг вперед по сравнению с предыдущей. Он еще не окончательный (на момент написания этой статьи последняя версия - 0.4.2), но вы уже можете безопасно использовать его в большинстве своих проектов.
Я лично переключил большую часть своего проекта на эту новую систему сборки, и у меня были некоторые проблемы из-за отсутствия поддержки в некоторых конкретных ситуациях. Одним из них является поддержка переименования полномочий ContentProvider
Новая встроенная система Android позволяет вам работать с различными типами вашего приложения, просто изменяя имя пакета во время сборки. Одним из главных преимуществ этого улучшения является то, что теперь вы можете иметь две разные версии приложения, установленные на одном устройстве одновременно. Например:
android {
compileSdkVersion 17
buildToolsVersion "17.0.0"
defaultConfig {
packageName "com.cyrilmottier.android.app"
versionCode 1
versionName "1"
minSdkVersion 14 // Listen to +Jeff Gilfelt advices :)
targetSdkVersion 17
}
buildTypes {
debug {
packageNameSuffix ".debug"
versionNameSuffix "-debug"
}
}
}
Используя такую конфигурацию Gradle, вы можете собрать два разных APK:
• Отладочный APK с именем пакета com.cyrilmottier.android.app.debug • Выпуск APK с именем пакета com.cyrilmottier.android.app
Единственная проблема заключается в том, что вы не сможете установить два APK одновременно, если они оба предоставляют ContentProvider с одинаковыми правами доступа. Довольно логично, что нам нужно переименовать полномочия в зависимости от текущего типа сборки... но это не поддерживается системой сборки Gradle (пока? ... Я уверен, что это будет исправлено в ближайшее время). Итак, вот путь:
Во-первых, нам нужно переместить декларацию ContentProvider для манифеста Android в соответствующий тип сборки. Для этого у нас будет просто:
SRC / отлаживать /AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.cyrilmottier.android.app"
android:versionCode="1"
android:versionName="1">
<application>
<provider
android:name=".provider.Provider1"
android:authorities="com.cyrilmottier.android.app.debug.provider"
android:exported="false" />
</application>
</manifest>
SRC / выпуск /AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.cyrilmottier.android.app"
android:versionCode="1"
android:versionName="1">
<application>
<provider
android:name=".provider.Provider1"
android:authorities="com.cyrilmottier.android.app.provider"
android:exported="false" />
</application>
</manifest>
Убедитесь, что удалили объявление ContentProvider из AndroidManifest.xml в src/main/, потому что Gradle не знает, как объединить ContentProvider с тем же именем, но с другими полномочиями.
Наконец, нам может понадобиться доступ к авторитету в коде. Это можно сделать довольно легко, используя файл BuildConfig и метод buildConfig:
android {
// ...
final PROVIDER_DEBUG = "com.cyrilmottier.android.app.debug.provider"
final PROVIDER_RELEASE = "com.cyrilmottier.android.app.provider"
buildTypes {
debug {
// ...
buildConfigField "String", "PROVIDER_AUTHORITY", PROVIDER_DEBUG
}
release {
buildConfigField "String", "PROVIDER_AUTHORITY", PROVIDER_RELEASE
}
}
}
Благодаря этому обходному пути вы сможете использовать BuildConfig.PROVIDER_AUTHORITY в вашем ProviderContract и установить две разные версии вашего приложения одновременно.
Первоначально в Google+: https://plus.google.com/u/0/118417777153109946393/posts/EATUmhntaCQ
Хотя пример Кирилла отлично работает, если у вас есть только несколько типов сборки, он быстро усложняется, если у вас много типов сборки и / или разновидностей продукта, так как вам нужно поддерживать множество различных AndroidManifest.xml.
Наш проект состоит из 3 разных типов сборки и 6 разновидностей, всего 18 вариантов сборки, поэтому вместо этого мы добавили поддержку ".res-auto" в полномочия ContentProvider, которые расширяют текущее имя пакета и избавляют от необходимости поддерживать разные AndroidManifest.xml.
/**
* Version 1.1.
*
* Add support for installing multiple variants of the same app which have a
* content provider. Do this by overriding occurrences of ".res-auto" in
* android:authorities with the current package name (which should be unique)
*
* V1.0 : Initial version
* V1.1 : Support for ".res-auto" in strings added,
* eg. use "<string name="auth">.res-auto.path.to.provider</string>"
*
*/
def overrideProviderAuthority(buildVariant) {
def flavor = buildVariant.productFlavors.get(0).name
def buildType = buildVariant.buildType.name
def pathToManifest = "${buildDir}/manifests/${flavor}/${buildType}/AndroidManifest.xml"
def ns = new groovy.xml.Namespace("http://schemas.android.com/apk/res/android", "android")
def xml = new XmlParser().parse(pathToManifest)
def variantPackageName = xml.@package
// Update all content providers
xml.application.provider.each { provider ->
def newAuthorities = provider.attribute(ns.authorities).replaceAll('.res-auto', variantPackageName)
provider.attributes().put(ns.authorities, newAuthorities)
}
// Save modified AndroidManifest back into build dir
saveXML(pathToManifest, xml)
// Also make sure that all strings with ".res-auto" are expanded automagically
def pathToValues = "${buildDir}/res/all/${flavor}/${buildType}/values/values.xml"
xml = new XmlParser().parse(pathToValues)
xml.findAll{it.name() == 'string'}.each{item ->
if (!item.value().isEmpty() && item.value()[0].startsWith(".res-auto")) {
item.value()[0] = item.value()[0].replace(".res-auto", variantPackageName)
}
}
saveXML(pathToValues, xml)
}
def saveXML(pathToFile, xml) {
def writer = new FileWriter(pathToFile)
def printer = new XmlNodePrinter(new PrintWriter(writer))
printer.preserveWhitespace = true
printer.print(xml)
}
// Post processing of AndroidManifest.xml for supporting provider authorities
// across build variants.
android.applicationVariants.all { variant ->
variant.processManifest.doLast {
overrideProviderAuthority(variant)
}
}
Пример кода можно найти здесь: https://gist.github.com/cmelchior/6988275
Начиная с версии плагина 0.8.3 (на самом деле 0.8.1, но он не работал должным образом), вы можете определить ресурсы в файле сборки, так что это может быть более чистым решением, потому что вам не нужно создавать строковые файлы или дополнительную отладку / выпуск папки.
build.gradle
android {
buildTypes {
debug{
resValue "string", "authority", "com.yourpackage.debug.provider"
}
release {
resValue "string", "authority", "com.yourpackage.provider"
}
}
}
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.yourpackage"
android:versionCode="1"
android:versionName="1">
<application>
<provider
android:name=".provider.Provider1"
android:authorities="@string/authority"
android:exported="false" />
</application>
</manifest>
Я не знаю, упоминал ли кто-нибудь об этом. На самом деле после плагина Android Grade 0.10+, слияние манифеста обеспечит официальную поддержку этой функции: http://tools.android.com/tech-docs/new-build-system/user-guide/manifest-merger
В AndroidManifest.xml вы можете использовать ${packageName} следующим образом:
<provider
android:name=".provider.DatabasesProvider"
android:authorities="${packageName}.databasesprovider"
android:exported="true"
android:multiprocess="true" />
И в вашем build.gradle вы можете иметь:
productFlavors {
free {
packageName "org.pkg1"
}
pro {
packageName "org.pkg2"
}
}
Смотрите полный пример здесь: https://code.google.com/p/anymemo/source/browse/AndroidManifest.xml#152
и здесь: https://code.google.com/p/anymemo/source/browse/build.gradle
Использование ${applicationId}
заполнители в XML и BuildConfig.APPLICATION_ID
в коде.
Вам нужно будет расширить скрипт сборки, чтобы включить заполнители в XML-файлах, отличных от манифеста. Вы можете использовать исходный каталог для каждого варианта сборки, чтобы предоставить разные версии XML-файлов, но обслуживание станет очень громоздким.
AndroidManifest.xml
Вы можете использовать заполнитель applicationId из коробки в манифесте. Объявите своего провайдера следующим образом:
<provider
android:name=".provider.DatabaseProvider"
android:authorities="${applicationId}.DatabaseProvider"
android:exported="false" />
Обратите внимание ${applicationId}
немного. Это заменяется во время сборки фактическим applicationId для варианта сборки, который собирается.
В коде
Ваш ContentProvider должен создать авторитетную строку в коде. Он может использовать класс BuildConfig.
public class DatabaseContract {
/** The authority for the database provider */
public static final String AUTHORITY = BuildConfig.APPLICATION_ID + ".DatabaseProvider";
// ...
}
Обратите внимание BuildConfig.APPLICATION_ID
немного. Это сгенерированный класс с фактическим applicationId для создаваемого варианта сборки.
res / xml / files, например syncadapter.xml, accountauthenticator.xml
Если вы хотите использовать адаптер синхронизации, вам необходимо предоставить метаданные для ContentProvider и AccountManager в файлах xml в каталоге res / xml /. Заполнитель applicationId здесь не поддерживается. Но вы можете расширить сценарий сборки, чтобы взломать его.
<sync-adapter xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="${applicationId}"
android:allowParallelSyncs="false"
android:contentAuthority="${applicationId}.DatabaseProvider"
android:isAlwaysSyncable="true"
android:supportsUploading="true"
android:userVisible="true" />
<account-authenticator xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="${applicationId}"
android:icon="@drawable/ic_launcher"
android:label="@string/account_authenticator_label"
android:smallIcon="@drawable/ic_launcher" />
Опять же, обратите внимание на ${applicationId}
, Это работает, только если вы добавили приведенный ниже скрипт gradle в корень вашего модуля и применили его из build.gradle.
build.gradle
Примените дополнительный скрипт сборки из скрипта модуля build.gradle. Хорошее место под плагином Android Gradle.
apply plugin: 'com.android.application'
apply from: './build-processApplicationId.gradle'
android {
compileSdkVersion 21
// etc.
строить-processApplicationId.gradle
Ниже приведен рабочий источник для скрипта сборки res / xml / placeholder. Лучшая документированная версия доступна на github. Улучшения и расширения приветствуются.
def replace(File file, String target, String replacement) {
def result = false;
def reader = new FileReader(file)
def lines = reader.readLines()
reader.close()
def writer = new FileWriter(file)
lines.each { line ->
String replacedLine = line.replace(target, replacement)
writer.write(replacedLine)
writer.write("\n")
result = result || !replacedLine.equals(line)
}
writer.close()
return result
}
def processXmlFile(File file, String applicationId) {
if (replace(file, "\${applicationId}", applicationId)) {
logger.info("Processed \${applicationId} in $file")
}
}
def processXmlDir(File dir, String applicationId) {
dir.list().each { entry ->
File file = new File(dir, entry)
if (file.isFile()) {
processXmlFile(file, applicationId)
}
}
}
android.applicationVariants.all { variant ->
variant.mergeResources.doLast {
def applicationId = variant.mergedFlavor.applicationId + (variant.buildType.applicationIdSuffix == null ? "" : variant.buildType.applicationIdSuffix)
def path = "${buildDir}/intermediates/res/${variant.dirName}/xml/"
processXmlDir(new File(path), applicationId)
}
}
strings.xml
На мой взгляд, нет необходимости добавлять поддержку заполнителей для строк ресурсов. Для приведенного выше варианта использования, по крайней мере, он не нужен. Однако вы можете легко изменить скрипт, чтобы он заменял не только заполнители в каталоге res / xml /, но также и в каталоге res/values /.
Я бы предпочел смесь между Кириллом и Рчиовати. Я думаю, что проще, у вас есть только две модификации.
build.gradle
похоже:
android {
...
productFlavors {
production {
packageName "package.name.production"
resValue "string", "authority", "package.name.production.provider"
buildConfigField "String", "AUTHORITY", "package.name.production.provider"
}
testing {
packageName "package.name.debug"
resValue "string", "authority", "package.name.debug.provider"
buildConfigField "String", "AUTHORITY", "package.name.debug.provider"
}
}
...
}
И AndroidManifest.xml
:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="package.name" >
<application
...>
<provider android:name=".contentprovider.Provider" android:authorities="@string/authority" />
</application>
</manifest>
gradle.build
android {
compileSdkVersion 23
buildToolsVersion "23.0.1"
defaultConfig {
applicationId "com.example.awsomeapp"
minSdkVersion 9
targetSdkVersion 23
versionCode 1
versionName "1.0.0"
}
productFlavors
{
prod {
applicationId = "com.example.awsomeapp"
}
demo {
applicationId = "com.example.awsomeapp.demo"
versionName = defaultConfig.versionName + ".DEMO"
}
}
buildTypes {
release {
signingConfig signingConfigs.release
debuggable false
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt'
}
debug {
applicationIdSuffix ".debug"
versionNameSuffix = ".DEBUG"
debuggable true
}
}
applicationVariants.all { variant ->
variant.outputs.each { output ->
// rename the apk
def file = output.outputFile;
def newName;
newName = file.name.replace(".apk", "-" + defaultConfig.versionName + ".apk");
newName = newName.replace(project.name, "awsomeapp");
output.outputFile = new File(file.parent, newName);
}
//Generate values Content Authority and Account Type used in Sync Adapter, Content Provider, Authenticator
def valueAccountType = applicationId + '.account'
def valueContentAuthority = applicationId + '.authority'
//generate fields in Resource string file generated.xml
resValue "string", "content_authority", valueContentAuthority
resValue "string", "account_type", valueAccountType
//generate fields in BuildConfig class
buildConfigField "String", "ACCOUNT_TYPE", '"'+valueAccountType+'"'
buildConfigField "String", "CONTENT_AUTHORITY", '"'+valueContentAuthority+'"'
//replace field ${valueContentAuthority} in AndroidManifest.xml
mergedFlavor.manifestPlaceholders = [ valueContentAuthority: valueContentAuthority ]
}
}
authenticator.xml
<?xml version="1.0" encoding="utf-8"?>
<account-authenticator xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="@string/account_type"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:smallIcon="@drawable/ic_launcher" />
sync_adapter.xml
<?xml version="1.0" encoding="utf-8"?>
<sync-adapter xmlns:android="http://schemas.android.com/apk/res/android"
android:contentAuthority="@string/content_authority"
android:accountType="@string/account_type"
android:userVisible="true"
android:allowParallelSyncs="false"
android:isAlwaysSyncable="true"
android:supportsUploading="true"/>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" android:versionCode="1" android:versionName="1.0.0" package="com.example.awsomeapp">
<uses-permission android:name="android.permission.GET_ACCOUNTS"/><!-- SyncAdapter and GCM requires a Google account. -->
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS"/>
<uses-permission android:name="android.permission.USE_CREDENTIALS"/>
<!-- GCM Creates a custom permission so only this app can receive its messages. -->
<permission android:name="${applicationId}.permission.C2D_MESSAGE" android:protectionLevel="signature"/>
<uses-permission android:name="${applicationId}.permission.C2D_MESSAGE"/>
<application....
.......
<!-- Stub Authenticator -->
<service
android:name="com.example.awsomeapp.service.authenticator.CAuthenticatorService"
android:exported="true">
<intent-filter>
<action android:name="android.accounts.AccountAuthenticator"/>
</intent-filter>
<meta-data android:name="android.accounts.AccountAuthenticator" android:resource="@xml/authenticator"/>
</service>
<!-- -->
<!-- Sync Adapter -->
<service
android:name="com.example.awsomeapp.service.sync.CSyncService"
android:exported="true"
android:process=":sync">
<intent-filter>
<action android:name="android.content.SyncAdapter"/>
</intent-filter>
<meta-data android:name="android.content.SyncAdapter" android:resource="@xml/sync_adapter" />
</service>
<!-- -->
<!-- Content Provider -->
<provider android:authorities="${valueContentAuthority}"
android:exported="false"
android:name="com.example.awsomeapp.database.contentprovider.CProvider">
</provider>
<!-- -->
</application>
</manifest>
Код:
public static final String CONTENT_AUTHORITY = BuildConfig.CONTENT_AUTHORITY;
public static final String ACCOUNT_TYPE = BuildConfig.ACCOUNT_TYPE;
Основываясь на примере @ChristianMelchior, вот мое решение, которое устраняет две проблемы в предыдущих решениях:
Решения, которые изменяют values.xml в каталоге компоновки, приводят к полной перестройке ресурсов (в том числе по всем чертежам)
по неизвестной причине IntelliJ (и, вероятно, Android Studio) не надежно обрабатывают ресурсы, в результате чего сборка содержит незамещенные
.res-auto
органы власти провайдера
Это новое решение делает больше, чем Gradle, создавая новую задачу, и допускает инкрементные сборки, определяя входные и выходные файлы.
создать файл (в примере я положил его в
variants
каталог), отформатированный как файл ресурсов XML, который содержит строковые ресурсы. Они будут объединены с ресурсами приложения, и любое возникновение.res-auto
в значениях будет заменено имя пакета варианта, например<string name="search_provider">.res-auto.MySearchProvider</string>
добавить
build_extras.gradle
файл из этой сути вашего проекта и ссылки на него из основногоbuild.gradle
добавляяapply from: './build_extras.gradle'
где-то надandroid
блокубедитесь, что вы установили имя пакета по умолчанию, добавив его в
android.defaultConfig
блок изbuild.gradle
в
AndroidManifest.xml
и другие файлы конфигурации (такие какxml/searchable.xml
для поставщиков поиска автозаполнения), обратитесь к поставщику (например,@string/search_provider
)если вам нужно получить одно и то же имя, вы можете использовать
BuildConfig.PACKAGE_NAME
переменная, напримерBuildConfig.PACKAGE_NAME + ".MySearchProvider"
https://gist.github.com/paour/9189462
Обновление: этот метод работает только на Android 2.2.1 и выше. Для более ранних платформ посмотрите этот ответ, у которого есть свой собственный набор проблем, так как новое объединение манифеста все еще очень грубо по краям...
Я написал блог-пост с примером проекта Github, который решает эту проблему (и другие подобные проблемы) немного иначе, чем Сирил.
http://brad-android.blogspot.com/2013/08/android-gradle-building-unique-build.html
К сожалению, текущая версия (0.4.1) плагина для Android, похоже, не является хорошим решением для этого. У меня еще не было времени попробовать это, но возможное решение этой проблемы - использование строкового ресурса. @string/provider_authority
и используйте это в манифесте: android:authority="@string/provider_authority"
, Затем у вас есть res/values/provider.xml
в папке res каждого типа сборки, который должен переопределить полномочия, в вашем случае это будет src/debug/res
Я посмотрел на создание XML-файла на лету, но опять же, похоже, что в текущей версии плагина нет хороших зацепок для него. Я бы порекомендовал добавить запрос, но я могу представить, что больше людей столкнется с этой же проблемой.
Ответ в этом посте у меня работает.
Я использую 3 разных варианта, поэтому я создаю 3 манифеста с поставщиком контента в каждом варианте, как сказал Кевинршульц:
productFlavors {
free {
packageName "your.package.name.free"
}
paid {
packageName "your.package.name.paid"
}
other {
packageName "your.package.name.other"
}
}
Ваш основной манифест не включает поставщиков:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" >
<!-- Permissions -->
<application>
<!-- Nothing about Content Providers at all -->
<!-- Activities -->
...
<!-- Services -->
...
</application>
И ваш манифест в каждом вашем вкусе, включая поставщика.
Свободно:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" >
<application>
<!-- Content Providers -->
<provider
android:name="your.package.name.Provider"
android:authorities="your.package.name.free"
android:exported="false" >
</provider>
</application>
</manifest>
Выплачено:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" >
<application>
<!-- Content Providers -->
<provider
android:name="your.package.name.Provider"
android:authorities="your.package.name.paid"
android:exported="false" >
</provider>
</application>
</manifest>
Другой:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" >
<application>
<!-- Content Providers -->
<provider
android:name="your.package.name.Provider"
android:authorities="your.package.name.other"
android:exported="false" >
</provider>
</application>
</manifest>
Мое решение заключается в использовании замены заполнителя в AndroidManifest.xml
, Это также обрабатывает packageNameSuffix
атрибуты, чтобы вы могли иметь debug
а также release
а также любые другие пользовательские сборки на том же устройстве.
applicationVariants.all { variant ->
def flavor = variant.productFlavors.get(0)
def buildType = variant.buildType
variant.processManifest.doLast {
println '################# Adding Package Names to Manifest #######################'
replaceInManifest(variant,
'PACKAGE_NAME',
[flavor.packageName, buildType.packageNameSuffix].findAll().join()) // ignores null
}
}
def replaceInManifest(variant, fromString, toString) {
def flavor = variant.productFlavors.get(0)
def buildtype = variant.buildType
def manifestFile = "$buildDir/manifests/${flavor.name}/${buildtype.name}/AndroidManifest.xml"
def updatedContent = new File(manifestFile).getText('UTF-8').replaceAll(fromString, toString)
new File(manifestFile).write(updatedContent, 'UTF-8')
}
У меня это на gist
тоже, если вы хотите увидеть, развивается ли это позже.
Я обнаружил, что это более элегантный подход, чем подходы с несколькими ресурсами и анализом XML.
Почему бы просто не добавить это?
type.packageNameSuffix = ". $ type.name"