From 84934c28185c55c42ee2f232469a4419b1bb789c Mon Sep 17 00:00:00 2001 From: xfileFIN Date: Thu, 3 Sep 2020 00:07:52 +0300 Subject: [PATCH 1/3] New shield button POC --- .../apps/youtube/app/YouTubeApplication.java | 15 ++ .../main/java/pl/jakubweg/ShieldButton.java | 132 ++++++++++++++++++ .../pl/jakubweg/SponsorBlockSettings.java | 4 +- .../java/pl/jakubweg/SponsorBlockUtils.java | 8 +- 4 files changed, 153 insertions(+), 6 deletions(-) create mode 100644 app/src/main/java/com/google/android/apps/youtube/app/YouTubeApplication.java create mode 100644 app/src/main/java/pl/jakubweg/ShieldButton.java diff --git a/app/src/main/java/com/google/android/apps/youtube/app/YouTubeApplication.java b/app/src/main/java/com/google/android/apps/youtube/app/YouTubeApplication.java new file mode 100644 index 00000000..ef8400e4 --- /dev/null +++ b/app/src/main/java/com/google/android/apps/youtube/app/YouTubeApplication.java @@ -0,0 +1,15 @@ +package com.google.android.apps.youtube.app; + +import android.app.Application; +import android.content.Context; +import android.os.Bundle; + +public class YouTubeApplication extends Application { + protected void onCreate(final Bundle bundle) { + super.onCreate(); + } + + public static Context getAppContext() { + return null; + } +} diff --git a/app/src/main/java/pl/jakubweg/ShieldButton.java b/app/src/main/java/pl/jakubweg/ShieldButton.java new file mode 100644 index 00000000..f6b7bc38 --- /dev/null +++ b/app/src/main/java/pl/jakubweg/ShieldButton.java @@ -0,0 +1,132 @@ +package pl.jakubweg; + +import android.content.Context; +import android.content.SharedPreferences; +import android.util.Log; +import android.view.View; +import android.view.animation.Animation; +import android.view.animation.AnimationUtils; +import android.widget.ImageView; +import android.widget.RelativeLayout; + +import com.google.android.apps.youtube.app.YouTubeApplication; + +import java.lang.ref.WeakReference; + +public class ShieldButton { + static boolean debug = true; + static String TAG = "SHIELD"; + static RelativeLayout _youtubeControlsLayout; + static WeakReference _shieldBtn = new WeakReference<>(null); + static int fadeDurationFast; + static int fadeDurationScheduled; + static Animation fadeIn; + static Animation fadeOut; + static boolean isShowing; + + public static void initialize(Object viewStub) { + try { + if(debug){ + Log.d(TAG, "initializing shield button"); + } + + _youtubeControlsLayout = (RelativeLayout) viewStub; + initButtonVisibilitySettings(); + + ImageView imageView = (ImageView)_youtubeControlsLayout + .findViewById(getIdentifier("sponsorblock_button", "id")); + + if (debug && imageView == null){ + Log.d(TAG, "Couldn't find imageView with tag \"sponsorblock_button\""); + } + if (imageView == null) return; + imageView.setOnClickListener(SponsorBlockUtils.sponsorBlockBtnListener); + _shieldBtn = new WeakReference<>(imageView); + + // Animations + fadeDurationFast = getInteger("fade_duration_fast"); + fadeDurationScheduled = getInteger("fade_duration_scheduled"); + fadeIn = getAnimation("fade_in"); + fadeIn.setDuration(fadeDurationFast); + fadeOut = getAnimation("fade_out"); + fadeOut.setDuration(fadeDurationScheduled); + isShowing = true; + changeVisibilityImmediate(false); + } + catch (Exception ex) { + Log.e("XError", "Unable to set RelativeLayout", ex); + } + } + + public static void changeVisibilityImmediate(boolean visible) { + changeVisibility(visible, true); + } + + public static void changeVisibilityNegatedImmediate(boolean visible) { + changeVisibility(!visible, true); + } + + public static void changeVisibility(boolean visible) { + changeVisibility(visible, false); + } + + public static void changeVisibility(boolean visible, boolean immediate) { + if (isShowing == visible) return; + isShowing = visible; + + ImageView iView = _shieldBtn.get(); + if (_youtubeControlsLayout == null || iView == null) return; + + if (visible && shouldBeShown()) { + if (debug){ + Log.d(TAG, "Fading in"); + } + iView.setVisibility(View.VISIBLE); + if (!immediate) + iView.startAnimation(fadeIn); + return; + } + + if (iView.getVisibility() == View.VISIBLE) { + if (debug){ + Log.d(TAG, "Fading out"); + } + if (!immediate) + iView.startAnimation(fadeOut); + iView.setVisibility(shouldBeShown() ? View.INVISIBLE : View.GONE); + } + } + + private static boolean shouldBeShown() { + return SponsorBlockSettings.isSponsorBlockEnabled && SponsorBlockSettings.isAddNewSegmentEnabled; + } + + private static void initButtonVisibilitySettings() { + Context context = YouTubeApplication.getAppContext(); + if(context == null){ + Log.e(TAG, "context is null"); + SponsorBlockSettings.isSponsorBlockEnabled = false; + SponsorBlockSettings.isAddNewSegmentEnabled = false; + return; + } + + SharedPreferences sharedPreferences = context.getSharedPreferences(SponsorBlockSettings.PREFERENCES_NAME, Context.MODE_PRIVATE); + SponsorBlockSettings.isSponsorBlockEnabled = sharedPreferences.getBoolean(SponsorBlockSettings.PREFERENCES_KEY_SPONSOR_BLOCK_ENABLED, false); + SponsorBlockSettings.isAddNewSegmentEnabled = sharedPreferences.getBoolean(SponsorBlockSettings.PREFERENCES_KEY_NEW_SEGMENT_ENABLED, false); + } + + //region Helpers + private static int getIdentifier(String name, String defType) { + Context context = YouTubeApplication.getAppContext(); + return context.getResources().getIdentifier(name, defType, context.getPackageName()); + } + + private static int getInteger(String name) { + return YouTubeApplication.getAppContext().getResources().getInteger(getIdentifier(name, "integer")); + } + + private static Animation getAnimation(String name) { + return AnimationUtils.loadAnimation(YouTubeApplication.getAppContext(), getIdentifier(name, "anim")); + } + //endregion +} diff --git a/app/src/main/java/pl/jakubweg/SponsorBlockSettings.java b/app/src/main/java/pl/jakubweg/SponsorBlockSettings.java index 24ef9d5e..27b1939f 100644 --- a/app/src/main/java/pl/jakubweg/SponsorBlockSettings.java +++ b/app/src/main/java/pl/jakubweg/SponsorBlockSettings.java @@ -75,12 +75,12 @@ public class SponsorBlockSettings { NewSegmentHelperLayout.hide(); SponsorBlockUtils.hideButton(); PlayerController.sponsorSegmentsOfCurrentVideo = null; - } else if (isAddNewSegmentEnabled) { + } else if (/*isAddNewSegmentEnabled*/false) { SponsorBlockUtils.showButton(); } isAddNewSegmentEnabled = preferences.getBoolean(PREFERENCES_KEY_NEW_SEGMENT_ENABLED, isAddNewSegmentEnabled); - if (!isAddNewSegmentEnabled) { + if (!/*isAddNewSegmentEnabled*/false) { NewSegmentHelperLayout.hide(); SponsorBlockUtils.hideButton(); } else { diff --git a/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java b/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java index 31968819..0d3c1f9b 100644 --- a/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java +++ b/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java @@ -61,7 +61,7 @@ public abstract class SponsorBlockUtils { @SuppressLint("SimpleDateFormat") public static final SimpleDateFormat dateFormatter = new SimpleDateFormat(DATE_FORMAT); private static final int sponsorBtnId = 1234; - private static final View.OnClickListener sponsorBlockBtnListener = new View.OnClickListener() { + public static final View.OnClickListener sponsorBlockBtnListener = new View.OnClickListener() { @Override public void onClick(View v) { NewSegmentHelperLayout.toggle(); @@ -292,7 +292,7 @@ public abstract class SponsorBlockUtils { if (existingSponsorBtn != null) { if (VERBOSE) Log.d(TAG, "addImageButton: sponsorBtn exists"); - if (SponsorBlockSettings.isAddNewSegmentEnabled) + if (/*isAddNewSegmentEnabled*/false) showButton(); return; } @@ -365,7 +365,7 @@ public abstract class SponsorBlockUtils { instance.setOnClickListener(sponsorBlockBtnListener); sponsorBlockBtn = new WeakReference<>(instance); isShown = true; - if (!SponsorBlockSettings.isAddNewSegmentEnabled) + if (!/*isAddNewSegmentEnabled*/false) hideButton(); if (VERBOSE) Log.i(TAG, "Image Button added"); @@ -444,7 +444,7 @@ public abstract class SponsorBlockUtils { } public static void notifyShareBtnVisibilityChanged(View v) { - if (v.getId() != shareBtnId || !SponsorBlockSettings.isAddNewSegmentEnabled) return; + if (v.getId() != shareBtnId || !/*SponsorBlockSettings.isAddNewSegmentEnabled*/false) return; // if (VERBOSE) // Log.d(TAG, "VISIBILITY CHANGED of view " + v); ImageView sponsorBtn = sponsorBlockBtn.get(); From 863bad149bd499a59535e156f3ea2f290dbebd09 Mon Sep 17 00:00:00 2001 From: xfileFIN Date: Thu, 15 Oct 2020 18:53:00 +0300 Subject: [PATCH 2/3] Player overlay buttons redone --- .../java/fi/razerman/youtube/XGlobals.java | 6 + .../libraries/youtube/player/PlayerType.java | 9 + .../sponsors/player/ui/NewSegmentLayout.java | 153 +++++++++++++++++ .../sponsors/player/ui/SkipSponsorButton.java | 134 +++++++++++++++ .../sponsors/player/ui/SponsorBlockView.java | 157 ++++++++++++++++++ .../pl/jakubweg/NewSegmentHelperLayout.java | 123 +------------- .../java/pl/jakubweg/PlayerController.java | 20 +-- .../main/java/pl/jakubweg/ShieldButton.java | 9 +- .../java/pl/jakubweg/SkipSegmentView.java | 69 ++------ .../java/pl/jakubweg/SponsorBlockUtils.java | 100 +---------- .../quantum_ic_fast_forward_grey600_36.png | Bin 0 -> 736 bytes .../quantum_ic_fast_forward_white_36.png | Bin 0 -> 690 bytes .../quantum_ic_fast_rewind_grey600_36.png | Bin 0 -> 642 bytes .../quantum_ic_fast_rewind_white_36.png | Bin 0 -> 644 bytes .../main/res/drawable/player_fast_forward.xml | 7 + .../main/res/drawable/player_fast_rewind.xml | 7 + .../res/layout/inline_sponsor_overlay.xml | 9 + app/src/main/res/layout/new_segment.xml | 16 ++ .../main/res/layout/skip_sponsor_button.xml | 11 ++ app/src/main/res/values/strings.xml | 2 + 20 files changed, 541 insertions(+), 291 deletions(-) create mode 100644 app/src/main/java/fi/razerman/youtube/XGlobals.java create mode 100644 app/src/main/java/fi/vanced/libraries/youtube/player/PlayerType.java create mode 100644 app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/NewSegmentLayout.java create mode 100644 app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SkipSponsorButton.java create mode 100644 app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SponsorBlockView.java create mode 100644 app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_grey600_36.png create mode 100644 app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_white_36.png create mode 100644 app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_rewind_grey600_36.png create mode 100644 app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_rewind_white_36.png create mode 100644 app/src/main/res/drawable/player_fast_forward.xml create mode 100644 app/src/main/res/drawable/player_fast_rewind.xml create mode 100644 app/src/main/res/layout/inline_sponsor_overlay.xml create mode 100644 app/src/main/res/layout/new_segment.xml create mode 100644 app/src/main/res/layout/skip_sponsor_button.xml diff --git a/app/src/main/java/fi/razerman/youtube/XGlobals.java b/app/src/main/java/fi/razerman/youtube/XGlobals.java new file mode 100644 index 00000000..7bc8ec22 --- /dev/null +++ b/app/src/main/java/fi/razerman/youtube/XGlobals.java @@ -0,0 +1,6 @@ +package fi.razerman.youtube; + +// Ignore this file, the implementation is in another repository +public class XGlobals { + public static Boolean debug = false; +} diff --git a/app/src/main/java/fi/vanced/libraries/youtube/player/PlayerType.java b/app/src/main/java/fi/vanced/libraries/youtube/player/PlayerType.java new file mode 100644 index 00000000..448ebe0c --- /dev/null +++ b/app/src/main/java/fi/vanced/libraries/youtube/player/PlayerType.java @@ -0,0 +1,9 @@ +package fi.vanced.libraries.youtube.player; + +import fi.vanced.libraries.youtube.sponsors.player.ui.SponsorBlockView; + +public class PlayerType { + public static void playerTypeChanged(String playerType) { + SponsorBlockView.playerTypeChanged(playerType); + } +} diff --git a/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/NewSegmentLayout.java b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/NewSegmentLayout.java new file mode 100644 index 00000000..72eadd73 --- /dev/null +++ b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/NewSegmentLayout.java @@ -0,0 +1,153 @@ +package fi.vanced.libraries.youtube.sponsors.player.ui; + +import android.content.Context; +import android.content.res.ColorStateList; +import android.content.res.Resources; +import android.graphics.drawable.RippleDrawable; +import android.util.AttributeSet; +import android.util.Log; +import android.util.TypedValue; +import android.view.LayoutInflater; +import android.view.View; +import android.widget.FrameLayout; +import android.widget.ImageButton; +import android.widget.LinearLayout; + +import pl.jakubweg.NewSegmentHelperLayout; +import pl.jakubweg.PlayerController; +import pl.jakubweg.SponsorBlockSettings; +import pl.jakubweg.SponsorBlockUtils; + +import static fi.razerman.youtube.XGlobals.debug; + +public class NewSegmentLayout extends FrameLayout { + static String TAG = "NewSegmentLayout"; + + private LinearLayout newSegmentContainer; + public int defaultBottomMargin; + public int ctaBottomMargin; + public ImageButton rewindButton; + public ImageButton forwardButton; + public ImageButton adjustButton; + public ImageButton compareButton; + public ImageButton editButton; + public ImageButton publishButton; + private int rippleEffectId; + + public NewSegmentLayout(Context context) { + super(context); + this.initialize(context); + } + + public NewSegmentLayout(Context context, AttributeSet attributeSet) { + super(context, attributeSet); + this.initialize(context); + } + + public NewSegmentLayout(Context context, AttributeSet attributeSet, int defStyleAttr) { + super(context, attributeSet, defStyleAttr); + this.initialize(context); + } + + public NewSegmentLayout(Context context, AttributeSet attributeSet, int defStyleAttr, int defStyleRes) { + super(context, attributeSet, defStyleAttr, defStyleRes); + this.initialize(context); + } + + private final void initialize(Context context) { + LayoutInflater.from(context).inflate(getIdentifier(context, "new_segment", "layout"), this, true); + Resources resources = context.getResources(); + + TypedValue rippleEffect = new TypedValue(); + getContext().getTheme().resolveAttribute(android.R.attr.selectableItemBackground, rippleEffect, true); + rippleEffectId = rippleEffect.resourceId; + + this.newSegmentContainer = (LinearLayout)this.findViewById(getIdentifier(context, "new_segment_container", "id")); + + this.rewindButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_rewind", "id")); + if (this.rewindButton != null) { + setClickEffect(this.rewindButton); + this.rewindButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Rewind button clicked"); } + PlayerController.skipRelativeMilliseconds(-SponsorBlockSettings.adjustNewSegmentMillis); + } + }); + } + this.forwardButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_forward", "id")); + if (this.forwardButton != null) { + setClickEffect(this.forwardButton); + this.forwardButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Forward button clicked"); } + PlayerController.skipRelativeMilliseconds(SponsorBlockSettings.adjustNewSegmentMillis); + } + }); + } + this.adjustButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_adjust", "id")); + if (this.adjustButton != null) { + setClickEffect(this.adjustButton); + this.adjustButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Adjust button clicked"); } + SponsorBlockUtils.onMarkLocationClicked(NewSegmentHelperLayout.context); + } + }); + } + this.compareButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_compare", "id")); + if (this.compareButton != null) { + setClickEffect(this.compareButton); + this.compareButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Compare button clicked"); } + SponsorBlockUtils.onPreviewClicked(NewSegmentHelperLayout.context); + } + }); + } + this.editButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_edit", "id")); + if (this.editButton != null) { + setClickEffect(this.editButton); + this.editButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Edit button clicked"); } + SponsorBlockUtils.onEditByHandClicked(NewSegmentHelperLayout.context); + } + }); + } + this.publishButton = (ImageButton)this.findViewById(getIdentifier(context, "new_segment_publish", "id")); + if (this.publishButton != null) { + setClickEffect(this.publishButton); + this.publishButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { Log.d(TAG, "Publish button clicked"); } + SponsorBlockUtils.onPublishClicked(NewSegmentHelperLayout.context); + } + }); + } + + this.defaultBottomMargin = resources.getDimensionPixelSize(getIdentifier(context, "brand_interaction_default_bottom_margin", "dimen")); + this.ctaBottomMargin = resources.getDimensionPixelSize(getIdentifier(context, "brand_interaction_cta_bottom_margin", "dimen")); + } + + private void setClickEffect(ImageButton btn) { + btn.setBackgroundResource(rippleEffectId); + + RippleDrawable rippleDrawable = (RippleDrawable)btn.getBackground(); + + int[][] states = new int[][] { new int[] { android.R.attr.state_enabled } }; + int[] colors = new int[] { 0x33ffffff }; // sets the ripple color to white + + ColorStateList colorStateList = new ColorStateList(states, colors); + rippleDrawable.setColor(colorStateList); + } + + private int getIdentifier(Context context, String name, String defType) { + return context.getResources().getIdentifier(name, defType, context.getPackageName()); + } +} diff --git a/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SkipSponsorButton.java b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SkipSponsorButton.java new file mode 100644 index 00000000..17446b90 --- /dev/null +++ b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SkipSponsorButton.java @@ -0,0 +1,134 @@ +package fi.vanced.libraries.youtube.sponsors.player.ui; + +import android.content.Context; +import android.content.res.ColorStateList; +import android.content.res.Resources; +import android.graphics.Canvas; +import android.graphics.Paint; +import android.graphics.drawable.ColorDrawable; +import android.graphics.drawable.RippleDrawable; +import android.os.Build; +import android.util.AttributeSet; +import android.util.Log; +import android.util.TypedValue; +import android.view.LayoutInflater; +import android.view.View; +import android.widget.FrameLayout; +import android.widget.ImageButton; +import android.widget.ImageView; +import android.widget.LinearLayout; +import android.widget.TextView; + +import pl.jakubweg.PlayerController; + +import static fi.razerman.youtube.XGlobals.debug; + +public class SkipSponsorButton extends FrameLayout { + String TAG = "SkipSponsorButton"; + public CharSequence skipSponsorTextViewText; + public CharSequence skipSponsorText; + public ImageView skipSponsorButtonIcon; + public TextView skipSponsorTextView; + public int currentTextColor; + public int invertedButtonForegroundColor; + public int backgroundColor; + public int invertedBackgroundColor; + public ColorDrawable backgroundColorDrawable; + public int defaultBottomMargin; + public int ctaBottomMargin; + private LinearLayout skipSponsorBtnContainer; + private final Paint background; + private final Paint border; + private boolean highContrast = true; + + public SkipSponsorButton(Context context) { + super(context); + this.background = new Paint(); + this.border = new Paint(); + this.initialize(context); + } + + public SkipSponsorButton(Context context, AttributeSet attributeSet) { + super(context, attributeSet); + this.background = new Paint(); + this.border = new Paint(); + this.initialize(context); + } + + public SkipSponsorButton(Context context, AttributeSet attributeSet, int defStyleAttr) { + super(context, attributeSet, defStyleAttr); + this.background = new Paint(); + this.border = new Paint(); + this.initialize(context); + } + + public SkipSponsorButton(Context context, AttributeSet attributeSet, int defStyleAttr, int defStyleRes) { + super(context, attributeSet, defStyleAttr, defStyleRes); + this.background = new Paint(); + this.border = new Paint(); + this.initialize(context); + } + + private final void initialize(Context context) { + LayoutInflater.from(context).inflate(getIdentifier(context, "skip_sponsor_button", "layout"), this, true); // layout:skip_ad_button + this.setMinimumHeight(this.getResources().getDimensionPixelSize(getIdentifier(context, "ad_skip_ad_button_min_height", "dimen"))); // dimen:ad_skip_ad_button_min_height + this.skipSponsorBtnContainer = (LinearLayout)this.findViewById(getIdentifier(context, "skip_sponsor_button_container", "id")); // id:skip_ad_button_container + this.skipSponsorButtonIcon = (ImageView)this.findViewById(getIdentifier(context, "skip_sponsor_button_icon", "id")); // id:skip_ad_button_icon + this.backgroundColor = getColor(context, getIdentifier(context, "skip_ad_button_background_color", "color")); // color:skip_ad_button_background_color + this.invertedBackgroundColor = getColor(context, getIdentifier(context, "skip_ad_button_inverted_background_color", "color")); // color:skip_ad_button_inverted_background_color + this.background.setColor(this.backgroundColor); + this.background.setStyle(Paint.Style.FILL); + int borderColor = getColor(context, getIdentifier(context, "skip_ad_button_border_color", "color")); // color:skip_ad_button_border_color + this.border.setColor(borderColor); + float borderWidth = this.getResources().getDimension(getIdentifier(context, "ad_skip_ad_button_border_width", "dimen")); // dimen:ad_skip_ad_button_border_width + this.border.setStrokeWidth(borderWidth); + this.border.setStyle(Paint.Style.STROKE); + TextView skipSponsorText = (TextView)this.findViewById(getIdentifier(context, "skip_sponsor_button_text", "id")); // id:skip_ad_button_text + this.skipSponsorTextView = skipSponsorText; + this.skipSponsorTextViewText = skipSponsorText.getText(); + this.currentTextColor = this.skipSponsorTextView.getCurrentTextColor(); + this.invertedButtonForegroundColor = getColor(context, getIdentifier(context, "skip_ad_button_inverted_foreground_color", "color")); // color:skip_ad_button_inverted_foreground_color + this.backgroundColorDrawable = new ColorDrawable(this.backgroundColor); + Resources resources = context.getResources(); + this.defaultBottomMargin = resources.getDimensionPixelSize(getIdentifier(context, "skip_button_default_bottom_margin", "dimen")); // dimen:skip_button_default_bottom_margin + this.ctaBottomMargin = resources.getDimensionPixelSize(getIdentifier(context, "skip_button_cta_bottom_margin", "dimen")); // dimen:skip_button_cta_bottom_margin + this.skipSponsorText = resources.getText(getIdentifier(context, "skip_sponsor", "string")); // string:skip_ads "Skip ads" + + this.skipSponsorBtnContainer.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + if (debug) { + Log.d(TAG, "Skip button clicked"); + } + PlayerController.onSkipSponsorClicked(); + } + }); + } + + @Override // android.view.ViewGroup + protected final void dispatchDraw(Canvas canvas) { + int width = this.skipSponsorBtnContainer.getWidth(); + int height = this.skipSponsorBtnContainer.getHeight(); + int top = this.skipSponsorBtnContainer.getTop(); + int left = this.skipSponsorBtnContainer.getLeft(); + float floatLeft = (float)left; + float floatTop = (float)top; + float floatWidth = (float)(left + width); + float floatHeight = (float)(top + height); + canvas.drawRect(floatLeft, floatTop, floatWidth, floatHeight, this.background); + if (!this.highContrast) { + canvas.drawLines(new float[] { floatWidth, floatTop, floatLeft, floatTop, floatLeft, floatTop, floatLeft, floatHeight, floatLeft, floatHeight, floatWidth, floatHeight }, this.border); + } + + super.dispatchDraw(canvas); + } + + + public static int getColor(Context context, int arg3) { + return Build.VERSION.SDK_INT < 23 ? context.getResources().getColor(arg3) : context.getColor(arg3); + } + + private int getIdentifier(Context context, String name, String defType) { + return context.getResources().getIdentifier(name, defType, context.getPackageName()); + } +} diff --git a/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SponsorBlockView.java b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SponsorBlockView.java new file mode 100644 index 00000000..af1d011f --- /dev/null +++ b/app/src/main/java/fi/vanced/libraries/youtube/sponsors/player/ui/SponsorBlockView.java @@ -0,0 +1,157 @@ +package fi.vanced.libraries.youtube.sponsors.player.ui; + +import android.content.Context; +import android.util.Log; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.RelativeLayout; + +import com.google.android.apps.youtube.app.YouTubeApplication; + +import java.lang.ref.WeakReference; + +import static fi.razerman.youtube.XGlobals.debug; + +public class SponsorBlockView { + static String TAG = "SponsorBlockView"; + static RelativeLayout inlineSponsorOverlay; + static ViewGroup _youtubeOverlaysLayout; + static WeakReference _skipSponsorButton = new WeakReference<>(null); + static WeakReference _newSegmentLayout = new WeakReference<>(null); + static boolean shouldShowOnPlayerType = true; + + public static void initialize(Object viewGroup) { + try { + if(debug){ + Log.d(TAG, "initializing"); + } + + _youtubeOverlaysLayout = (ViewGroup) viewGroup; + + addView(); + } + catch (Exception ex) { + Log.e(TAG, "Unable to set ViewGroup", ex); + } + } + + public static void showSkipButton() { + skipSponsorButtonVisibility(true); + } + public static void hideSkipButton() { + skipSponsorButtonVisibility(false); + } + + public static void showNewSegmentLayout() { + newSegmentLayoutVisibility(true); + } + public static void hideNewSegmentLayout() { + newSegmentLayoutVisibility(false); + } + + public static void playerTypeChanged(String playerType) { + try { + shouldShowOnPlayerType = playerType.equalsIgnoreCase("WATCH_WHILE_FULLSCREEN") || playerType.equalsIgnoreCase("WATCH_WHILE_MAXIMIZED"); + + if (playerType.equalsIgnoreCase("WATCH_WHILE_FULLSCREEN")) { + setSkipBtnMargins(true); + setNewSegmentLayoutMargins(true); + return; + } + + setSkipBtnMargins(false); + setNewSegmentLayoutMargins(false); + } + catch (Exception ex) { + Log.e(TAG, "Player type changed caused a crash.", ex); + } + } + + private static void addView() { + inlineSponsorOverlay = new RelativeLayout(YouTubeApplication.getAppContext()); + setLayoutParams(inlineSponsorOverlay); + LayoutInflater.from(YouTubeApplication.getAppContext()).inflate(getIdentifier("inline_sponsor_overlay", "layout"), inlineSponsorOverlay); + + _youtubeOverlaysLayout.addView(inlineSponsorOverlay, _youtubeOverlaysLayout.getChildCount() - 2); + + SkipSponsorButton skipSponsorButton = (SkipSponsorButton) inlineSponsorOverlay.findViewById(getIdentifier("skip_sponsor_button", "id")); + _skipSponsorButton = new WeakReference<>(skipSponsorButton); + + NewSegmentLayout newSegmentView = (NewSegmentLayout) inlineSponsorOverlay.findViewById(getIdentifier("new_segment_view", "id")); + _newSegmentLayout = new WeakReference<>(newSegmentView); + } + + private static void setLayoutParams(RelativeLayout relativeLayout) { + relativeLayout.setLayoutParams(new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.MATCH_PARENT)); + } + + private static void setSkipBtnMargins(boolean fullScreen) { + SkipSponsorButton skipSponsorButton = _skipSponsorButton.get(); + if (skipSponsorButton == null) { + Log.e(TAG, "Unable to setSkipBtnMargins"); + return; + } + + RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) skipSponsorButton.getLayoutParams(); + if (params == null) { + Log.e(TAG, "Unable to setSkipBtnMargins"); + return; + } + params.bottomMargin = fullScreen ? skipSponsorButton.ctaBottomMargin : skipSponsorButton.defaultBottomMargin; + skipSponsorButton.setLayoutParams(params); + } + + private static void skipSponsorButtonVisibility(boolean visible) { + SkipSponsorButton skipSponsorButton = _skipSponsorButton.get(); + if (skipSponsorButton == null) { + Log.e(TAG, "Unable to skipSponsorButtonVisibility"); + return; + } + + visible &= shouldShowOnPlayerType; + + skipSponsorButton.setVisibility(visible ? View.VISIBLE : View.GONE); + bringLayoutToFront(); + } + + private static void setNewSegmentLayoutMargins(boolean fullScreen) { + NewSegmentLayout newSegmentLayout = _newSegmentLayout.get(); + if (newSegmentLayout == null) { + Log.e(TAG, "Unable to setNewSegmentLayoutMargins"); + return; + } + + RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) newSegmentLayout.getLayoutParams(); + if (params == null) { + Log.e(TAG, "Unable to setNewSegmentLayoutMargins"); + return; + } + params.bottomMargin = fullScreen ? newSegmentLayout.ctaBottomMargin : newSegmentLayout.defaultBottomMargin; + newSegmentLayout.setLayoutParams(params); + } + + private static void newSegmentLayoutVisibility(boolean visible) { + NewSegmentLayout newSegmentLayout = _newSegmentLayout.get(); + if (newSegmentLayout == null) { + Log.e(TAG, "Unable to newSegmentLayoutVisibility"); + return; + } + + visible &= shouldShowOnPlayerType; + + newSegmentLayout.setVisibility(visible ? View.VISIBLE : View.GONE); + bringLayoutToFront(); + } + + private static void bringLayoutToFront() { + inlineSponsorOverlay.bringToFront(); + inlineSponsorOverlay.requestLayout(); + inlineSponsorOverlay.invalidate(); + } + + private static int getIdentifier(String name, String defType) { + Context context = YouTubeApplication.getAppContext(); + return context.getResources().getIdentifier(name, defType, context.getPackageName()); + } +} diff --git a/app/src/main/java/pl/jakubweg/NewSegmentHelperLayout.java b/app/src/main/java/pl/jakubweg/NewSegmentHelperLayout.java index 5c0ac4c3..6aca22f9 100644 --- a/app/src/main/java/pl/jakubweg/NewSegmentHelperLayout.java +++ b/app/src/main/java/pl/jakubweg/NewSegmentHelperLayout.java @@ -1,139 +1,28 @@ package pl.jakubweg; -import android.annotation.SuppressLint; import android.content.Context; -import android.util.TypedValue; -import android.view.Gravity; -import android.view.View; -import android.widget.FrameLayout; -import android.widget.ImageView; -import android.widget.LinearLayout; -import java.lang.ref.WeakReference; +import static fi.vanced.libraries.youtube.sponsors.player.ui.SponsorBlockView.hideNewSegmentLayout; +import static fi.vanced.libraries.youtube.sponsors.player.ui.SponsorBlockView.showNewSegmentLayout; -public class NewSegmentHelperLayout extends LinearLayout implements View.OnClickListener { - private static final int rewindBtnId = 1235; - private static final int forwardBtnId = 1236; - private static final int publishBtnId = 1237; - private static final int hideBtnId = 1238; - private static final int markLocationBtnId = 1239; - private static final int previewBtnId = 1240; - private static final int editByHandBtnId = 1241; - private static WeakReference INSTANCE = new WeakReference<>(null); +public class NewSegmentHelperLayout { + public static Context context; private static boolean isShown = false; - private final int padding; - private final int iconSize; - private final int rippleEffectId; - private final String packageName; - - @SuppressLint({"DefaultLocale", "SetTextI18n"}) - public NewSegmentHelperLayout(Context context) { - super(context); - INSTANCE = new WeakReference<>(this); - isShown = false; - setVisibility(GONE); - - packageName = context.getPackageName(); - padding = (int) SkipSegmentView.convertDpToPixel(4f, context); - iconSize = (int) SkipSegmentView.convertDpToPixel(40f, context); - - TypedValue rippleEffect = new TypedValue(); - getContext().getTheme().resolveAttribute(android.R.attr.selectableItemBackground, rippleEffect, true); - rippleEffectId = rippleEffect.resourceId; - - - setOrientation(VERTICAL); - @SuppressLint("RtlHardcoded") - FrameLayout.LayoutParams layoutParams = new FrameLayout.LayoutParams( - FrameLayout.LayoutParams.WRAP_CONTENT, - FrameLayout.LayoutParams.WRAP_CONTENT, - Gravity.START | Gravity.LEFT | Gravity.CENTER_VERTICAL - ); - this.setBackgroundColor(0x66000000); - this.bringToFront(); - this.setLayoutParams(layoutParams); - this.setPadding(padding, padding, padding, padding); - - final LinearLayout topLayout = new LinearLayout(context); - final LinearLayout bottomLayout = new LinearLayout(context); - topLayout.setOrientation(HORIZONTAL); - bottomLayout.setOrientation(HORIZONTAL); - this.addView(topLayout); - this.addView(bottomLayout); - - topLayout.addView(createTextViewBtn(rewindBtnId, "player_fast_rewind")); - topLayout.addView(createTextViewBtn(forwardBtnId, "player_fast_forward")); - topLayout.addView(createTextViewBtn(markLocationBtnId, "ic_sb_adjust")); - bottomLayout.addView(createTextViewBtn(previewBtnId, "ic_sb_compare")); - bottomLayout.addView(createTextViewBtn(editByHandBtnId, "ic_sb_edit")); - bottomLayout.addView(createTextViewBtn(publishBtnId, "ic_sb_publish")); -// bottomLayout.addView(createTextViewBtn(hideBtnId,"btn_close_light")); - } public static void show() { if (isShown) return; isShown = true; - NewSegmentHelperLayout i = INSTANCE.get(); - if (i == null) return; - i.setVisibility(VISIBLE); - i.bringToFront(); - i.requestLayout(); - i.invalidate(); + showNewSegmentLayout(); } public static void hide() { if (!isShown) return; isShown = false; - NewSegmentHelperLayout i = INSTANCE.get(); - if (i != null) - i.setVisibility(GONE); + hideNewSegmentLayout(); } public static void toggle() { if (isShown) hide(); else show(); } - - private View createTextViewBtn(int id, String drawableName) { - int drawableId = getResources().getIdentifier(drawableName, "drawable", packageName); - final ImageView view = new ImageView(getContext()); - view.setPadding(padding, padding, padding, padding); - view.setLayoutParams(new LayoutParams(iconSize, iconSize, 1)); - view.setScaleType(ImageView.ScaleType.CENTER_INSIDE); - view.setImageResource(drawableId); - view.setId(id); - view.setClickable(true); - view.setFocusable(true); - view.setBackgroundResource(rippleEffectId); - view.setOnClickListener(this); - return view; - } - - - @Override - public void onClick(View v) { - switch (v.getId()) { - case forwardBtnId: - PlayerController.skipRelativeMilliseconds(SponsorBlockSettings.adjustNewSegmentMillis); - break; - case rewindBtnId: - PlayerController.skipRelativeMilliseconds(-SponsorBlockSettings.adjustNewSegmentMillis); - break; - case markLocationBtnId: - SponsorBlockUtils.onMarkLocationClicked(getContext()); - break; - case publishBtnId: - SponsorBlockUtils.onPublishClicked(getContext()); - break; - case previewBtnId: - SponsorBlockUtils.onPreviewClicked(getContext()); - break; - case editByHandBtnId: - SponsorBlockUtils.onEditByHandClicked(getContext()); - break; - case hideBtnId: - hide(); - break; - } - } } diff --git a/app/src/main/java/pl/jakubweg/PlayerController.java b/app/src/main/java/pl/jakubweg/PlayerController.java index 94a3d131..fe547ac8 100644 --- a/app/src/main/java/pl/jakubweg/PlayerController.java +++ b/app/src/main/java/pl/jakubweg/PlayerController.java @@ -104,11 +104,6 @@ public class PlayerController { SkipSegmentView.hide(); NewSegmentHelperLayout.hide(); - - // add image button when starting new video - Activity activity = playerActivity.get(); - if (activity != null) - SponsorBlockUtils.addImageButton(activity, 5); } catch (Exception e) { Log.e(TAG, "Exception while initializing skip method", e); } @@ -344,14 +339,13 @@ public class PlayerController { playerActivity = new WeakReference<>((Activity) view.getContext()); if (VERBOSE) Log.d(TAG, "addSkipSponsorView15: view=" + view.toString()); + new Handler(Looper.getMainLooper()).postDelayed(new Runnable() { @Override public void run() { final ViewGroup viewGroup = (ViewGroup) ((ViewGroup) view).getChildAt(2); Activity context = ((Activity) viewGroup.getContext()); - viewGroup.addView(new SkipSegmentView(context)); - viewGroup.addView(new NewSegmentHelperLayout(context)); - SponsorBlockUtils.addImageButton(context, 40); + NewSegmentHelperLayout.context = context; } }, 500); } @@ -365,15 +359,7 @@ public class PlayerController { public void run() { final ViewGroup viewGroup = (ViewGroup) view.getParent(); Activity activity = (Activity) viewGroup.getContext(); - viewGroup.addView(new SkipSegmentView(activity)); - viewGroup.addView(new NewSegmentHelperLayout(activity)); - - // add image button when creating new activity - SponsorBlockUtils.addImageButton(activity, 5); - -// InjectedPlugin.printViewStack(viewGroup, 0); - -// SponsorBlockUtils.addImageButton(activity); + NewSegmentHelperLayout.context = activity; } }, 500); } diff --git a/app/src/main/java/pl/jakubweg/ShieldButton.java b/app/src/main/java/pl/jakubweg/ShieldButton.java index f6b7bc38..fb9e7c3f 100644 --- a/app/src/main/java/pl/jakubweg/ShieldButton.java +++ b/app/src/main/java/pl/jakubweg/ShieldButton.java @@ -13,8 +13,9 @@ import com.google.android.apps.youtube.app.YouTubeApplication; import java.lang.ref.WeakReference; +import static fi.razerman.youtube.XGlobals.debug; + public class ShieldButton { - static boolean debug = true; static String TAG = "SHIELD"; static RelativeLayout _youtubeControlsLayout; static WeakReference _shieldBtn = new WeakReference<>(null); @@ -54,7 +55,7 @@ public class ShieldButton { changeVisibilityImmediate(false); } catch (Exception ex) { - Log.e("XError", "Unable to set RelativeLayout", ex); + Log.e(TAG, "Unable to set RelativeLayout", ex); } } @@ -78,7 +79,7 @@ public class ShieldButton { if (_youtubeControlsLayout == null || iView == null) return; if (visible && shouldBeShown()) { - if (debug){ + if (debug) { Log.d(TAG, "Fading in"); } iView.setVisibility(View.VISIBLE); @@ -88,7 +89,7 @@ public class ShieldButton { } if (iView.getVisibility() == View.VISIBLE) { - if (debug){ + if (debug) { Log.d(TAG, "Fading out"); } if (!immediate) diff --git a/app/src/main/java/pl/jakubweg/SkipSegmentView.java b/app/src/main/java/pl/jakubweg/SkipSegmentView.java index 2c9edeae..cd1a891b 100644 --- a/app/src/main/java/pl/jakubweg/SkipSegmentView.java +++ b/app/src/main/java/pl/jakubweg/SkipSegmentView.java @@ -4,69 +4,28 @@ import android.annotation.SuppressLint; import android.content.Context; import android.util.DisplayMetrics; import android.util.Log; -import android.view.Gravity; -import android.view.View; -import android.widget.FrameLayout; -import android.widget.TextView; import android.widget.Toast; +import com.google.android.apps.youtube.app.YouTubeApplication; + import java.lang.ref.WeakReference; +import static fi.vanced.libraries.youtube.sponsors.player.ui.SponsorBlockView.hideSkipButton; +import static fi.vanced.libraries.youtube.sponsors.player.ui.SponsorBlockView.showSkipButton; import static pl.jakubweg.PlayerController.VERBOSE; import static pl.jakubweg.StringRef.str; @SuppressLint({"RtlHardcoded", "SetTextI18n", "LongLogTag", "AppCompatCustomView"}) -public class SkipSegmentView extends TextView implements View.OnClickListener { +public class SkipSegmentView { public static final String TAG = "jakubweg.SkipSegmentView"; - private static boolean isVisible = false; - private static WeakReference view = new WeakReference<>(null); private static SponsorSegment lastNotifiedSegment; - public SkipSegmentView(Context context) { - super(context); - isVisible = false; - setVisibility(GONE); - view = new WeakReference<>(this); - - FrameLayout.LayoutParams layoutParams = new FrameLayout.LayoutParams( - FrameLayout.LayoutParams.WRAP_CONTENT, - FrameLayout.LayoutParams.WRAP_CONTENT, - Gravity.END | Gravity.RIGHT | Gravity.CENTER_VERTICAL - ); - this.setLayoutParams(layoutParams); - this.setBackgroundColor(0x66000000); -// this.setBackgroundColor(Color.MAGENTA); - this.setTextColor(0xFFFFFFFF); - int padding = (int) convertDpToPixel(4, context); - setPadding(padding, padding, padding, padding); - - this.setText(str("tap_skip")); - - setOnClickListener(this); - } - public static void show() { - if (isVisible) return; - SkipSegmentView view = SkipSegmentView.view.get(); - if (VERBOSE) - Log.d(TAG, "show; view=" + view); - if (view != null) { - view.setVisibility(VISIBLE); - view.bringToFront(); - view.requestLayout(); - view.invalidate(); - } - isVisible = true; + showSkipButton(); } public static void hide() { - if (!isVisible) return; - SkipSegmentView view = SkipSegmentView.view.get(); - if (VERBOSE) - Log.d(TAG, "hide; view=" + view); - if (view != null) - view.setVisibility(GONE); - isVisible = false; + hideSkipButton(); } public static void notifySkipped(SponsorSegment segment) { @@ -77,19 +36,15 @@ public class SkipSegmentView extends TextView implements View.OnClickListener { } lastNotifiedSegment = segment; String skipMessage = segment.category.skipMessage.toString(); - SkipSegmentView view = SkipSegmentView.view.get(); + Context context = YouTubeApplication.getAppContext(); if (VERBOSE) - Log.d(TAG, String.format("notifySkipped; view=%s, message=%s", view, skipMessage)); - if (view != null) - Toast.makeText(view.getContext(), skipMessage, Toast.LENGTH_SHORT).show(); + Log.d(TAG, String.format("notifySkipped; message=%s", skipMessage)); + + if (context != null) + Toast.makeText(context, skipMessage, Toast.LENGTH_SHORT).show(); } public static float convertDpToPixel(float dp, Context context) { return dp * ((float) context.getResources().getDisplayMetrics().densityDpi / DisplayMetrics.DENSITY_DEFAULT); } - - @Override - public void onClick(View v) { - PlayerController.onSkipSponsorClicked(); - } } diff --git a/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java b/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java index 0d3c1f9b..844c7253 100644 --- a/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java +++ b/app/src/main/java/pl/jakubweg/SponsorBlockUtils.java @@ -46,6 +46,7 @@ import java.util.TimeZone; import static android.view.View.GONE; import static android.view.View.VISIBLE; +import static fi.razerman.youtube.XGlobals.debug; import static pl.jakubweg.PlayerController.VERBOSE; import static pl.jakubweg.PlayerController.getCurrentVideoId; import static pl.jakubweg.PlayerController.getLastKnownVideoTime; @@ -64,6 +65,9 @@ public abstract class SponsorBlockUtils { public static final View.OnClickListener sponsorBlockBtnListener = new View.OnClickListener() { @Override public void onClick(View v) { + if (debug) { + Log.d(TAG, "Shield button clicked"); + } NewSegmentHelperLayout.toggle(); } }; @@ -280,102 +284,6 @@ public abstract class SponsorBlockUtils { i.setVisibility(GONE); } - @SuppressLint("LongLogTag") - public static void addImageButton(final Activity activity, final int attemptsWhenFail) { - if (VERBOSE) - Log.d(TAG, "addImageButton activity=" + activity + ",attemptsWhenFail=" + attemptsWhenFail); - - if (activity == null) - return; - - final View existingSponsorBtn = activity.findViewById(sponsorBtnId); - if (existingSponsorBtn != null) { - if (VERBOSE) - Log.d(TAG, "addImageButton: sponsorBtn exists"); - if (/*isAddNewSegmentEnabled*/false) - showButton(); - return; - } - - String packageName = activity.getPackageName(); - Resources R = activity.getResources(); - shareBtnId = R.getIdentifier("player_share_button", "id", packageName); -// final int addToBtnId = R.getIdentifier("player_addto_button", "id", packageName); - final int addToBtnId = R.getIdentifier("live_chat_overlay_button", "id", packageName); - int titleViewId = R.getIdentifier("player_video_title_view", "id", packageName); -// final int iconId = R.getIdentifier("player_fast_forward", "drawable", packageName); - final int iconId = R.getIdentifier("ic_sb_logo", "drawable", packageName); - - - final View addToBtn = activity.findViewById(addToBtnId); - final ImageView shareBtn = activity.findViewById(shareBtnId); - final TextView titleView = activity.findViewById(titleViewId); - - if (addToBtn == null || shareBtn == null || titleView == null) { - if (VERBOSE) - Log.e(TAG, String.format("one of following is null: addToBtn=%s shareBtn=%s titleView=%s", - addToBtn, shareBtn, titleView)); - - if (attemptsWhenFail > 0) - new Handler(Looper.getMainLooper()).postDelayed(new Runnable() { - @Override - public void run() { - if (VERBOSE) - Log.i(TAG, "Retrying addImageButton"); - addImageButton(PlayerController.playerActivity.get(), attemptsWhenFail - 1); - } - }, 5000); - return; - } - - new Handler(Looper.getMainLooper()).post(new Runnable() { - @Override - public void run() { - try { - - Class touchImageViewClass = Class.forName("com.google.android.libraries.youtube.common.ui.TouchImageView"); - Constructor constructor = touchImageViewClass.getConstructor(Context.class); - final ImageView instance = ((ImageView) constructor.newInstance(activity)); - instance.setImageResource(iconId); - instance.setId(sponsorBtnId); - - RelativeLayout.LayoutParams layoutParams = new RelativeLayout.LayoutParams(shareBtn.getLayoutParams()); - layoutParams.addRule(RelativeLayout.LEFT_OF, addToBtnId); - - instance.setLayoutParams(layoutParams); - ((ViewGroup) shareBtn.getParent()).addView(instance, 0); - - - instance.setPadding(shareBtn.getPaddingLeft(), - shareBtn.getPaddingTop(), - shareBtn.getPaddingRight(), - shareBtn.getPaddingBottom()); - - - RelativeLayout.LayoutParams titleViewLayoutParams = (RelativeLayout.LayoutParams) titleView.getLayoutParams(); - titleViewLayoutParams.addRule(RelativeLayout.START_OF, sponsorBtnId); - titleView.requestLayout(); - - instance.setClickable(true); - instance.setFocusable(true); - Drawable.ConstantState constantState = shareBtn.getBackground().mutate().getConstantState(); - if (constantState != null) - instance.setBackground(constantState.newDrawable()); - - instance.setOnClickListener(sponsorBlockBtnListener); - sponsorBlockBtn = new WeakReference<>(instance); - isShown = true; - if (!/*isAddNewSegmentEnabled*/false) - hideButton(); - if (VERBOSE) - Log.i(TAG, "Image Button added"); - } catch (Exception e) { - Log.e(TAG, "Error while adding button", e); - } - } - }); - } - @SuppressLint("DefaultLocale") public static void onMarkLocationClicked(Context context) { newSponsorSegmentDialogShownMillis = PlayerController.getLastKnownVideoTime(); diff --git a/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_grey600_36.png b/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_grey600_36.png new file mode 100644 index 0000000000000000000000000000000000000000..9bf1507501324cf9518a361b99ae259ce1e19252 GIT binary patch literal 736 zcmeAS@N?(olHy`uVBq!ia0vp^6F``Q8Ax83A=Cw=ssnsNT!D0HX=y`41CRuQl9H0* z;$jFJP67pi0LTR@12W3W${C#@C)Mjv*C{Zw@vBO<-tP@OP#7N+u9bf8GYHP#g?xq>G2 z^DTp8I2{dV7cXTw`YCFGP*c&e9YRfywjK83I5m?AL^y$nBnS)WjDAf;jr1-}U5&Qn zC2mdu=R|^|0(73aoPFjLu#a8&RaDD@Ihsqh0A10s>ut+|n&)m;J}R(0^-)~PdiR}A zq+>sCz=PtRXZ42zEFCwi2Nl0Q|Id__t1jwBhUUtmCcXS6Y?pSF*?W9lGhKeIS-4xk zU)NP3eivS{i`1#Jv|JU<2fYK@hL$r>tosEv-@m3JW)78&qol`;+03l^9VE_OC literal 0 HcmV?d00001 diff --git a/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_white_36.png b/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_forward_white_36.png new file mode 100644 index 0000000000000000000000000000000000000000..6900f4365f3b1c76dcc0f20fc130c262b46b2bb6 GIT binary patch literal 690 zcmeAS@N?(olHy`uVBq!ia0vp^6F``Q4M;wBd$farfvM8d#WAFU@$GHKl+Z$nV-FQu z1Qc2rof-rbm|7SFm<|XiurP2iC^EEwF;Id*vBh9d;I$2x-q}6B`}KV#L&Nm*Z|2>% zWdFktG!-2*sGOf1v$31|q`v&5J8>n~*D`OL9DIMW%*~^IMb~Dln7?@x^C>-kl1)s> z_2%ZYmJ5XSC&g^;-hXO;tyfX{!B~0Y`>yAnTb#UXY~TG>X?NxHCl~bUPQTlyR+%Dh z|MXFu>dt)+)?K%LqhvgP@}1468GU8mCzpFYv)U~aaLo3s9=pkVzcR}O!TIcWPXak_ zGl87S@=d21&sNWh<$H0Ya8kPGx*J<3TsgCQGKjzZ^1GDj%vvStyJapcu)g`Ufeozh z1;`Lbu)bZBV|N!je15;<(!%9S)9%fEbE}bgR_*bre73sPng3+Nt-fp#5!rRQ324B9 zKi2C?US!_%{`~$jNcGYSbHnaFZLq5UCcl3FX~x`#wUZ8e-+C(lN<^gY*SdW`kq;l< zzH=?BT=AkIPHpEl_1!7GA|gyrJ^{`8Bib>$``s?ZWqH4au1$6=m#v(;_?y&(2w-TV3PC#kT7^|mj{IJs)?@oY)e>WdrA79>xNse8W>7&_M1*h3xJONHV$fMTF|#d&$>>!v5xY z0wv4W@X4-Dxl;XR(fy59KvN((-S!^W1L|IUW#fzTHGM(Hlb{N`WV?Y1G)i^0y|XQ2 eL{Ak!{s-pi-_D2V$+z4DNqV~axvXJ)BMc`6&;%vCS|WuxO$iSK-_}J{Yj?F*!J>WY`M2Ue8s_B zC&Q~#OHCDKiM(@&d%j=3xY6^Nul<=BGrn?ft2y*gK4!D&24#Yy-%e-|1B52ydmkh%|X)-l_$gff?`_FHqD>&WCxqt`Jaa}r>*{Ak#an9*6IQc zvzq6=^Rg2@%qqU;yl8br)wDksK5g0ER$womdAR(l@cr|qe=9?-|M>5Jnp>nG>hF|# z`^a^IwNXv07x{Y^^uD+(v;vAHE-qIM0Ad56!~!5raG5{tK!=@fV;2wubuftnvGT7i zELu?P_9c@u|Xj48{ zab&^4r|eDaW-`Uyj?Jg}SqhFaid%q`6`Y@>`W>vHar-m&uHB~g0e9B^Y-Gwm%P&%% zUAN#s^k?RycW23Ke7N?L>{iDQYeo4sZV#()xfhivRFU24dtWSP(c0>PNlrEkA=leI*B07p00i_>zopr0DQj}4FCWD literal 0 HcmV?d00001 diff --git a/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_rewind_white_36.png b/app/src/main/res/drawable-xxxhdpi/quantum_ic_fast_rewind_white_36.png new file mode 100644 index 0000000000000000000000000000000000000000..e1baaa331270db7ab69a9a5e3c7c89230c1270f8 GIT binary patch literal 644 zcmeAS@N?(olHy`uVBq!ia0vp^6F``Q1xWh(YZ)^zFnN2rIEGX(zB$+k6k}*u@OP#7 zN+u8!3}*Z`I?}s||M7!2lkM4>ots#kk6sILT(w)gVXwnxxs6i6&2jA+PWczOuRJjO zY3Ea#8q3?ER=bnA(kj3H46B-?`IpTJY0Jdl{Af8Gze7xT^QXq*xQ!LN)*Y;|dCWau z^Y&Ds7rOT&?P|jf`K7<<2*)tbpI-A=l*?)d7f69K!axL4b zJ@Z|zci8D3486Eqb%)!ROwN!GnVKPQGCg}vTozh+;Bt^;;_{`zk;|VNZ}K~xo-%iu zf9l+6=TBJa&OZgj6HBA)rb4mu>n$}}zqdT&H+g8txP*1rg9dGus+c;39S>*de&;CJ z@yFrA0ZI25aMQnc7$Hp>UK;z-H zKMsl8pRsf8Hnk79vG!-Qf`Vg{0EA-U jbOP}}!W<9^Bs)ie>HWFS{QYi1+92JYu6{1-oD!M + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/player_fast_rewind.xml b/app/src/main/res/drawable/player_fast_rewind.xml new file mode 100644 index 00000000..dcfeee5b --- /dev/null +++ b/app/src/main/res/drawable/player_fast_rewind.xml @@ -0,0 +1,7 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/inline_sponsor_overlay.xml b/app/src/main/res/layout/inline_sponsor_overlay.xml new file mode 100644 index 00000000..9d67ac57 --- /dev/null +++ b/app/src/main/res/layout/inline_sponsor_overlay.xml @@ -0,0 +1,9 @@ + + \ No newline at end of file diff --git a/app/src/main/res/layout/new_segment.xml b/app/src/main/res/layout/new_segment.xml new file mode 100644 index 00000000..18fad916 --- /dev/null +++ b/app/src/main/res/layout/new_segment.xml @@ -0,0 +1,16 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/skip_sponsor_button.xml b/app/src/main/res/layout/skip_sponsor_button.xml new file mode 100644 index 00000000..fc7e3e65 --- /dev/null +++ b/app/src/main/res/layout/skip_sponsor_button.xml @@ -0,0 +1,11 @@ + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 05de3704..71e9c244 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -214,6 +214,8 @@ Already read Show me + Skip sponsor + From d013f95111ea8fb1db8526de999a29a085d0faf1 Mon Sep 17 00:00:00 2001 From: xfileFIN Date: Thu, 15 Oct 2020 18:56:39 +0300 Subject: [PATCH 3/3] Skip sponsor to Skip segment --- app/src/main/res/values/strings.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 71e9c244..c5e342c8 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -214,7 +214,7 @@ Already read Show me - Skip sponsor + Skip segment