|
| 1 | +package org.schabi.newpipe.fragments.list.search.filter; |
| 2 | + |
| 3 | +import org.schabi.newpipe.App; |
| 4 | +import org.schabi.newpipe.R; |
| 5 | +import org.schabi.newpipe.extractor.NewPipe; |
| 6 | +import org.schabi.newpipe.extractor.exceptions.ExtractionException; |
| 7 | +import org.schabi.newpipe.extractor.search.filter.FilterContainer; |
| 8 | +import org.schabi.newpipe.extractor.search.filter.FilterGroup; |
| 9 | +import org.schabi.newpipe.extractor.search.filter.FilterItem; |
| 10 | +import org.schabi.newpipe.extractor.search.filter.LibraryStringIds; |
| 11 | +import org.schabi.newpipe.extractor.services.youtube.search.filter.YoutubeFilters; |
| 12 | + |
| 13 | +import java.util.List; |
| 14 | + |
| 15 | +import androidx.annotation.NonNull; |
| 16 | + |
| 17 | +/** |
| 18 | + * Inject a {@link FilterItem} that actually should not be a real filter. |
| 19 | + * <p> |
| 20 | + * This base class is meant to inject eg {@link DividerItem} (that inherits {@link FilterItem}) |
| 21 | + * as Divider between {@link FilterItem}. It will be shown in the UI's. |
| 22 | + * <p> |
| 23 | + * Of course you have to handle {@link DividerItem} or whatever in the Ui's. |
| 24 | + * For that for example have a look at {@link SearchFilterDialogSpinnerAdapter}. |
| 25 | + */ |
| 26 | +public abstract class InjectFilterItem { |
| 27 | + |
| 28 | + protected InjectFilterItem( |
| 29 | + @NonNull final String serviceName, |
| 30 | + final int injectedAfterFilterWithId, |
| 31 | + @NonNull final FilterItem toBeInjectedFilterItem) { |
| 32 | + |
| 33 | + prepareAndInject(serviceName, injectedAfterFilterWithId, toBeInjectedFilterItem); |
| 34 | + } |
| 35 | + |
| 36 | + // Please refer a static boolean to determine if already injected |
| 37 | + protected abstract boolean isAlreadyInjected(); |
| 38 | + |
| 39 | + // Please refer a static boolean to determine if already injected |
| 40 | + protected abstract void setAsInjected(); |
| 41 | + |
| 42 | + private void prepareAndInject( |
| 43 | + @NonNull final String serviceName, |
| 44 | + final int injectedAfterFilterWithId, |
| 45 | + @NonNull final FilterItem toBeInjectedFilterItem) { |
| 46 | + |
| 47 | + if (isAlreadyInjected()) { // already run |
| 48 | + return; |
| 49 | + } |
| 50 | + |
| 51 | + try { // using serviceName to test if we are trying to inject into the right service |
| 52 | + final List<FilterGroup> groups = NewPipe.getService(serviceName) |
| 53 | + .getSearchQHFactory().getAvailableContentFilter().getFilterGroups(); |
| 54 | + injectFilterItemIntoGroup( |
| 55 | + groups, |
| 56 | + injectedAfterFilterWithId, |
| 57 | + toBeInjectedFilterItem); |
| 58 | + setAsInjected(); |
| 59 | + } catch (final ExtractionException ignored) { |
| 60 | + // no the service we want to prepareAndInject -> so ignore |
| 61 | + } |
| 62 | + } |
| 63 | + |
| 64 | + private void injectFilterItemIntoGroup( |
| 65 | + @NonNull final List<FilterGroup> groups, |
| 66 | + final int injectedAfterFilterWithId, |
| 67 | + @NonNull final FilterItem toBeInjectedFilterItem) { |
| 68 | + |
| 69 | + int indexForFilterId = 0; |
| 70 | + boolean isFilterItemFound = false; |
| 71 | + FilterGroup groupWithTheSearchFilterItem = null; |
| 72 | + |
| 73 | + for (final FilterGroup group : groups) { |
| 74 | + for (final FilterItem item : group.getFilterItems()) { |
| 75 | + if (item.getIdentifier() == injectedAfterFilterWithId) { |
| 76 | + isFilterItemFound = true; |
| 77 | + break; |
| 78 | + } |
| 79 | + indexForFilterId++; |
| 80 | + } |
| 81 | + |
| 82 | + if (isFilterItemFound) { |
| 83 | + groupWithTheSearchFilterItem = group; |
| 84 | + break; |
| 85 | + } |
| 86 | + } |
| 87 | + |
| 88 | + if (isFilterItemFound) { |
| 89 | + // we want to insert after the FilterItem we've searched |
| 90 | + indexForFilterId++; |
| 91 | + groupWithTheSearchFilterItem.getFilterItems() |
| 92 | + .add(indexForFilterId, toBeInjectedFilterItem); |
| 93 | + } |
| 94 | + } |
| 95 | + |
| 96 | + /** |
| 97 | + * Inject DividerItem between YouTube content filters and YoutubeMusic content filters. |
| 98 | + */ |
| 99 | + public static class DividerBetweenYoutubeAndYoutubeMusic extends InjectFilterItem { |
| 100 | + |
| 101 | + private static boolean isYoutubeMusicDividerInjected = false; |
| 102 | + |
| 103 | + protected DividerBetweenYoutubeAndYoutubeMusic() { |
| 104 | + super(App.getApp().getApplicationContext().getString(R.string.youtube), |
| 105 | + YoutubeFilters.ID_CF_MAIN_PLAYLISTS, |
| 106 | + new DividerItem(R.string.search_filters_youtube_music) |
| 107 | + ); |
| 108 | + } |
| 109 | + |
| 110 | + /** |
| 111 | + * Have a static runner method to avoid creating unnecessary objects if already inserted. |
| 112 | + */ |
| 113 | + public static void run() { |
| 114 | + if (!isYoutubeMusicDividerInjected) { |
| 115 | + new DividerBetweenYoutubeAndYoutubeMusic(); |
| 116 | + } |
| 117 | + } |
| 118 | + |
| 119 | + @Override |
| 120 | + protected boolean isAlreadyInjected() { |
| 121 | + return isYoutubeMusicDividerInjected; |
| 122 | + } |
| 123 | + |
| 124 | + @Override |
| 125 | + protected void setAsInjected() { |
| 126 | + isYoutubeMusicDividerInjected = true; |
| 127 | + } |
| 128 | + } |
| 129 | + |
| 130 | + /** |
| 131 | + * Used to have a title divider between regular {@link FilterItem}s. |
| 132 | + */ |
| 133 | + public static class DividerItem extends FilterItem { |
| 134 | + |
| 135 | + private final int resId; |
| 136 | + |
| 137 | + public DividerItem(final int resId) { |
| 138 | + // the LibraryStringIds.. is not needed at all I just need one to satisfy FilterItem. |
| 139 | + super(FilterContainer.ITEM_IDENTIFIER_UNKNOWN, LibraryStringIds.SEARCH_FILTERS_ALL); |
| 140 | + this.resId = resId; |
| 141 | + } |
| 142 | + |
| 143 | + public int getStringResId() { |
| 144 | + return this.resId; |
| 145 | + } |
| 146 | + } |
| 147 | +} |
0 commit comments