android সবর অ্যান্ড্রয়েড: ScrollView এর ভেতর একটি দৃশ্য দৃশ্যমান কিনা তা কিভাবে পরীক্ষা করবেন?




সবর (10)

public static int getVisiblePercent(View v) {
        if (v.isShown()) {
            Rect r = new Rect();
            v.getGlobalVisibleRect(r);
            double sVisible = r.width() * r.height();
            double sTotal = v.getWidth() * v.getHeight();
            return (int) (100 * sVisible / sTotal);
        } else {
            return -1;
        }
    }

আমি একটি ScrollView যা Views সিরিজের একটি ধারাবাহিক। কোনও দৃশ্য বর্তমানে দৃশ্যমান কিনা তা নির্ধারণ করতে আমি সক্ষম হব (যদি এটির কোনও অংশ বর্তমানে ScrollView দ্বারা প্রদর্শিত হয়)। আমি নীচের কোডটি এই কাজ করতে আশা করি, আশ্চর্যজনকভাবে এটি হয় না:

Rect bounds = new Rect();
view.getDrawingRect(bounds);

Rect scrollBounds = new Rect(scroll.getScrollX(), scroll.getScrollY(), 
        scroll.getScrollX() + scroll.getWidth(), scroll.getScrollY() + scroll.getHeight());

if(Rect.intersects(scrollBounds, bounds))
{
    //is  visible
}

আমি আপনার View সম্পূর্ণরূপে visible কিনা তা সনাক্ত করতে চাই, এই পদ্ধতির সাথে চেষ্টা করুন:

private boolean isViewVisible(View view) {
    Rect scrollBounds = new Rect();
    mScrollView.getDrawingRect(scrollBounds);
    float top = view.getY();
    float bottom = top + view.getHeight();
    if (scrollBounds.top < top && scrollBounds.bottom > bottom) {
        return true; //View is visible.
    } else {
        return false; //View is NOT visible.
    }
}

কঠোরভাবে বলার সাথে সাথে আপনি একটি দৃশ্যের দৃশ্যমানতা পেতে পারেন:

if (myView.getVisibility() == View.VISIBLE) {
    //VISIBLE
} else {
    //INVISIBLE
}

একটি দৃশ্যের দৃশ্যমানতার ধ্রুবক মানগুলি হল:

VISIBLE এই দৃশ্যটি দৃশ্যমান। SetVisibility (int) এবং অ্যান্ড্রয়েড: দৃশ্যমানতা সঙ্গে ব্যবহার করুন।

INVISIBLE এই দৃশ্য অদৃশ্য, কিন্তু এটি এখনও লেআউট উদ্দেশ্যে স্থান নেয়। SetVisibility (int) এবং অ্যান্ড্রয়েড: দৃশ্যমানতা সঙ্গে ব্যবহার করুন।

GONE এই দৃশ্যটি অদৃশ্য, এবং এটি লেআউট উদ্দেশ্যে কোনো স্থান নেয় না। SetVisibility (int) এবং অ্যান্ড্রয়েড: দৃশ্যমানতা সঙ্গে ব্যবহার করুন।


আমি আজ একই সমস্যা সম্মুখীন। গুগলিং এবং অ্যান্ড্রয়েড রেফারেন্স পড়ার সময় আমি এই পোস্টটি পেয়েছি এবং পদ্ধতির পরিবর্তে আমি শেষ হয়ে গেছি;

public final boolean getLocalVisibleRect (Rect r)

তাদের শুধুই রেক্ট প্রদান করা নয় বরং বুলিয়ান নির্দেশ করে যে দৃশ্যমান দৃশ্যটি দেখানো হচ্ছে। নেতিবাচক দিকে এই পদ্ধতিটি নথিভুক্ত করা হয় :(


এই এক্সটেনশান সম্পূর্ণরূপে দৃশ্যমান দেখুন সনাক্ত সাহায্য।
ScrollView (যেমন: ScrollView -> ScrollView -> ScrollView -> ... -> YourView ) এর শিশুটির শিশুটি যদি আপনার সন্তানের সন্তান হয় তবে এটিও কাজ করে।

fun ScrollView.isViewVisible(view: View): Boolean {
    val scrollBounds = Rect()
    this.getDrawingRect(scrollBounds)
    var top = 0f
    var temp = view
    while (temp !is ScrollView){
        top += (temp).y
        temp = temp.parent as View
    }
    val bottom = top + view.height
    return scrollBounds.top < top && scrollBounds.bottom > bottom
}

বিঃদ্রঃ

1) view.getY() এবং view.getX() x, y মানটি FIRST PARENT তে ফেরত দিন।

2) এখানে কিভাবে getDrawingRect ফিরে getDrawingRect তার উদাহরণ Link


@ কবার্টিকাস উত্তরটি ব্যবহার করে যা বিন্দুতে ছিল কিন্তু মহান বিটিউইউ, আমি কোডগুলির একটি গুচ্ছ সংকলন করেছিলাম যখন কোনও স্ক্রলভিউ কল করা হয় এবং স্ক্রোল করা হয় কিনা তা @ কবার্টিকাসের উত্তরটিকে ট্রিগার করে এবং আপনি যা করতে চান তা করতে পারেন, আমার ক্ষেত্রে আমার সামাজিক নেটওয়ার্কে ভিডিও ধারণ করে যাতে স্ক্রীনে দৃশ্যটি আঁকতে থাকে তবে আমি ভিডিওটি ফেসবুক এবং Instagram মত একই ধারণাটি খেলি। এখানে কোডটি রয়েছে:

mainscrollview.getViewTreeObserver().addOnScrollChangedListener(new OnScrollChangedListener() {

                    @Override
                    public void onScrollChanged() {
                        //mainscrollview is my scrollview that have inside it a linearlayout containing many child views.
                        Rect bounds = new Rect();
                         for(int xx=1;xx<=postslayoutindex;xx++)
                         {

                          //postslayoutindex is the index of how many posts are read.
                          //postslayoutchild is the main layout for the posts.
                        if(postslayoutchild[xx]!=null){

                            postslayoutchild[xx].getHitRect(bounds);

                        Rect scrollBounds = new Rect();
                        mainscrollview.getDrawingRect(scrollBounds);

                        if(Rect.intersects(scrollBounds, bounds))
                        {
                            vidPreview[xx].startPlaywithoutstoppping();
                         //I made my own custom video player using textureview and initialized it globally in the class as an array so I can access it from anywhere.
                        }
                        else
                        {

                        }


                        }
                    }
                    }
                });

আমার সমাধান NestedScrollView স্ক্রোল উপাদান ব্যবহার করা হয়:

    final Rect scrollBounds = new Rect();
    scroller.getHitRect(scrollBounds);

    scroller.setOnScrollChangeListener(new NestedScrollView.OnScrollChangeListener() {
        @Override
        public void onScrollChange(NestedScrollView v, int scrollX, int scrollY, int oldScrollX, int oldScrollY) {

            if (myBtn1 != null) {

                if (myBtn1.getLocalVisibleRect(scrollBounds)) {
                    if (!myBtn1.getLocalVisibleRect(scrollBounds)
                            || scrollBounds.height() < myBtn1.getHeight()) {
                        Log.i(TAG, "BTN APPEAR PARCIALY");
                    } else {
                        Log.i(TAG, "BTN APPEAR FULLY!!!");
                    }
                } else {
                    Log.i(TAG, "No");
                }
            }

        }
    });
}

আমি খুব দেরী জানি। কিন্তু আমি একটি ভাল সমাধান আছে। নীচে স্ক্রোল ভিউতে দৃশ্যমানতার দৃশ্য দেখতে পাওয়ার জন্য কোড স্নিপেট।

স্ক্রোল স্টপের জন্য কলব্যাক পাওয়ার জন্য স্ক্রোল ভিউতে সমস্ত সেট স্পর্শ শ্রোতাদের প্রথম।

@Override
public boolean onTouch(View v, MotionEvent event) {
    switch ( event.getAction( ) ) {
        case MotionEvent.ACTION_CANCEL:
        case MotionEvent.ACTION_UP:
            new Handler().postDelayed(new Runnable() {
                @Override
                public void run() {
                    if(mScrollView == null){
                        mScrollView = (ScrollView) findViewById(R.id.mScrollView);
                    }
                    int childCount = scrollViewRootChild.getChildCount();

                    //Scroll view location on screen
                    int[] scrollViewLocation = {0,0};
                    mScrollView.getLocationOnScreen(scrollViewLocation);

                    //Scroll view height
                    int scrollViewHeight = mScrollView.getHeight();
                    for (int i = 0; i < childCount; i++){
                        View child = scrollViewRootChild.getChildAt(i);
                        if(child != null && child.getVisibility() == View.VISIBLE){
                            int[] viewLocation = new int[2];
                            child.getLocationOnScreen(viewLocation);
                            int viewHeight = child.getHeight();
                            getViewVisibilityOnScrollStopped(scrollViewLocation, scrollViewHeight,
                                    viewLocation, viewHeight, (String) child.getTag(), (childCount - (i+1)));
                        }
                    }
                }
            }, 150);
            break;
    }
    return false;
}

উপরের কোড স্নিপেটে, আমরা স্ক্রোল ভিউ স্পর্শ ইভেন্টগুলির জন্য কল ব্যাকগুলি পেয়ে যাচ্ছি এবং স্ক্রোল বন্ধ করার জন্য কলব্যাক পাওয়ার পরে 150 মিলি (বাধ্যতামূলক নয়) পরে রানযোগ্য। যে runnable আমরা পর্দায় স্ক্রল ভিউ অবস্থান পাবেন এবং দেখার উচ্চতা স্ক্রোল পাবেন। তারপরে স্ক্রোল দৃশ্যের সরাসরি সন্তানের ভিউগ্রুপের উদাহরণটি পান এবং শিশুটিকে গণনা করুন। আমার ক্ষেত্রে স্ক্রোল ভিউয়ের সরাসরি সন্তানটি স্ক্রল ভিউউইটChild নামে লিনিয়ারআউট । তারপর scrollViewRootChild এর সব সন্তানের মতামত পুনরাবৃত্তি করুন । উপরের কোড স্নিপেটে আপনি দেখতে পারেন যে আমি স্ক্রিনে সন্তানের অবস্থানটি লোকেশন নামে একটি পূর্ণসংখ্যা অ্যারে পেয়ে যাচ্ছি , পরিবর্তনশীল নাম দর্শনে দেখুন উচ্চতা পান । তারপর আমি একটি ব্যক্তিগত পদ্ধতি পেতে ভিভিভিভেসিবিলিটিঅনস্ক্রোলসস্টপড । আপনি ডকুমেন্টেশন পড়া এই পদ্ধতির অভ্যন্তরীণ কাজ বোঝার পেতে পারেন।

/**
 * getViewVisibilityOnScrollStopped
 * @param scrollViewLocation location of scroll view on screen
 * @param scrollViewHeight height of scroll view
 * @param viewLocation location of view on screen, you can use the method of view claas's getLocationOnScreen method.
 * @param viewHeight height of view
 * @param tag tag on view
 * @param childPending number of views pending for iteration.
 */
void getViewVisibilityOnScrollStopped(int[] scrollViewLocation, int scrollViewHeight, int[] viewLocation, int viewHeight, String tag, int childPending) {
    float visiblePercent = 0f;
    int viewBottom = viewHeight + viewLocation[1]; //Get the bottom of view.
    if(viewLocation[1] >= scrollViewLocation[1]) {  //if view's top is inside the scroll view.
        visiblePercent = 100;
        int scrollBottom = scrollViewHeight + scrollViewLocation[1];    //Get the bottom of scroll view 
        if (viewBottom >= scrollBottom) {   //If view's bottom is outside from scroll view
            int visiblePart = scrollBottom - viewLocation[1];  //Find the visible part of view by subtracting view's top from scrollview's bottom  
            visiblePercent = (float) visiblePart / viewHeight * 100;
        }
    }else{      //if view's top is outside the scroll view.
        if(viewBottom > scrollViewLocation[1]){ //if view's bottom is outside the scroll view
            int visiblePart = viewBottom - scrollViewLocation[1]; //Find the visible part of view by subtracting scroll view's top from view's bottom
            visiblePercent = (float) visiblePart / viewHeight * 100;
        }
    }
    if(visiblePercent > 0f){
        visibleWidgets.add(tag);        //List of visible view.
    }
    if(childPending == 0){
        //Do after iterating all children.
    }
}

যদি আপনি এই কোড কোন উন্নতি অনুভব অনুগ্রহ করে অবদান।


View#getHitRect ব্যবহার View#getHitRect আপনি যে দর্শনের পরীক্ষা করছেন। আপনি স্পষ্টভাবে গণনা করার পরিবর্তে ScrollView View#getDrawingRect ব্যবহার করতে পারেন।

কোড থেকে View#getDrawingRect :

 public void getDrawingRect(Rect outRect) {
        outRect.left = mScrollX;
        outRect.top = mScrollY;
        outRect.right = mScrollX + (mRight - mLeft);
        outRect.bottom = mScrollY + (mBottom - mTop);
 }

কোড থেকে View#getHitRect :

public void getHitRect(Rect outRect) {
        outRect.set(mLeft, mTop, mRight, mBottom);
}

আপনি FocusAwareScrollView ব্যবহার করতে পারেন যা FocusAwareScrollView দৃশ্যমান হওয়ার পরে সূচিত হয়:

FocusAwareScrollView focusAwareScrollView = (FocusAwareScrollView) findViewById(R.id.focusAwareScrollView);
    if (focusAwareScrollView != null) {

        ArrayList<View> viewList = new ArrayList<>();
        viewList.add(yourView1);
        viewList.add(yourView2);

        focusAwareScrollView.registerViewSeenCallBack(viewList, new FocusAwareScrollView.OnViewSeenListener() {

            @Override
            public void onViewSeen(View v, int percentageScrolled) {

                if (v == yourView1) {

                    // user have seen view1

                } else if (v == yourView2) {

                    // user have seen view2
                }
            }
        });

    }

এখানে শ্রেণী:

import android.content.Context;
import android.graphics.Rect;
import android.support.v4.widget.NestedScrollView;
import android.util.AttributeSet;
import android.view.View;

import java.util.ArrayList;
import java.util.List;

public class FocusAwareScrollView extends NestedScrollView {

    private List<OnScrollViewListener> onScrollViewListeners = new ArrayList<>();

    public FocusAwareScrollView(Context context) {
        super(context);
    }

    public FocusAwareScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public FocusAwareScrollView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    public interface OnScrollViewListener {
        void onScrollChanged(FocusAwareScrollView v, int l, int t, int oldl, int oldt);
    }

    public interface OnViewSeenListener {
        void onViewSeen(View v, int percentageScrolled);
    }

    public void addOnScrollListener(OnScrollViewListener l) {
        onScrollViewListeners.add(l);
    }

    public void removeOnScrollListener(OnScrollViewListener l) {
        onScrollViewListeners.remove(l);
    }

    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        for (int i = onScrollViewListeners.size() - 1; i >= 0; i--) {
            onScrollViewListeners.get(i).onScrollChanged(this, l, t, oldl, oldt);
        }
        super.onScrollChanged(l, t, oldl, oldt);
    }

    @Override
    public void requestChildFocus(View child, View focused) {
        super.requestChildFocus(child, focused);
    }

    private boolean handleViewSeenEvent(View view, int scrollBoundsBottom, int scrollYOffset,
                                        float minSeenPercentage, OnViewSeenListener onViewSeenListener) {
        int loc[] = new int[2];
        view.getLocationOnScreen(loc);
        int viewBottomPos = loc[1] - scrollYOffset + (int) (minSeenPercentage / 100 * view.getMeasuredHeight());
        if (viewBottomPos <= scrollBoundsBottom) {
            int scrollViewHeight = this.getChildAt(0).getHeight();
            int viewPosition = this.getScrollY() + view.getScrollY() + view.getHeight();
            int percentageSeen = (int) ((double) viewPosition / scrollViewHeight * 100);
            onViewSeenListener.onViewSeen(view, percentageSeen);
            return true;
        }
        return false;
    }

    public void registerViewSeenCallBack(final ArrayList<View> views, final OnViewSeenListener onViewSeenListener) {

        final boolean[] viewSeen = new boolean[views.size()];

        FocusAwareScrollView.this.postDelayed(new Runnable() {
            @Override
            public void run() {

                final Rect scrollBounds = new Rect();
                FocusAwareScrollView.this.getHitRect(scrollBounds);
                final int loc[] = new int[2];
                FocusAwareScrollView.this.getLocationOnScreen(loc);

                FocusAwareScrollView.this.setOnScrollChangeListener(new NestedScrollView.OnScrollChangeListener() {

                    boolean allViewsSeen = true;

                    @Override
                    public void onScrollChange(NestedScrollView v, int x, int y, int oldx, int oldy) {

                        for (int index = 0; index < views.size(); index++) {

                            //Change this to adjust criteria
                            float viewSeenPercent = 1;

                            if (!viewSeen[index])
                                viewSeen[index] = handleViewSeenEvent(views.get(index), scrollBounds.bottom, loc[1], viewSeenPercent, onViewSeenListener);

                            if (!viewSeen[index])
                                allViewsSeen = false;
                        }

                        //Remove this if you want continuous callbacks
                        if (allViewsSeen)
                            FocusAwareScrollView.this.setOnScrollChangeListener((NestedScrollView.OnScrollChangeListener) null);
                    }
                });
            }
        }, 500);
    }
}

GetLocalVisibleRect ব্যবহার করে বিল মোটের উত্তরের একটি বিট প্রসারিত করতে, আপনি দেখতে পারেন যে দৃশ্য শুধুমাত্র আংশিকভাবে দৃশ্যমান কিনা:

Rect scrollBounds = new Rect();
scrollView.getHitRect(scrollBounds);
if (!imageView.getLocalVisibleRect(scrollBounds)
    || scrollBounds.height() < imageView.getHeight()) {
    // imageView is not within or only partially within the visible window
} else {
    // imageView is completely visible
}




visible