Include layout with custom attributes

后端 未结 7 864
我寻月下人不归
我寻月下人不归 2020-12-02 22:24

I\'m building a complex layout and I want to use include tag for my custom component, like this:



        
相关标签:
7条回答
  • 2020-12-02 22:54

    I ran into this issue today. For whatever it is worth, I think there is a straight-forward work around. Instead of adding attributes to the include tag, create a custom wrapper view for the include and add attributes to that. Then, do the include from the wrapper. Have the wrapper class implementation extract the attributes and pass along to its single child, which is the root view of the include layout.

    So, say we declare some custom attributes for a wrapper called SingleSettingWrapper like this -

    <declare-styleable name="SingleSettingWrapper">
        <attr name="labelText" format="string"/>
    </declare-styleable>
    

    Then, we create two custom view classes - one for the wrapper (SingleSettingWrapper) and one for the child (SingleSettingChild) that will be included -

    <!-- You will never end up including this wrapper - it will be pasted where ever you wanted to include. But since the bulk of the XML is in the child, that's ok -->
    <com.something.SingleSettingWrapper
        android:id="@+id/wrapper"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        custom:labelText="@string/my_label_string">
    
        <!-- Include the child layout -->
        <include layout="@layout/setting_single_item"/>
    
    </com.something.SingleSettingWrapper>
    

    For the child, we can put whatever complex layout in there that we want. I'll just put something basic, but really you can include whatever -

    <com.something.SingleSettingItem
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">
        <RelativeLayout >
            <!-- add whatever custom stuff here -->
            <!-- in this example there would be a text view for the label and maybe a bunch of other stuff -->
            <!-- blah blah blah -->
        </RelativeLayout>
    </com.something.SingleSettingItem>
    

    For the wrapper (this is the key), we read all of our custom attributes in the constructor. Then, we override onViewAdded() and pass those custom attributes to our child.

    public class SingleSettingWrapper extends FrameLayout 
    {
        private String mLabel;
    
        public SingleSettingWrapper(Context context, AttributeSet attrs)
        {
            super(context, attrs);
    
            TypedArray a = context.getTheme().obtainStyledAttributes(attrs,
                                                                 R.styleable.SingleSettingWrapper,
                                                                 0, 0);
    
            mLabel = a.getString(R.styleable.SingleSettingWrapper_labelText);
            a.recycle();
        }
    
        public void onViewAdded(View child)
        {
            super.onViewAdded(child);
            if (!(child instanceof SingleSettingItem))
                return;
    
           ((TextView)child.findViewById(R.id.setting_single_label)).setText(mLabel);
            /*
            Or, alternatively, call a custom method on the child implementation -
            ((SingleSettingItem)child)setLabel(mLabel);
            */
        }
    }
    

    Optionally, you can implement the child too and have it receive messages from the wrapper and modify itself (instead of having the wrapper modify the child as I did above).

    public class SingleSettingItem extends LinearLayout
    {
        public SingleSettingItem(Context context, AttributeSet attrs)
        {
            super(context, attrs);
        }
    
        public void setLabel(String l)
        {
            // set the string into the resource here if desired, for example
        }
    }
    

    At the end of the day, each of the XML files where you wanted to <include> your layout will contain about 7 lines of XML for the wrapper+include instead of the single include that you wanted, but if the included view contains hundreds of lines you're still way better off. For example -

    <LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:custom="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical" >
    
        <!-- this is the beginning of your custom attribute include -->
        <com.something.SingleSettingWrapper
            android:id="@+id/my_wrapper"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            custom:labelText="@string/auto_lock_heading">
            <include layout="@layout/setting_single_item"/>
        </com.something.SingleSettingWrapper>
        <!-- this is the end of your custom attribute include -->
    </LinearLayout>
    

    In practice, this seems to work pretty well and is relatively simple to set up. I hope it helps someone.

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