Kivy:如何在RecycleView中正确制作标签

问题描述

我试图使用recycleview在kivy上构建日志查看器,因为日志可能很大。我为每行分配一个标签小部件,以便将来可以更好地控制文本。有些行会比其他行包含更多文本,因此需要对Label小部件进行调整以根据其大小调整大小,但是当将其放入recycleview时,似乎似乎无法再控制每行小部件的高度,则其大小将保持不变。我期望的是标签可以包裹在文本上并调整高度,因为不需要行之间的多余空间。如果只有很少的文本,则会显示大量可用空间,如果我在标签中放入过多的文本,则会淹没标签,并且标签不会增长。

我尝试使用不同代码的一种解决方法是,每个标签至少分配200行,这似乎可行,但是我确实需要对文本的每一行进行更多控制。

这是示例代码:

from kivy.app import App
from kivy.lang import Builder
from kivy.uix.boxlayout import BoxLayout
from kivy.properties import ObjectProperty

Builder.load_string('''
<Row@BoxLayout>:
    canvas:
        Color:
            rgba: 1,0.1,0.5 #Red Marker
        Rectangle:
            size: self.size
            pos: self.pos
    value: ''
    orientation: 'vertical'
    Label:
        text: root.value
        text_size: self.width,None
        size_hint_y: None
        height: self.texture_size[1]
        font_size: 20

<LogDisplayWidget>:
    rv: rv
    orientation: 'vertical'
                
    RecycleView:
        id: rv
        scroll_type: ['bars','content']
        scroll_wheel_distance: dp(114)
        bar_width: dp(10)
        viewclass: 'Row'
        RecycleBoxLayout:
            default_size_hint: 1,None
            size_hint_y: None
            height: self.minimum_height
            orientation: 'vertical'
            spacing: dp(2)
''')


class LogDisplayWidget(BoxLayout):
    rv = ObjectProperty()

    def __init__(self):
        super(LogDisplayWidget,self).__init__()
        self.load_text()

    def load_text(self):
        for i in range(10):
            line = str(i) + 'This is a test of a bunch of text'
            self.rv.data.append({'value': line})


class TestApp(App):
    def build(self):
        return LogDisplayWidget()


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

enter image description here enter image description here

解决方法

进行了代码重写,标签在第一页中显示为正确调整了大小,但是滚动后却得到了意想不到的结果,它有时显示正确的标签大小,然后又有些变大,并且滚动跳过,如尝试自行调整并固定大小再次。有没有人有更好的方法来实现这一点,或者我错过了什么?我怀疑这与视图刷新的方式有关

这是新代码:

from kivy.app import App
from kivy.lang import Builder
from kivy.uix.recycleview import RecycleView
import random

Builder.load_string('''
<Row@Label>:
    canvas.before:
        Color:
            rgba: 0.8,0.1,0.5 #Red Marker
        Rectangle:
            size: self.size
            pos: self.pos
    text_size: self.width,None
    size_hint_y: None
    height: self.texture_size[1]
    font_size: dp(20)

<RV>:
    viewclass: 'Row'
    RecycleBoxLayout:
        default_size: None,dp(20)
        default_size_hint: 1,None
        size_hint_y: None
        height: self.minimum_height
        orientation: 'vertical'
        spacing: dp(3)
        
''')


class RV(RecycleView):
    def __init__(self,**kwargs):
        super(RV,self).__init__(**kwargs)
        line = ''
        for i in range(50):
            n = random.randint(0,1)
            if n:
                j = random.randint(5,30)
                line = 'Line: ' + str(i+1) + ' This is a test of a bunch of text' * j
            else:
                line = 'Line: ' + str(i+1) + ' This is a test of a bunch of text'
            self.data.append({'text': line})


class TestApp(App):
    def build(self):
        return RV()


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

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...