Fragment Inside Fragment


137

ฉันต้องการความช่วยเหลือเกี่ยวกับการทำงานกับชิ้นส่วนภายในชิ้นส่วนที่จริงฉันกำลังประสบปัญหาในการกดปุ่มย้อนกลับ แอปพลิเคชันหน้าจอหลักมีปุ่มและการกดบนมุมมองแต่ละปุ่มแทนที่ด้วยแฟรกเมนต์ใหม่ (และแฟรกเมนต์นั้นมีอยู่ในแฟรกเมนต์อื่น) การเพิ่ม / แทนที่แฟรกเมนต์แบบไดนามิกทำงานได้ดีโดยการกด button1 แฟรกเมนต์แทน ปุ่มอีกครั้งมีข้อยกเว้น:

"Duplicate id 0x7f05000a, tag null, or parent id 0x7f050009 with
another fragment for com........ fragmentname"

หมายความว่ามีการเพิ่มแฟรกเมนต์หรือแฟรกเมนต์ภายในแล้วและฉันพยายามเพิ่มอีกครั้งใคร ๆ ก็มีความคิดว่าจะทำงานกับแฟรกเมนต์ภายในแฟรกเมนต์ได้อย่างไรและเคลื่อนที่ไปมาโดยไม่มีปัญหาขอบคุณสำหรับการสนับสนุน

MainActivity ที่ชิ้นส่วนจะถูกเพิ่มและแทนที่แบบไดนามิก

public class FragmentInsideFragmentTestActivity extends Activity {

    private Button button1;
    private Button button2;
    private Button button3;
    private Button button4;


    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        button1 =(Button) this.findViewById(R.id.button1);
        button1.setOnClickListener(new View.OnClickListener() {
           public void onClick(View view) {
               onButtonClick(view);
            }
        });

        button2 =(Button) this.findViewById(R.id.button2);
        button2.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view) {
                onButtonClick(view);
            }
        });

        button3 =(Button) this.findViewById(R.id.button3);
        button3.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view) {
               onButtonClick(view);
            }
        });

        button4 =(Button) this.findViewById(R.id.button4);
        button4.setOnClickListener(new View.OnClickListener() {
           public void onClick(View view) {
               onButtonClick(view);
           }
        });
    }

    public void onButtonClick(View v) {
        Fragment fg;

        switch (v.getId()) {
           case R.id.button1:
                   fg=FirstFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button2:
                   fg=SecondFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button3:
                   fg=FirstFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button4:
                   fg=SecondFragment.newInstance();
                   replaceFragment(fg);
                   break;
        }
    }

    private void replaceFragment(Fragment newFragment) {
       FragmentTransaction trasection = getFragmentManager().beginTransaction();

        if(!newFragment.isAdded()) {
            try {
                //FragmentTransaction trasection =
                getFragmentManager().beginTransaction();
                trasection.replace(R.id.linearLayout2, newFragment);
                trasection.addToBackStack(null);
                trasection.commit();
            } catch (Exception e) {
                // TODO: handle exception
                // AppConstants.printLog(e.getMessage());
            } else {
                trasection.show(newFragment);
            }
        }
    }

นี่คือเค้าโครง: main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical">

    <LinearLayout
        android:id="@+id/linearLayout1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">

        <Button
            android:id="@+id/button1"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Button1" />

        <Button
            android:id="@+id/button2"
            android:text="Button2"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />

        <Button
            android:id="@+id/button3"
            android:text="Button3"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />

        <Button
            android:id="@+id/button4"
            android:text="Button4"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />
    </LinearLayout>

    <LinearLayout
        android:id="@+id/linearLayout2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal" />
</LinearLayout>

หวังว่าฉันพยายามที่จะแก้ปัญหาของฉัน


1
รหัสดังกล่าวทำงานได้อย่างสมบูรณ์แบบดีสำหรับฉันกับ Android 3.1
Vivek


สำหรับรายละเอียดเพิ่มเติมดูstackoverflow.com/a/11020531/1219456
Raneez Ahmed

คำตอบ:


284

AFAIK ชิ้นส่วนไม่สามารถเก็บชิ้นส่วนอื่นได้


UPDATE

กับรุ่นปัจจุบันของแพคเกจสนับสนุน Android - หรือเศษพื้นเมือง API ระดับ 17 และสูงกว่า - getChildFragmentManager()คุณสามารถเศษรังโดยใช้วิธีการ โปรดทราบว่านี้หมายถึงว่าคุณต้องใช้รุ่นแพคเกจสนับสนุน Android ของชิ้นส่วนในระดับ API 11-16 getChildFragmentManager()เพราะแม้ว่าจะมีรุ่นที่ชนพื้นเมืองของเศษบนอุปกรณ์เหล่านั้นรุ่นที่ไม่ได้มี


30
แพลตฟอร์มดูดจริงๆที่นี่ ฉันใช้เวลาแก้จุดบกพร่องนี้สามชั่วโมงเพราะชิ้นส่วนด้านในจะแสดงผลได้ดีในครั้งแรก แต่จะหายไปหลังจากการเปลี่ยนแนวหน้าจอ ไม่มีข้อยกเว้นบันทึกข้อมูลไม่มีอะไร การสลับไปมาgetChildFragmentManager()และการถอดออกsetRetainInstance(true)จากชิ้นส่วนด้านใน (สงสาร) แก้ไขมัน ขอบคุณสำหรับการบันทึกเบคอนของฉันอีกครั้ง @CommonsWare
เฟลิกซ์

82

ป้อนคำอธิบายรูปภาพที่นี่

ฉันต้องการบริบทเพิ่มเติมดังนั้นฉันจึงทำตัวอย่างเพื่อแสดงให้เห็นว่าการทำเช่นนี้เป็นอย่างไร สิ่งที่มีประโยชน์ที่สุดที่ฉันอ่านขณะเตรียมคือ:

กิจกรรม

activity_main.xml

เพิ่ม a FrameLayoutไปยังกิจกรรมของคุณเพื่อเก็บชิ้นส่วนหลัก

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Activity"/>

    <FrameLayout
        android:id="@+id/parent_fragment_container"
        android:layout_width="match_parent"
        android:layout_height="200dp"/>

 </LinearLayout>

MainActivity.java

โหลดแฟรกเมนต์พาเรนต์และนำแฟรกเมนต์ listeners (ดูการสื่อสารชิ้นส่วน )

import android.support.v4.app.FragmentTransaction;
import android.support.v7.app.AppCompatActivity;

public class MainActivity extends AppCompatActivity implements ParentFragment.OnFragmentInteractionListener, ChildFragment.OnFragmentInteractionListener {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // Begin the transaction
        FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
        ft.replace(R.id.parent_fragment_container, new ParentFragment());
        ft.commit();
    }

    @Override
    public void messageFromParentFragment(Uri uri) {
        Log.i("TAG", "received communication from parent fragment");
    }

    @Override
    public void messageFromChildFragment(Uri uri) {
        Log.i("TAG", "received communication from child fragment");
    }
}

ผู้ปกครองส่วน

fragment_parent.xml

เพิ่มFrameLayoutคอนเทนเนอร์อื่นสำหรับแฟรกเมนต์ชายด์

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:layout_margin="20dp"
              android:background="#91d0c2">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Parent fragment"/>

    <FrameLayout
        android:id="@+id/child_fragment_container"
        android:layout_width="match_parent"
        android:layout_height="match_parent">

    </FrameLayout>

</LinearLayout>

ParentFragment.java

ใช้getChildFragmentManagerในonViewCreatedการตั้งค่าส่วนย่อยของเด็ก

import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentTransaction;    

public class ParentFragment extends Fragment {

    private OnFragmentInteractionListener mListener;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.fragment_parent, container, false);
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        Fragment childFragment = new ChildFragment();
        FragmentTransaction transaction = getChildFragmentManager().beginTransaction();
        transaction.replace(R.id.child_fragment_container, childFragment).commit();
    }


    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof OnFragmentInteractionListener) {
            mListener = (OnFragmentInteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }

    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        void messageFromParentFragment(Uri uri);
    }
}

ชิ้นส่วนของเด็ก

fragment_child.xml

ไม่มีอะไรพิเศษที่นี่

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:layout_margin="20dp"
              android:background="#f1ff91">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Child fragment"/>
</LinearLayout>

ChildFragment.java

ไม่มีอะไรพิเศษเกินไปที่นี่เช่นกัน

import android.support.v4.app.Fragment;

public class ChildFragment extends Fragment {

    private OnFragmentInteractionListener mListener;


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        return inflater.inflate(R.layout.fragment_child, container, false);
    }

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof OnFragmentInteractionListener) {
            mListener = (OnFragmentInteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }


    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        void messageFromChildFragment(Uri uri);
    }
}

หมายเหตุ


67

เนื่องจาก Android 4.2 (API 17) มีแฟรกเมนต์ซ้อนกันให้ใช้งานได้http://developer.android.com/about/versions/android-4.2.html#NestedFragments

เมื่อต้องการวางแฟรกเมนต์ในแฟรกเมนต์อื่นให้ใช้ getChildFragmentManager ()

มันยังมีอยู่ในห้องสมุดสนับสนุน!


โปรดอัปเดตคำตอบของคุณเพื่อกล่าวถึงการสนับสนุน lib ที่สนับสนุน fragements ซ้อนกันจาก Android 1.6+ แล้ว
FindOutIslamNow

13

สามารถเพิ่มชิ้นส่วนในชิ้นส่วนอื่น ๆ ได้ แต่คุณจะต้องลบออกจากชิ้นส่วนหลักทุกครั้งที่onDestroyView()มีการเรียกใช้เมธอดชิ้นส่วนหลัก และเพิ่มอีกครั้งในonCreateView()วิธีการของส่วนผู้ปกครอง

ทำแบบนี้:

@Override
    public void onDestroyView()
    {
                FragmentManager mFragmentMgr= getFragmentManager();
        FragmentTransaction mTransaction = mFragmentMgr.beginTransaction();
                Fragment childFragment =mFragmentMgr.findFragmentByTag("qa_fragment")
        mTransaction.remove(childFragment);
        mTransaction.commit();
        super.onDestroyView();
    }

4
สิ่งนี้ไม่ได้ผลสำหรับฉัน ฉันมีชิ้นส่วนที่ขยายมุมมองที่มีเด็กสองคนมอง ในมันเพิ่มอีกสองชิ้นส่วนหนึ่งในแต่ละมุมมองที่ใช้onActivityCreated() getFragmentManager()สิ่งนี้ใช้งานได้เป็นครั้งแรก แต่เมื่อหมุนและกลับมามองเห็นชิ้นส่วนเพียงชิ้นเดียว พฤติกรรมนั้นถูกต้องด้วยgetChildFragmentManager()แทน วิธีการที่แนะนำในที่นี้มีข้อยกเว้นเนื่องจากonSaveInstanceState()มีการเรียกกิจกรรมแล้ว การทำธุรกรรมโดยใช้commitAllowingStateLoss()หลีกเลี่ยงข้อยกเว้น แต่ไม่ได้แก้ปัญหาเดิม
user1978019

8

ฉันแก้ไขปัญหานี้แล้ว ViewPagerคุณสามารถใช้ห้องสมุดการสนับสนุนและ หากคุณไม่ต้องการรูดด้วยท่าทางคุณสามารถปิดการรูดได้ ดังนั้นนี่คือรหัสเพื่อปรับปรุงโซลูชันของฉัน:

public class TestFragment extends Fragment{
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View v = inflater.inflate(R.layout.frag, container, false);
    final ArrayList<Fragment> list = new ArrayList<Fragment>();

    list.add(new TrFrag());
    list.add(new TrFrag());
    list.add(new TrFrag());

    ViewPager pager = (ViewPager) v.findViewById(R.id.pager);
    pager.setAdapter(new FragmentPagerAdapter(getChildFragmentManager()) {
        @Override
        public Fragment getItem(int i) {
            return list.get(i);
        }

        @Override
        public int getCount() {
            return list.size();
        }
    });
    return v;
}
}

PS เป็นโค้ดที่น่าเกลียดสำหรับการทดสอบ แต่ปรับปรุงว่าเป็นไปได้

ควรส่งต่อชิ้นส่วนภายใน PPSChildFragmentManagerViewPagerAdapter


8

คุณสามารถใช้getChildFragmentManager()ฟังก์ชั่น

ตัวอย่าง:

ส่วนย่อยของผู้ปกครอง:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {

    rootView = inflater.inflate(R.layout.parent_fragment, container,
            false);


    }

    //child fragment 
    FragmentManager childFragMan = getChildFragmentManager();
    FragmentTransaction childFragTrans = childFragMan.beginTransaction();
    ChildFragment fragB = new ChildFragment ();
    childFragTrans.add(R.id.FRAGMENT_PLACEHOLDER, fragB);
    childFragTrans.addToBackStack("B");
    childFragTrans.commit();        


    return rootView;
}

เค้าโครงหลัก ( parent_fragment.xml):

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/white">



    <FrameLayout
        android:id="@+id/FRAGMENT_PLACEHOLDER"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>




</LinearLayout>

ส่วนย่อยของเด็ก:

public class ChildFragment extends Fragment implements View.OnClickListener{

    View v ;
    @Override
    public View onCreateView(LayoutInflater inflater,
                             @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        View rootView = inflater.inflate(R.layout.child_fragment, container, false);


        v = rootView;


        return rootView;
    }



    @Override
    public void onClick(View view) {


    }


} 

4

มันไม่มีอะไรซับซ้อน เราไม่สามารถใช้getFragmentManager()ที่นี่ สำหรับการใช้ชิ้นส่วนภายในส่วนgetChildFragmentManager()ที่เราใช้ ส่วนที่เหลือจะเหมือนกัน


3

คุณสามารถเพิ่มลงFrameLayoutในแฟรกเมนต์และแทนที่ด้วยแฟรกเมนต์อื่นเมื่อเริ่มต้น

ด้วยวิธีนี้คุณสามารถพิจารณาชิ้นส่วนอื่น ๆ ที่จะอยู่ในส่วนแรก



2

มีการสนับสนุนอย่างชัดเจนในส่วนที่ซ้อนกันหนึ่งรายการที่ซ้อนกันจะได้รับการสนับสนุนก็ต่อเมื่อมีการสร้างโดยทางโปรแกรม! ดังนั้นในเวลานี้จึงไม่มีโครงร่างของแฟรกเมนต์ซ้อนกันที่สนับสนุนในโครงร่าง xml!


1

ที่อาจช่วยผู้ที่ทำงานบนKotlinคุณสามารถใช้ฟังก์ชั่นเสริมดังนั้นสร้างไฟล์ kotlin สมมติว่า"util.kt"และเพิ่มรหัสชิ้นนี้

fun Fragment.addChildFragment(fragment: Fragment, frameId: Int) {

    val transaction = childFragmentManager.beginTransaction()
    transaction.replace(frameId, fragment).commit()
}

สมมุติว่านี่คือคลาสของเด็ก

class InputFieldPresentation: Fragment()
{
    var views: View? = null
    override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?,
                              savedInstanceState: Bundle?): View? {
        views = inflater!!.inflate(R.layout.input_field_frag, container, false)
        return views
    }

    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        ...
    }
    ...
}

ตอนนี้คุณสามารถเพิ่มเด็ก ๆ ลงในส่วนพ่อแบบนี้

 FatherPresentation:Fragment()
{
  ...

    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        val fieldFragment= InputFieldPresentation()
        addChildFragment(fieldFragment,R.id.fragmet_field)
    }
...
}

โดยที่R.id.fragmet_fieldเป็นรหัสของเลย์เอาต์ที่จะมีส่วนย่อยสิ่งนี้น่าพิศวงอยู่ในส่วนของพ่อแน่นอน นี่คือตัวอย่าง

father_fragment.xml :

<LinearLayout android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    xmlns:android="http://schemas.android.com/apk/res/android"
    >

    ...

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:gravity="center"
            android:id="@+id/fragmet_field"
            android:orientation="vertical"
            >
        </LinearLayout>
    ...

    </LinearLayout>

1

ไม่มีการสนับสนุนMapFragmentทีม Android กล่าวว่าทำงานได้ตั้งแต่ Android 3.0 นี่คือข้อมูลเพิ่มเติมเกี่ยวกับปัญหา MapActivityแต่สิ่งที่คุณสามารถทำมันได้โดยการสร้างส่วนที่ผลตอบแทน นี่คือตัวอย่างรหัส ขอบคุณ inazaruk:

มันทำงานอย่างไร :

  • MainFragmentActivity เป็นกิจกรรมที่ขยายFragmentActivityและโฮสต์ MapFragments สองรายการ
  • MyMapActivity ขยาย MapActivity และมี MapView.
  • LocalActivityManagerFragment โฮสต์ LocalActivityManager
  • MyMapFragment ขยายLocalActivityManagerFragmentและด้วยความช่วยเหลือในการTabHostสร้างอินสแตนซ์ภายในของ MyMapActivity

หากคุณมีข้อสงสัยโปรดแจ้งให้เราทราบ


0

สวัสดีฉันแก้ไขปัญหานี้โดยใส่ส่วนย่อยลงในเลย์เอาต์ที่แตกต่างกันและฉันสร้างเลย์เอาต์ที่เกี่ยวข้องให้เห็นและทำให้การมองเห็นอื่น ๆ หายไป

ฉันหมายถึง:

<?xml version="1.0" encoding="utf-8"?>
 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:orientation="vertical"
 android:layout_width="fill_parent"
 android:layout_height="fill_parent">

     <LinearLayout android:id="@+id/linearLayout1"
           android:layout_width="match_parent"
           android:layout_height="wrap_content"
           android:orientation="horizontal">
           <Button android:layout_width="wrap_content"
                   android:id="@+id/button1"
                   android:layout_height="wrap_content"
                   android:text="Button1"></Button>
           <Button android:text="Button2"
                   android:id="@+id/button2"
                   android:layout_width="wrap_content"
                   android:layout_height="wrap_content"></Button>
           <Button android:text="Button3"
                   android:id="@+id/button3"
                   android:layout_width="wrap_content"
                   android:layout_height="wrap_content"></Button>
           <Button android:text="Button4"
                   android:id="@+id/button4"
                   android:layout_width="wrap_content"
                   android:layout_height="wrap_content"></Button>
   </LinearLayout>
   <LinearLayout android:layout_width="full_screen"
              android:layout_height="0dp"
              android:layout_weight="1"
              android:id="action_For_Button1"
              android:visibility="visible">
                    <Fragment android:layout_width="full_screen"
                              android:layout_height="full_screen"
                              android:id="fragment1"
                                        .
                                        .
                                        .
             / >
     </LinearLayout>

     <LinearLayout android:layout_width="full_screen"
              android:layout_height="0dp"
              android:id="action_For_Button1"
              android:layout_weight="1"
              android:visibility="gone">
                       <Fragment android:layout_width="full_screen"
                                 android:layout_height="full_screen"
                                 android:id="fragment2"
                                           .
                                           .
                                           .
             / >
        </LinearLayout>
                     .
                     .
                     .
        </LinearLayout>

ฉันสันนิษฐานว่าคุณจะเปิดหน้าเว็บของคุณเนื่องจากมีการคลิกปุ่ม 1 คุณสามารถควบคุมการมองเห็นชิ้นส่วนของคุณในการดำเนินการคลิกคุณสามารถทำให้เค้าโครงที่เกี่ยวข้องปรากฏให้เห็นได้ และเนื่องจาก มุมมองที่มองเห็นได้: หายไปแล้วจะมองไม่เห็นและไม่มีพื้นที่ว่างสำหรับจุดประสงค์ในการจัดวางฉันคิดว่าวิธีการนี้ไม่ทำให้เกิดปัญหาพื้นที่

PS: ฉันแค่พยายามอธิบายรหัสโซลูชันของฉันอาจมีข้อผิดพลาดทางไวยากรณ์หรือโครงสร้างที่ไม่สมบูรณ์

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.