Значение текстового поля Kivy повторяется для нескольких строк в Recycleview - PullRequest
0 голосов
/ 07 августа 2020

Моя проблема в том, что когда у меня есть, скажем, 20 элементов в данных recycleview. Для каждого элемента данных будет новая строка. Каждая строка имеет текстовое поле, поэтому вы можете изменить значение.

Если я изменяю значение строки индекса 0, оно повторяется до первой строки, которая будет загружена лениво, и если я прокручу вниз, чтобы эта первая ленивая загруженная строка была первой строка в представлении, затем снова первый элемент вне поля зрения, который загружается лениво, получит то же значение и т. д. помощь будет очень признательна. Спасибо!

from kivy.app import App
from kivy.lang import Builder
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.gridlayout import GridLayout
from kivy.uix.recycleview import RecycleView
from kivy.uix.recycleview.views import RecycleDataViewBehavior
from kivy.uix.label import Label
from kivy.properties import BooleanProperty
from kivy.uix.recycleboxlayout import RecycleBoxLayout
from kivy.uix.behaviors import FocusBehavior, ButtonBehavior
from kivy.uix.recycleview.layout import LayoutSelectionBehavior
from kivy.uix.screenmanager import ScreenManager, Screen
 
 
 
class MainScreen(Screen):
    pass
 
 
 
class SelectableRecycleBoxLayout(FocusBehavior, LayoutSelectionBehavior,
                                 RecycleBoxLayout):
    ''' Adds selection and focus behaviour to the view. '''
 
 
class SelectableLabel(RecycleDataViewBehavior, GridLayout, ButtonBehavior):
    ''' Add selection support to the Label '''
    index = None
    selected = BooleanProperty(False)
    selectable = BooleanProperty(True)
    cols = 2
 
    def refresh_view_attrs(self, rv, index, data):
        ''' Catch and handle the view changes '''
        self.index = index
        self.quantity_text = data['quantity']
        self.name_text = data['name']
        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
        if is_selected:
            print("selection changed to {0}".format(rv.data[index]))
        else:
            print("selection removed for {0}".format(rv.data[index]))
 
 
class RV(RecycleView):
 
    name_list = []
    quantity_list = []
 
 
    def __init__(self, **kwargs):
        super(RV, self).__init__(**kwargs)
 
        for x in range(15):
            self.name_list.append('item'+str(x))
 
        for i in range(len(self.name_list)):
            self.quantity_list.append('1')
 
        paired_iter = zip(self.quantity_list,  self.name_list)
        self.data = []
        for i1, i2 in paired_iter:
            d = {'quantity': i1, 'name': i2}
            self.data.append(d)
 
 
 
 
    def update_quantity(self, text, index):
        self.data[index]['quantity'] = text
        self.refresh_from_data()
        print(self.data)
 
 
class ScreenManagement(ScreenManager):
    pass
 
with open('main.kv', encoding='utf-8') as f:
    KV = Builder.load_string(f.read())
 
class TestApp(App):
    def build(self):
        return KV
 
if __name__ == '__main__':
    TestApp().run()
 
 
 
 
 
 
 
KV FILE BEGINS _______________________________
 
 
ScreenManagement:
    MainScreen:
        name:"MainScreen"
        id:MainScreen
 
 
<MainScreen>:
    rvcreatewidget : rvcreatewidget
    BoxLayout:
        orientation:'vertical'
        RV:
            id:rvcreatewidget
            viewclass: 'SelectableLabel'
            SelectableRecycleBoxLayout:
                default_size: None, dp(56)
                default_size_hint: 1, None
                size_hint_y: None
                height: self.minimum_height
                orientation: 'vertical'
                multiselect: True
                touch_multiselect: True
                scroll_type:['content','bars']
 
        Button:
            size_hint_y:0.3
            text:'test'
 
 
 
 
 
 
 
<SelectableLabel>:
    # Draw a background to indicate selection
    canvas.before:
        Color:
            rgba: (.0, 0.1, .1, .8) if self.selected else (0, 0, 0, 0.5)
        Rectangle:
            pos: self.pos
            size: self.size
    quantity_text:'1'
    name_text:'dummyname'
 
 
 
    TextInput:
        size_hint_x:0.1
        id:quantity
        text: root.quantity_text
        background_color: 0,0,0,0.4
        foreground_color: 1,1,1,1
        multiline:False
        halign:'center'
        input_filter:"int"
        write_tab:False
        on_text_validate:
            app.root.ids.MainScreen.ids.rvcreatewidget.update_quantity(args[0].text, root.index)
 
 
    Label:
        id: id_name
        text: root.name_text
        background_color: 1,0,0,0.4

...