Управление одной строкой в ​​ListView - PullRequest
1 голос
/ 19 марта 2012

Я пытаюсь добавить анимацию в свой ListView, чтобы текст исчезал в отдельных строках по мере поступления результатов из запроса HTTP GET. Я знаю, как сделать эффект затухания, и у меня уже есть собственный адаптер ListView, но проблема в том, что ListView обновляет все строки каждый раз, когда приходит результат, таким образом вызывая эффект затухания каждый раз для всего списка.

Как бы я мог контролировать одну строку, чтобы ListView не анимировал каждую строку при каждом изменении данных?

Это код, который я использую для заполнения ListView:

    private class CustomAdapter extends ArrayAdapter<Row> {

    public CustomAdapter() {
        super(Results.this, R.layout.row, dataList);
    }

    public View getView(int position, View convertView, ViewGroup parent) {
        View row = convertView;
        RowHolder holder = null;
        if (row == null) {
            LayoutInflater inflater = getLayoutInflater();

            row = inflater.inflate(R.layout.row, parent, false);
            holder = new RowHolder(row);
            row.setTag(holder);
        } else {
            holder = (RowHolder) row.getTag();
        }
        try {
            holder.populateRow(dataList.get(position));
        } catch (NullPointerException e) {
            e.printStackTrace();
        }
        return row;
    }

}

    private class RowHolder {
        private TextView label = null;
        private TextView count = null;
        private TextView result = null;

        public RowHolder(View row) {
            label = (TextView) row.findViewById(R.id.list_label);
            count = (TextView) row.findViewById(R.id.list_count);
            result = (TextView) row.findViewById(R.id.list_result);
        }

        public void populateRow(Row r) {
            label.setText(r.getLabel());
            count.setText(r.getCount());
            result.setText(r.getResult());
            label.startAnimation(fadeIn);
            count.startAnimation(fadeIn);
            result.startAnimation(fadeIn);
        }
    }

Любая помощь приветствуется, спасибо заранее!

Редактировать 1:

Мой AsyncTask:

    private class CheckSource extends AsyncTask<String, Void, String> {

    protected void onPreExecute() {
        results.setUnixTime(getUnixTime());
        results.setLabel(getString(R.string.label));
        results.setCount(null);
        results.setResult(null);
        results.setResultLabel("");
        results.setShowProgress(true);
        results.setIconType(null);
        results.setShowIcon(false);
        results.setHasResults(false);
        adapter.notifyDataSetChanged();
    }

    @Override
    protected String doInBackground(String... params) {

        String query = params[0];
        String httpResults = null;

        try {
            httpResults = getResults(query, "source");
            jsonObject = new JSONObject(httpResults);
        } catch (IllegalStateException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (JSONException e) {
            e.printStackTrace();
        } catch (URISyntaxException e) {
            e.printStackTrace();
        }
        return httpResults;
    }

    protected void onPostExecute(String results) {
        try {
            parseJSON(results);
        } catch (JSONException e) {
            e.printStackTrace();
            results.setResultLabel("<br />"
                    + getString(R.string.source_not_available) + "<br />");
        }
        results.setShowProgress(false);
        adapter.notifyDataSetChanged();
    }

    // Parse the retrieved json results
    private void parseJSON(String jsonResults) throws JSONException {
        if (jsonResults == null) {
            results.setResult(null);
            results.setHasResults(false);
            results.setResultLabel("<br />"
                    + getString(R.string.source_not_available) + "<br />");
            return;
        }
        jsonObject = new JSONObject(jsonResults);
        String result = null;
        String resultLabel = null;
        switch (jsonObject.getInt("count")) {
        case -1:
            results.setCount(null);
            results.setHasResults(false);
            resultLabel = getString(R.string.no_results);
            break;
        case 0:
            results.setCount(null);
            results.setHasResults(false);
            resultLabel = getString(R.string.no_results);
            break;
        case 1:
            results.setHasResults(true);
            results.setCount(jsonObject.get("count").toString() + " "
                    + getString(R.string.one_result));
            result = jsonObject.get("url").toString();
            resultLabel = getString(R.string.hyperlink_text);
            break;
        default:
            results.setHasResults(true);
            results.setCount(jsonObject.get("count").toString() + " "
                    + getString(R.string.multiple_results));
            result = jsonObject.get("url").toString();
            resultLabel = getString(R.string.hyperlink_text);
            break;
        }
        results.setResult(result);
        results.setResultLabel("<br />" + resultLabel + "<br />");
    }
}

Метод, который выполняет HTTP-запрос:

private String getResults(String query, String source)
        throws IllegalStateException, IOException, URISyntaxException {

    /* Method variables */
    StringBuilder builder = new StringBuilder();
    String URL = "url";
    URI uri;
    String phrase = "phrase";
    List<NameValuePair> params = new ArrayList<NameValuePair>();

    /* HTTP variables */
    HttpGet httpGet;
    DefaultHttpClient httpClient;
    HttpResponse httpResponse;
    HttpEntity httpEntity;
    HttpParams httpParams;

    int socketTimeout = 10000;
    int connectionTimeout = 10000;

    // Set the socket and connection timeout values
    httpParams = new BasicHttpParams();
    HttpConnectionParams
            .setConnectionTimeout(httpParams, connectionTimeout);
    HttpConnectionParams.setSoTimeout(httpParams, socketTimeout);
    httpClient = new DefaultHttpClient(httpParams);
    // Add parameters to the GET request
    params.add(new BasicNameValuePair("query", query));
    params.add(new BasicNameValuePair("type", source));
    String paramString = URLEncodedUtils.format(params, "utf-8");
    uri = new URI(URL + paramString);
    httpGet = new HttpGet(uri);
    // Execute the GET request
    httpResponse = httpClient.execute(httpGet);

    /* Read http response if http status = 200 */
    if (httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
        httpEntity = httpResponse.getEntity();
        InputStream content = httpEntity.getContent();
        BufferedReader reader = new BufferedReader(new InputStreamReader(
                content));
        String line;
        while ((line = reader.readLine()) != null) {
            builder.append(line);
        }
    }
    return builder.toString();
}

Ответы [ 2 ]

5 голосов
/ 03 апреля 2012

Как недавно объяснил Ромен Гай во время сеанса ввода-вывода Google, наиболее эффективный способ обновить только одно представление в представлении списка - это что-то вроде следующего (в этом случае обновляются данные всего представления):

ListView list = getListView();
int start = list.getFirstVisiblePosition();
for(int i=start, j=list.getLastVisiblePosition();i<=j;i++)
    if(target==list.getItemAtPosition(i)){
        View view = list.getChildAt(i-start);
        list.getAdapter().getView(i, view, list);
        break;
    }

Предполагается, что target является одним элементом адаптера.

Этот код извлекает ListView, затем просматривает отображаемые в настоящий момент представления, сравнивает искомый элемент target с каждым отображаемым элементом представления, и, если ваша цель находится среди них, получите включающее представление и выполните адаптер getView в этом представлении для обновления дисплея.

В качестве примечания invalidate не работает так, как ожидают некоторые люди, и не обновляет представление, как это делает getView, notifyDataSetChanged перестроит весь список и в итоге вызовет getview для всех отображаемых элементов и invalidateViews также повлияет на группу.

И последнее: можно получить дополнительную производительность, если ему нужно только изменить дочерний элемент представления строки, а не всю строку, как это делает getView. В этом случае следующий код может заменить list.getAdapter().getView(i, view, list); (пример изменения текста TextView):

((TextView)view.findViewById(R.id.myid)).setText("some new text");

В коде мы доверяем.

0 голосов
/ 19 марта 2012

Метод notifyDataSetChanged принудительно вызывает метод getView для всех видимых элементов ListView. Если вы хотите обновить только 1 конкретный элемент из ListView, вам необходимо указать путь к AsynhTask.

...