Kivy:滚动缩放

Posted

技术标签:

【中文标题】Kivy:滚动缩放【英文标题】:Kivy: scroll to zoom 【发布时间】:2018-09-23 05:26:13 【问题描述】:

有没有办法在桌面 kivy 应用程序上放大图像(例如用鼠标滚轮缩放)?它似乎在这里讨论:https://github.com/kivy/kivy/issues/3563 但我不知道是否有解决办法。

我从我的 kivy 应用程序中的静态图像开始。我想添加缩放/平移图像的功能。我不希望图像框架的实际大小发生变化,只需添加缩放/平移功能,就像您可能期望与谷歌地图交互一样。

可能的编程方向

根据我的阅读,我应该使用 Scatter(?),并且我发现我可以手动设置 Scatter 比例以放大/缩小图像。

我最初的想法是,我必须添加一个带有滚动视图的单独小部件以包含分散布局,这将使图像框架保持一致的大小。然后我需要添加一个动态改变比例值的事件。

问题

我找不到用于更改比例值的适当事件。 on_motion 事件似乎很有希望。我的应用可以使用滑块更新值,但是当我尝试使用 on_motion, 的类似方法时,我得到 AttributeError: motion 错误

我正在努力创建小部件。大多数文档似乎在 python 文件中使用.add_widget(...)。是否可以从 kv 文件中执行此操作?我想这个过程类似于屏幕和屏幕管理器,但我正在努力寻找一种可行的方法。

有没有更直接的方法可以做到这一点?

有没有办法可以在我的 kv 文件中使用 on_motion 类型事件来使用鼠标滚轮调整此值?

我给出了我的 kivy 应用程序结构的淡化示例 - 以及我尝试添加 Scatter 的方法。我想我需要将它放入它自己的小部件中以保持图像大小相同?

玩具示例

import kivy
from kivy.lang import Builder
from kivy.core.window import Window


kivy.require('1.1.0')

from kivy.app import App

presentation = Builder.load_file("scatter.kv")
class TestApp(App):
    def build(self):
        Window.clearcolor = (1, 1, 1, 1)
        return presentation

    # def foo():
    #    print("You've reached foo")    

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

#:kivy 1.10.0
GridLayout:
    cols: 2

    Scatter:
        scale: 5
        # on_motion: root.foo()
        Image :
            source: 'foo.png'
            allow_stretch: True
            keep_ratio: True

    Button:
        text: 'Hello World'

产生:

相关种类:

Scatter Covers whole display kivy(未答复) https://github.com/kivy/kivy/issues/3563

【问题讨论】:

从未理解过ScatterScatterLayout,但您可以在app.build() 方法中使用Window.bind(on_motion=on_motion)on_motion() 方法中获取滚动事件。或者,更好的是,您可以将绑定命令放在自定义布局小部件中,并使用实例 on_motion 方法。 请参考我的解决方案detect mouse wheel scrolling 【参考方案1】:

为了实现我的目标,我结合了kivy python3 detect mousewheel 中的信息,正如ikolim 指出的那样,以及此处给出的代码:https://github.com/kivy/kivy/wiki/Draggable-Scalable-Button

为了让我的回答简短,这里是扩展 Scatter 对象的简约对象。

class ResizableDraggablePicture(Scatter):
    def on_touch_down(self, touch):
        # Override Scatter's `on_touch_down` behavior for mouse scroll
        if touch.is_mouse_scrolling:
            if touch.button == 'scrolldown':
                if self.scale < 10:
                    self.scale = self.scale * 1.1
            elif touch.button == 'scrollup':
                if self.scale > 1:
                    self.scale = self.scale * 0.8
        # If some other kind of "touch": Fall back on Scatter's behavior
        else:
            super(ResizableDraggablePicture, self).on_touch_down(touch)

布局略有不同,我更改了按钮上的文本,但我的代码的功能可以在以下 gif 中看到:

对于任何想查看我的整个玩具项目以适应自己的目的的人,整个代码都在我的 github 上:https://github.com/melissadale/Learning-Kivy/tree/master/ZoomPanning

更新我的代码已经被编辑,从面向对象的方法来看更加正确,所以我不能问心无愧地拒绝编辑。然而,当我第一次开始使用 kivy 时,我会发现这段代码令人困惑。如果你只想看简单的版本可以直接申请验证相关,这是我的原始代码:

    if touch.is_mouse_scrolling:
    if touch.button == 'scrolldown':
        print('down')
        ## zoom in
        if self.scale < 10:
            self.scale = self.scale * 1.1

    elif touch.button == 'scrollup':
        ## zoom out
        print('up')
        if self.scale > 1:
            self.scale = self.scale * 0.8

【讨论】:

【参考方案2】:

@mdoc-2011 的other answer 的问题是它没有锚定在鼠标指针上,这意味着使用起来很烦人,因为内容会滚动很多。这可以通过不使用.scale 属性而是直接调用apply_transform 来解决:

if touch.is_mouse_scrolling:
    factor = None
    if touch.button == 'scrolldown':
        if self.scale < self.scale_max:
            factor = 1.1
    elif touch.button == 'scrollup':
        if self.scale > self.scale_min:
            factor = 1 / 1.1
    if factor is not None:
        self.apply_transform(Matrix().scale(factor, factor, factor),
                             anchor=touch.pos)

此版本还使用.scale_(min/max) 属性,而不是1.10.8 使用1/1.1,这样可以更直观地滚动进出。

【讨论】:

以上是关于Kivy:滚动缩放的主要内容,如果未能解决你的问题,请参考以下文章

(垂直)滚动视图包含另一个(水平)滚动视图 - Kivy

KIVY:缺少滚动条

在kivy中禁用弹跳滚动效果

Kivy Scrollview 自动滚动到新文本。防止向上滚动

Kivy 滚动浏览数千行文本

带有 kivy 语言和自定义小部件的 Kivy 滚动视图