Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Pack all resources to assets on the android target. #4965

Merged
merged 13 commits into from
Jun 20, 2024
Merged

Conversation

terrakok
Copy link
Member

@terrakok terrakok commented Jun 13, 2024

The PR changes the android resources packaging. Now all resources are packed to the android assets (not only fonts). It unblocks usage android URIs to the resources in a WebView or other external resource consumers.
Additionally the PR fixes Android Studio Compose Previews work with multiplatform resources:

For a backward compatibility the resources library tries to read resources in java resources if assets were not found.

Fixes #4877
Fixes #4503
Fixes #4932
Fixes #4476
Fixes #3828

Release Notes

Highlights - Resources

  • Android Studio Preview works with Compose Multiplatform resources now
  • Compose Multiplatform resources are stored in the android assets now. This fixes such cases as a rendering resource files in WebViews or Media Players

@terrakok terrakok requested a review from igordmn June 13, 2024 15:37
@sergeshustoff
Copy link

I might be wrong here, but it looks like it might lead to fixing android previews with multiplatform resources (#4932, #4476) eventually... If CopyResourcesToAndroidAssetsTask is run before whatewer Android Studio does to render previews and AndroidContextProvider.ANDROID_CONTEXT is set somehow from the preview (or ContextLocal is used instead)

@terrakok
Copy link
Member Author

terrakok commented Jun 17, 2024

I checked a preview with the new packaging but it doesn't work because android assets is semi available in the preview mode. It works strange for a regular android project too:

  1. Assets are not available unless you run the project first time.
  2. Assets are only available in the same module where your preview is.

UPDATE: the problem is addGeneratedSourceDirectory function doesn't work with AS Compose Preview :feelsgood:

ISSUE: https://issuetracker.google.com/348208777

@terrakok
Copy link
Member Author

Now Android Studio Compose Preview works :godmode:

image

@terrakok terrakok force-pushed the k.tskh/android-ass branch from 6140e1b to 3d6788a Compare June 19, 2024 21:47
@AhmedTawfek
Copy link

AhmedTawfek commented Jun 24, 2024

Now Android Studio Compose Preview works :godmode:

image

Sorry for the question, but Iam currently having these versions in my compose multiplaform project, what versions do I need to use in case of solving the preview issues in AS ?

compose-plugin = "1.6.11" androidx-compose = "1.6.8" kotlin = "1.9.23" jetbrainsCompose = { id = "org.jetbrains.compose", version.ref = "compose-plugin" } androidx-ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling", version.ref = "androidx-compose" } androidx-ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview", version.ref = "androidx-compose" }

@PaoloMazza1995
Copy link

@terrakok Do you have any update about this? What step we have to follow to solve the issue? Thanks

@rschattauer
Copy link

@joreilly
Copy link

joreilly commented Jul 3, 2024

Should this have also resolved https://issuetracker.google.com/issues/345275143? I did quick test and it didn't seem to but I might have missed something.

@kaidotarma
Copy link

Demo ImageResPreview doesn't seem to work and gives error:
java.lang.NoClassDefFoundError: org/xml/sax/InputSource
at org.jetbrains.compose.resources.ImageResources_jvmAndAndroidKt.toXmlElement(ImageResources.jvmAndAndroid.kt:13)
at org.jetbrains.compose.resources.ImageResourcesKt$vectorResource$imageVector$3$cached$1.invoke(ImageResources.kt:86)
at org.jetbrains.compose.resources.ImageResourcesKt$vectorResource$imageVector$3$cached$1.invoke(ImageResources.kt:85)
at org.jetbrains.compose.resources.ImageResourcesKt$loadImage$2.invokeSuspend(ImageResources.kt:150)
at org.jetbrains.compose.resources.ImageResourcesKt$loadImage$2.invoke(ImageResources.kt)
at org.jetbrains.compose.resources.ImageResourcesKt$loadImage$2.invoke(ImageResources.kt)
at org.jetbrains.compose.resources.AsyncCache$getOrLoad$2$deferred$1$1.invokeSuspend(AsyncCache.kt:19)
at layoutlib.internal.kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
at layoutlib.internal.kotlinx.coroutines.DispatchedTask.run(DispatchedTask.kt:104)
at layoutlib.internal.kotlinx.coroutines.EventLoopImplBase.processNextEvent(EventLoop.common.kt:277)
at layoutlib.internal.kotlinx.coroutines.BlockingCoroutine.joinBlocking(Builders.kt:95)
at layoutlib.internal.kotlinx.coroutines.BuildersKt__BuildersKt.runBlocking(Builders.kt:69)
at layoutlib.internal.kotlinx.coroutines.BuildersKt.runBlocking(Unknown Source)
at layoutlib.internal.kotlinx.coroutines.BuildersKt__BuildersKt.runBlocking$default(Builders.kt:48)
at layoutlib.internal.kotlinx.coroutines.BuildersKt.runBlocking$default(Unknown Source)
at org.jetbrains.compose.resources.ResourceState_blockingKt.rememberResourceState(ResourceState.blocking.kt:46)
at org.jetbrains.compose.resources.ImageResourcesKt.vectorResource(ImageResources.kt:83)
at org.jetbrains.compose.resources.ImageResourcesKt.painterResource(ImageResources.kt:40)
at org.jetbrains.compose.resources.demo.shared.ComposableSingletons$ImagesResKt$lambda-2$1.invoke(ImagesRes.kt:53)
at org.jetbrains.compose.resources.demo.shared.ComposableSingletons$ImagesResKt$lambda-2$1.invoke(ImagesRes.kt:46)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:118)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.material3.CardKt$Card$1.invoke(Card.kt:886)
at androidx.compose.material3.CardKt$Card$1.invoke(Card.kt:93)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.material3.SurfaceKt$Surface$1.invoke(Surface.kt:134)
at androidx.compose.material3.SurfaceKt$Surface$1.invoke(Surface.kt:115)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:228)
at androidx.compose.material3.SurfaceKt.Surface-T9BRK9s(Surface.kt:112)
at androidx.compose.material3.CardKt.Card(Card.kt:85)
at androidx.compose.material3.CardKt.OutlinedCard(Card.kt:284)
at org.jetbrains.compose.resources.demo.shared.ImagesResKt.ImagesRes(ImagesRes.kt:46)
at org.jetbrains.compose.resources.demo.shared.Main_androidKt.ImagesResPreview(main.android.kt:22)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:77)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:568)
at androidx.compose.ui.tooling.ComposableInvoker.invokeComposableMethod(ComposableInvoker.jvm.kt:181)
at androidx.compose.ui.tooling.ComposableInvoker.invokeComposable(ComposableInvoker.jvm.kt:221)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3$1$composable$1.invoke(ComposeViewAdapter.android.kt:504)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3$1$composable$1.invoke(ComposeViewAdapter.android.kt:502)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3$1.invoke(ComposeViewAdapter.android.kt:539)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3$1.invoke(ComposeViewAdapter.android.kt:497)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:228)
at androidx.compose.ui.tooling.InspectableKt.Inspectable(Inspectable.android.kt:61)
at androidx.compose.ui.tooling.ComposeViewAdapter$WrapPreview$1.invoke(ComposeViewAdapter.android.kt:444)
at androidx.compose.ui.tooling.ComposeViewAdapter$WrapPreview$1.invoke(ComposeViewAdapter.android.kt:443)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:228)
at androidx.compose.ui.tooling.ComposeViewAdapter.WrapPreview(ComposeViewAdapter.android.kt:438)
at androidx.compose.ui.tooling.ComposeViewAdapter.access$WrapPreview(ComposeViewAdapter.android.kt:124)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3.invoke(ComposeViewAdapter.android.kt:497)
at androidx.compose.ui.tooling.ComposeViewAdapter$init$3.invoke(ComposeViewAdapter.android.kt:494)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.ui.platform.ComposeView.Content(ComposeView.android.kt:428)
at androidx.compose.ui.platform.AbstractComposeView$ensureCompositionCreated$1.invoke(ComposeView.android.kt:252)
at androidx.compose.ui.platform.AbstractComposeView$ensureCompositionCreated$1.invoke(ComposeView.android.kt:251)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:228)
at androidx.compose.ui.platform.CompositionLocalsKt.ProvideCommonCompositionLocals(CompositionLocals.kt:186)
at androidx.compose.ui.platform.AndroidCompositionLocals_androidKt$ProvideAndroidCompositionLocals$3.invoke(AndroidCompositionLocals.android.kt:119)
at androidx.compose.ui.platform.AndroidCompositionLocals_androidKt$ProvideAndroidCompositionLocals$3.invoke(AndroidCompositionLocals.android.kt:118)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:228)
at androidx.compose.ui.platform.AndroidCompositionLocals_androidKt.ProvideAndroidCompositionLocals(AndroidCompositionLocals.android.kt:110)
at androidx.compose.ui.platform.WrappedComposition$setContent$1$1$2.invoke(Wrapper.android.kt:139)
at androidx.compose.ui.platform.WrappedComposition$setContent$1$1$2.invoke(Wrapper.android.kt:138)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.CompositionLocalKt.CompositionLocalProvider(CompositionLocal.kt:248)
at androidx.compose.ui.platform.WrappedComposition$setContent$1$1.invoke(Wrapper.android.kt:138)
at androidx.compose.ui.platform.WrappedComposition$setContent$1$1.invoke(Wrapper.android.kt:123)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:109)
at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:35)
at androidx.compose.runtime.ActualJvm_jvmKt.invokeComposable(ActualJvm.jvm.kt:90)
at androidx.compose.runtime.ComposerImpl.doCompose(Composer.kt:3302)
at androidx.compose.runtime.ComposerImpl.composeContent$runtime_release(Composer.kt:3235)
at androidx.compose.runtime.CompositionImpl.composeContent(Composition.kt:725)
at androidx.compose.runtime.Recomposer.composeInitial$runtime_release(Recomposer.kt:1071)
at androidx.compose.runtime.CompositionImpl.composeInitial(Composition.kt:633)
at androidx.compose.runtime.CompositionImpl.setContent(Composition.kt:619)
at androidx.compose.ui.platform.WrappedComposition$setContent$1.invoke(Wrapper.android.kt:123)
at androidx.compose.ui.platform.WrappedComposition$setContent$1.invoke(Wrapper.android.kt:114)
at androidx.compose.ui.platform.AndroidComposeView.setOnViewTreeOwnersAvailable(AndroidComposeView.android.kt:1289)
at androidx.compose.ui.platform.WrappedComposition.setContent(Wrapper.android.kt:114)
at androidx.compose.ui.platform.WrappedComposition.onStateChanged(Wrapper.android.kt:164)
at androidx.lifecycle.LifecycleRegistry$ObserverWithState.dispatchEvent(LifecycleRegistry.kt:314)
at androidx.lifecycle.LifecycleRegistry.addObserver(LifecycleRegistry.kt:192)
at androidx.compose.ui.platform.WrappedComposition$setContent$1.invoke(Wrapper.android.kt:121)
at androidx.compose.ui.platform.WrappedComposition$setContent$1.invoke(Wrapper.android.kt:114)
at androidx.compose.ui.platform.AndroidComposeView.onAttachedToWindow(AndroidComposeView.android.kt:1364)
at android.view.View.dispatchAttachedToWindow(View.java:21980)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3490)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3497)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3497)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3497)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3497)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:3497)
at android.view.AttachInfo_Accessor.setAttachInfo(AttachInfo_Accessor.java:44)
at com.android.layoutlib.bridge.impl.RenderSessionImpl.inflate(RenderSessionImpl.java:372)
at com.android.layoutlib.bridge.Bridge.createSession(Bridge.java:454)
at com.android.tools.idea.layoutlib.LayoutLibrary.createSession(LayoutLibrary.java:120)
at com.android.tools.rendering.RenderTask.createRenderSession(RenderTask.java:784)
at com.android.tools.rendering.RenderTask.lambda$inflate$6(RenderTask.java:934)
at com.android.tools.rendering.RenderExecutor$runAsyncActionWithTimeout$3.run(RenderExecutor.kt:202)
at com.android.tools.rendering.RenderExecutor$PriorityRunnable.run(RenderExecutor.kt:316)
at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1136)
at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:635)
at java.base/java.lang.Thread.run(Thread.java:840)
Caused by: java.lang.ClassNotFoundException: org.xml.sax.InputSource
at com.android.tools.rendering.classloading.loaders.DelegatingClassLoader.findClass(DelegatingClassLoader.kt:76)
at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:592)
at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:525)
at com.android.tools.rendering.classloading.loaders.DelegatingClassLoader.loadClass(DelegatingClassLoader.kt:62)
... 118 more

@terrakok
Copy link
Member Author

terrakok commented Jul 5, 2024

I don't see the problem. Please, file a new issue with reproducer steps and a full build log.
image

@kaidotarma
Copy link

Found the root cause:
I was using an old version of Android Studio (Iguana), and after updating (to Koala) everything seems to work.
Well done, guys! =)

@SirFilbido
Copy link

Hello everyone, is this feature only available for projects that have a shared folder? I tested it on my project that uses composeApp/commonMain and it didn't work.

I'm using Android Studio Koala version 2024.1.1

@Varun-Sethi-Dev
Copy link

Varun-Sethi-Dev commented Jul 12, 2024

I am trying to get the preview feature work on my android studio but i am not even being able to see the icons for preview
https://drive.google.com/file/d/1-tsYdfsNlegyo8oYGUndO6rsYbDLVAlX/view?usp=sharing

@terrakok
Copy link
Member Author

Android Studio preview works in the android source set only

@Varun-Sethi-Dev
Copy link

Android Studio preview works in the android source set only

Okay so can you guide me on how to set that up or provide any guide to it

@terrakok
Copy link
Member Author

terrakok commented Jul 12, 2024

just wait the official release 1.7.0 with proper documentation
(or take a look at the library demo app https://github.com/JetBrains/compose-multiplatform/tree/master/components/resources/demo)

@NaiRo94s
Copy link

NaiRo94s commented Aug 1, 2024

@Varun-Sethi-Dev You can reference any composable from commonMain in your androidMain sourceSet. Just make sure to use the following dependencies in your androidMain and to set buildFeatures.compose = true.

androidx-ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling", version.ref = "androidx-compose" }
androidx-ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview", version.ref = "androidx-compose" }

Note: I am using Android Studio Ladybug | 2024.1.3 Canary 3

@robertjamison
Copy link

This is probably user error, but...

Lets say I save an MP3 in my shared Compose Resources:

composeApp/src/commonMain/composeResources/files/click.mp3

and then I get the url:

Res.getUri("files/test.mp3")

And the result is so:

/android_asset/composeResources/com.example.test/files/click.mp3

Is that an intended behavior? I ask because I'm not able to find files there.

@terrakok
Copy link
Member Author

terrakok commented Sep 9, 2024

File an issue on the youtrack, please

@JetBrains JetBrains locked and limited conversation to collaborators Sep 9, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.