Fix loading placeholder image for artist
This commit is contained in:
parent
6966351113
commit
b41ef5bff5
8 changed files with 96 additions and 115 deletions
|
@ -141,8 +141,6 @@ dependencies {
|
||||||
implementation "com.squareup.retrofit2:retrofit:$retrofit_version"
|
implementation "com.squareup.retrofit2:retrofit:$retrofit_version"
|
||||||
implementation "com.squareup.retrofit2:converter-gson:$retrofit_version"
|
implementation "com.squareup.retrofit2:converter-gson:$retrofit_version"
|
||||||
|
|
||||||
implementation 'com.jakewharton.retrofit:retrofit2-kotlin-coroutines-adapter:0.9.2'
|
|
||||||
|
|
||||||
def material_dialog_version = "3.1.1"
|
def material_dialog_version = "3.1.1"
|
||||||
implementation "com.afollestad.material-dialogs:core:$material_dialog_version"
|
implementation "com.afollestad.material-dialogs:core:$material_dialog_version"
|
||||||
implementation "com.afollestad.material-dialogs:input:$material_dialog_version"
|
implementation "com.afollestad.material-dialogs:input:$material_dialog_version"
|
||||||
|
|
|
@ -215,9 +215,7 @@ public class MainActivity extends AbsSlidingMusicPanelActivity
|
||||||
menu.add(0, R.id.action_search, 0, getString(R.string.action_search))
|
menu.add(0, R.id.action_search, 0, getString(R.string.action_search))
|
||||||
.setIcon(R.drawable.ic_mic_white_24dp)
|
.setIcon(R.drawable.ic_mic_white_24dp)
|
||||||
.setShowAsActionFlags(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
.setShowAsActionFlags(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
||||||
menu.add(0, R.id.action_settings, 0, getString(R.string.action_settings))
|
|
||||||
.setIcon(R.drawable.ic_settings_white_24dp)
|
|
||||||
.setShowAsActionFlags(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
|
||||||
}
|
}
|
||||||
if (isFolderPage()) {
|
if (isFolderPage()) {
|
||||||
menu.add(0, R.id.action_scan, 0, R.string.scan_media)
|
menu.add(0, R.id.action_scan, 0, R.string.scan_media)
|
||||||
|
@ -249,6 +247,9 @@ public class MainActivity extends AbsSlidingMusicPanelActivity
|
||||||
menu.removeItem(R.id.action_grid_size);
|
menu.removeItem(R.id.action_grid_size);
|
||||||
menu.removeItem(R.id.action_sort_order);
|
menu.removeItem(R.id.action_sort_order);
|
||||||
}
|
}
|
||||||
|
menu.add(0, R.id.action_settings, 0, getString(R.string.action_settings))
|
||||||
|
.setIcon(R.drawable.ic_settings_white_24dp)
|
||||||
|
.setShowAsActionFlags(MenuItem.SHOW_AS_ACTION_IF_ROOM);
|
||||||
return super.onCreateOptionsMenu(menu);
|
return super.onCreateOptionsMenu(menu);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,6 @@ import code.name.monkey.retromusic.extensions.appHandleColor
|
||||||
import code.name.monkey.retromusic.glide.palette.BitmapPaletteTranscoder
|
import code.name.monkey.retromusic.glide.palette.BitmapPaletteTranscoder
|
||||||
import code.name.monkey.retromusic.glide.palette.BitmapPaletteWrapper
|
import code.name.monkey.retromusic.glide.palette.BitmapPaletteWrapper
|
||||||
import code.name.monkey.retromusic.loaders.AlbumLoader
|
import code.name.monkey.retromusic.loaders.AlbumLoader
|
||||||
import code.name.monkey.retromusic.rest.LastFMRestClient
|
|
||||||
import code.name.monkey.retromusic.util.ImageUtil
|
import code.name.monkey.retromusic.util.ImageUtil
|
||||||
import code.name.monkey.retromusic.util.RetroColorUtil.generatePalette
|
import code.name.monkey.retromusic.util.RetroColorUtil.generatePalette
|
||||||
import code.name.monkey.retromusic.util.RetroColorUtil.getColor
|
import code.name.monkey.retromusic.util.RetroColorUtil.getColor
|
||||||
|
@ -82,7 +81,6 @@ class AlbumTagEditorActivity : AbsTagEditorActivity(), TextWatcher {
|
||||||
|
|
||||||
private var albumArtBitmap: Bitmap? = null
|
private var albumArtBitmap: Bitmap? = null
|
||||||
private var deleteAlbumArt: Boolean = false
|
private var deleteAlbumArt: Boolean = false
|
||||||
private var lastFMRestClient: LastFMRestClient? = null
|
|
||||||
|
|
||||||
private fun setupToolbar() {
|
private fun setupToolbar() {
|
||||||
toolbar.setBackgroundColor(ATHUtil.resolveColor(this, R.attr.colorSurface))
|
toolbar.setBackgroundColor(ATHUtil.resolveColor(this, R.attr.colorSurface))
|
||||||
|
@ -93,7 +91,6 @@ class AlbumTagEditorActivity : AbsTagEditorActivity(), TextWatcher {
|
||||||
setDrawUnderStatusBar()
|
setDrawUnderStatusBar()
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
window.sharedElementsUseOverlay = true
|
window.sharedElementsUseOverlay = true
|
||||||
lastFMRestClient = LastFMRestClient(this)
|
|
||||||
imageContainer?.transitionName = "${getString(R.string.transition_album_art)}_$id"
|
imageContainer?.transitionName = "${getString(R.string.transition_album_art)}_$id"
|
||||||
windowEnterTransition()
|
windowEnterTransition()
|
||||||
setUpViews()
|
setUpViews()
|
||||||
|
|
|
@ -27,7 +27,9 @@ import com.bumptech.glide.load.model.GlideUrl
|
||||||
import com.bumptech.glide.load.model.ModelLoader
|
import com.bumptech.glide.load.model.ModelLoader
|
||||||
import com.bumptech.glide.load.model.ModelLoaderFactory
|
import com.bumptech.glide.load.model.ModelLoaderFactory
|
||||||
import com.bumptech.glide.load.model.stream.StreamModelLoader
|
import com.bumptech.glide.load.model.stream.StreamModelLoader
|
||||||
|
import okhttp3.Interceptor
|
||||||
import okhttp3.OkHttpClient
|
import okhttp3.OkHttpClient
|
||||||
|
import okhttp3.logging.HttpLoggingInterceptor
|
||||||
import java.io.IOException
|
import java.io.IOException
|
||||||
import java.io.InputStream
|
import java.io.InputStream
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
|
@ -76,9 +78,15 @@ class ArtistImageFetcher(
|
||||||
return try {
|
return try {
|
||||||
val deezerResponse = response.body();
|
val deezerResponse = response.body();
|
||||||
val imageUrl = deezerResponse?.data?.get(0)?.let { getHighestQuality(it) }
|
val imageUrl = deezerResponse?.data?.get(0)?.let { getHighestQuality(it) }
|
||||||
val glideUrl = GlideUrl(imageUrl)
|
// Fragile way to detect a place holder image returned from Deezer:
|
||||||
urlFetcher = urlLoader.getResourceFetcher(glideUrl, width, height)
|
// ex: "https://e-cdns-images.dzcdn.net/images/artist//250x250-000000-80-0-0.jpg"
|
||||||
urlFetcher?.loadData(priority)
|
// the double slash implies no artist identified
|
||||||
|
val placeHolder = imageUrl?.contains("/images/artist//") ?: false
|
||||||
|
if (!placeHolder) {
|
||||||
|
val glideUrl = GlideUrl(imageUrl)
|
||||||
|
urlFetcher = urlLoader.getResourceFetcher(glideUrl, width, height)
|
||||||
|
urlFetcher?.loadData(priority)
|
||||||
|
} else null
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
|
@ -132,10 +140,17 @@ class Factory(
|
||||||
.connectTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
.connectTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
||||||
.readTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
.readTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
||||||
.writeTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
.writeTimeout(TIMEOUT, TimeUnit.MILLISECONDS)
|
||||||
|
.addInterceptor(createLogInterceptor())
|
||||||
.build()
|
.build()
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun createLogInterceptor(): Interceptor {
|
||||||
|
val interceptor = HttpLoggingInterceptor()
|
||||||
|
interceptor.level = HttpLoggingInterceptor.Level.BODY
|
||||||
|
return interceptor
|
||||||
|
}
|
||||||
|
|
||||||
override fun build(
|
override fun build(
|
||||||
context: Context?,
|
context: Context?,
|
||||||
factories: GenericLoaderFactory?
|
factories: GenericLoaderFactory?
|
||||||
|
|
|
@ -23,7 +23,7 @@ import code.name.monkey.retromusic.adapter.HomeAdapter
|
||||||
import code.name.monkey.retromusic.loaders.*
|
import code.name.monkey.retromusic.loaders.*
|
||||||
import code.name.monkey.retromusic.model.*
|
import code.name.monkey.retromusic.model.*
|
||||||
import code.name.monkey.retromusic.providers.interfaces.Repository
|
import code.name.monkey.retromusic.providers.interfaces.Repository
|
||||||
import code.name.monkey.retromusic.rest.LastFMRestClient
|
import code.name.monkey.retromusic.rest.LastFmClient
|
||||||
import code.name.monkey.retromusic.rest.model.LastFmAlbum
|
import code.name.monkey.retromusic.rest.model.LastFmAlbum
|
||||||
import code.name.monkey.retromusic.rest.model.LastFmArtist
|
import code.name.monkey.retromusic.rest.model.LastFmArtist
|
||||||
import java.io.IOException
|
import java.io.IOException
|
||||||
|
@ -259,7 +259,7 @@ class RepositoryImpl @Inject constructor(private val context: Context) : Reposit
|
||||||
cache: String?
|
cache: String?
|
||||||
): Result<LastFmArtist> = safeApiCall(
|
): Result<LastFmArtist> = safeApiCall(
|
||||||
call = {
|
call = {
|
||||||
Success(LastFMRestClient(context).apiService.artistInfo(name, lang, cache))
|
Success(LastFmClient.getApiService().artistInfo(name, lang, cache))
|
||||||
},
|
},
|
||||||
errorMessage = "Error"
|
errorMessage = "Error"
|
||||||
|
|
||||||
|
@ -270,7 +270,7 @@ class RepositoryImpl @Inject constructor(private val context: Context) : Reposit
|
||||||
album: String
|
album: String
|
||||||
): Result<LastFmAlbum> = safeApiCall(
|
): Result<LastFmAlbum> = safeApiCall(
|
||||||
call = {
|
call = {
|
||||||
Success(LastFMRestClient(context).apiService.albumInfo(artist, album))
|
Success(LastFmClient.getApiService().albumInfo(artist, album))
|
||||||
},
|
},
|
||||||
errorMessage = "Error"
|
errorMessage = "Error"
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,99 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) 2019 Hemanth Savarala.
|
|
||||||
*
|
|
||||||
* Licensed under the GNU General Public License v3
|
|
||||||
*
|
|
||||||
* This is free software: you can redistribute it and/or modify it under
|
|
||||||
* the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation either version 3 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
||||||
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
||||||
* See the GNU General Public License for more details.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package code.name.monkey.retromusic.rest;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import code.name.monkey.retromusic.rest.service.LastFMService;
|
|
||||||
import okhttp3.Cache;
|
|
||||||
import okhttp3.Call;
|
|
||||||
import okhttp3.ConnectionPool;
|
|
||||||
import okhttp3.Interceptor;
|
|
||||||
import okhttp3.OkHttpClient;
|
|
||||||
import okhttp3.Request;
|
|
||||||
import okhttp3.logging.HttpLoggingInterceptor;
|
|
||||||
import retrofit2.Retrofit;
|
|
||||||
import retrofit2.converter.gson.GsonConverterFactory;
|
|
||||||
|
|
||||||
|
|
||||||
public class LastFMRestClient {
|
|
||||||
|
|
||||||
private static final String BASE_URL = "https://ws.audioscrobbler.com/2.0/";
|
|
||||||
|
|
||||||
private LastFMService apiService;
|
|
||||||
|
|
||||||
public LastFMRestClient(@NonNull Context context) {
|
|
||||||
this(createDefaultOkHttpClientBuilder(context).build());
|
|
||||||
}
|
|
||||||
|
|
||||||
private LastFMRestClient(@NonNull Call.Factory client) {
|
|
||||||
Retrofit restAdapter = new Retrofit.Builder()
|
|
||||||
.baseUrl(BASE_URL)
|
|
||||||
.callFactory(client)
|
|
||||||
.addConverterFactory(GsonConverterFactory.create())
|
|
||||||
.build();
|
|
||||||
|
|
||||||
apiService = restAdapter.create(LastFMService.class);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Interceptor createCacheControlInterceptor() {
|
|
||||||
return chain -> {
|
|
||||||
Request modifiedRequest = chain.request().newBuilder()
|
|
||||||
.addHeader("Cache-Control", "max-age=31536000, max-stale=31536000")
|
|
||||||
.build();
|
|
||||||
return chain.proceed(modifiedRequest);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
private static Cache createDefaultCache(Context context) {
|
|
||||||
File cacheDir = new File(context.getCacheDir().getAbsolutePath(), "/okhttp-lastfm/");
|
|
||||||
if (cacheDir.mkdirs() || cacheDir.isDirectory()) {
|
|
||||||
return new Cache(cacheDir, 1024 * 1024 * 10);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
private static OkHttpClient.Builder createDefaultOkHttpClientBuilder(@NonNull Context context) {
|
|
||||||
return new OkHttpClient.Builder()
|
|
||||||
.connectionPool(new ConnectionPool(0, 1, TimeUnit.NANOSECONDS))
|
|
||||||
.retryOnConnectionFailure(true)
|
|
||||||
.connectTimeout(1, TimeUnit.MINUTES) // connect timeout
|
|
||||||
.writeTimeout(1, TimeUnit.MINUTES) // write timeout
|
|
||||||
.readTimeout(1, TimeUnit.MINUTES) // read timeout
|
|
||||||
.cache(createDefaultCache(context))
|
|
||||||
.addInterceptor(createCacheControlInterceptor())
|
|
||||||
.addInterceptor(createLogInterceptor());
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
private static Interceptor createLogInterceptor() {
|
|
||||||
HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
|
|
||||||
interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
|
|
||||||
return interceptor;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
public LastFMService getApiService() {
|
|
||||||
return apiService;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,70 @@
|
||||||
|
package code.name.monkey.retromusic.rest
|
||||||
|
|
||||||
|
import code.name.monkey.retromusic.App
|
||||||
|
import code.name.monkey.retromusic.rest.service.LastFMService
|
||||||
|
import com.google.gson.Gson
|
||||||
|
import okhttp3.Cache
|
||||||
|
import okhttp3.ConnectionPool
|
||||||
|
import okhttp3.Interceptor
|
||||||
|
import okhttp3.Interceptor.Chain
|
||||||
|
import okhttp3.OkHttpClient
|
||||||
|
import okhttp3.logging.HttpLoggingInterceptor
|
||||||
|
import retrofit2.Retrofit
|
||||||
|
import retrofit2.converter.gson.GsonConverterFactory
|
||||||
|
import java.io.File
|
||||||
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
|
object LastFmClient {
|
||||||
|
private const val baseUrl = "https://ws.audioscrobbler.com/2.0/"
|
||||||
|
|
||||||
|
private var lastFMService: LastFMService
|
||||||
|
|
||||||
|
fun getApiService(): LastFMService {
|
||||||
|
return lastFMService
|
||||||
|
}
|
||||||
|
|
||||||
|
init {
|
||||||
|
val retrofit = Retrofit.Builder()
|
||||||
|
.baseUrl(baseUrl)
|
||||||
|
.callFactory(createDefaultOkHttpClientBuilder().build())
|
||||||
|
.addConverterFactory(GsonConverterFactory.create(Gson()))
|
||||||
|
.build()
|
||||||
|
|
||||||
|
lastFMService = retrofit.create(LastFMService::class.java)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createDefaultOkHttpClientBuilder(): OkHttpClient.Builder {
|
||||||
|
return OkHttpClient.Builder()
|
||||||
|
.connectionPool(ConnectionPool(0, 1, TimeUnit.NANOSECONDS))
|
||||||
|
.retryOnConnectionFailure(true)
|
||||||
|
.connectTimeout(1, TimeUnit.MINUTES) // connect timeout
|
||||||
|
.writeTimeout(1, TimeUnit.MINUTES) // write timeout
|
||||||
|
.readTimeout(1, TimeUnit.MINUTES) // read timeout
|
||||||
|
.cache(createDefaultCache())
|
||||||
|
.addInterceptor(createCacheControlInterceptor())
|
||||||
|
.addInterceptor(createLogInterceptor())
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createLogInterceptor(): Interceptor {
|
||||||
|
val interceptor = HttpLoggingInterceptor()
|
||||||
|
interceptor.level = HttpLoggingInterceptor.Level.BASIC
|
||||||
|
return interceptor
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createCacheControlInterceptor(): Interceptor {
|
||||||
|
return Interceptor { chain: Chain ->
|
||||||
|
val modifiedRequest = chain.request().newBuilder()
|
||||||
|
.addHeader("Cache-Control", "max-age=31536000, max-stale=31536000")
|
||||||
|
.build()
|
||||||
|
chain.proceed(modifiedRequest)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createDefaultCache(): Cache? {
|
||||||
|
val cacheDir = File(App.getContext().cacheDir.absolutePath, "/okhttp-lastfm/")
|
||||||
|
if (cacheDir.mkdirs() || cacheDir.isDirectory) {
|
||||||
|
return Cache(cacheDir, 1024 * 1024 * 10)
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,9 +25,8 @@ import retrofit2.http.Query
|
||||||
*/
|
*/
|
||||||
|
|
||||||
interface LastFMService {
|
interface LastFMService {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
const val API_KEY = "c679c8d3efa84613dc7dcb2e8d42da4c"
|
private const val API_KEY = "c679c8d3efa84613dc7dcb2e8d42da4c"
|
||||||
const val BASE_QUERY_PARAMETERS = "?format=json&autocorrect=1&api_key=$API_KEY"
|
const val BASE_QUERY_PARAMETERS = "?format=json&autocorrect=1&api_key=$API_KEY"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue