我有一个自定义光标适配器,其中每一行都包含一个复选框、 TextView 和按钮。 该按钮最初设置为 GONE,当用户选中我希望按钮显示的复选框,以便用户可以单击它并删除相应的行。
我的问题是,在我的 getView() 中,我无法获得正确的光标,所以不断获得 cursorOutOfBoundsExceptions。
如果我不再需要它,我还想删除 bindView()。
public class DeleteCampaignCustomCursorAdapater extends CursorAdapter {
protected static final String TAG = null;
DatabaseHelper myDbHelper;
protected SQLiteDatabase db;
private final Activity context;
private LayoutInflater mInflater;
protected ListView mListView;
private Cursor c;
public static int[] checked;
TextView merchantNameView;
Button deleteButton;
CheckBox selectForDelete;
//ImageView moveButton;
public DeleteCampaignCustomCursorAdapater(Context context, Cursor c) {
super(context, c);
this.context = (Activity) context;
mInflater = LayoutInflater.from(context);
this.c = c;
initializeChecked();
}
@Override
public void bindView(View view, Context context, final Cursor cursor) {
/*myDbHelper = new DatabaseHelper(context); //Links to DatabaseHelper class
merchantNameView = (TextView) view.findViewById(R.id.deleteMerchantNameView);
deleteButton = (Button) view.findViewById(R.id.deleteButton);
//selectForDelete = (CheckBox) view.findViewById(R.id.deleteCheckBox);
//moveButton = (ImageView) view.findViewById(R.id.moveButton);
merchantNameView.setText(cursor.getString(cursor.getColumnIndex("merchantName")));
final int rowID = cursor.getInt(cursor.getColumnIndex("_id"));
//final long rowID = mListView.getSelectedItemId();
if (checked[rowID] == 0) {
selectForDelete.setChecked(false);
} else {
selectForDelete.setChecked(true);
}
selectForDelete.setOnCheckedChangeListener(new OnCheckedChangeListener()
{
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked)
{
if ( isChecked )
{
// Make deleteButton appear
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
checked[rowID] = 1;
deleteButton.setVisibility(0);
}
else {
//Hide deleteButton
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
checked[rowID] = 1;
deleteButton.setVisibility(8);
}
}
});
deleteButton.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
if(rowID > 0) {
//Log.d(TAG, "Button Click. rowID = "+rowID);
myDbHelper.deleteRecordWithID(rowID);
cursor.requery();
notifyDataSetChanged();
initializeChecked(); // re-set the checked array
}
}
}); */
}
@Override
public View newView(Context context, Cursor cursor, ViewGroup parent) {
return mInflater.inflate(R.layout.delete_campaign_row_layout, null);
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
myDbHelper = new DatabaseHelper(context); //Links to DatabaseHelper class
if (convertView == null) {
mInflater = context.getLayoutInflater();
convertView = mInflater.inflate(R.layout.delete_campaign_row_layout, null);
}
merchantNameView = (TextView) convertView.findViewById(R.id.deleteMerchantNameView);
deleteButton = (Button) convertView.findViewById(R.id.deleteButton);
selectForDelete = (CheckBox) convertView.findViewById(R.id.deleteCheckBox);
merchantNameView.setText(c.getString(c.getColumnIndex("merchantName")));
final int rowID = c.getInt(c.getColumnIndex("_id"));
//final long rowID = mListView.getSelectedItemId();
if (checked[position] == 0) {
selectForDelete.setChecked(false);
} else {
selectForDelete.setChecked(true);
}
selectForDelete.setOnCheckedChangeListener(new OnCheckedChangeListener()
{
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked)
{
if ( isChecked )
{
// Make deleteButton appear
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
checked[position] = 1;
deleteButton.setVisibility(0);
}
else {
//Hide deleteButton
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
checked[position] = 1;
deleteButton.setVisibility(8);
}
}
});
deleteButton.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
if(rowID > 0) {
//Log.d(TAG, "Button Click. rowID = "+rowID);
myDbHelper.deleteRecordWithID(rowID);
c.requery();
notifyDataSetChanged();
initializeChecked(); // re-set the checked array
}
}
});
return convertView;
}
public void initializeChecked() {
checked = new int[c.getCount()];
int i = 0;
while (i < c.getCount()) {
checked[i] = 0;
i++;
}
}
日志:
05-12 00:36:23.232: W/KeyCharacterMap(15777): Can't open keycharmap file
05-12 00:36:23.232: W/KeyCharacterMap(15777): Error loading keycharmap file '/system/usr/keychars/synaptics-rmi-touchscreen.kcm.bin'. hw.keyboards.65537.devname='synaptics-rmi-touchscreen'
05-12 00:36:23.232: I/KeyCharacterMap(15777): Using default keymap: /system/usr/keychars/qwerty.kcm.bin
05-12 00:36:23.332: D/WindowManagerImpl(15777): addView, new view, mViews[1]: com.android.internal.policy.impl.PhoneWindow$DecorView@4053e328
05-12 00:36:23.903: D/View(15777): onTouchEvent: viewFlags: 0x18005001
05-12 00:36:23.903: D/View(15777): onTouchEvent: isFocusable: true, isFocusableInTouchMode: false, isFocused: false; focusTaken: false
05-12 00:36:23.913: D/WindowManagerImpl(15777): finishRemoveViewLocked, mViews[1]: com.android.internal.policy.impl.PhoneWindow$DecorView@4053e328
05-12 00:36:23.973: D/Database(15777): dbopen(): path = /data/data/com.roardog.FidelrBasic/databases/Fidelr_memory, flag = 6, file size = 5120
05-12 00:36:23.973: D/Database(15777): dbopen(): path = /data/data/com.roardog.FidelrBasic/databases/Fidelr_memory, mode: wal, disk free size: 253 M, handle: 0x387968
05-12 00:36:23.983: D/ATRecorder(15777): com.htc.autotest.dlib.RecordEngine in loader dalvik.system.DexClassLoader@40550268
05-12 00:36:23.983: D/WindowManagerImpl(15777): addView, new view, mViews[1]: com.android.internal.policy.impl.PhoneWindow$DecorView@40548d90
05-12 00:36:23.993: D/AndroidRuntime(15777): Shutting down VM
05-12 00:36:23.993: W/dalvikvm(15777): threadid=1: thread exiting with uncaught exception (group=0x4001d5a0)
05-12 00:36:24.023: E/AndroidRuntime(15777): FATAL EXCEPTION: main
05-12 00:36:24.023: E/AndroidRuntime(15777): android.database.CursorIndexOutOfBoundsException: Index -1 requested, with a size of 3
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.database.AbstractCursor.checkPosition(AbstractCursor.java:585)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.database.AbstractWindowedCursor.checkPosition(AbstractWindowedCursor.java:214)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.database.AbstractWindowedCursor.getString(AbstractWindowedCursor.java:41)
05-12 00:36:24.023: E/AndroidRuntime(15777): at com.roardog.FidelrBasic.DeleteCampaignCustomCursorAdapater.getView(DeleteCampaignCustomCursorAdapater.java:125)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.AbsListView.obtainView(AbsListView.java:1449)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.ListView.measureHeightOfChildren(ListView.java:1265)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.ListView.onMeasure(ListView.java:1128)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.View.measure(View.java:8553)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3261)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1017)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.LinearLayout.measureVertical(LinearLayout.java:386)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.LinearLayout.onMeasure(LinearLayout.java:309)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.View.measure(View.java:8553)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3261)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.View.measure(View.java:8553)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:3261)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.widget.FrameLayout.onMeasure(FrameLayout.java:250)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.View.measure(View.java:8553)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.ViewRoot.performTraversals(ViewRoot.java:915)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.view.ViewRoot.handleMessage(ViewRoot.java:1991)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.os.Handler.dispatchMessage(Handler.java:99)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.os.Looper.loop(Looper.java:150)
05-12 00:36:24.023: E/AndroidRuntime(15777): at android.app.ActivityThread.main(ActivityThread.java:4385)
05-12 00:36:24.023: E/AndroidRuntime(15777): at java.lang.reflect.Method.invokeNative(Native Method)
05-12 00:36:24.023: E/AndroidRuntime(15777): at java.lang.reflect.Method.invoke(Method.java:507)
05-12 00:36:24.023: E/AndroidRuntime(15777): 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:849) 05-12 00:36:24.023: E/AndroidRuntime(15777): 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:607) 05-12 00:36:24.023:E/AndroidRuntime(15777):在 dalvik.system.NativeStart.main( native 方法) 05-12 00:36:25.304: D/进程 (15777): killProcess, pid=15777 05-12 00:36:25.304:D/进程(15777):dalvik.system.VMStack.getThreadStackTrace( native 方法) 05-12 00:36:25.304: D/进程 (15777): java.lang.Thread.getStackTrace(Thread.java:745) 05-12 00:36:25.304: D/进程 (15777): android.os.Process.killProcess(Process.java:797) 05-12 00:36:25.304: D/进程(15777):com.android.internal.os.RuntimeInit$UncaughtHandler.uncaughtException(RuntimeInit.java:108) 05-12 00:36:25.304: D/进程(15777):java.lang.ThreadGroup.uncaughtException(ThreadGroup.java:854) 05-12 00:36:25.314: D/进程(15777):java.lang.ThreadGroup.uncaughtException(ThreadGroup.java:851) 05-12 00:36:25.314:D/进程(15777):dalvik.system.NativeStart.main( native 方法) 05-12 00:36:25.314: I/Process(15777): 发送信号。 PID:15777 SIG:9
最终工作解决方案
public class DeleteCampaignCustomCursorAdapater extends CursorAdapter {
protected static final String TAG = null;
DatabaseHelper myDbHelper;
protected SQLiteDatabase db;
private final Activity context;
private LayoutInflater mInflater;
protected ListView mListView;
private Cursor c;
public static int[] checked;
TextView merchantNameView;
Button deleteButton;
CheckBox selectForDelete;
//ImageView moveButton;
public DeleteCampaignCustomCursorAdapater(Context context, Cursor c) {
super(context, c);
this.context = (Activity) context;
mInflater = LayoutInflater.from(context);
this.c = c;
c.moveToFirst();
initializeChecked();
}
@Override
public void bindView(View view, Context context, final Cursor cursor) {
}
@Override
public View newView(Context context, Cursor cursor, ViewGroup parent) {
return mInflater.inflate(R.layout.delete_campaign_row_layout, null);
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
c.moveToPosition(position);
myDbHelper = new DatabaseHelper(context); //Links to DatabaseHelper class
if (convertView == null) {
mInflater = context.getLayoutInflater();
convertView = mInflater.inflate(R.layout.delete_campaign_row_layout, null);
}
merchantNameView = (TextView) convertView.findViewById(R.id.deleteMerchantNameView);
deleteButton = (Button) convertView.findViewById(R.id.deleteButton);
selectForDelete = (CheckBox) convertView.findViewById(R.id.deleteCheckBox);
merchantNameView.setText(c.getString(c.getColumnIndex("merchantName")));
final int rowID = c.getInt(c.getColumnIndex("_id"));
if (checked[position] == 0) {
selectForDelete.setChecked(false);
} else {
selectForDelete.setChecked(true);
}
selectForDelete.setOnCheckedChangeListener(new OnCheckedChangeListener()
{
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked)
{
if ( isChecked )
{
// Make deleteButton appear
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
checked[position] = 1;
deleteButton.setVisibility(0);
}
else {
//Hide deleteButton
RelativeLayout ll = (RelativeLayout) buttonView.getParent();
deleteButton = (Button) ll.findViewById(R.id.deleteButton);
//checked[position] = 1;
deleteButton.setVisibility(8);
}
}
});
deleteButton.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
if(rowID > 0) {
//Log.d(TAG, "Button Click. rowID = "+rowID);
myDbHelper.deleteRecordWithID(rowID); //Call deleteRecord on selected row
c.requery(); //requery the cursor
notifyDataSetChanged(); //notify the view the data set has changed
initializeChecked(); // re-set the checked array
}
}
});
return convertView;
}
public void initializeChecked() {
checked = new int[c.getCount()];
int i = 0;
while (i < c.getCount()) {
checked[i] = 0;
i++;
}
}
最佳答案
在 initializeChecked(); 之前将 c.moveToFirst(); 添加到你的构造函数中。
编辑
首先在您的 getView 中添加:
c.moveToPosition(position);
关于android - 在 CustomCursor Adapter getView() 中获取正确的光标,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10553389/
我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击
有没有办法在这个简单的get方法中添加超时选项?我正在使用法拉第3.3。Faraday.get(url)四处寻找,我只能先发起连接后应用超时选项,然后应用超时选项。或者有什么简单的方法?这就是我现在正在做的:conn=Faraday.newresponse=conn.getdo|req|req.urlurlreq.options.timeout=2#2secondsend 最佳答案 试试这个:conn=Faraday.newdo|conn|conn.options.timeout=20endresponse=conn.get(url
我有一个存储主机名的Ruby数组server_names。如果我打印出来,它看起来像这样:["hostname.abc.com","hostname2.abc.com","hostname3.abc.com"]相当标准。我想要做的是获取这些服务器的IP(可能将它们存储在另一个变量中)。看起来IPSocket类可以做到这一点,但我不确定如何使用IPSocket类遍历它。如果它只是尝试像这样打印出IP:server_names.eachdo|name|IPSocket::getaddress(name)pnameend它提示我没有提供服务器名称。这是语法问题还是我没有正确使用类?输出:ge
我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c
question的一些答案关于redirect_to让我想到了其他一些问题。基本上,我正在使用Rails2.1编写博客应用程序。我一直在尝试自己完成大部分工作(因为我对Rails有所了解),但在需要时会引用Internet上的教程和引用资料。我设法让一个简单的博客正常运行,然后我尝试添加评论。靠我自己,我设法让它进入了可以从script/console添加评论的阶段,但我无法让表单正常工作。我遵循的其中一个教程建议在帖子Controller中创建一个“评论”操作,以添加评论。我的问题是:这是“标准”方式吗?我的另一个问题的答案之一似乎暗示应该有一个CommentsController参
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
我喜欢使用Textile或Markdown为我的项目编写自述文件,但是当我生成RDoc时,自述文件被解释为RDoc并且看起来非常糟糕。有没有办法让RDoc通过RedCloth或BlueCloth而不是它自己的格式化程序运行文件?它可以配置为自动检测文件后缀的格式吗?(例如README.textile通过RedCloth运行,但README.mdown通过BlueCloth运行) 最佳答案 使用YARD直接代替RDoc将允许您包含Textile或Markdown文件,只要它们的文件后缀是合理的。我经常使用类似于以下Rake任务的东西:
假设我有这个范围:("aaaaa".."zzzzz")如何在不事先/每次生成整个项目的情况下从范围中获取第N个项目? 最佳答案 一种快速简便的方法:("aaaaa".."zzzzz").first(42).last#==>"aaabp"如果出于某种原因你不得不一遍又一遍地这样做,或者如果你需要避免为前N个元素构建中间数组,你可以这样写:moduleEnumerabledefskip(n)returnto_enum:skip,nunlessblock_given?each_with_indexdo|item,index|yieldit
我一直致力于让我们的Rails2.3.8应用程序在JRuby下正确运行。一切正常,直到我启用config.threadsafe!以实现JRuby提供的并发性。这导致lib/中的模块和类不再自动加载。使用config.threadsafe!启用:$rubyscript/runner-eproduction'pSim::Sim200Provisioner'/Users/amchale/.rvm/gems/jruby-1.5.1@web-services/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:105:in`co
我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent