如何在我的android应用中修复布局视图?

时间:2019-06-18 12:22:00

标签: java android

我正在尝试使用我的api创建按钮。我使用Retrofit调用了api,然后创建了按钮。但是有一个代码错误! API响应中返回的数据是正确的,但是创建视图是一个问题。

它说我需要removeView()。我这样做了,但是现在没有数据打印。

我的代码:

package com.example.englishapp;

import android.annotation.SuppressLint;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.TextView;

import java.util.List;

import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;


public class LessonsActivity extends AppCompatActivity {

    private TextView textViewResult;
    String URL = "http://kurchanovenglish.ru/data/";

    private String[] name = new String[18];



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


        final LinearLayout layout = new LinearLayout(this);

        final LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(
                LinearLayout.LayoutParams.MATCH_PARENT,
                LinearLayout.LayoutParams.MATCH_PARENT
        );
        params.setMargins(250, 10, 250, 50);
        layout.setOrientation(LinearLayout.VERTICAL);
        int i;

        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(URL)
                .addConverterFactory(GsonConverterFactory.create())
                .build();

        JsonPlaceHolderApi jsonPlaceHolderApi = retrofit.create(JsonPlaceHolderApi.class);

        Call<List<Lesson>> call = jsonPlaceHolderApi.getPosts();


        final LinearLayout row = new LinearLayout(this);
        final Button button = new Button(this);
        call.enqueue(new Callback<List<Lesson>>() {


            @Override
            public void onResponse(Call<List<Lesson>> call, Response<List<Lesson>> response) {


                if (!response.isSuccessful()) {
                    textViewResult.setText("Code: " + response.code());
                    return;
                }

                List<Lesson> lessons = response.body();

                int b = 0;

                String[] name = new String[18];

                for (Lesson lesson : lessons) {

                    button.setText(lesson.getName());
                    button.setId(lesson.getNumber());
                    button.setLayoutParams(params);
                    button.setTextSize(20);
                    /*button.setBackground(this.getResources().getDrawable(R.drawable.buttons));*/
                    button.setOnClickListener(new View.OnClickListener() {
                        @SuppressLint("ShowToast")
                        @Override
                        public void onClick(View v) {
                            Intent intent = new Intent(LessonsActivity.this, WelcomeActivity.class);
                            startActivity(intent);
                        }
                    });
                    row.removeView(layout);
                    row.addView(button);
                }
                layout.addView(row);


                }


            @Override
            public void onFailure(Call<List<Lesson>> call, Throwable t) {
                textViewResult.setText(t.getMessage());
            }
        });


    }
}

错误:

/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.example.englishapp, PID: 19385
    java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.
        at android.view.ViewGroup.addViewInner(ViewGroup.java:4915)
        at android.view.ViewGroup.addView(ViewGroup.java:4746)
        at android.view.ViewGroup.addView(ViewGroup.java:4686)
        at android.view.ViewGroup.addView(ViewGroup.java:4659)
        at com.example.englishapp.LessonsActivity$1.onResponse(LessonsActivity.java:92)
        at retrofit2.DefaultCallAdapterFactory$ExecutorCallbackCall$1$1.run(DefaultCallAdapterFactory.java:83)
        at android.os.Handler.handleCallback(Handler.java:789)
        at android.os.Handler.dispatchMessage(Handler.java:98)
        at android.os.Looper.loop(Looper.java:164)
        at android.app.ActivityThread.main(ActivityThread.java:6541)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)

请帮帮我。我不知道该怎么办。

1 个答案:

答案 0 :(得分:1)

这是您在做错的事情;您正在使用相同的Button实例,然后将其再次添加到rowLinearLayout)。

如果我理解正确,这就是您想要的视图层次结构。

LinearLayout

    --LinearLayout

        --Button

        --Button

        .

    --LinearLayout

        --Button

        --Button

        .
.

首先需要了解的是,您不能在视图层次结构的多个位置重复使用View。其次,View不能是多个父母的孩子,即同一Button不能出现在两行中。让我们考虑这种限制的原因;例如,您被允许将相同的Button添加到另一位父母中,那么您如何确定他们中的点击,如何知道您点击了哪一个。此外,假设您要在第二个Button上显示一个文本,在另一个文本上显示一个文本,如果/当您共享引用时,每个文本将如何更改。你看到问题了吗?


但是,由于要在Java代码中实例化并添加Button。您需要首先创建一个Button的新实例,以添加到父row中。因此,如果要添加多个行,则还需要一个新的row实例以将其添加到父layout中。

您的代码将变为:

for (Lesson lesson : lessons) {

    Button button = new Button(LessonsActivity.this);
    button.setText(lesson.getName());
    button.setId(lesson.getNumber());
    button.setLayoutParams(params);
    button.setTextSize(20);
    /*button.setBackground(this.getResources().getDrawable(R.drawable.buttons));*/
    button.setOnClickListener(new View.OnClickListener() {
    @SuppressLint("ShowToast")
        @Override
        public void onClick(View v) {
            Intent intent = new Intent(LessonsActivity.this, WelcomeActivity.class);
            startActivity(intent);
            }
        });

        row.addView(button);
    }
    layout.addView(row);
}

您将需要相应地调整layout.addView(row);,但遵循相同的原理。