How to pass view reference to android custom view?

后端 未结 3 633
無奈伤痛
無奈伤痛 2020-12-30 00:14

I did as follows

1) Creating a styleable


    


        
相关标签:
3条回答
  • 2020-12-30 00:38

    Your described android:id is set to app:linkedView="@+id/tvTest. However, @+id/tvTest is used to create a new id with name "tvTest". What you want to do is use app:linkedView="@id/tvTest.

    0 讨论(0)
  • 2020-12-30 00:51

    I found the answer!

    The issue was with findViewById(id) and where I called it. findViewById only looks for a child view not a view exist on upper hierarchy level as documentation says . So I have to call something like getRootView().findViewById(id) but that also returns null becase where I called it was not corrent.

    In Viewee constractor Viewee itself has not attached to its root yet so that call causes NullPointerException.

    So If I call to getRootView().findViewById(id) somewhere else after constraction, it works fine and both "@+id/tvTest" and "@id/tvTest" are correct. I've tested it!

    the answer is as follows

    public class Viewee extends LinearLayout
    {
        public Viewee(Context context, AttributeSet a)
        {
            super(context, attributeSet);
            View.inflate(context, R.layout.main6, this);
            TextView textView = (TextView) findViewById(R.id.custom_text);
            TypedArray t = context.obtainStyledAttributes(a, R.styleable.Viewee);
            int id = t.getResourceId(R.styleable.Viewee_linkedView, 0);
            if (id != 0)
            {
                _id = id;
            }
    
            t.recycle();
        }
    
        private int _id;
    
        public void Foo()
        {
            TextView textView = (TextView) findViewById(R.id.custom_text);
            View view = getRootView().findViewById(_id);
            textView.setText(((TextView) view).getText().toString());
        }
    }
    

    and Foo is called when it is required to process the attached view via its reference id somewhere else in your activity and the like.

    The credit completely goes to those guys contributed in this post. I had not seen that post before submitting the question.

    0 讨论(0)
  • 2020-12-30 01:03

    I know this is an old question, but I thought I would add another way of doing this as I wanted to encapsulate everything into my custom view.

    Instead of calling from the outside, another way of getting a view higher up in the hierarchy, I've hooked into onAttachedToWindow() instead:

    public class MyCustomView extends LinearLayout {
        private int siblingResourceId;
        private View siblingView;
    
        public MyCustomView(Context context, AttributeSet a) {
            super(context, attributeSet);
            inflate(context, R.layout.main6, this);
            TextView textView = (TextView) findViewById(R.id.custom_text);
            TypedArray t = context.obtainStyledAttributes(a, R.styleable.Viewee);
            siblingResourceId = t.getResourceId(R.styleable.MyCustomView_siblingResourceId, NO_ID);
            t.recycle();
        }
    
        @Override
        public void onAttachedToWindow() {
            super.onAttachedToWindow();
            if (siblingResourceId != NO_ID) {
                siblingView = ((View) getParent()).findViewById(siblingResourceId);
            }
        }
    }
    

    onAttachedToWindow is called quite early, but apparently late enough for the whole view hierarchy to have settled. It works flawless for my needs at least and is a little bit more controlled and doesn't need interaction from outside to work ;-)

    EDIT: Kotlin code added

    class MyCustomView(context: Context, attributeSet: AttributeSet) : LinearLayout(context, attributeSet) {
        private val siblingResourceId: Int
        private lateinit var siblingView: View
    
        // All other constructors left out for brevity.
    
        init {
            inflate(context, R.layout.main6, this)
            val textView = findViewById<TextView>(R.id.custom_text)
            val t = context.obtainStyledAttributes(a, R.styleable.Viewee)
            siblingResourceId = t.getResourceId(R.styleable.MyCustomView_siblingResourceId, NO_ID)
            t.recycle()
        }
    
        override fun onAttachedToWindow() {
            super.onAttachedToWindow()
            if (siblingResourceId != NO_ID) {
                siblingView = (parent as View).findViewById(siblingResourceId) 
            }
        }
    }
    

    Note: We assume that the parent of this custom View is a View itself.

    0 讨论(0)
提交回复
热议问题