私はKivyメディアプレーヤーを開発中で、RecycleViewを使用してメディアリストを表示しています。次へ/前へKivy RecycleViewボタンを押してボタンを押します
私が達成しようとしているのは、既存の行選択をボタン選択からRecycleViewリストの次または前の行に変更することです。
私はLayoutSelectionBehaviorクラスが適切な方法のように見えますが、私はこれを使用する方法を見つけるのに苦労しています。以下
はRecycleView任意の助言又は支援が理解されるであろうkivy.uix.recycleview例
# -*- coding: utf-8 -*-
from kivy.app import App
from kivy.lang import Builder
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.recycleview import RecycleView
from kivy.uix.recycleview.views import RecycleDataViewBehavior
from kivy.uix.label import Label
from kivy.properties import BooleanProperty, StringProperty, NumericProperty
from kivy.uix.recycleboxlayout import RecycleBoxLayout
from kivy.uix.behaviors import FocusBehavior
from kivy.uix.recycleview.layout import LayoutSelectionBehavior
kv = """
<SelectableLabel>:
# Draw a background to indicate selection
canvas.before:
Color:
rgba: (0.4, 0.4, 0.4, 1) if self.selected else (0.5, 0.5, 0.5, 1)
Rectangle:
pos: self.pos
size: self.size
<KivyPlayer>:
canvas:
Color:
rgba: 0.3, 0.3, 0.3, 1
Rectangle:
size: self.size
pos: self.pos
orientation: 'vertical'
BoxLayout:
orientation: 'vertical'
BoxLayout:
Button:
id: next_track
text: "Next Track"
Button:
id: previous_track
text: "Previous Track"
BoxLayout:
RecycleView:
id: media_list
viewclass: 'SelectableLabel'
scroll_type: ['bars', 'content']
scroll_wheel_distance: dp(114)
bar_width: dp(10)
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
spacing: dp(2)
"""
Builder.load_string(kv)
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
if is_selected:
print("selection changed to {0}".format(rv.data[index]))
else:
print("selection removed for {0}".format(rv.data[index]))
class KivyPlayer(BoxLayout):
''' Main Kivy class for creating the initial BoxLayout '''
def __init__(self, **kwargs):
super(KivyPlayer, self).__init__(**kwargs)
# Set media_list data
self.ids.media_list.data = [{'text': str(x)} for x in range(100)]
class KivyApp(App):
def build(self):
return KivyPlayer()
if __name__ == '__main__':
KivyApp().run()
から変更の簡略化した例です。
この投稿を読んでいただきありがとうございます。このような詳細な返信@mattをご提供いただきありがとうございます。あなたのソリューションは完璧に動作しています。これまで見たことのないように、 'ids'の使用法を読み上げます。 SOモデレーター、新しい質問を投稿する必要がある場合、返信で追加の質問をする方針が不明です。 @mattには、KivyPlayerクラスのinitメソッドから最初/次のノードを簡単に選択する方法はありますか?あなたがそこにハンドルを持っているなら、#kivy IRCで議論してもよろしいですか? – campervancoder
私は、ButtonBehaviorクラスとtrigger_actionメソッドの 'self.next_track.trigger_action(duration = 0.1)'を使うというハッキーなアプローチを見つけました。私はもっとクリーンな方法があると思っています:) – campervancoder
このためにクロックを使用する必要があります。しかし、 'Clock.schedule_once(lambda * x:self.ids.controller.select_node(0)、0.1)'のようなものです。しかし、IRCがうまく動作しないかどうかを尋ねるべきです(私も 'matham'です)。 – Matt