Blackberry: изменить сортировку KeywordFilterField - PullRequest
3 голосов
/ 15 апреля 2011

Я подготовил простой тестовый пример, демонстрирующий мою проблему.

Это всего лишь 1 файл (CheckMenu.java, указанный ниже), который можно запустить.

Как только пользователь изменит метод сортировки, выбрав 1 из 2 элементов меню, как мне перерисовать базовый KeywordFilterField ? Я пытался вызвать myList.updateList () и myList.invalidate () - это не помогает - элементы myList не реорганизованы.

Также мне интересно, что использовать вместо устаревшего MenuItem.setText (String)?

enter image description here

package mypackage;

import net.rim.device.api.system.*;
import net.rim.device.api.ui.*;
import net.rim.device.api.ui.component.*;
import net.rim.device.api.ui.container.*;
import net.rim.device.api.util.Comparator;
import net.rim.device.api.util.StringUtilities;
import net.rim.device.api.collection.util.*;

public class CheckMenu extends UiApplication {
    public static void main(String[] args) {
        CheckMenu myApp = new CheckMenu();       
        myApp.enterEventDispatcher();
    }

    public CheckMenu() {        
        pushScreen(new MyScreen());
    }    
}

class MyScreen extends MainScreen {
    KeywordFilterField myList = new KeywordFilterField();
    MyItemList myItems = new MyItemList();

    public MyScreen() {
        setTitle(myList.getKeywordField());

        myItems.doAdd(new MyItem(1, "Eins"));
        myItems.doAdd(new MyItem(2, "Zwei"));
        myItems.doAdd(new MyItem(3, "Drei"));
        myItems.doAdd(new MyItem(4, "Vier"));

        myList.setSourceList(myItems, new MyItem.MyProvider());
        // XXX commenting the line below does not help
        myList.setCallback(new MyListFieldCallback());
        add(myList);
    }

    private MenuItem numMenu = new MenuItem("num sort", 0, 0) {  
        public void run() {
            MyItem.NUMERIC_SORT = true;
            Status.show("Use " + toString());
            setText("num sort \u221A");
            alphaMenu.setText("alpha sort");
            myList.updateList();
        }
    };
    private MenuItem alphaMenu = new MenuItem("alpha sort", 1, 0) {  
        public void run() {
            MyItem.NUMERIC_SORT = false;
            Status.show("Use " + toString());
            setText("alpha sort \u221A");
            numMenu.setText("num sort");
            myList.updateList();
        }
    };

    protected void makeMenu(Menu menu, int instance) {
        super.makeMenu(menu, instance);
        menu.add(numMenu);
        menu.add(alphaMenu);
    }

    private class MyListFieldCallback implements ListFieldCallback {

        public void drawListRow(ListField list, Graphics g, int index, int y, int width) {
            Object obj = ((KeywordFilterField)list).getElementAt(index);
            if(obj != null && obj instanceof MyItem) {
                MyItem item = (MyItem) obj;
                g.drawText(item.toString(), 20 * (1 + index), y);
            } else if(index == 0) {
                g.drawText(list.getEmptyString(), 0, y);
            }
        }

        public Object get(ListField list, int index) { 
            return null; 
        }

        public int getPreferredWidth(ListField list) { 
            return 0; 
        }

        public int indexOfList(ListField list, String prefix, int start) { 
            return 0; 
        }
    }
}

class MyItemList extends SortedReadableList {
    public MyItemList() {
        super(new MyItem.MyComparator());        
    } 

    protected void doAdd(Object obj) {
        super.doAdd(obj);   
    }

    protected boolean doRemove(Object obj) {
        return super.doRemove(obj);        
    }
}

class MyItem {
    public static boolean NUMERIC_SORT;

    private int _num;
    private String _name;

    public MyItem(int num, String name) {
        _num = num;
        _name = name;
    }

    public String toString() {
        return _num + ": " + _name;
    }

    static class MyComparator implements Comparator {
        public int compare(Object obj1, Object obj2) {
            if (! (obj1 instanceof MyItem && obj2 instanceof MyItem))
                throw new IllegalArgumentException("Cannot compare non-MyItems");

            MyItem item1 = (MyItem) obj1;
            MyItem item2 = (MyItem) obj2;

            if (MyItem.NUMERIC_SORT) {
                if (item1._num == item2._num)
                    return 0;
                return (item1._num > item2._num ? 1 : -1);
            }

            return item1._name.compareTo(item2._name);
        }
    }

    static class MyProvider implements KeywordProvider {
        public String[] getKeywords(Object obj) {
            if (obj instanceof MyItem) {
                MyItem item = (MyItem) obj;
                return new String[]{ Integer.toString(item._num), item._name };
            }
            return null;
        }
    }
}

ОБНОВЛЕНИЕ: Я обновил код для использования makeMenu () в соответствии с рекомендациями Архимеда. Я также разместил свой вопрос на форуме BlackBerry .

Спасибо! Alex

Ответы [ 2 ]

1 голос
/ 19 апреля 2011

Может быть, я не понимаю вашу проблему ..... Но, как я вижу, этот код отлично работает на симуляторе

import net.rim.device.api.system.*;
import net.rim.device.api.ui.*;
import net.rim.device.api.ui.component.*;
import net.rim.device.api.ui.container.*;
import net.rim.device.api.util.Comparator;
import net.rim.device.api.util.StringUtilities;
import net.rim.device.api.collection.util.*;

public class CheckMenu extends UiApplication {
    public static void main(String[] args) {
        CheckMenu myApp = new CheckMenu();       
        myApp.enterEventDispatcher();
    }

    public CheckMenu() {        
        pushScreen(new MyScreen());
    }    
}

class MyScreen extends MainScreen {
    KeywordFilterField myList = new KeywordFilterField();
    MyItemList myItems = new MyItemList();

    public MyScreen() {
        setTitle(myList.getKeywordField());

        myItems.doAdd(new MyItem(1, "Eins"));
        myItems.doAdd(new MyItem(2, "Zwei"));
        myItems.doAdd(new MyItem(3, "Drei"));
        myItems.doAdd(new MyItem(4, "Vier"));

        myList.setSourceList(myItems, new MyItem.MyProvider());
        // XXX commenting the line below does not help
        myList.setCallback(new MyListFieldCallback());
        add(myList);
    }

    private MenuItem numMenu = new MenuItem("num sort", 0, 0) {  
    public void run() {
        MyItem.NUMERIC_SORT = true;
        Status.show("Use " + toString());
        setText("num sort \u221A");
        alphaMenu.setText("alpha sort");
        myItems.sort();
        myList.updateList();
    }
};
private MenuItem alphaMenu = new MenuItem("alpha sort", 1, 0) {  
    public void run() {
        MyItem.NUMERIC_SORT = false;
        Status.show("Use " + toString());
        setText("alpha sort \u221A");
        numMenu.setText("num sort");
        myItems.sort();
        myList.updateList();
    }
};

protected void makeMenu(Menu menu, int instance) {
    super.makeMenu(menu, instance);
    menu.add(numMenu);
    menu.add(alphaMenu);
}

private class MyListFieldCallback implements ListFieldCallback {

    public void drawListRow(ListField listField, Graphics g, int index, int y, int    width) {
        Object obj = ((KeywordFilterField)listField).getElementAt(index);
        if(obj != null && obj instanceof MyItem) {
            MyItem item = (MyItem) obj;
            g.drawText(item.toString(), 20 * (1 + index), y);
        } else if(index == 0) {
            g.drawText("ssssssss", 0, y);
        }
    }

    public Object get(ListField listField, int index) { 
        return null; 
    }

    public int getPreferredWidth(ListField listField) { 
        return 0; 
    }

    public int indexOfList(ListField listField, String prefix, int start) { 
        return 0; 
    }
}
}

class MyItemList extends SortedReadableList {
    public MyItemList() {
        super(new MyItem.MyComparator());        
    } 

    protected void doAdd(Object obj) {
        super.doAdd(obj);   
    }

    protected boolean doRemove(Object obj) {
        return super.doRemove(obj);        
    }
 }

class MyItem {
    public static boolean NUMERIC_SORT;

    private int _num;
    private String _name;

    public MyItem(int num, String name) {
        _num = num;
        _name = name;
    }

    public String toString() {
        return _num + ": " + _name;
    }

    static class MyComparator implements Comparator {
        public int compare(Object obj1, Object obj2) {
            if (! (obj1 instanceof MyItem && obj2 instanceof MyItem))
                throw new IllegalArgumentException("Cannot compare non-MyItems");

            MyItem item1 = (MyItem) obj1;
            MyItem item2 = (MyItem) obj2;

            if (MyItem.NUMERIC_SORT) {
                if (item1._num == item2._num)
                    return 0;
                return (item1._num > item2._num ? 1 : -1);
            }

            return item1._name.compareTo(item2._name);
        }
    }

    static class MyProvider implements KeywordProvider {
        public String[] getKeywords(Object obj) {
            if (obj instanceof MyItem) {
                MyItem item = (MyItem) obj;
                return new String[]{ Integer.toString(item._num), item._name };
            }
        return null;
    }
}
}
0 голосов
/ 15 апреля 2011

1). Как я могу изменить текст MenuItem? Я бы нравится отображать / скрывать галочку символ \ u221A - в зависимости от того, какой порядок сортировки в настоящее время выбирается Пользователь. SetText (текст строки) метод устарел, и я не уверен как использовать setText (StringProvider текст)

Вместо использования addMenuItem(myMenu1); переопределить makeMenu(Menu menu, int instance):

protected void makeMenu(Menu menu, int instance) {
    super.makeMenu(menu, instance);

    // add menu items on-the-fly
    // generating their labels in accordance to some conditions
    menu.add(new MenuItem("any label", 1, 1) {
        public void run() {
            // do smth useful
        }
    });
}

2). После того, как пользователь изменит метод сортировки, выбрав 1 из 2 MenuItems, как мне перерисовать лежащий в основе KeywordFilterField? Я попытался вызвать myList.updateList () и myList.invalidate () - это не помогает.

Попробуйте закомментировать строку myList.setCallback(new MyListFieldCallback());. Я считаю, что KeywordFilterField уже реализует этот обратный вызов каким-то особым образом, поэтому, установив свой пользовательский ListFieldCallback, вы нарушите поведение по умолчанию / ожидаемое поведение.

ОБНОВЛЕНИЕ о пунктах меню:

Полностью избавьтесь от полей private MenuItem alphaMenu и private MenuItem numMenu. Создайте эти пункты меню на лету:

protected void makeMenu(Menu menu, int instance) {
    super.makeMenu(menu, instance);

    String label = "num sort" + (MyItem.NUMERIC_SORT ? " \u221A" : "");

    menu.add(new MenuItem(label, 0, 0) {  
        public void run() {
            MyItem.NUMERIC_SORT = true;
            Status.show("Use " + toString());
            // setText("num sort \u221A");
            // alphaMenu.setText("alpha sort");
            myList.updateList();
        }
    });

    label = "alpha sort" + (MyItem.NUMERIC_SORT ? "" : " \u221A");

    menu.add(new MenuItem(label, 1, 0) {
        public void run() {
            MyItem.NUMERIC_SORT = false;
            Status.show("Use " + toString());
            // setText("alpha sort \u221A");
            // numMenu.setText("num sort");
            myList.updateList();
        }
    });
}

ОБНОВЛЕНИЕ по основной проблеме:

class MyScreen extends MainScreen {
    KeywordFilterField myList = new KeywordFilterField();
    MyItemList myItems = new MyItemList();

    public MyScreen() {
        setTitle(myList.getKeywordField());

        myItems.doAdd(new MyItem(1, "Eins"));
        myItems.doAdd(new MyItem(2, "Zwei"));
        myItems.doAdd(new MyItem(3, "Drei"));
        myItems.doAdd(new MyItem(4, "Vier"));

        myList.setSourceList(myItems, new MyItem.MyProvider());

        // XXX commenting the line below does not help
        myList.setCallback(new MyListFieldCallback());
        add(myList);
    }

    protected void makeMenu(Menu menu, int instance) {
        super.makeMenu(menu, instance);

        String label = "num sort" + (MyItem.NUMERIC_SORT ? " \u221A" : "");

        menu.add(new MenuItem(label, 0, 0) {  
            public void run() {
                MyItem.NUMERIC_SORT = true;
                Status.show("Use " + toString());
                // setText("num sort \u221A");
                // alphaMenu.setText("alpha sort");

                myItems.setComparator(new MyItem.MyComparator());
                myList.updateList();
            }
        });

        label = "alpha sort" + (MyItem.NUMERIC_SORT ? "" : " \u221A");

        menu.add(new MenuItem(label, 1, 0) {
            public void run() {
                MyItem.NUMERIC_SORT = false;
                Status.show("Use " + toString());
                // setText("alpha sort \u221A");
                // numMenu.setText("num sort");

                myItems.setComparator(new MyItem.MyComparator());
                myList.updateList();
            }
        });

    }

// .. the rest of the code is the same
...