Kivy tutorial 008: More kv language

Kivy tutorial 008: More kv language – Kivy Blog

Central themes: Event binding and canvas instructions in kv language
中心主题: 事件绑定 和 kv语言里的画布结构

This tutorial directly follows on from the previous, so start by retrieving the previous code, as below:
这节导师课直接地跟随上节的,因此从上次的代码找了些代码开始。

main.py:

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.slider import Sliderfrom kivy.uix.boxlayout import BoxLayout
from kivy.uix.label import Label
from kivy.uix.slider import Sliderfrom kivy.uix.widget import Widget
from kivy.graphics import Rectangle, Color, Linefrom random import randomclass DrawingWidget(Widget):def __init__(self):super(DrawingWidget, self).__init__()with self.canvas:Color(1, 1, 1, 1)self.rect = Rectangle(size=self.size,pos=self.pos)self.bind(pos=self.update_rectangle,size=self.update_rectangle)def update_rectangle(self, instance, value):self.rect.pos = self.posself.rect.size = self.sizedef on_touch_down(self, touch):super(DrawingWidget, self).on_touch_down(touch)if not self.collide_point(*touch.pos):returnwith self.canvas:Color(random(), random(), random())self.line = Line(points=[touch.pos[0], touch.pos[1]], width=2)def on_touch_move(self, touch):if not self.collide_point(*touch.pos):returnself.line.points = self.line.points + [touch.pos[0], touch.pos[1]]class Interface(BoxLayout):passclass DrawingApp(App):def build(self):root_widget = Interface()return root_widgetDrawingApp().run()

drawing.kv:

<Interface>:orientation: 'vertical'DrawingWidget:Slider:min: 0max: 1value: 0.5size_hint_y: Noneheight: 80Slider:min: 0max: 1value: 0.5size_hint_y: Noneheight: 80Slider:min: 0max: 1value: 0.5size_hint_y: Noneheight: 80BoxLayout:orientation: 'horizontal'size_hint_y: Noneheight: 80Label:text: 'output colour:'Widget:

The first thing to do is draw the coloured Rectangle that the final Widget uses to display an output colour, and for this we need to know how to draw canvas instructions in kv language. The syntax is as below:
第一件事 是 画多彩的矩形, 这矩形是最后的组件 用来演示一个output颜色, 对这个颜色,我们需要知道kv语言中画图结构如何。

Widget:canvas:Color:rgb: 0, 1, 0  # using a fixed colour for nowRectangle:size: self.sizepos: self.pos

Run the code, and you’ll see another of kv language’s most important features; automatic event binding. In the original Python code of tutorial 7 we needed an extra .bind(...) call to make the be updated to always be placed within its Widget. In kv language this is not necessary, the dependency on self.size and self.pos is automatically detected, and a binding automatically created!
运行代码, 并且你将看到kv语言另一个最重要的特征;自动地事件绑定。 在导师课7中的Python源始代码,我们需要一个巨大的  .extra() 召唤, 让更新 总是在它组件内。 在kv语言中是不必要的, self.size 和self.pos独自自动地执行并且绑定了自动地创建!

This is also the generic syntax for canvas instructions; first add canvas: (or canvas.before or canvas.after), then, indent by 4 spaces, and add canvas instructions much like you would Widgets. However, note that canvas instructions are not widgets.
这也是通用的语法给canvas结构;第一 添加canvas: (或者canvas.before 或者 canvas.after )然后, 4个空格缩进, 并且增加尽可能你想要的组件到canvas结构。 然而, 注意canvas 结构不是组件。

The only thing now missing from the original Python interface implementation in tutorial 7 is having the Sliders automatically update the output colour rectangle. Change the <Interface>: rule to the following:
在导师课第7节中从源码界面执行中唯一缺少的事情是 有sliders 自动地更新到 输出颜色矩形。 改变<interface>: 规则在下面:

<Interface>:orientation: 'vertical'DrawingWidget:Slider:id: red_slidermin: 0max: 1value: 0.5size_hint_y: Noneheight: 80Slider:id: green_slidermin: 0max: 1value: 0.5size_hint_y: Noneheight: 80Slider:id: blue_slidermin: 0max: 1value: 0.5size_hint_y: Noneheight: 80BoxLayout:orientation: 'horizontal'size_hint_y: Noneheight: 80Label:text: 'output colour:'Widget:canvas:Color:rgb: red_slider.value, green_slider.value, blue_slider.valueRectangle:size: self.sizepos: self.pos

There are actually only two changes here; we gave each Slider an id declaration, and in the canvas Color referred to the sliders with this name. Giving a widget an id is just like naming it in Python so that you can refer to it elsewhere.
这只有2个改变: 我们给每个slider 一个id声明, 并且在canvas 颜色中提及了sliders 和这个名字。 给与一个widget组件一个id 就像在python中命名它一样,因此你可以在其他的地方提及它。

Thanks to kv’s automatic binding, this is all we need to do to have the Color update automatically whenever a slider value changes. Run the code, and you should see that things work exactly as they did in the original Python interface.
感谢kv自动地绑定, 这是所有我们需要做的是 让颜色更新自动地当一个slider值改变时。 运行代码,并且你应该看到事儿们严谨地像在源码python界面中做的一样。

We can finish this tutorial with a couple of extra kv conveniences. First, just as we added an automatically updating Rectangle in the Widget kv, we can do the same for the background of the DrawingWidget. Delete the __init__ and update_rectangle methods in the Python DrawingWidget code, and add a new rule in the kv file:
我们可以完成这节导师课同一些kv的巨大方便性。 首先,像我们添加的一个自动地更新的矩形在widget kv中一样, 我们可以在DrawingWidget的背景中做相同的事情。 删除__init__ 和  update_rectangle 方法在python 的DrawingWidget代码中, 并且添加一个新的规则在kv文件中:

<DrawingWidget>:canvas:Color:rgba: 1, 1, 1, 1Rectangle:pos: self.possize: self.size

Second, you might have noticed that there’s a lot of code duplication in each of the Slider rules - we set the same min, max, initial value`, ``size_hint_y` and height for every one. As is normal in Python, it would be natural to abstract this in a new class, so as to set each value only once. You can probably already see how to do this with what we’ve learned so far (make a new class YourSlider(Slider): in the Python and add a new <YourSlider>: rule in the kv), but I’ll note that you can even do this entirely in kv:
其次, 你可能已经注意到了那有一些复制代码在每个slider规则中,我们设置了相同的 min, initial value 'size_hint_y'  和高度给每一个Slider。 像平常在Python中一样, 它总是本能的使分离在一个新的类中,因此像每次设置每个值。你大概可以已经看到如何同我们学到的来这么做, (制作一个新的类: YourSlider(Slider): 在python 代码中,并且增加一个新的<YourSlider>: 规则在kv中, 但是我将意识到 你可以在kv中全部这么干:

<ColourSlider@Slider>:min: 0max: 1value: 0.5size_hint_y: Noneheight: 80<Interface>:orientation: 'vertical'DrawingWidget:ColourSlider:id: red_sliderColourSlider:id: green_sliderColourSlider:id: blue_sliderBoxLayout:orientation: 'horizontal'size_hint_y: Noneheight: 80Label:text: 'output colour:'Widget:canvas:Color:rgb: red_slider.value, green_slider.value, blue_slider.valueRectangle:size: self.sizepos: self.pos

The new <ColourSlider@Slider>: rule defines a dynamic class, a Python class kv rule without a corresponding Python code definition. This is convenient if you want to do something repeatedly only in kv, and never access it from Python.

新的 <ColourSlider@Slider>: 规则定义了一个动力的类, 一个python 类kv规则没有一个相关的python编码定义。 这是方便的如果你只想做些重复的事在kv中, 并且绝不从python中访问它。

At this point, we’ve reached feature parity with the original Python code, and seen all the basics of kv language. In the next tutorial we’ll finish off the original purpose of all these sliders; letting the user set the colour of line that is drawn by the DrawingWidget.
这时候,我们已经达到了和源码python代码相同的效果, 并且看到了所有的kv的基础元素。 在下面的导师课中,我们将完成这些sliders 原始的目的,让用户设置DrawingWidget画的线的颜色。

Full code

main.py:

from kivy.app import App
from kivy.uix.boxlayout import BoxLayout
from kivy.uix.slider import Sliderfrom kivy.uix.boxlayout import BoxLayout
from kivy.uix.label import Label
from kivy.uix.slider import Sliderfrom kivy.uix.widget import Widget
from kivy.graphics import Rectangle, Color, Linefrom random import randomclass DrawingWidget(Widget):def on_touch_down(self, touch):super(DrawingWidget, self).on_touch_down(touch)if not self.collide_point(*touch.pos):returnwith self.canvas:Color(random(), random(), random())self.line = Line(points=[touch.pos[0], touch.pos[1]], width=2)def on_touch_move(self, touch):if not self.collide_point(*touch.pos):returnself.line.points = self.line.points + [touch.pos[0], touch.pos[1]]class Interface(BoxLayout):passclass DrawingApp(App):def build(self):root_widget = Interface()return root_widgetDrawingApp().run()

drawing.kv:

<DrawingWidget>:canvas:Color:rgba: 1, 1, 1, 1Rectangle:pos: self.possize: self.size<ColourSlider@Slider>:min: 0max: 1value: 0.5size_hint_y: Noneheight: 80<Interface>:orientation: 'vertical'DrawingWidget:ColourSlider:id: red_sliderColourSlider:id: green_sliderColourSlider:id: blue_sliderBoxLayout:orientation: 'horizontal'size_hint_y: Noneheight: 80Label:text: 'output colour:'Widget:canvas:Color:rgb: red_slider.value, green_slider.value, blue_slider.valueRectangle:size: self.sizepos: self.pos

这段代码是使用Kivy框架(一个用于开发多触摸应用的Python库)来创建一个可以绘制线条的DrawingWidget类。让我们逐步分析这段代码:

  1. 类定义

class DrawingWidget(Widget):

这里定义了一个名为DrawingWidget的类,它继承自Kivy的Widget类。
      2. on_touch_down方法
当触摸屏幕并按下时,这个方法会被调用

* 首先,它调用了`super(DrawingWidget, self).on_touch_down(touch)`,这是为了确保父类(即`Widget`类)的`on_touch_down`方法也能得到执行(如果有需要的话)。

* 接着,它检查触摸点是否在`DrawingWidget`的区域内(使用`self.collide_point(*touch.pos)`)。如果不在,方法就返回,不执行后续的绘制代码。

* 如果在区域内,它会进入`with self.canvas:`块,这意味着它将对`canvas`(即绘图表面)执行一些操作。 + 它首先使用`Color(random(), random(), random())`随机选择一个颜色。 + 然后,它创建了一个`Line`对象,该对象从触摸点开始,但只有一个点(即触摸点的位置)。注意这里有个小问题:`Line`对象的`points`应该是一个二维列表,即列表的每个元素都是一个坐标(x, y)。但在这里,它只是一个包含两个元素的列表,这可能会导致问题。

小问题:如果触摸点不在DrawingWidget区域内,return 返回的是什么?

在给出的代码片段中,如果触摸点不在DrawingWidget的区域内,return语句将直接返回None,因为它没有指定要返回的具体值。在Python中,如果函数或方法中没有明确的return语句或return后面没有跟任何值,那么该函数或方法将默认返回None

on_touch_downon_touch_move方法中,当self.collide_point(*touch.pos)返回False(即触摸点不在DrawingWidget区域内)时,方法会立即通过return退出,不执行后续的代码。由于没有指定返回值,所以返回的是None

这种设计是为了避免在不必要的情况下执行绘图或更新操作,从而提高代码效率和响应速度。

        3. on_touch_move方法
当触摸点在屏幕上移动时,这个方法会被调用。

* 同样,它首先检查触摸点是否在`DrawingWidget`的区域内。

* 如果在区域内,它会更新`line`对象的`points`属性,将新的触摸点坐标添加到列表的末尾。但这里也有一个小问题:由于`line.points`是一个二维列表(至少它应该是一个),但之前我们设置它时似乎没有将其作为一个二维列表。所以,这里应该是`self.line.points += [[touch.pos[0], touch.pos[1]]]`,以确保我们添加的是一个坐标点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/35957.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

meizu M10 魅蓝 10 mblu10 root 解锁 安装LSPosed框架 紫光展锐改串 AT命令 一键新机 改机软件 硬改 改参数

meizu M10 魅蓝 10 mblu10 root 解锁 安装LSPosed框架 紫光展锐改串 AT命令 一键新机 改机软件 硬改 改参数 ro.system.build.version.release11 ro.system.build.version.release_or_codename11 ro.system.build.version.sdk30 ro.system.custom.versionAndroid_M01 ro.prod…

python目录树生成器

代码如下&#xff1a; import os from colorama import Fore, Style, init from tkinter import Tk, Label, Entry, Button, Text, Scrollbar, RIGHT, Y, END# 初始化 colorama init(autoresetTrue)def print_directory_tree(root_dir, text_widget, indent, lastTrue):"…

NISP国家信息安全水平考试

国家信息安全水平考试(NISP)是中国信息安全测评中心考试、发证&#xff0c;由国家网络空间安全人才培养基地运营管理&#xff0c;并授权网安世纪科技有限公司为NISP证书管理中心。 中国信息安全测评中心开展国家信息安全水平考试(NISP)考试项目&#xff0c;是为普及信息安全/网…

Sklearn 深入教程

Sklearn 深入教程 Scikit-learn&#xff08;sklearn&#xff09;是Python中功能强大且广泛使用的机器学习库。本教程将深入探讨sklearn中的高级主题&#xff0c;涵盖高级数据预处理、高级特征工程、复杂模型调参、高级集成学习、模型持久化与部署等内容。 目录 高级数据预处…

Python实例:faker生成数据集、文件

原文链接&#xff1a;http://www.juzicode.com/python-example-faker-create-data 本文的几个例子用faker生成数据并用openpyxl生成表格&#xff0c;或者生成csv、xml、图像文件。 A&#xff09;生成个人数据 #juzicode.com/VX公众号:juzicode from openpyxl import Workboo…

IOS Swift 从入门到精通:BlurEffect BlendMode stroke

文章目录 UIBlurEffectBlendModestroke基本用法:描边样式:与strokeBorder的区别:组合使用:自定义形状:UIBlurEffect 在Swift中,实现模糊效果通常是通过UIKit框架中的UIBlurEffect类来完成的,这通常被称作毛玻璃效果。 **创建UIBlurEffect实例:**选择一个模糊效果的样…

探索AudioLM的音频源识别能力:解锁声音的奥秘

&#x1f3b5; 探索AudioLM的音频源识别能力&#xff1a;解锁声音的奥秘 &#x1f50d; 在音频处理领域&#xff0c;识别不同的音频源是一项挑战性任务&#xff0c;涉及到音频信号处理、模式识别和机器学习等多个领域。AudioLM&#xff0c;作为一个假想的先进的音频处理模型&a…

docker 安装MySQL8以上

1.新建目录 mkdir -p /usr/local/src/mysql80 2.安装最新mysql # 拉取镜像 docker pull mysql 3.运行镜像创建容器&#xff0c;实现数据库和日志的持久化等命令 docker run \ -p 3380:3306 \ --name mysql80 \ --privilegedtrue \ --restartalways \ -v /usr/local/src/my…

Spring Boot如何实现跨域资源共享(CORS)?

&#x1f345; 作者简介&#xff1a;哪吒&#xff0c;CSDN2021博客之星亚军&#x1f3c6;、新星计划导师✌、博客专家&#x1f4aa; &#x1f345; 哪吒多年工作总结&#xff1a;Java学习路线总结&#xff0c;搬砖工逆袭Java架构师 &#x1f345; 技术交流&#xff1a;定期更新…

1.k8s:架构,组件,基础概念

目录 一、k8s了解 1.什么是k8s 2.为什么要k8s &#xff08;1&#xff09;部署方式演变 &#xff08;2&#xff09;k8s作用 &#xff08;3&#xff09;Mesos&#xff0c;Swarm&#xff0c;K8S三大平台对比 二、k8s架构、组件 1.k8s架构 2.k8s基础组件 3.k8s附加组件 …

5种算法简单介绍:贪心算法、分治法、回溯法、动态规划法、最大流算法

1. 贪心算法&#xff08;Greedy Algorithm&#xff09; 定义&#xff1a; 贪心算法是一种在每一步选择中都采取在当前状态下最好或最优&#xff08;即最有利&#xff09;的选择&#xff0c;从而希望导致结果是全局最好或最优的算法。贪心算法并不是对所有问题都能得到整体最优…

【STM32-存储器映射】

STM32-存储器映射 ■ STM32F1-4G地址空间分成8个块■ STM32F1-Block0■ STM32F1-Block1■ STM32F1-Block2■ STM32F1- ■ STM32F1-4G地址空间分成8个块 ■ STM32F1-Block0 有出厂 BootLoader 就可以使用串口下载程序。如Keil5图中IROM地址是0x8000000 开始 就是flash地址 ■ S…

使用方法——注意事项及好处

public class MethodDemo01 {public static void main(String[] args) {// 目标&#xff1a;掌握定义方法的完整性&#xff0c;清楚使用方法的好处。// 需求&#xff1a;假如现在有很多程序员都要进行2个整数求和的操作。//1、李工。int rs sun(10,20);System.out.println(&q…

低代码开发平台赋能智慧警务管理:创新引领下的安全新篇章

随着信息技术的飞速发展&#xff0c;智慧警务管理已经成为维护社会治安、保障人民安全的重要手段。在这一背景下&#xff0c;低代码开发平台以其高效、灵活、易用的特性&#xff0c;为智慧警务管理注入了新的活力。本文将探讨低代码开发平台在智慧警务管理中的应用&#xff0c;…

vue-codemirror的简单使用

功能 代码编辑器 安装 命令行&#xff1a;npm install codemirror vue-codemirror --save 单文件引入 import { Codemirror } from vue-codemirror 单文件展示 <codemirrorv-model"localCode"class"code-mirror":placeholder"placeholder&qu…

[vscode] 自定义log快捷生成代码

1、进入设置页面&#xff1a;文件>首选项>用户代码片段>选择设置的语言。 2. 关于代码段显示位置的调整设置 文件>首选项>设置&#xff0c;搜索代码段或snippetSuggestions&#xff0c;修改为”top”; 参考&#xff1a; vscode自定义log快捷生成代码

vue 设置定时器在某个时间段执行

业务需求&#xff1a;数据大屏页面中的某个方法需要在今天12:00执行一次&#xff0c;或是在今天的17:00执行一次&#xff0c; 这里用到 setTimeout定时器来实现。 //获取某个时间的时间戳 getCurrentDate(time) {let now new Date();let year now.getFullYear();let mont…

直流电机双闭环调速Simulink仿真

直流电机参数&#xff1a; 仿真模型算法介绍&#xff1a; 1&#xff09;三相整流桥&#xff0c;采用半控功率器件SCR晶闸管&#xff1b; 2&#xff09;采用转速环、电流环 双闭环控制算法&#xff1b; 3&#xff09;外环-转速环&#xff0c;采用PI 比例积分控制&#xff1b;…

Java进阶-Lambda

Java进阶-Lambda 前言Lambda表达式什么是Lambda表达式初识Lambda表达式Lambda表达式的简单使用Lambda表达式格式分析与传统接口方法实现的比较 理解Lambda表达式函数式编程非纯函数实例纯函数示例函数式编程在Lambda表达式中的体现 闭包闭包与Lambda表达式的示例 类型推导-匿名…

633. 平方数之和(中等)

633. 平方数之和 1. 题目描述2.详细题解3.代码实现3.1 Python3.2 Java内存溢出溢出代码正确代码与截图 1. 题目描述 题目中转&#xff1a;633. 平方数之和 2.详细题解 本题是167. 两数之和 II - 输入有序数组&#xff08;中等&#xff09;题目的变型&#xff0c;由两数之和变…