你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32 TouchGFX经验分享(三)

[复制链接]
STMCU小助手 发布时间:2023-3-5 12:20
界面事件: ~) D$ I0 w  H5 J
之前和大家共享了一些基本的操作,这章会给大家共享下界面使用的一些小技巧及方法。5 w# [1 X4 [/ d+ ^& P  n. C
除去现成控件的事件外,不知道大家有没有考虑过其他一些界面的实现方式,例如,类似手机下滑菜单,或者上滑菜单,再比如一个动态背景。或者自己做的地图的量算尺子。
! O* f- F( h2 @) O1 L# A7 S这些方法都涉及到界面的滑动事件或者界面的定时刷新事件又或者点击事件之类。TouchGFX给大家提供了UIEventListener类用于事件监听。8 r8 X; r$ Y( d. b& N
此类为用户界面事件(即由用户与设备交互生成的事件)声明处理程序接口。除系统计时器刻度外,所有其他与用户界面设备外围设备(显示器,按键等)无关的其他系统事件均不属于该界面。% b8 S4 L& Z/ k& O8 s, ?
下方是类中函数的介绍。! ^! a& l9 n$ m# u
$ c. z! P+ v; C% m
6013)0@~BT]M(I)I@$G805X.png ; F, r% Q1 K( x. w  D/ R, p
/ j+ A% [+ G0 e8 v8 P' w
5 x- O0 d$ \* c
UIEventListener类在我们创建窗口界面时该类已经继承过来了,实际上在view类中并没有这些方法的实现,所以需要自己根据需要实现方法。; R4 N+ |* Q* j! B" ]4 N
之后会给大家展示例子。
: I. Z2 X5 P: V& P9 i5 g- X  S0 u
! n: s9 }$ F$ N, b0 x" p
9 J1 E; f* X& h. {! [" h
如意控件
/ j0 |2 D7 M) r. _& M/ }# o
上面介绍了界面事件类,现在介绍修改控件的方法以及如何自定义控件,在正常的开发中,TouchGFX Designer提供的控件经常满足不了我们的需求,此时就需要动态的修改控件,或者自己定义控件。首先先介绍动态的修改控件。
  m! m: O5 {1 K# V% Y! d# {还记得第二章讲到的Base类吗,其中有个add()函数,该函数的作用是在窗口上添加一个控件,这就是修改的一种,除此之外,还可以使用控件自身的属性进行更改,例如换图片,移动,渐变,更改文字,具体的需要根据控件的API进行开发。但是需要注意的是控件更改后为了保证可以及时刷新,需要使用invalidate()函数。
0 ]5 P+ o$ G$ k& s9 n
2 d5 X  g% ]) ^* ^

0 E9 k* S' u$ v- |3 Y: U+ UContainer2 Y) l# l4 y4 S* c5 O
Container容器,容器这个控件可以说在TouchGFX里使用量最大的控件,容器可以理解为一个透明的窗口,可以将其他控件放到容器里,再将容器作为单独控件放到窗口中,由此可以实现多个界面切换,或者也可以在容器中添加动画,或者事件监听。* e) ~0 A4 J: g
添加位置:
9 [1 R. P: J1 I1 Y& J7 W可以在TouchGFX Designer的“容器”小部件组中找到“容器”。+ I, s7 @$ C0 L4 Y* X  Q
4 J# y: p) r7 p) L8 w4 C5 B+ D
20200618110912157.png
: x" c% u0 H' y: J  b+ ~( Q. [8 m/ X$ a
您可以在用户代码中使用这些函数以及Container类中可用的其他函数。container1.invalidate()如果更改小部件的外观,请记住要通过调用来强制重绘。
1 ?. S  W0 I$ Z5 z" M2 g+ m7 G( m( B5 T/ L

+ z6 I- D+ n# s% {, x重载容器
# D% Z! L, g/ p  z1 h
该方法是将容器重载,在容器中创建自己了的逻辑,并且将子控件加入容器之中,针对自己的需要编写对外的接口。1 P) l- k) x' @1 D' y3 J
下面看一个例子。
* c* f% f7 C, Z; D0 q# n0 K3 L
  1. #ifndef SLIDEMENU_CONTAINER_HPP
    5 m; f/ E4 E9 A- m
  2. #define SLIDEMENU_CONTAINER_HPP
    & d0 v5 R7 H( E( L
  3. 5 D+ t& ?9 k. ?9 b) r) e. H' s1 P+ d# ~
  4. #include <touchgfx/containers/Container.hpp>
    ; |8 N* \; d: a* \* ]* F
  5. #include <touchgfx/widgets/Image.hpp>
    3 u5 n8 P  m, c* T
  6. #include <touchgfx/containers/ListLayout.hpp>
    # {2 ?: t% F- K0 T) m& S- o: H

  7. . @9 _6 \. r: R6 U" a' X' f
  8. using namespace touchgfx;9 N+ e9 n' Y2 `$ h
  9. - Z/ n* U. P9 V6 x
  10. class SlideMenuContainer : public touchgfx::Container 继承容器
    3 e  \/ O! j% g5 O
  11. {$ j5 D. g3 L1 R
  12. public:
    # K+ J6 D' A8 F4 |) |. F
  13.         SlideMenuContainer();
    4 k! q2 j0 G# m8 o. Q7 M
  14.         virtual ~SlideMenuContainer();& M0 g* h7 y' T  A: K8 F
  15.         将事件添加到容器中,这部分的事件上述提到了
    8 x1 f8 o* c' h, H
  16.         virtual void handleTickEvent();
    " S4 ^1 X( ^0 c3 `& J
  17.         virtual void handleClickEvent(const touchgfx::ClickEvent& evt);
    . S! W7 J, A& ]
  18.         virtual void handleDragEvent(const touchgfx::DragEvent& evt);. v" P* H- c- B6 L
  19.         virtual void handleGestureEvent(const touchgfx::GestureEvent& evt);
    4 L8 S  p0 T7 g# J8 J
  20. & q% w2 R4 d; r8 w; C
  21.         : s+ X( u4 t) \! K2 K. S/ j
  22.         uint8_t getSelectedScreen()
    2 ~% F+ k8 ]$ W# _5 h& k) y: f
  23.         {
    , k% k3 I5 B* I, P
  24.                 return currentState;
    : W' j- V! l( m. ~- v
  25.         }$ y+ ~' {& e! y0 x2 ]# Q% c
  26.         对外设置回调函数的接口
    ! W  `: \  I+ R: v/ @% c0 ~% ]
  27.         void setMenuChangeEndedCallback(touchgfx::GenericCallback<const SlideMenuContainer&>& callback)
    ; P$ [* i! ^2 l0 K9 q4 B( J% C+ {7 ?) c
  28.         {5 g- r  p. u5 p0 r' p
  29.                 menuChangeAction = &callback;
    7 F8 i, X6 J# ~- U# t
  30.         }# b! M, l& ^) T9 R/ Z
  31. private:
    1 ?2 r' b4 e' w0 ~8 ~
  32.         enum States$ g! _2 `2 K/ q, S# Y) j7 G/ t$ V
  33.         {
    " M! i" {+ l+ u8 m% |
  34.                 ANIMATE_COLLAPSE,
    % K5 c! `% E" Q, Q
  35.                 ANIMATE_SHOW,
    ! ^8 D6 ~) _0 r+ m: c% K4 Z
  36.                 NO_ANIMATION5 M$ \* I+ n2 o/ Y# @
  37.         } currentState;
    % ^$ U4 q& s) S4 J$ Z

  38. % j4 L$ U( U. n4 x) x
  39.         GenericCallback<const SlideMenuContainer&>* menuChangeAction;7 r* O3 H% S* {5 g- n3 ~7 q
  40. 2 O2 V$ c/ B3 Y' w. t
  41. };  N$ `$ K7 v, K5 F
复制代码

3 f: B+ d  ?; `9 @# D5 F! m* g接下来看一下cpp文件9 X+ x" a- p) a: d
  1. #include <gui/common/SlideMenuContainer.hpp>5 m) `6 L0 _7 Z( v
  2. #include <touchgfx/EasingEquations.hpp>
      }0 D# h3 U) Q7 V! A3 W+ F6 {/ N  @
  3. #include "BitmapDatabase.hpp"
    3 o4 v; x& N( z+ n- {* Q6 N
  4. #include <touchgfx/Color.hpp>+ D( B; S3 D. K; Q- b: u8 N

  5. & W' x+ ?% G4 M9 f6 k# {* D
  6. using namespace touchgfx;
    , C; B6 q2 S+ x( b4 G

  7. . @- P+ m1 q/ O
  8. SlideMenuContainer::SlideMenuContainer():# S; @# I' A* H; W8 [- v8 Z* a+ x
  9.         currentState(NO_ANIMATION),6 t$ c5 a) ?# ?/ n* ?- v
  10.         menuChangeAction(0), ^* X- b& L1 {
  11. {
    4 E' T1 W( n& B/ x
  12.         touchgfx::Application::getInstance()->registerTimerWidget(this);) C. {% g& _. V* v5 n. s4 |
  13.         setTouchable(true);/ ^# j7 [$ H# n% Z  O

  14. ( i) U3 ]2 G, s* M2 x
  15. }/ X$ b$ A+ g9 N  r4 B: r
  16. ) u6 t; p& [+ j4 N1 N0 T& b
  17. SlideMenuContainer::~SlideMenuContainer()3 U  h3 O# [3 G: A1 W. _6 J% Z
  18. {. R# |2 F8 v2 p* _9 u
  19.         touchgfx::Application::getInstance()->unregisterTimerWidget(this);
    " D  o- t+ G, m+ E: [! ?' J' B
  20. }
    / B# A0 o8 n$ a9 x  _: @
  21. - S- j, |- y/ d" e
  22. void SlideMenuContainer::handleTickEvent()
    0 Q7 j0 J1 w0 T5 g; L
  23. {
    # n! [0 M% h) H- T: e
  24. }
    3 E5 z+ W: v% k. \& I
  25. void SlideMenuContainer::handleClickEvent(const ClickEvent& evt)" [( h$ G1 @) W7 X
  26. {
    % N3 s4 K0 F# K  Z6 w

  27. " w* u- F, r. ~
  28. }
    4 \2 P# _& F3 C3 Y
  29. void SlideMenuContainer::handleDragEvent(const DragEvent& evt)3 x( p/ F2 o- I- F3 f! `9 b
  30. {
    ( r8 d+ H" M: k; w: c
  31.         //touchgfx_printf("SlideMenuContainer->getDeltaX :%d   \r\n", evt.getDeltaX());
    3 z, H( g# ^  b0 c$ F
  32.         
    " y" M& i2 R$ G* R7 U7 c4 Q
  33. }3 {# V9 P& L; [) y% G
  34. 手势响应函数,上滑收起下滑弹出
    $ J1 F) G7 H, k( Q% |. g
  35. void SlideMenuContainer::handleGestureEvent(const GestureEvent& evt)( G2 n. b" d3 l0 |
  36. {
    + ]% _4 u$ ^! o: n! \9 l( z
  37.         if (evt.getType() == evt.SWIPE_VERTICAL)4 I! F, C: {: f# P7 o3 K- ~) Z
  38.         {2 P7 B2 w5 N: K
  39.                 //touchgfx_printf("SlideMenuContainer->handleGestureEvent   \r\n");$ {6 q/ m, |" B1 t& {3 J6 I/ y
  40.                 if (evt.getVelocity() < 0). ^) X% o- m- x( y8 ~) m) _8 q! y
  41.                 {5 D  X0 [5 @, I+ E
  42.                         currentState = ANIMATE_COLLAPSE;
    ( N9 y8 b: h5 u/ {2 A: A7 |
  43.                         if (menuChangeAction && menuChangeAction->isValid())
    * o" C  ^7 P; R0 }. {  S# G
  44.                         {
    8 l+ x: n1 }9 k2 ]
  45.                                 menuChangeAction->execute(*this);) \! g1 j! b' _2 s2 n
  46.                         }9 r6 o1 b. }! p/ I7 _1 _
  47. ' G, S7 c& y1 c: n3 W  l- n' Q
  48.                 }5 t  w& {  e( @9 r' g0 n% o
  49.                 else if (evt.getVelocity() > 0)
    4 V; q: N0 j) }4 f5 S" ~
  50.                 {
    & b& g+ B0 s" W. B4 ]+ U
  51.                         currentState = ANIMATE_SHOW;
    ' @4 H2 l/ R: x+ V  t6 \( ]
  52.                         if (menuChangeAction && menuChangeAction->isValid())
    . D" ]) O* ]4 _( y8 k! J( l0 g
  53.                         {
    * T" L) D" v7 d! V+ E# ?  P0 [
  54.                                 menuChangeAction->execute(*this);7 H2 y, h2 z6 Q2 d! V
  55.                         }
    ( D  D* y- D( l2 V, c
  56.                 }; V4 [5 v4 I# X* G8 |7 Y1 [1 ]
  57.         }2 U$ g' S* `' f& o
  58. }
    / E1 f$ p8 j( e' n0 W% L+ _" R5 F+ I) q
复制代码
4 r( ]6 C; n9 M# t- @6 x( u1 \& W
自定义控件
7 D* S% t1 ^& X% V除了之上的两种方法,如果需要更为具体的定义控件,或者想要节省空间可以选择自定义控件的方法。8 z% _  I& n2 u; ~+ F
TouchGFX Designer当前不支持自定义窗口小部件的创建。因此,您将需要手动编写自定义小部件的代码,然后将小部件插入View的用户代码部分。. A4 p* c7 }& ~3 F
您可以通过扩展Widget类来创建自己的自定义窗口小部件。这样做需要在用户方面付出更多的努力,但也将完全控制小部件绘制的所有像素。您的自定义窗口小部件不一定使用任何现有的窗口小部件,而是通过指定像素的颜色来定义应如何绘制。自定义窗口小部件方法通常还将具有较小的内存占用空间,这在某些情况下非常重要。
2 M6 T" g7 f& `& x- E/ O$ y
  1. #ifndef QR_CODE_WIDGET_HPP" i. Q* ?( c5 D& {9 T; [, U
  2. #define QR_CODE_WIDGET_HPP
    ; _, |' [1 @7 q0 D* T

  3. 7 a3 k, Y* X3 [  P% N. t- ^
  4. #include <touchgfx/widgets/Widget.hpp>& W; |7 B" I9 r
  5. 以这种方法重载widget窗口,但是如果不是有特殊需要的话还是建议用容器去做。
    3 B. B7 Z  z) ]% Z( n( o
  6. class QRCodeWidget : public touchgfx::Widget( _; K1 t1 O  A8 P0 w7 }
  7. {
    2 ~3 O& m8 N. b/ d1 L
  8. public:
    / C4 |1 J* n/ j- \% m
  9.     QRCodeWidget();8 Z" f( W, P0 c/ S! \( a. r4 l( q$ G

  10. 9 r/ `; l! g8 ]( _! V# J9 U
  11.     virtual void draw(const touchgfx::Rect& invalidatedArea) const;* W. M8 [# W6 ~9 u
  12.     virtual touchgfx::Rect getSolidRect() const;
    " _* f! X0 J; Y" ^, c* T. a

  13. * m3 e# ~$ J  d3 E
  14.     void setQRCodeData(QRCodeData* data);
    ( D2 \& y; ~2 p& ?+ a( D6 ~/ [% D0 t
  15.     void setScale(uint8_t s);
    7 d+ h' ^& O! t! Y' L' y+ e

  16. ; M2 p# w: j* {
  17. private:
    9 d# r  q, {8 x8 n- U. c# t# e# [! [, v
  18.     void updateSize();
    % a' I* e9 V; f* m9 ]. h2 I4 C

  19. * ]; C/ n# c2 P& }0 k" _
  20.     QRCodeData* data;( o2 t8 t. o- Q* a8 J
  21.     uint8_t scale;
    2 ?7 H6 ?. ]4 }: l7 l
  22. };
    : c7 Y2 A, }0 y4 v, G7 K
  23. #endif
    ; n+ J: f5 _- T) P0 E
复制代码
: b4 l" c! ^3 r' G0 g( |
具体类中的继承和重载请参照官方的API文档。9 W( l2 ^6 J9 T; a! ~, n

2 ?/ N7 c* j$ d* Z; D8 r图片等资源放置位置" ?* Y. D1 r3 d
好了说了这么多也可以大致的制作界面了,但是界面在开发过程中需要放入一些图片资源或者字库资源,这些地方放在TouchGFX\assets中,进入后可以看见放入资源的文件夹,但是注意图片格式为png。) l" F! V& |/ {/ t- _
, A9 j' I% \1 @) |- `3 `: I
2020061813293165.png $ D2 M6 G9 P" J" c  ?

0 @4 M. Y& v! J6 }- a2 T( s& K  o 20200618132950218.png
$ D; W, \4 f9 e& T
& i1 {7 @. ]! O5 N
到这里大体的框架都给大家介绍完毕了,之后会给大家介绍几个细节性的控件。3 U1 D; z! s. Q# Y% F* C( w
————————————————0 R% H8 c4 P; |0 I; q7 [
版权声明:空闲的程序员5 V4 @" S( t, C* @2 w) F
& v0 ]0 z* ]5 g& d
7 M8 I% ]4 i5 h! T0 a+ i1 u
收藏 评论0 发布时间:2023-3-5 12:20

举报

0个回答
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版