how to reference an asset in a library project
Asked Answered
S

7

54

In a class belonging to a Library project I call:

webview.loadUrl("file:///android_asset/info.html", null);

Unfortunately, this only works if I duplicate the file info.html into the Application's project asset folder as well.

Is there a way to tell an Android library code: "look for this file in the library's assets folder, not in the application's assets folder" ?

Swept answered 14/6, 2011 at 16:32 Comment(1)
Three years later, this is finally possible. See my answer: https://mcmap.net/q/335152/-how-to-reference-an-asset-in-a-library-projectDemagnetize
H
80

This answer is out of date, the gradle build system and AAR files support assets.


From the Android Docs:

Library projects cannot include raw assets

The tools do not support the use of raw asset files (saved in the assets/ directory) in a library project. Any asset resources used by an application must be stored in the assets/ directory of the application project itself. However, resource files saved in the res/ directory are supported.

If you want to include files from a Library project, you'll need to put it in the resources instead of the assets. If you're trying to load HTML files from your library project into a WebView, this means that you'll have to go a more roundabout method than the usual asset URL. Instead you'll have to read the resource data and use something like loadData.

Hegyera answered 14/6, 2011 at 16:50 Comment(4)
The problem with loadData is, that images and other page -element referenced in the loaded html can not be loaded without a lot of hacking.Stack
@Hegyera Google explanation is not clear to me. I have a project that has a library dependency. I have a path assets/dir/index.html, js, css files in this library project and this is loading from my application. I am using IntelliJ and there is an option to tell compiler to "Include assets from dependencies into APK". While compiling, it merges both assets. This is a common practice that developers do. Make sure that you have distinct directory and files names that won't conflict with application's files.Idealist
@Idealist It's entirely possible that this is a feature that is included in IntelliJ. I don't use IntelliJ or Android Studio regularly, so I can't say much on that front. I was purely speaking from experience (2.5 years ago to boot) using Eclipse and Ant as well as noting Google's own documentation. I would imagine IntelliJ is just doing exactly what OP described: copying the files over, albeit only in the build process.Hegyera
Thanks to update the answer. Now the gradle build system allow us to add assets into Android libraries, is there anyway to access them directly from the Android library or they just get merged into the app module and we access them separately?Circumrotate
D
24

This is now possible using the Gradle build system.

Testing with Android Studio 0.5.0 and v0.9 of the Android Gradle plugin, I've found that files such as

MyLibProject/src/main/assets/test.html

are correctly packaged in the final application and can be accessed at runtime via the expected URL:

file:///android_asset/test.html
Demagnetize answered 7/3, 2014 at 15:44 Comment(7)
You're putting a folder called assets in the src directory, which isn't standard. Assets folders go in the root of the eclipse project / intellij moduleMetro
@StealthRabbi Wrong. See #18303103Demagnetize
This code does not work. Got exception "no such file" from url.openStream. URL url = new URL("file:///android_asset/test.html"); BufferedReader in = new BufferedReader(new InputStreamReader(url.openStream()));Felid
I am building library module. Not getting how to open existing database in library module assets.Its the problem with path.Earlier in MainApplication i used to access it through applicationContext.getAssets().open(dbName) but now i don't know proper way to implement the same in library modulePoesy
@VeereshP, I build a library and want to access to asset files via getAssets() method but it's not working in library are you find any solution for this?Battleship
@saeedmasoomi , I resolved path issue like this 'DB_PATH = Environment.getDataDirectory() + "/data/" + this.myContext.getPackageName() + "/databases/";'Poesy
@VeereshP,Thanks :)Battleship
L
14

You can achieve this by creating a symbolic link in the project's asset folder that points to the directory in the library project.

Then you can access as below:

webview.loadUrl("file:///android_asset/folder_in_a_libary_project/info.html", null);
Laundress answered 10/9, 2012 at 7:1 Comment(2)
This is the most useful answer here!Bearnard
+1 for a real answer :D in eclipse , just drag the assets folder from the explorer/finder/whatever into the project and chose to link it .. fastest way to do itFulvous
D
12

Okay. Ive been stressing out and losing sleep about this for a while. Im the type of person that loves API creation, and HATES complicated integration.

There arent many solutions around on the internet, so im quite proud of what Ive discovered with a bit of Eclipse Hackery.

It turns out that when you put a file in the Android Lib's /assets folder. The target apk will capture this and place it on the root of the APK archive. Thus, making general access fail. This can be resolved by simply creating a Raw Java Library, and placing all assets in there, ie (JAVALIB)/assets/fileX.txt.

You can in turn then include this as a Java Build Path Folder Source in Project > Properties > Java Build Path > Source > Link Source.

  1. Link Source
  2. Click on Variables. and Add New Variable, ie VAR_NAME_X. location : ../../(relative_path_to_assets_project)
  3. Click Ok

Now, when you build and run your app, the assets folder in the APK will contain your (GLOBAL Library) files as you intended.

No need to reconfigure android internals or nothing. Its all capable within a few clicks of Eclipse.

Donatelli answered 26/4, 2012 at 9:54 Comment(3)
+1 for a great workaround, but what do you mean by "a Raw Java Library"? How do you create one on Eclipse? Thanks!Swept
You should write a proof of concept. I am doubtful of whether this actually works in the way someone familiar with assets in an android project would expect it to. For example, can the resultant assets be referenced with the "file:///android_asset" url?Piggott
This worked for me, @TomDignan, so I posted a sample project that uses a linked resource for its assets folder.Cacus
K
4

I confirm that Daniel Grant's approach works for at least the following situation: target project does NOT have an asset folder (or the folder is empty, so you can safely delete it). I did not setup any variable. Simply setup a LinkSource as follows (just an example) Linked folder location: /home/matthew/workspace_moonblink/assetsForAdvocacy/assets Folder name : assets

The "assetsForAdvocacy" is a Java project, (created with New- Project - Java Project) with empty src folder, and a new folder named "assets", which now provides the entire assets folder for the target project.

This is a fairly straightforward way within Eclipse to provide assets re-use across many different projects IF they do not already have assets, good enough to get going with. I would probably want to enhance it to become a content provider in the long run, but that is a lot more development.

My project accesses the assets with the following code: String advocacyFolderInAssets = "no_smoking/"; //a folder underneath assets/ String fn =advocacyFolderInAssets+imageFilename; Bitmap pristineBitmapForAdvocacy = getBitmapFromAsset(context, fn);

I use Motodev Studio 3.1.0 on Ubuntu. It would not let me 'merge' a new assets folder in the new assets-only project onto an existing assets folder in the target project.

Kevel answered 28/6, 2012 at 18:24 Comment(0)
T
1

If you want to use a setup where multiple derivate products are created from one library you might consider using svn:externals or similar solution in your SCM system. This will also do the trick that static assets like online help may be versioned seperately from the android source code.

Templet answered 26/5, 2013 at 14:45 Comment(0)
E
0

I found this older question, it might help you, too.

This is the official way Google uses to archive this (from the above post): Link

Elbrus answered 14/6, 2011 at 16:39 Comment(3)
Thanks. Unfortunately, that thread doesn't explain how to reference an asset in a library project.Swept
Yes, it does... even with pictures: LinkElbrus
No it doesn't. Only how to reference a library project, not an asset in the library project.Ressieressler

© 2022 - 2024 — McMap. All rights reserved.