- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在尝试实现一个可以出现在左侧或右侧的聊天气泡。我阅读了很多已在此处发布的答案,但我没有设法让其中任何一个为我工作,也许我遗漏了什么。
如果我手动将 row_item xml 中 linearLayout 的 layout_gravity 从左更改为右,然后查看图形布局,我可以看到它按我想要的方式工作。问题是我无法控制此属性。据我了解,问题与存在的 LayoutParams 属于 listview.LayoutParams 这一事实有关,如果我尝试将 LayoutParams 获取到 listview.LayoutParams 对象,它会起作用。但这不是我想要实现的目标。
代码如下:
Activity xml
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:ignore="MergeRootFrame" >
<ListView
android:id="@+id/listViewMessages"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="0.98" >
</ListView>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_weight="0.02"
android:background="#D3D3D3"
android:orientation="horizontal" >
<EditText
android:id="@+id/editTextMessage"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_toLeftOf="@+id/imageButtonSend"
android:ems="10"
android:hint="@string/textFieldMessage"
android:textSize="14sp"
android:visibility="visible" >
<requestFocus />
</EditText>
<ImageButton
android:id="@+id/imageButtonSend"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:layout_centerVertical="true"
android:layout_marginRight="14dp"
android:background="?android:attr/selectableItemBackground"
android:contentDescription="@string/imageButtonSendNow_contentDescription"
android:src="@drawable/ic_action_send_now"
android:onClick="onClick" />
</RelativeLayout>
</LinearLayout>
row_item xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/singleMessageContainer"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="left"
android:orientation="horizontal"
android:padding="6dp" >
<TextView
android:id="@+id/textViewMessage"
android:layout_width="wrap_content"
android:layout_height="20dp"
android:layout_weight="0.85"
android:layout_marginBottom="2dp"
android:layout_marginLeft="5dp"
android:layout_marginTop="2dp"
android:gravity="center_vertical"
android:text="Message"
android:textSize="16sp"
tools:ignore="HardcodedText" />
<TextView
android:id="@+id/TextViewTime"
android:layout_width="wrap_content"
android:layout_height="14dp"
android:layout_weight="0.1"
android:layout_gravity="bottom"
android:layout_marginLeft="22dp"
android:gravity="center_vertical"
android:text="Time"
android:textSize="12sp"
tools:ignore="HardcodedText" />
<ImageView
android:id="@+id/imageViewSuccess"
android:layout_width="12dp"
android:layout_height="12dp"
android:layout_weight="0.05"
android:layout_gravity="bottom"
android:contentDescription="successIcon"
android:src="@drawable/ic_action_accept"
tools:ignore="HardcodedText" />
</LinearLayout>
适配器代码:
public class ConversationAdapter extends ArrayAdapter<Message> {
private final Context context;
static class ViewHolder {
public TextView message;
public TextView time;
public ImageView success;
public LinearLayout singleMessageContainer;
}
public ConversationAdapter(Context context, ArrayList<Message> values) {
super(context, R.layout.conversation_list_item, values);
this.context = context;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View rowView = convertView;
// reuse views
if (rowView == null) {
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
rowView = inflater.inflate(R.layout.conversation_list_item, parent,false);
// configure view holder
ViewHolder viewHolder = new ViewHolder();
viewHolder.message = (TextView) rowView.findViewById(R.id.textViewMessage);
viewHolder.time = (TextView) rowView.findViewById(R.id.TextViewTime);
viewHolder.success = (ImageView) rowView.findViewById(R.id.imageViewSuccess);
viewHolder.singleMessageContainer = (LinearLayout) rowView.findViewById(R.id.singleMessageContainer);
rowView.setTag(viewHolder);
}
// fill data
ViewHolder holder = (ViewHolder) rowView.getTag();
holder.message.setText(getItem(position).getMessage());
holder.time.setText(getItem(position).getUniversalTimeString());
holder.success.setImageResource(R.drawable.ic_action_accept);
boolean isMine = (getItem(position).getSender() == 0);
holder.singleMessageContainer.setBackgroundResource(
isMine ? R.drawable.bubble_yellow : R.drawable.bubble_green);
// This part doesn't work
LinearLayout.LayoutParams params = (LinearLayout.LayoutParams)
holder.singleMessageContainer.getLayoutParams();
params.gravity=(isMine ? Gravity.LEFT : Gravity.RIGHT);
holder.singleMessageContainer.setLayoutParams(params);
return rowView;
}
}
最佳答案
解决方案
我继续阅读有关 gravity 和 layout_gravity 的内容,并遇到了帮助我解决此问题的更详细信息。在这里问题的几个答案中,有一个使用 warpper 布局的解决方案,但没有写下需要注意的重要事项。
首先要知道的是,如果您将线性布局设置为水平布局,您将无法将重力属性用作左或右仅顶部或底部因此外层容器必须是垂直的,而内层容器必须是水平的,以便在一行中一个接一个地显示消息和其他数据。
然后您可以使用外部容器的方法 setGravity(Gravity.LEFT) 或 setGravity(Gravity.RIGHT) 将内部 warpper 正确对齐到左侧或右侧。您还需要注意 fill_parent 和 wrap_content 的属性
这里是更新的代码:row_item xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/singleMessageContainer"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:gravity="left"
android:orientation="vertical"
android:padding="6dp" >
<LinearLayout
android:id="@+id/warpper"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal" >
<TextView
android:id="@+id/textViewMessage"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_marginBottom="2dp"
android:layout_marginLeft="5dp"
android:layout_marginRight="5dp"
android:layout_marginTop="2dp"
android:text="Message"
android:textSize="14sp"
tools:ignore="HardcodedText" />
<TextView
android:id="@+id/TextViewTime"
android:layout_width="wrap_content"
android:layout_height="14dp"
android:layout_gravity="bottom"
android:text="Time"
android:textSize="12sp"
tools:ignore="HardcodedText" />
<ImageView
android:id="@+id/imageViewSuccess"
android:layout_width="wrap_content"
android:layout_height="12dp"
android:layout_gravity="bottom"
android:contentDescription="successIcon"
android:src="@drawable/ic_action_accept"
tools:ignore="HardcodedText" />
</LinearLayout>
</LinearLayout>
和更新后的适配器代码:
public class ConversationAdapter extends ArrayAdapter<Message> {
private final Context context;
static class ViewHolder {
public TextView message;
public TextView time;
public ImageView success;
public LinearLayout singleMessageContainer;
// Added
public LinearLayout warpper;
}
public ConversationAdapter(Context context, ArrayList<Message> values) {
super(context, R.layout.conversation_list_item, values);
this.context = context;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View rowView = convertView;
// reuse views
if (rowView == null) {
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
rowView = inflater.inflate(R.layout.conversation_list_item, parent,false);
// configure view holder
ViewHolder viewHolder = new ViewHolder();
viewHolder.message = (TextView) rowView.findViewById(R.id.textViewMessage);
viewHolder.time = (TextView) rowView.findViewById(R.id.TextViewTime);
viewHolder.success = (ImageView) rowView.findViewById(R.id.imageViewSuccess);
viewHolder.singleMessageContainer = (LinearLayout) rowView.findViewById(R.id.singleMessageContainer);
// Added
viewHolder.warpper = (LinearLayout) rowView.findViewById(R.id.warpper);
rowView.setTag(viewHolder);
}
// fill data
ViewHolder holder = (ViewHolder) rowView.getTag();
holder.message.setText(getItem(position).getMessage());
holder.time.setText(getItem(position).getUniversalTimeString());
holder.success.setImageResource(R.drawable.ic_action_accept);
boolean isMine = (getItem(position).getSender() == 0);
// Changed to warpper
holder.warpper.setBackgroundResource(
isMine ? R.drawable.bubble_yellow : R.drawable.bubble_green);
// Added
holder.singleMessageContainer.setGravity(isMine ? Gravity.LEFT : Gravity.RIGHT);
return rowView;
}
}
它应该可以工作。唯一的问题是,您可能会在 xml 中看到一条关于线性布局之一无用的警告,但没有它就无法工作。
希望对你有所帮助。
关于android:如何更改 ListView 中 row_item 的 layout_gravity 属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25351057/
你能比较一下属性吗 我想禁用文本框“txtName”。有两种方式 使用javascript,txtName.disabled = true 使用 ASP.NET, 哪种方法更好,为什么? 最佳答案 我
Count 属性 返回一个集合或 Dictionary 对象包含的项目数。只读。 object.Count object 可以是“应用于”列表中列出的任何集合或对
CompareMode 属性 设置并返回在 Dictionary 对象中比较字符串关键字的比较模式。 object.CompareMode[ = compare] 参数
Column 属性 只读属性,返回 TextStream 文件中当前字符位置的列号。 object.Column object 通常是 TextStream 对象的名称。
AvailableSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。 object.AvailableSpace object 应为 Drive 
Attributes 属性 设置或返回文件或文件夹的属性。可读写或只读(与属性有关)。 object.Attributes [= newattributes] 参数 object
AtEndOfStream 属性 如果文件指针位于 TextStream 文件末,则返回 True;否则如果不为只读则返回 False。 object.A
AtEndOfLine 属性 TextStream 文件中,如果文件指针指向行末标记,就返回 True;否则如果不是只读则返回 False。 object.AtEn
RootFolder 属性 返回一个 Folder 对象,表示指定驱动器的根文件夹。只读。 object.RootFolder object 应为 Dr
Path 属性 返回指定文件、文件夹或驱动器的路径。 object.Path object 应为 File、Folder 或 Drive 对象的名称。 说明 对于驱动器,路径不包含根目录。
ParentFolder 属性 返回指定文件或文件夹的父文件夹。只读。 object.ParentFolder object 应为 File 或 Folder 对象的名称。 说明 以下代码
Name 属性 设置或返回指定的文件或文件夹的名称。可读写。 object.Name [= newname] 参数 object 必选项。应为 File 或&
Line 属性 只读属性,返回 TextStream 文件中的当前行号。 object.Line object 通常是 TextStream 对象的名称。 说明 文件刚
Key 属性 在 Dictionary 对象中设置 key。 object.Key(key) = newkey 参数 object 必选项。通常是 Dictionary 
Item 属性 设置或返回 Dictionary 对象中指定的 key 对应的 item,或返回集合中基于指定的 key 的&
IsRootFolder 属性 如果指定的文件夹是根文件夹,返回 True;否则返回 False。 object.IsRootFolder object 应为&n
IsReady 属性 如果指定的驱动器就绪,返回 True;否则返回 False。 object.IsReady object 应为 Drive&nbs
FreeSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。只读。 object.FreeSpace object 应为 Drive 对象的名称。
FileSystem 属性 返回指定的驱动器使用的文件系统的类型。 object.FileSystem object 应为 Drive 对象的名称。 说明 可
Files 属性 返回由指定文件夹中所有 File 对象(包括隐藏文件和系统文件)组成的 Files 集合。 object.Files object&n
我是一名优秀的程序员,十分优秀!