Установить поля в LinearLayout программно

266

Я пытаюсь использовать Java ( не XML ) для создания LinearLayout с кнопками, которые заполняют экран и имеют поля. Вот код, который работает без полей:

LinearLayout buttonsView = new LinearLayout(this);
buttonsView.setOrientation(LinearLayout.VERTICAL);
for (int r = 0; r < 6; ++r) {
    Button btn = new Button(this);
    btn.setText("A");

    LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.FILL_PARENT, LinearLayout.LayoutParams.FILL_PARENT); // Verbose!
    lp.weight = 1.0f; // This is critical. Doesn't work without it.
    buttonsView.addView(btn, lp);
}

ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT, ViewGroup.LayoutParams.FILL_PARENT);
setContentView(buttonsView, lp);

Так что это прекрасно работает, но как же вы задаете поля кнопок, чтобы между ними было пространство? Я пытался использовать LinearLayout.MarginLayoutParams, но это не имеет weightчлена, так что это не хорошо. И это не сработает, если вы передадите его lpв конструктор.

Это невозможно? Потому что это выглядит так, и это будет не первая задача макета Android, которую вы можете выполнить только в XML.

Timmmm
источник
1
developer.android.com/reference/android/view/… , int, int, int) Вместо layoutParams.setMargins (30, 20, 30, 0); Вы кладете layoutParams.setMargins (30dp, 20dp, 30dp, 0dp);
DarthestVader
3
setMargins принимает int в качестве аргументов. вы можете указать «dp» только с помощью атрибутов xml
jyavenard

Ответы:

496

Вот небольшой код для этого:

LinearLayout ll = new LinearLayout(this);
ll.setOrientation(LinearLayout.VERTICAL);

LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(
     LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT);

layoutParams.setMargins(30, 20, 30, 0);

Button okButton=new Button(this);
okButton.setText("some text");
ll.addView(okButton, layoutParams);
Mauro
источник
6
спасибо за примерный пример кода - гораздо полезнее, чем просто описания
Кто-то где-то
25
Похоже, параметры поля заданы в пикселях. Есть ли способ установить их в дпс?
Артем Руссаковский
30
@ArtemRussakovskii Как и большинство функций в Android, они принимают пиксели. Я предлагаю глобальную функцию полезности, которая преобразует провалы в px. Это то, что я сделал во всех моих приложениях. Android API отстой.
mxcl
7
@MaxHowell Это то, что я в итоге сделал. Невероятно, что такие базовые вещи иногда отсутствуют.
Артем Руссаковский
5
@ArtemRussakovskii - используйте TypedValue.applyDimension (...) для преобразования DIP в PX. См. Stackoverflow.com/questions/2238883/…
Гаутам
64

Так что это прекрасно работает, но как же вы задаете поля кнопок, чтобы между ними было пространство?

Вы звоните setMargins()на LinearLayout.LayoutParamsобъекте.

Я пытался использовать LinearLayout.MarginLayoutParams, но у него нет весового члена, так что это бесполезно.

LinearLayout.LayoutParamsявляется подклассом LinearLayout.MarginLayoutParams, как указано в документации .

Это невозможно?

Нет.

это была бы не первая задача макета Android, которую можно выполнить только в XML

Вы можете предоставить доказательство этого требования.

Лично я ничего не знаю о том, что может быть достигнуто только с помощью XML, а не с помощью методов Java в SDK. Фактически, по определению, все должно быть выполнимо через Java (хотя не обязательно через методы, достигаемые SDK), поскольку XML не является исполняемым кодом. Но, если вы знаете что-то, укажите на это, потому что это ошибка в SDK, которая должна быть исправлена ​​когда-нибудь.

CommonsWare
источник
2
Ха, хорошо, теперь я понимаю. Был немного потерян в системе макета, что довольно сложно. Я не думаю, что в любом случае можно установить layout_widgth / height во время выполнения.
Тимммм
2
И нет никакого способа установить стиль: stackoverflow.com/questions/2016249/…
Timmmm
1
У меня не было проблем с настройкой ширины и высоты во время выполнения через LayoutPararms. Я с точки зрения стилей ... да, ну, есть причина, по которой я их не использую. :-)
CommonsWare
6
Я хочу знать, почему это такая PITA для программного создания пользовательского интерфейса
кто-то где-то
3
Мне потребовалось много времени, чтобы осознать, что несколько дней назад я импортировал неправильные LayoutParams, которые не поддерживали поля ... Читая эти ответы, я решил удалить свой импорт. Есть более 10 вариантов D: я выбрал linearLayout.LayoutParams. Работает.
Doomsknight
31

Чтобы добавить поля непосредственно к элементам (некоторые элементы допускают непосредственное редактирование полей), вы можете сделать:

LayoutParams lp = ((ViewGroup) something).getLayoutParams();
if( lp instanceof MarginLayoutParams )
{
    ((MarginLayoutParams) lp).topMargin = ...;
    ((MarginLayoutParams) lp).leftMargin = ...;
    //... etc
}
else
    Log.e("MyApp", "Attempted to set the margins on a class that doesn't support margins: "+something.getClass().getName() );

... это работает без необходимости знать / редактировать окружающий макет. Обратите внимание на проверку «instanceof» на случай, если вы попытаетесь выполнить это для чего-то, что не поддерживает поля.

Адам
источник
2
Это именно то, что я искал.
djdance
21

Из-за различий в плотности пикселей экрана устройства хорошо всегда использовать DIPединицу для установки поля программно. Как ниже

//get resources
Resources r = getResources();
float pxLeftMargin = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 10, r.getDisplayMetrics());
float pxTopMargin = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 10, r.getDisplayMetrics());
float pxRightMargin = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 10, r.getDisplayMetrics());
float pxBottomMargin = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 20, r.getDisplayMetrics());

//get layout params...
LayoutParams params=new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
params.setMargins(Math.round(pxLeftMargin), Math.round(pxTopMargin), Math.round(pxRightMargin), Math.round(pxBottomMargin));

//set margin...
yourLayoutTOsetMargin.setLayoutParams(params); 

Надеюсь, это поможет.

Рупеш Ядав
источник
10

Я настроил поля напрямую, используя приведенный ниже код

LinearLayout layout = (LinearLayout)findViewById(R.id.yourrelative_layout);
LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT,LayoutParams.WRAP_CONTENT);            
params.setMargins(3, 300, 3, 3); 
layout.setLayoutParams(params);

Единственное, что здесь нужно отметить, это то, что LayoutParamsследует импортировать следующий пакет android.widget.RelativeLayout.LayoutParams, иначе произойдет ошибка.

Janith
источник
7

Попробуй это

 LayoutParams params = new LayoutParams(
            LayoutParams.WRAP_CONTENT,      
            LayoutParams.WRAP_CONTENT
    );
    params.setMargins(left, top, right, bottom);
    yourbutton.setLayoutParams(params);
Каран Бхалодия
источник
7
 MarginLayoutParams layoutParams = (MarginLayoutParams) getLayoutParams();
 layoutParams.setMargins(leftMargin, topMargin, rightMargin, bottomMargin);
Ginni
источник
2
«Хотя этот код может ответить на вопрос, лучше включить сюда объяснение кода.
xenteros
4

Попробуй это:

MarginLayoutParams params = (MarginLayoutParams) view.getLayoutParams();
params.width = 250;
params.leftMargin = 50;
params.topMargin = 50;
Makvin
источник
3
/*
 * invalid margin
 */
private void invalidMarginBottom() {
    RelativeLayout.LayoutParams lp = (RelativeLayout.LayoutParams) frameLayoutContent.getLayoutParams();
    lp.setMargins(0, 0, 0, 0);
    frameLayoutContent.setLayoutParams(lp);
}

вы должны знать тип viewGroup вида. Например, в приведенном выше коде я хочу изменить поле frameLayout, а группа представлений frameLayout - RelativeLayout,, поэтому вам необходимо преобразовать в (RelativeLayout.LayoutParams)

Фрэнсис Ши
источник
0

Здесь в одной строке Решение:

((LinearLayout.LayoutParams) yourLinearLayout.getLayoutParams()).marginToAdd = ((int)(Resources.getSystem().getDisplayMetrics().density * yourDPValue));
Stappo
источник