瀏覽代碼

MAL: add way to search by list items' titles

arkon 4 年之前
父節點
當前提交
8a792e6d76

+ 7 - 0
app/src/main/java/eu/kanade/tachiyomi/data/track/myanimelist/MyAnimeList.kt

@@ -22,6 +22,7 @@ class MyAnimeList(private val context: Context, id: Int) : TrackService(id) {
         const val REREADING = 7
 
         private const val SEARCH_ID_PREFIX = "id:"
+        private const val SEARCH_LIST_PREFIX = "my:"
     }
 
     private val json: Json by injectLazy()
@@ -88,6 +89,12 @@ class MyAnimeList(private val context: Context, id: Int) : TrackService(id) {
             }
         }
 
+        if (query.startsWith(SEARCH_LIST_PREFIX)) {
+            query.substringAfter(SEARCH_LIST_PREFIX).let { title ->
+                return api.findListItems(title)
+            }
+        }
+
         return api.search(query)
     }
 

+ 51 - 22
app/src/main/java/eu/kanade/tachiyomi/data/track/myanimelist/MyAnimeListApi.kt

@@ -164,6 +164,56 @@ class MyAnimeListApi(private val client: OkHttpClient, interceptor: MyAnimeListI
     }
 
     suspend fun findListItem(track: Track, offset: Int = 0): Track? {
+        val json = getListPage(offset)
+        val obj = json.jsonObject
+        val trackedManga = obj["data"]!!.jsonArray.find { data ->
+            data.jsonObject["node"]!!.jsonObject["id"]!!.jsonPrimitive.int == track.media_id
+        }
+
+        return when {
+            // Found the item in the list
+            trackedManga != null -> {
+                parseMangaItem(trackedManga.jsonObject["list_status"]!!.jsonObject, track)
+            }
+            // Check next page if there's more
+            !obj["paging"]!!.jsonObject["next"]?.jsonPrimitive?.contentOrNull.isNullOrBlank() -> {
+                findListItem(track, offset + listPaginationAmount)
+            }
+            // No more pages to check, item wasn't found
+            else -> {
+                null
+            }
+        }
+    }
+
+    suspend fun findListItems(query: String, offset: Int = 0): List<TrackSearch> {
+        return withContext(Dispatchers.IO) {
+            val json = getListPage(offset)
+            val obj = json.jsonObject
+
+            val matches = obj["data"]!!.jsonArray
+                .filter {
+                    it.jsonObject["node"]!!.jsonObject["title"]!!.jsonPrimitive.content.contains(
+                        query,
+                        ignoreCase = true
+                    )
+                }
+                .map {
+                    val id = it.jsonObject["node"]!!.jsonObject["id"]!!.jsonPrimitive.int
+                    async { getMangaDetails(id) }
+                }
+                .awaitAll()
+
+            // Check next page if there's more
+            if (!obj["paging"]!!.jsonObject["next"]?.jsonPrimitive?.contentOrNull.isNullOrBlank()) {
+                matches + findListItems(query, offset + listPaginationAmount)
+            } else {
+                matches
+            }
+        }
+    }
+
+    private suspend fun getListPage(offset: Int): JsonObject {
         return withContext(Dispatchers.IO) {
             val urlBuilder = "$baseApiUrl/users/@me/mangalist".toUri().buildUpon()
                 .appendQueryParameter("fields", "list_status")
@@ -178,28 +228,7 @@ class MyAnimeListApi(private val client: OkHttpClient, interceptor: MyAnimeListI
                 .build()
             authClient.newCall(request)
                 .await()
-                .parseAs<JsonObject>()
-                .let {
-                    val obj = it.jsonObject
-                    val trackedManga = obj["data"]!!.jsonArray.find { data ->
-                        data.jsonObject["node"]!!.jsonObject["id"]!!.jsonPrimitive.int == track.media_id
-                    }
-
-                    when {
-                        // Found the item in the list
-                        trackedManga != null -> {
-                            parseMangaItem(trackedManga.jsonObject["list_status"]!!.jsonObject, track)
-                        }
-                        // Check next page if there's more
-                        !obj["paging"]!!.jsonObject["next"]?.jsonPrimitive?.contentOrNull.isNullOrBlank() -> {
-                            findListItem(track, offset + listPaginationAmount)
-                        }
-                        // No more pages to check, item wasn't found
-                        else -> {
-                            null
-                        }
-                    }
-                }
+                .parseAs()
         }
     }