Есть ScrollView, внутри него LinearLayout, внутри которого статически созданы 2 кнопки "Открыть" и "Создать" обе кнопки вызывают рекурсивный генератор интерфейса, сам интерфейс создает неограниченное количество однотипных элементов - LL внутри него EditText,ImageButton и ListView.
Генератор интерфейса⇓
Кликните здесь для просмотра всего текста
Java |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
| public void generateLevels(int levelId, String parentLVitem) {
LinearLayout.LayoutParams LLParams = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT);
RelativeLayout.LayoutParams RLparamsL = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
RelativeLayout.LayoutParams RLparamsR = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
final LinearLayout layvar = (LinearLayout) findViewById(R.id.LL1);
//уровень
final LinearLayout layout;
if ((LinearLayout) findViewById(levelId) == null) {
layout = new LinearLayout(this);
layout.setId(levelId);
layout.setLayoutParams(LLParams);
layout.setOrientation(LinearLayout.VERTICAL);
layvar.addView(layout);
TextView tvlevel = new TextView(this);
tvlevel.setLayoutParams(LLParams);
int k = 0;
// Log.d("id", String.valueOf(levelId));
for (int i = 0; i < 10; i++) {
//Log.d("arr", String.valueOf(arrId[i]));
if (arrId[i] == levelId) {
k = i + 1;
// Log.d("k", String.valueOf(k));
break;
}
}
tvlevel.setText("Уровень " + k);
tvlevel.setTextSize(24);
layout.addView(tvlevel);
} else {
layout = (LinearLayout) findViewById(levelId);
}
//ЛЛ элемента
final LinearLayout laycontainer = new LinearLayout(this);
laycontainer.setId(hash(parentLVitem + "lay"));
laycontainer.setLayoutParams(LLParams);
laycontainer.setOrientation(LinearLayout.VERTICAL);
TextView tv = new TextView(this);
tv.setId(hash(parentLVitem + "tv"));
RLparamsL.addRule(RelativeLayout.ALIGN_PARENT_LEFT);
RLparamsL.addRule(RelativeLayout.LEFT_OF, R.id.but_var);
RLparamsR.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);
final EditText et = new EditText(this);
et.setId(hash(parentLVitem + "et"));
final ImageButton ibut = new ImageButton(this);
ibut.setId(hash(parentLVitem + "but"));
final ListView lv = new ListView(this);
lv.setId(hash(parentLVitem + "lv"));
RelativeLayout laysub = new RelativeLayout(this);
InputFilter[] fa = new InputFilter[1];
fa[0] = new InputFilter.LengthFilter(25);
layout.setOrientation(LinearLayout.VERTICAL);
layout.setLayoutParams(LLParams);
tv.setText("Элементы критерия " + parentLVitem);
tv.setLayoutParams(LLParams);
tv.setTextSize(20);
et.setHint("Введите название");
et.setLayoutParams(RLparamsL);
et.setFilters(fa);
ibut.setLayoutParams(RLparamsR);
ibut.setImageResource(R.drawable.ic_action_add);
laysub.setLayoutParams(new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.MATCH_PARENT));
// Создаём пустой массив для хранения критериев
final ArrayList<String> ar = new ArrayList<>();
// Создаём адаптер ArrayAdapter, чтобы привязать массив к ListView
final ArrayAdapter<String> ad;
ad = new ArrayAdapter<>(this, android.R.layout.simple_list_item_1, ar);
// Привяжем массив через адаптер к ListView
lv.setAdapter(ad);
// Прослушиваем нажатия клавиш
ibut.setOnClickListener(new View.OnClickListener() {
public void onClick(View v) {
if (ar.size() < 10) {
if (et.length() != 0) {
ar.add(0, et.getText().toString());
int k = 0;
int ki = 0;
int idpar = ((LinearLayout) laycontainer.getParent()).getId();
// Log.d("laypar", String.valueOf(idpar));
for (int i = 0; i < 10; i++) {
if (arrId[i] == idpar) {
ki = i + 1;
if (ki != 10) {
k = arrId[i + 1];
break;
}
}
}
if (ki != 10)
generateLevels(k, et.getText().toString());
et.setText("");
setListViewHeightBasedOnChildren(lv);
ad.notifyDataSetChanged();
} else {
Toast.makeText(getApplicationContext(), "Введите название!", Toast.LENGTH_SHORT).show();
}
} else {
Toast.makeText(getApplicationContext(), "Больше 10 нельзя=)", Toast.LENGTH_SHORT).show();
et.setText("");
}
}
});
//ОБРАБОТКА КЛИКА НА ИТЕМ ЛИСТВИЕВА КРИТЕРИЕВ
lv.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View itemClicked, final int position, long id) {
AlertDialog.Builder adb = new AlertDialog.Builder(MainActivity.this);
adb.setTitle("Удалить?");
adb.setMessage("Удалить критерий "" + ar.get(position) + "" ?");
final int positionToRemove = position;
adb.setNegativeButton("Отмена", null);
adb.setPositiveButton("Ок", new AlertDialog.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
LinearLayout lllevel = (LinearLayout) ((LinearLayout) lv.getParent()).getParent();
LinearLayout ll;
String str;
int k = 0;
for (int i = 0; i < arrId.length; i++) {
if (lllevel.getId() == arrId[i]) {
k = arrId[i + 1];
}
}
lllevel = (LinearLayout) findViewById(k);
for (int i = 0; i < (lllevel).getChildCount(); i++) {//Внутри уровня
if (lllevel.getChildAt(i) instanceof LinearLayout) {
ll = (LinearLayout) lllevel.getChildAt(i);
for (int j = 0; j < ll.getChildCount(); j++) {//Внутри ЛЛ критерия
if (ll.getChildAt(j) instanceof TextView) {
str = ((TextView) ll.getChildAt(j)).getText().toString();
str = str.substring(str.lastIndexOf("_")+1);
if (str.equals(ar.get(position))) {
lllevel.removeView(ll);
if ((lllevel.getChildCount()) == 1) {
layvar.removeView(lllevel);
}
break;
}
}
}
}
}
ar.remove(positionToRemove);
setListViewHeightBasedOnChildren(lv);
ad.notifyDataSetChanged();
}
});
adb.show();
}
});
laycontainer.addView(tv);
laycontainer.addView(laysub);
laysub.addView(et);
laysub.addView(ibut);
laycontainer.addView(lv, LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT);
setListViewHeightBasedOnChildren(lv);
layout.addView(laycontainer);
} |
|
Поскольку ListView конфликт с ScrollView'ом, методом гугления была найдена следующая функция, которая раскрывает полностью список)
Функция для "развертывания" списка⇓
Кликните здесь для просмотра всего текста
Java |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| public static void setListViewHeightBasedOnChildren(ListView listView) {
ListAdapter listAdapter = listView.getAdapter();
if (listAdapter == null) {
// pre-condition
return;
}
int totalHeight = 0;
int desiredWidth = View.MeasureSpec.makeMeasureSpec(listView.getWidth(), View.MeasureSpec.AT_MOST);
for (int i = 0; i < listAdapter.getCount(); i++) {
View listItem = listAdapter.getView(i, null, listView);
listItem.measure(desiredWidth, View.MeasureSpec.UNSPECIFIED);
totalHeight += listItem.getMeasuredHeight();
}
ViewGroup.LayoutParams params = listView.getLayoutParams();
params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));
listView.setLayoutParams(params);
listView.requestLayout();
} |
|
И при самостоятельном заполнении элементов все работает нормально,но при считывании из файла, которое представляет из себя последовательный вызов все того же рекурсивного генератора с необходимыми параметрами, мистическим (возможно и вполне очевидным, но я последние несколько недель нахожусь совершенно в неадекватном состоянии и не вижу явных на то причин) образом появляется пустое пространство (не пустые элементы, а именно пустое пространство).
Сами данные сохраняются в *.txt файл (не было времени гуглить про типизированные) с добавлением ключевых слов (для перехода на следующий уровень или на следующую группу дочерних элементов) и разделяется это все символом "_".