Android: कैसे देखें कि स्क्रोल व्यू के अंदर का दृश्य दिखाई दे रहा है या नहीं?


168

मेरे पास एक 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
}

मैं उत्सुक हूं कि आपको यह काम करने के लिए कैसे मिला। मैं एक ही काम करने की कोशिश कर रहा हूं, लेकिन एक स्क्रॉल दृश्य केवल 1 प्रत्यक्ष बच्चे की मेजबानी कर सकता है। क्या आपके "दृश्यों की श्रृंखला" स्क्रॉलव्यू के अंदर किसी अन्य लेआउट में लिपटी हुई है? इस तरह मेरा काम पूरा हो गया है, लेकिन जब मैं ऐसा करता हूं तो यहां दिए गए उत्तरों में से कोई भी मेरे लिए काम नहीं करता है।
मुर्गा 242

1
हां, मेरे सीरीज़ ऑफ़ व्यूज़ एक लीनियर लयआउट के अंदर हैं, जो कि स्क्रॉलव्यू का 1 बच्चा है। क्यूबर्टिकस के जवाब ने मेरे लिए काम किया।
ab11

जवाबों:


65

आपके द्वारा देखे जा रहे दृश्य के View#getHitRectबजाय उपयोग करें View#getDrawingRect। आप स्पष्ट रूप से गणना करने के बजाय View#getDrawingRectपर उपयोग कर सकते हैं ScrollView

से कोड 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);
}

35
मुझे इस विधियों को कहां से कॉल करना चाहिए?
तोतो

3
@Qberticus विधियों को कैसे कॉल करें? मैं इसका इस्तेमाल कर रहा हूं और यह हमेशा गलत है। कृपया मुझे बताएं
KK_07k11A0585

2
वास्तव में इन विधियों को कहां से कॉल करें?
जेमाइटिस

193

यह काम:

Rect scrollBounds = new Rect();
scrollView.getHitRect(scrollBounds);
if (imageView.getLocalVisibleRect(scrollBounds)) {
    // Any portion of the imageView, even a single pixel, is within the visible window
} else {
    // NONE of the imageView is within the visible window
}

1
अच्छी तरह से काम। इसे स्पष्ट करने के लिए: यदि दृश्य पूरी तरह से या आंशिक रूप से दिखाई देता है तो यह सच है; असत्य का अर्थ है कि दृश्य पूरी तरह से दिखाई नहीं दे रहा है।
qwertzguy

1
[1] मैं पाने के लिए इस कोड का इस्तेमाल किया GridView/ ListView/ GridViewWithHeaderके साथ काम SwipeRefreshLayout
कार्तिक

क्या कोई समझा सकता है कि यह क्यों काम करता है? getHitRectमूल निर्देशांक में getLocalVisibleRectएक आयत देता है , लेकिन स्क्रॉलव्यू के स्थानीय निर्देशांक में एक प्रतिफल देता है, है ना?
पिन

3
यह ओवरलैप को कवर नहीं करता है, अगर चाइल्ड व्यू किसी अन्य बाल तत्व द्वारा ओवरलैप किया जाता है, तो यह अभी भी सच हो जाएगा
प्रदीप

1
हां, हमें Rect की एक उदाहरण की आवश्यकता है। लेकिन यह आवश्यक है कि GetHitRect प्राप्त करें। क्या कोई भिन्न है अगर मैं एक Rect (0,0-0,0) का उपयोग करता हूं। हम getLocalVoubleRect कॉल getGlobalVanishRect देख सकते हैं। और Rect को r.set (0, 0, चौड़ाई, ऊंचाई) पर सेट किया गया है। @ BillMote
chefish

56

यदि आप यह देखना चाहते हैं कि दृश्य पूरी तरह से दिखाई दे रहा है:

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;
    } else {
        return false;
    }
}

6
यह सही उत्तर है =) मेरे मामले में मैंने इस तरह से परिवर्तन किया है: स्क्रॉलबॉन्ड्स ।टॉप = = टॉप एंड& स्क्रॉलबाउंड्स।बॉटम => नीचे
हेल्टन इसाक

2
+1 हेल्टन यदि आपका दृश्य या तो आपके स्क्रॉल दृश्य के ऊपर या नीचे के खिलाफ धकेल दिया जाता है तो आपको क्रमशः <= या> = की आवश्यकता होगी
जो मैहर

क्या आपने वास्तव में इसका परीक्षण किया है? यह हमेशा एक बच्चे के रूप में सबसे सरल लेआउट स्क्रॉलव्यू और टेक्स्ट व्यू में गलत होता है।
फरीद

1
GetHitRect () और getDrawingRect () के बीच क्या अंतर है? कृपया मार्गदर्शन करें
VVB

2
यह कोड केवल तभी काम करता है जब व्यू को स्क्रॉलव्यू कंटेनर के रूट में सीधे जोड़ा जाता है। यदि आप किसी चाइल्ड व्यू आदि में बच्चे के दृश्य को संभालना चाहते हैं तो फान वान लिन्ह के उत्तर की जाँच करें
Thijsonline

12

मेरा समाधान 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");
                }
            }

        }
    });
}

API की आवश्यकता है 23+
सॉलिडस्नेक

@SolidSnake, नहीं, आपको अलग-अलग वर्ग का आयात करने की आवश्यकता है, यह ठीक काम करता है
पार्थ अंजारिया

10

GetLocalVanishRect का उपयोग करके बिल मॉट के उत्तर पर थोड़ा विस्तार करने के लिए, आप जांचना चाहते हैं कि क्या दृश्य केवल सामान्य रूप से हो सकता है:

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
}

6
यह काम नहीं करता है .. यहां तक ​​कि आंशिक रूप से दिखाई देने वाला दृश्य भी पूरी तरह से दृश्यमान रूप में वर्गीकृत किया गया है
azfar

10

यह एक्सटेंशन पूरी तरह से दृश्यमान दृश्य का पता लगाने में मदद करता है ।
यह भी काम करता है यदि आपका Viewबच्चा ... ScrollView(जैसे: ScrollView-> LinearLayout-> ContraintLayout-> ... ... - 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()एक्स, करने के लिए y मान सबसे पहले PARENT

2) यहाँ उदाहरण है कि कैसे लिंकgetDrawingRect वापस आएगा यहां छवि विवरण दर्ज करें


मैं एक ऐसा समाधान चाहता था, जहां कीबोर्ड के तहत दृश्य छिपे होने पर विधि झूठी वापस आ जाए और यह काम करता है। धन्यवाद।
राहुल

8
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;
        }
    }

2
यह क्या ab11 के लिए कहा से अलग है। isShown () केवल दृश्यता ध्वज के लिए जाँच करता है, यह देखने के लिए कि स्क्रीन के दृश्य क्षेत्र में है या नहीं।
रोमेन गाइ

4
@Romain Guy कोड को कवर नहीं करता है जब कोई दृश्य स्क्रीन से दूर स्क्रॉल किया जाता है। यह `पब्लिक स्टैटिक इंट गेटवेबरेन्सेन्ट (दृश्य v) {if (v.isShown ()) {Rect r = new perect () होना चाहिए; बूलियन। अदृश्य = v.getGlobalVanishRect (r); if (अदृश्य) {double sV अदृश्य = r.width () * r.height (); double sTotal = v.getWidth () * v.getHeight (); वापसी (int) (100 * sV अदृश्य / sTotal); } और {वापसी -1; }} और {वापसी -1; }} `
रसोइया

6

मैंने आज उसी समस्या का सामना किया। Googling और Android संदर्भ पढ़ते हुए मुझे यह पोस्ट और एक विधि मिली जिसके बजाय मैंने इसका उपयोग किया;

public final boolean getLocalVisibleRect (Rect r)

उनमें से न केवल रेक्ट प्रदान करना, बल्कि बूलियन भी इंगित करता है कि यदि दृश्य बिल्कुल दिखाई दे। नकारात्मक पक्ष पर यह विधि अनिर्दिष्ट है :(


1
यह केवल आपको बताता है कि आइटम दृश्यता (सत्य) पर सेट है या नहीं। यह आपको नहीं बताता है कि "दृश्यमान" आइटम वास्तव में व्यूपोर्ट के भीतर दिखाई दे रहा है या नहीं।
बिल मोते

GetLocalVoubleRect का कोड आपके दावे का समर्थन नहीं करता है: `सार्वजनिक अंतिम बूलियन getLocalVoubleRect (Rect r) {अंतिम बिंदु ऑफसेट = mAttachInfo! = Null mAttachInfo.mPoint: नया बिंदु (); if (getGlobalVanishRect (r, ऑफसेट)) {r.offset (-offset.x, -offset.y); // r स्थानीय वापसी को सच करें; } विवरण झूठा है; } `
mbafford

6

यदि आप 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
}

दृश्य में दृश्यता के पॉसिबल स्थिर मान हैं:

दृश्य यह दृश्य दिखाई देता है। SetVisibility (int) और Android: दृश्यता के साथ उपयोग करें।

INVISIBLE यह दृश्य अदृश्य है, लेकिन यह अभी भी लेआउट उद्देश्यों के लिए जगह लेता है। SetVisibility (int) और Android: दृश्यता के साथ उपयोग करें।

चला गया यह दृश्य अदृश्य है, और यह लेआउट प्रयोजनों के लिए किसी भी स्थान नहीं ले करता है। SetVisibility (int) और Android: दृश्यता के साथ उपयोग करें।


3
धीमी ताली। ओपी क्या जानना चाहता था, दृश्य की दृश्यता को देखते हुए देखें # विज़िबल, यह कैसे पता चलेगा कि दृश्य स्वयं एक स्क्रॉलव्यू के भीतर दिखाई दे रहा है या नहीं।
जोआओ सूसा

1
मैंने सिर्फ एक साधारण प्रोजेक्ट पर जाँच की। लेआउट में एक बच्चे के रूप में स्क्रॉलव्यू और टेक्स्ट व्यू है; टेक्स्ट व्यू पूरी तरह से दिखाई देने पर भी हमेशा गलत होता है।
फरीद

यह हमेशा झूठा लौटता है।
राहुल

3

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);
    }
}

1

कोटलिन रास्ता;

स्क्रॉल दृश्य की स्क्रॉलिंग को सूचीबद्ध करने के लिए एक एक्सटेंशन और एक कार्रवाई प्राप्त करें यदि बच्चा स्क्रीन पर दिखाई देता है।

@SuppressLint("ClickableViewAccessibility")
fun View.setChildViewOnScreenListener(view: View, action: () -> Unit) {
    val visibleScreen = Rect()

    this.setOnTouchListener { _, motionEvent ->
        if (motionEvent.action == MotionEvent.ACTION_MOVE) {
            this.getDrawingRect(visibleScreen)

            if (view.getLocalVisibleRect(visibleScreen)) {
                action()
            }
        }

        false
    }
}

किसी भी स्क्रॉल दृश्य के लिए इस एक्सटेंशन फ़ंक्शन का उपयोग करें

nestedScrollView.setChildViewOnScreenListener(childView) {
               action()
            }

0

मुझे इसका पता बहुत देर से चला। लेकिन मेरे पास एक अच्छा उपाय है। नीचे स्क्रॉल दृश्य में दृश्यता प्रतिशत प्राप्त करने के लिए कोड स्निपेट है।

स्क्रॉल स्टॉप के लिए कॉलबैक प्राप्त करने के लिए स्क्रॉल व्यू पर सबसे पहले सेट टच श्रोता।

@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 मिली (अनिवार्य नहीं) के बाद एक रनवेबल पोस्ट कर सकते हैं। उस रननेबल में हमें स्क्रीन पर स्क्रॉल दृश्य और स्क्रॉल दृश्य ऊंचाई का स्थान मिलेगा। फिर स्क्रॉल दृश्य का प्रत्यक्ष चाइल्ड व्यूग्रुप उदाहरण प्राप्त करें और बच्चे को गिनें। मेरे मामले में स्क्रॉल दृश्य के प्रत्यक्ष बच्चे का नाम LinearLayout स्क्रॉलव्यूरूटचिल्ड है । तब के सभी बच्चे विचारों पुनरावृति scrollViewRootChild । उपरोक्त कोड स्निपेट में आप देख सकते हैं कि मैं स्क्रीन पर बच्चे का स्थान देख रहा हूं जिसे व्यूलेटरल नाम के एक पूर्णांक सरणी में देखा जा सकता है , चर नाम व्यूहाइट में दृश्य की ऊंचाई प्राप्त करें । तब मैंने एक निजी विधि getViewVisibilityOnScrollStontin को बुलाया। दस्तावेज़ीकरण पढ़कर आप इस पद्धति के आंतरिक कार्य की समझ प्राप्त कर सकते हैं।

/**
 * 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.
    }
}

यदि आप इस कोड में कोई सुधार महसूस करते हैं तो कृपया योगदान करें।


0

मैं एक जावा जवाब में से दो के संयोजन (@ बिल-धूलि का कण को लागू करने को समाप्त https://stackoverflow.com/a/12428154/3686125 और @ डेनिस-वैसिलेंको https://stackoverflow.com/a/25528434/3686125 में) कोटलिन एक्सटेंशन के एक सेट के रूप में मेरी परियोजना, जो मानक वर्टिकल स्क्रॉलव्यू या हॉरिज़ॉन्टलसर्कल व्यू नियंत्रण का समर्थन करती है।

मैंने एक्सटेंशन्स.kt नाम की एक कोटलिन फ़ाइल में बस इनको फेंक दिया, कोई क्लास नहीं, सिर्फ तरीके।

जब किसी उपयोगकर्ता ने मेरी परियोजना में विभिन्न स्क्रॉलव्यू में स्क्रॉल करना बंद कर दिया था, तो यह निर्धारित करने के लिए मैंने उनका उपयोग किया था:

fun View.isPartiallyOrFullyVisible(horizontalScrollView: HorizontalScrollView) : Boolean {
    @Suppress("CanBeVal") var scrollBounds = Rect()
    horizontalScrollView.getHitRect(scrollBounds)
    return getLocalVisibleRect(scrollBounds)
}

fun View.isPartiallyOrFullyVisible(scrollView: ScrollView) : Boolean {
    @Suppress("CanBeVal") var scrollBounds = Rect()
    scrollView.getHitRect(scrollBounds)
    return getLocalVisibleRect(scrollBounds)
}

fun View.isFullyVisible(horizontalScrollView: HorizontalScrollView) : Boolean {
    @Suppress("CanBeVal") var scrollBounds = Rect()
    horizontalScrollView.getDrawingRect(scrollBounds)
    val left = x
    val right = left + width
    return scrollBounds.left < left && scrollBounds.right > right
}

fun View.isFullyVisible(scrollView: ScrollView) : Boolean {
    @Suppress("CanBeVal") var scrollBounds = Rect()
    scrollView.getDrawingRect(scrollBounds)
    val top = y
    val bottom = top + height
    return scrollBounds.top < top && scrollBounds.bottom > bottom
}

fun View.isPartiallyVisible(horizontalScrollView: HorizontalScrollView) : Boolean = isPartiallyOrFullyVisible(horizontalScrollView) && !isFullyVisible(horizontalScrollView)
fun View.isPartiallyVisible(scrollView: ScrollView) : Boolean = isPartiallyOrFullyVisible(scrollView) && !isFullyVisible(scrollView)

उदाहरण का उपयोग, स्क्रॉलव्यू के रैखिकलेआउट बच्चों और लॉगिंग आउटपुट के माध्यम से पुनरावृति:

val linearLayoutChild: LinearLayout = getChildAt(0) as LinearLayout
val scrollView = findViewById(R.id.scroll_view) //Replace with your scrollview control or synthetic accessor
for (i in 0 until linearLayoutChild.childCount) {
    with (linearLayoutChild.getChildAt(i)) {
        Log.d("ScrollView", "child$i left=$left width=$width isPartiallyOrFullyVisible=${isPartiallyOrFullyVisible(scrollView)} isFullyVisible=${isFullyVisible(scrollView)} isPartiallyVisible=${isPartiallyVisible(scrollView)}")
    }
}

1
आप varide hint का उपयोग और दमन क्यों कर रहे हैं ?
फिलिप्पोविक्ज़

-1

@Qberticus उत्तर का उपयोग करते हुए जो बिंदु पर था, लेकिन महान btw, मैंने यह जांचने के लिए कोड का एक गुच्छा संकलित किया कि क्या जब भी एक स्क्रॉलव्यू कहा जाता है और स्क्रॉल किया जाता है तो यह @Qberticus उत्तर को ट्रिगर करता है और आप जो चाहें कर सकते हैं, मेरे पास मेरे पास है सोशल नेटवर्क जिसमें वीडियो होते हैं, जब स्क्रीन पर दृश्य खींचा जाता है तो मैं वीडियो को फेसबुक और इंस्टाग्राम की तरह ही चलाता हूं। यहाँ कोड है:

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
                        {

                        }


                        }
                    }
                    }
                });
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.