From a4d089ad89fea16c7d3fa4743355229d163c30c4 Mon Sep 17 00:00:00 2001 From: qua-non Date: Tue, 6 Nov 2012 02:12:15 +0530 Subject: [PATCH 1/8] Doc: fix running info in screen module closes #783 --- kivy/modules/screen.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/kivy/modules/screen.py b/kivy/modules/screen.py index 71ba23d48..3525a41a7 100644 --- a/kivy/modules/screen.py +++ b/kivy/modules/screen.py @@ -11,15 +11,15 @@ To see a list of the available screenid, just run:: Simulate a medium-density screen as Motolora Droid 2:: - python main.py -m screen,droid2 + python main.py -m screen:droid2 Simulate a high-density screen as HTC One X, in portrait:: - python main.py -m screen,onex,portrait + python main.py -m screen:onex,portrait Simulate the iPad 2 screen:: - python main.py -m screen,ipad + python main.py -m screen:ipad ''' import sys From e2211026589e4ece6d6497ea6053c1efb796d2bf Mon Sep 17 00:00:00 2001 From: qua-non Date: Wed, 7 Nov 2012 15:32:37 +0530 Subject: [PATCH 2/8] Doc: make the VM visible in creating apk docs --- doc/sources/guide/packaging-android.rst | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/doc/sources/guide/packaging-android.rst b/doc/sources/guide/packaging-android.rst index 4906a6f90..f95afc7bb 100644 --- a/doc/sources/guide/packaging-android.rst +++ b/doc/sources/guide/packaging-android.rst @@ -10,6 +10,12 @@ Create a package for Android .. _Packaging your application into APK: +TestDrive +--------- +There is a VirtualBox Image we provide with the prerequisites along with +Android SDK and NDK preinstalled to ease your installation woes. You can +download it from `here `_. + Packaging your application into APK ----------------------------------- @@ -29,7 +35,7 @@ First, follow the prerequisites needed for the project: http://python-for-android.readthedocs.org/en/latest/prerequisites/ -Then a console, and type:: +Then open a console, and type:: git clone git://github.com/kivy/python-for-android From afd670942af6838e7715206e54df0d6d664199d7 Mon Sep 17 00:00:00 2001 From: Gendo Ikari Date: Wed, 7 Nov 2012 19:02:57 +0100 Subject: [PATCH 3/8] performance fix in widget constructor for event binding --- kivy/uix/widget.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/kivy/uix/widget.py b/kivy/uix/widget.py index 0ce503f42..ae3153a4e 100644 --- a/kivy/uix/widget.py +++ b/kivy/uix/widget.py @@ -154,9 +154,9 @@ class Widget(EventDispatcher): # Builder.idmap.pop('root') # Bind all the events - for argument, value in kwargs.items(): - if argument.startswith('on_'): - self.bind(**{argument: value}) + for argument in kwargs: + if argument[:3] == 'on_': + self.bind(**{argument: kwargs[argument]}) # # Collision From 6ddafdbfa8de5bbaa4ddec5995ab2f06b9e816ac Mon Sep 17 00:00:00 2001 From: Gendo Ikari Date: Wed, 7 Nov 2012 19:19:27 +0100 Subject: [PATCH 4/8] examples: using new api for event binding in kwargs --- examples/animation/animate.py | 3 +-- examples/canvas/canvas_stress.py | 12 ++++++------ examples/demo/showcase/main.py | 4 ++-- examples/widgets/popup_with_kv.py | 3 +-- 4 files changed, 10 insertions(+), 12 deletions(-) diff --git a/examples/animation/animate.py b/examples/animation/animate.py index 0b642b036..cb28e48f8 100644 --- a/examples/animation/animate.py +++ b/examples/animation/animate.py @@ -28,8 +28,7 @@ class TestApp(App): def build(self): # create a button, and attach animate() method as a on_press handler - button = Button(size_hint=(None, None), text='plop') - button.bind(on_press=self.animate) + button = Button(size_hint=(None, None), text='plop', on_press=self.animate) return button if __name__ == '__main__': diff --git a/examples/canvas/canvas_stress.py b/examples/canvas/canvas_stress.py index 458559cc7..b52998f0c 100644 --- a/examples/canvas/canvas_stress.py +++ b/examples/canvas/canvas_stress.py @@ -34,14 +34,14 @@ class StressCanvasApp(App): label = Label(text='0') - btn_add100 = Button(text='+ 100 rects') - btn_add100.bind(on_press=partial(self.add_rects, label, wid, 100)) + btn_add100 = Button(text='+ 100 rects', + on_press=partial(self.add_rects, label, wid, 100)) - btn_add500 = Button(text='+ 500 rects') - btn_add500.bind(on_press=partial(self.add_rects, label, wid, 500)) + btn_add500 = Button(text='+ 500 rects', + on_press=partial(self.add_rects, label, wid, 500)) - btn_reset = Button(text='Reset') - btn_reset.bind(on_press=partial(self.reset_rects, label, wid)) + btn_reset = Button(text='Reset', + on_press=partial(self.reset_rects, label, wid)) layout = BoxLayout(size_hint=(1, None), height=50) layout.add_widget(btn_add100) diff --git a/examples/demo/showcase/main.py b/examples/demo/showcase/main.py index db966fcc4..3eba89786 100644 --- a/examples/demo/showcase/main.py +++ b/examples/demo/showcase/main.py @@ -259,8 +259,8 @@ class ShowcaseApp(App): auto_dismiss=False) btnclose.bind(on_release=popup.dismiss) button = Button(text='Open popup', size_hint=(None, None), - size=('150sp', '70dp')) - button.bind(on_release=popup.open) + size=('150sp', '70dp'), + on_release=popup.open) popup.open() col = AnchorLayout() col.add_widget(button) diff --git a/examples/widgets/popup_with_kv.py b/examples/widgets/popup_with_kv.py index 61bf7ca8d..0a42c4527 100644 --- a/examples/widgets/popup_with_kv.py +++ b/examples/widgets/popup_with_kv.py @@ -22,8 +22,7 @@ class CustomPopup(Popup): class TestApp(App): def build(self): - b = Button() - b.bind(on_press=self.show_popup) + b = Button(on_press=self.show_popup) return b def show_popup(self, b): From 508788833b9b293625cfb6ca7deee881df62acd9 Mon Sep 17 00:00:00 2001 From: Mathieu Virbel Date: Wed, 7 Nov 2012 22:24:57 +0100 Subject: [PATCH 5/8] carousel: fix swipe gesture detection. refs #737 --- kivy/uix/carousel.py | 144 ++++++++++++++++++++++++++++++++++++------- 1 file changed, 122 insertions(+), 22 deletions(-) diff --git a/kivy/uix/carousel.py b/kivy/uix/carousel.py index 3246d42fa..7fbd93ef9 100644 --- a/kivy/uix/carousel.py +++ b/kivy/uix/carousel.py @@ -23,10 +23,19 @@ Example:: Example1().run() +.. versionchanged:: 1.5.0 + + The carousel now support active children, as + :class:`~kivy.uix.scrollview.ScrollView`. It will detect a swipe gesture + according to :data:`Carousel.scroll_timeout` and + :data:`Carousel.scroll_distance`. ''' __all__ = ('Carousel', ) +from functools import partial +from kivy.clock import Clock +from kivy.config import Config from kivy.factory import Factory from kivy.animation import Animation from kivy.uix.stencilview import StencilView @@ -34,6 +43,11 @@ from kivy.uix.relativelayout import RelativeLayout from kivy.properties import BooleanProperty, OptionProperty, AliasProperty, \ NumericProperty, ListProperty, ObjectProperty +# When we are generating documentation, Config doesn't exist +_scroll_timeout = _scroll_distance = 0 +if Config: + _scroll_timeout = Config.getint('widgets', 'scroll_timeout') + _scroll_distance = Config.getint('widgets', 'scroll_distance') class Carousel(StencilView): '''Carousel class. See module documentation for more information. @@ -166,13 +180,39 @@ class Carousel(StencilView): :data:`previous_slide` is a :class:`~kivy.properties.AliasProperty`. ''' + scroll_timeout = NumericProperty(_scroll_timeout) + '''Timeout allowed to trigger the :data:`scroll_distance`, in milliseconds. + If the user has not moved :data:`scroll_distance` within the timeout, + the scrolling will be disabled, and the touch event will go to the children. + + :data:`scroll_timeout` is a :class:`~kivy.properties.NumericProperty`, + default to 55 (milliseconds), according to the default value in user + configuration. + + .. versionadded:: 1.5.0 + ''' + + scroll_distance = NumericProperty(_scroll_distance) + '''Distance to move before scrolling the :class:`ScrollView`, in pixels. As + soon as the distance has been traveled, the :class:`ScrollView` will start + to scroll, and no touch event will go to children. + It is advisable that you base this value on the dpi of your target device's + screen. + + :data:`scroll_distance` is a :class:`~kivy.properties.NumericProperty`, + default to 20 (pixels), according to the default value in user + configuration. + + .. versionadded:: 1.5.0 + ''' + #### private properties, for internal use only ### _index = NumericProperty(0) _prev = ObjectProperty(None, allownone=True) _current = ObjectProperty(None, allownone=True) _next = ObjectProperty(None, allownone=True) _offset = NumericProperty(0) - _drag_touches = ListProperty([]) + _touch = ObjectProperty(None, allownone=True) def _insert_visible_slides(self): self._prev = self.previous_slide @@ -248,10 +288,8 @@ class Carousel(StencilView): if self._offset >= self.height: self.index = self.index + 1 - def on__drag_touches(self, *args): + def _start_animation(self, *args): Animation.cancel_all(self) - if self._drag_touches: - return # compute target offset for ease back, next or prev new_offset = 0 @@ -282,35 +320,97 @@ class Carousel(StencilView): anim = Animation(_offset=new_offset, d=dur, t='out_quad') anim.start(self) + def _get_uid(self, prefix='sv'): + return '{0}.{1}'.format(prefix, self.uid) + def on_touch_down(self, touch): if not self.collide_point(*touch.pos): + touch.ud[self._get_uid('cavoid')] = True return - if not super(Carousel, self).on_touch_down(touch): - self._drag_touches.append(touch.uid) - touch.grab(self) + if self._touch: + return super(Carousel, self).on_touch_down(touch) + Animation.cancel_all(self) + self._touch = touch + uid = self._get_uid() + touch.grab(self) + touch.ud[uid] = { + 'mode': 'unknown', + 'time': touch.time_start} + Clock.schedule_once(self._change_touch_mode, + self.scroll_timeout / 1000.) return True def on_touch_move(self, touch): - if touch.grab_current is self: - if self._drag_touches.index(touch.uid) > 0: - return True - if self.direction in ['right', 'left']: - self._offset += touch.dx - if self.direction in ['top', 'bottom']: - self._offset += touch.dy + if self._get_uid('cavoid') in touch.ud: + return + if self._touch is not touch: + super(Carousel, self).on_touch_move(touch) + return self._get_uid() in touch.ud + if touch.grab_current is not self: return True - - if self.collide_point(*touch.pos): - return super(Carousel, self).on_touch_move(touch) + ud = touch.ud[self._get_uid()] + direction = self.direction + if ud['mode'] == 'unknown': + if direction in ('right', 'left'): + distance = abs(touch.ox - touch.x) + else: + distance = abs(touch.oy - touch.y) + if distance > self.scroll_distance: + Clock.unschedule(self._change_touch_mode) + ud['mode'] = 'scroll' + else: + if direction in ('right', 'left'): + self._offset += touch.dx + if direction in ('top', 'bottom'): + self._offset += touch.dy + return True def on_touch_up(self, touch): - if touch.grab_current is self: - self._drag_touches.remove(touch.uid) + if self._get_uid('cavoid') in touch.ud: + return + if self in [x() for x in touch.grab_list]: touch.ungrab(self) - return True + self._touch = None + ud = touch.ud[self._get_uid()] + if ud['mode'] == 'unknown': + super(Carousel, self).on_touch_down(touch) + Clock.schedule_once(partial(self._do_touch_up, touch), .1) + else: + self._start_animation() - if self.collide_point(*touch.pos): - return super(Carousel, self).on_touch_move(touch) + else: + if self._touch is not touch and self.uid not in touch.ud: + super(Carousel, self).on_touch_up(touch) + return self._get_uid() in touch.ud + + def _do_touch_up(self, touch, *largs): + super(Carousel, self).on_touch_up(touch) + # don't forget about grab event! + for x in touch.grab_list[:]: + touch.grab_list.remove(x) + x = x() + if not x: + continue + touch.grab_current = x + super(Carousel, self).on_touch_up(touch) + touch.grab_current = None + + def _change_touch_mode(self, *largs): + if not self._touch: + return + self._start_animation() + uid = self._get_uid() + touch = self._touch + ud = touch.ud[uid] + if ud['mode'] == 'unknown': + touch.ungrab(self) + self._touch = None + touch.push() + touch.apply_transform_2d(self.to_widget) + touch.apply_transform_2d(self.to_parent) + super(Carousel, self).on_touch_down(touch) + touch.pop() + return def add_widget(self, widget, index=0): slide = RelativeLayout(size=self.size, x=self.x - self.width, y=self.y) From 5cc1d26f071c312fe79e63f46fc8df57749c27f9 Mon Sep 17 00:00:00 2001 From: Mathieu Virbel Date: Thu, 8 Nov 2012 09:33:59 +0100 Subject: [PATCH 6/8] add carousel example with buttons --- examples/widgets/carousel_buttons.py | 37 ++++++++++++++++++++++++++++ 1 file changed, 37 insertions(+) create mode 100644 examples/widgets/carousel_buttons.py diff --git a/examples/widgets/carousel_buttons.py b/examples/widgets/carousel_buttons.py new file mode 100644 index 000000000..fbfb1c600 --- /dev/null +++ b/examples/widgets/carousel_buttons.py @@ -0,0 +1,37 @@ +''' +Carousel example with button inside. +This is a tiny test for testing the scroll distance/timeout +And ensure the down/up are dispatched if no gesture is done. +''' +from kivy.uix.carousel import Carousel +from kivy.uix.gridlayout import GridLayout +from kivy.app import App +from kivy.lang import Builder + +Builder.load_string(''' +: + cols: 3 + Label: + text: str(id(root)) + Button + Button + Button + Button + Button + Button + Button + Button +''') + +class Page(GridLayout): + pass + +class TestApp(App): + def build(self): + root = Carousel() + for x in xrange(10): + root.add_widget(Page()) + return root + +if __name__ == '__main__': + TestApp().run() From b2421128502a1bca00df4d678f370425affa90d4 Mon Sep 17 00:00:00 2001 From: Mathieu Virbel Date: Thu, 8 Nov 2012 09:47:50 +0100 Subject: [PATCH 7/8] carousel: dont use the same configuration as scrollview, the timing doesnt give a good behavior. --- kivy/uix/carousel.py | 17 +++++------------ 1 file changed, 5 insertions(+), 12 deletions(-) diff --git a/kivy/uix/carousel.py b/kivy/uix/carousel.py index 7fbd93ef9..39be205c4 100644 --- a/kivy/uix/carousel.py +++ b/kivy/uix/carousel.py @@ -35,7 +35,6 @@ __all__ = ('Carousel', ) from functools import partial from kivy.clock import Clock -from kivy.config import Config from kivy.factory import Factory from kivy.animation import Animation from kivy.uix.stencilview import StencilView @@ -43,11 +42,6 @@ from kivy.uix.relativelayout import RelativeLayout from kivy.properties import BooleanProperty, OptionProperty, AliasProperty, \ NumericProperty, ListProperty, ObjectProperty -# When we are generating documentation, Config doesn't exist -_scroll_timeout = _scroll_distance = 0 -if Config: - _scroll_timeout = Config.getint('widgets', 'scroll_timeout') - _scroll_distance = Config.getint('widgets', 'scroll_distance') class Carousel(StencilView): '''Carousel class. See module documentation for more information. @@ -180,19 +174,18 @@ class Carousel(StencilView): :data:`previous_slide` is a :class:`~kivy.properties.AliasProperty`. ''' - scroll_timeout = NumericProperty(_scroll_timeout) + scroll_timeout = NumericProperty(200) '''Timeout allowed to trigger the :data:`scroll_distance`, in milliseconds. If the user has not moved :data:`scroll_distance` within the timeout, the scrolling will be disabled, and the touch event will go to the children. :data:`scroll_timeout` is a :class:`~kivy.properties.NumericProperty`, - default to 55 (milliseconds), according to the default value in user - configuration. + default to 200 (milliseconds) .. versionadded:: 1.5.0 ''' - scroll_distance = NumericProperty(_scroll_distance) + scroll_distance = NumericProperty('20dp') '''Distance to move before scrolling the :class:`ScrollView`, in pixels. As soon as the distance has been traveled, the :class:`ScrollView` will start to scroll, and no touch event will go to children. @@ -200,8 +193,7 @@ class Carousel(StencilView): screen. :data:`scroll_distance` is a :class:`~kivy.properties.NumericProperty`, - default to 20 (pixels), according to the default value in user - configuration. + default to 20dp. .. versionadded:: 1.5.0 ''' @@ -373,6 +365,7 @@ class Carousel(StencilView): self._touch = None ud = touch.ud[self._get_uid()] if ud['mode'] == 'unknown': + Clock.unschedule(self._change_touch_mode) super(Carousel, self).on_touch_down(touch) Clock.schedule_once(partial(self._do_touch_up, touch), .1) else: From ad3f5f139ff1caefa963ee5c9d6b2541132a26f1 Mon Sep 17 00:00:00 2001 From: Gabriel Pettier Date: Thu, 8 Nov 2012 14:48:30 +0100 Subject: [PATCH 8/8] correct error in screenmanager docstring (sm->screen) --- kivy/uix/screenmanager.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kivy/uix/screenmanager.py b/kivy/uix/screenmanager.py index 7701c46fe..c5e0aad88 100644 --- a/kivy/uix/screenmanager.py +++ b/kivy/uix/screenmanager.py @@ -31,7 +31,7 @@ screen, you absolutely need to give a name to it:: # Add few screens for i in xrange(4): screen = Screen(name='Title %d' % i) - sm.add_widget(sm) + sm.add_widget(screen) # By default, the first screen added into the ScreenManager will be # displayed. Then, you can change to another screen: