Используется ли ListView так же, как RecycleView? - PullRequest
0 голосов
/ 08 июня 2019

Я пытаюсь напечатать self.search_results.data на моей консоли, но это не работает.Я правильно использую RecycleView?Учебник, с которым я учусь, из 2014 года и использует ListView.В интернете я обнаружил, что ListView устарел.Я не могу понять, как на самом деле работает RecycleView.Я прочитал документацию, но все еще не вижу.

Kivy:

WeatherRoot:

<WeatherRoot>:
    AddLocationForm


<AddLocationForm>: 



    orientation: "vertical"#

    search_input: search_input
    search_results: search_results_list

    BoxLayout:

        height: "40dp"
        size_hint_y:None

        TextInput:
            id: search_input
            size_hint_x: 50
            focus: True
            multiline: False
            on_text_validate: root.search_location()


         Button:
             text: "Search"
             size_hint_x: 25
             on_press: root.search_location()

        Button:
             text: "Current Location"
             size_hint_x: 25

     RecycleView:
        id: search_results_list
        data: []

Python:

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.properties import ObjectProperty, ListProperty
from kivy.network.urlrequest import UrlRequest
from kivy.factory import Factory
import json


class WeatherApp(App):
    pass

class WeatherRoot(BoxLayout):
    pass


class AddLocationForm(BoxLayout):

    search_input = ObjectProperty()
    search_results = ObjectProperty()

    # do something
    def search_location(self):

        search_template = "http://api.openweathermap.org/data/2.5/find?q= 
        {}&type=like"
        search_url = search_template.format(self.search_input.text)
        request = UrlRequest(search_url, self.found_location)

    def found_location(self, request, data):
        data =  json.loads(data.decode()) if not isinstance(data, dict) 
        else data
        cities = ["{} ({})".format(d['name'], d['sys']['country'])
              for d in data['list']]

        self.search_results.data= cities
        print(self.search_results.data)

if __name__ =='__main__':
    WeatherApp().run()

Строки в списке должны быть напечатанына консоли

1 Ответ

0 голосов
/ 08 июня 2019

Kivy ListView »Устаревший

ListView больше не определяется в недавно выпущенной стабильной версии Kivy 1.11.0.

Kivy RecycleView »MVC ( Model-View-Controller )

Представление генерируется путем обработки данных , по сути, списка диктов, и использует эти диктанты для создания экземпляров viewclass по мере необходимости. Его дизайн основан на MVC (Модель-представление-контроллер) шаблон.

  • Модель : модель состоит из данных , которые вы передаете через список диктовок.
  • Представление : представление разделено по макету и представлениям и реализовано с использованием адаптеров.
  • Контроллер : Контроллер определяет логическое взаимодействие и реализуется с помощью RecycleViewBehavior .

Решение

Чтобы создать RecycleView для выбираемого элемента, необходимо реализовать следующие классы как часть viewclass . Предметом обычно является виджет, например Метка , Кнопка или группа / ряд виджетов в макете ( BoxLayout или GridLayout ).

viewclass

  • Класс макета рециркуляции Selectabel, например, SelectableRecycleBoxLayout() или SelectableRecycleGridLayout()
  • Выбираемый класс виджетов, например SelectableLabel(), SelectableButton() или SelectableRow()

данные

* * Пример тысяча семьдесят-одна * * тысяча семьдесят две

Следующий пример иллюстрирует эквивалентность ListView с помощью RecycleView. viewclass - это выбираемый RecycleBoxLayout из Label виджет. Приложение использует OpenWeatherMap API для получения данных о погоде в Лондоне, Великобритания (Великобритания).

Примечание:

Чтобы совершать вызовы в OpenWeatherMap с использованием реальной точки API, вам необходим ключ API (APPID) .

main.py

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.recycleview.views import RecycleDataViewBehavior
from kivy.uix.label import Label
from kivy.properties import BooleanProperty, ObjectProperty
from kivy.uix.recycleboxlayout import RecycleBoxLayout
from kivy.uix.behaviors import FocusBehavior
from kivy.uix.recycleview.layout import LayoutSelectionBehavior
from kivy.network.urlrequest import UrlRequest
from kivy.lang import Builder

import json


class SelectableRecycleBoxLayout(FocusBehavior, LayoutSelectionBehavior,
                                 RecycleBoxLayout):
    ''' Adds selection and focus behaviour to the view. '''


class SelectableLabel(RecycleDataViewBehavior, Label):
    ''' Add selection support to the Label '''
    index = None
    selected = BooleanProperty(False)
    selectable = BooleanProperty(True)

    def refresh_view_attrs(self, rv, index, data):
        ''' Catch and handle the view changes '''
        self.index = index
        return super(SelectableLabel, self).refresh_view_attrs(
            rv, index, data)

    def on_touch_down(self, touch):
        ''' Add selection on touch down '''
        if super(SelectableLabel, self).on_touch_down(touch):
            return True
        if self.collide_point(*touch.pos) and self.selectable:
            return self.parent.select_with_touch(self.index, touch)

    def apply_selection(self, rv, index, is_selected):
        ''' Respond to the selection of items in the view. '''
        self.selected = is_selected


class AddLocationForm(BoxLayout):
    search_input = ObjectProperty()
    search_results = ObjectProperty()

    def search_location(self):
        search_template = "https://samples.openweathermap.org/data/2.5/find?q={}&appid=b6907d289e10d714a6e88b30761fae22"
        # search_template = "https://api.openweathermap.org/data/2.5/find?q={}&typle=like&appid=xyz"
        search_url = search_template.format(self.search_input.text)
        request = UrlRequest(search_url, self.found_location)

    def found_location(self, request, data):
        data = json.loads(data.decode()) if not isinstance(data, dict) else data
        cities = ["{} ({})".format(d['name'], d['sys']['country']) for d in data['list']]
        self.search_results.data = [{'text': str(x)} for x in cities]
        print(f"self.search_results.data={self.search_results.data}")


class WeatherRoot(BoxLayout):
    pass


class TestApp(App):
    title = "Weather App"

    def build(self):
        return Builder.load_file("main.kv")


if __name__ == '__main__':
    TestApp().run()

main.kv

WeatherRoot:

<WeatherRoot>:
    AddLocationForm:

<SelectableLabel>:
    # Draw a background to indicate selection
    canvas.before:
        Color:
            rgba: (1, 0, 0, 1) if self.selected else (.0, 0.9, .1, .3)
        Rectangle:
            pos: self.pos
            size: self.size
        Color:
            rgba: (0, 0.9, .1, .3)
        Rectangle:
            pos: self.pos
            size: self.size

<AddLocationForm>:
    orientation: "vertical"

    search_input: search_input
    search_results: search_results_list

    BoxLayout:
        height: "40dp"
        size_hint_y:None

        TextInput:
            id: search_input
            size_hint_x: 50
            focus: True
            multiline: False
            hint_text: 'Your city name'
            on_text_validate: root.search_location()


        Button:
            text: "Search"
            size_hint_x: 25
            on_press: root.search_location()

        Button:
            text: "Current Location"
            size_hint_x: 25

    RecycleView:
        id: search_results_list

        viewclass: 'SelectableLabel'

        SelectableRecycleBoxLayout:
            default_size: None, dp(26)
            default_size_hint: 1, None
            size_hint_y: None
            height: self.minimum_height
            orientation: 'vertical'
            multiselect: True
            touch_multiselect: True

выход

Data - Unselected Data - Selected

...