Commit 61e3478a6898fa42d40d4abdfcabe95fc69aada1
1 parent
642081ad
feat:封装具备答错摇晃、答对缩放动效能力的选择题选项组件
Showing
2 changed files
with
244 additions
and
135 deletions
lib/common/widgets/option_widget.dart
0 → 100644
| 1 | +import 'package:flutter/material.dart'; | |
| 2 | +import 'package:wow_english/common/widgets/throttledGesture_gesture_detector.dart'; | |
| 3 | + | |
| 4 | +import '../../utils/log_util.dart'; | |
| 5 | + | |
| 6 | +/// 选择题选项组件,具备答错摇晃、答对缩放动效能力 | |
| 7 | +class OptionWidget extends StatefulWidget { | |
| 8 | + final Widget child; | |
| 9 | + | |
| 10 | + //正确缩放动画;错误摇晃动画;为空即非选中选项,无操作 | |
| 11 | + final bool? isCorrect; | |
| 12 | + final bool isClickable; | |
| 13 | + final VoidCallback? onTap; | |
| 14 | + | |
| 15 | + const OptionWidget({ | |
| 16 | + Key? key, | |
| 17 | + required this.child, | |
| 18 | + this.isCorrect, | |
| 19 | + this.isClickable = true, | |
| 20 | + this.onTap, | |
| 21 | + }) : super(key: key); | |
| 22 | + | |
| 23 | + @override | |
| 24 | + _OptionWidgetState createState() => _OptionWidgetState(); | |
| 25 | +} | |
| 26 | + | |
| 27 | +class _OptionWidgetState extends State<OptionWidget> | |
| 28 | + with SingleTickerProviderStateMixin { | |
| 29 | + late AnimationController _controller; | |
| 30 | + late Animation<double> _shakeAnimation; | |
| 31 | + late Animation<double> _scaleAnimation; | |
| 32 | + static const String TAG = 'OptionWidget'; | |
| 33 | + | |
| 34 | + @override | |
| 35 | + void initState() { | |
| 36 | + super.initState(); | |
| 37 | + | |
| 38 | + _controller = AnimationController( | |
| 39 | + vsync: this, | |
| 40 | + duration: const Duration(milliseconds: 500), | |
| 41 | + ); | |
| 42 | + | |
| 43 | + _shakeAnimation = TweenSequence([ | |
| 44 | + TweenSequenceItem( | |
| 45 | + tween: Tween(begin: 0.0, end: 10.0) | |
| 46 | + .chain(CurveTween(curve: Curves.easeInOut)), | |
| 47 | + weight: 1, | |
| 48 | + ), | |
| 49 | + TweenSequenceItem( | |
| 50 | + tween: Tween(begin: 10.0, end: -10.0) | |
| 51 | + .chain(CurveTween(curve: Curves.easeInOut)), | |
| 52 | + weight: 1, | |
| 53 | + ), | |
| 54 | + TweenSequenceItem( | |
| 55 | + tween: Tween(begin: -10.0, end: 10.0) | |
| 56 | + .chain(CurveTween(curve: Curves.easeInOut)), | |
| 57 | + weight: 1, | |
| 58 | + ), | |
| 59 | + TweenSequenceItem( | |
| 60 | + tween: Tween(begin: 10.0, end: -10.0) | |
| 61 | + .chain(CurveTween(curve: Curves.easeInOut)), | |
| 62 | + weight: 1, | |
| 63 | + ), | |
| 64 | + TweenSequenceItem( | |
| 65 | + tween: Tween(begin: -10.0, end: 0.0) | |
| 66 | + .chain(CurveTween(curve: Curves.easeInOut)), | |
| 67 | + weight: 1, | |
| 68 | + ), | |
| 69 | + ]).animate(_controller); | |
| 70 | + | |
| 71 | + _scaleAnimation = TweenSequence<double>([ | |
| 72 | + TweenSequenceItem(tween: Tween(begin: 1.0, end: 0.8), weight: 1), | |
| 73 | + TweenSequenceItem(tween: Tween(begin: 0.8, end: 1.25), weight: 1), | |
| 74 | + TweenSequenceItem(tween: Tween(begin: 1.25, end: 1.0), weight: 1), | |
| 75 | + ]).animate(CurvedAnimation( | |
| 76 | + parent: _controller, | |
| 77 | + curve: Curves.easeInOut, | |
| 78 | + )); | |
| 79 | + } | |
| 80 | + | |
| 81 | + @override | |
| 82 | + void didUpdateWidget(OptionWidget oldWidget) { | |
| 83 | + super.didUpdateWidget(oldWidget); | |
| 84 | + Log.d( | |
| 85 | + '$TAG didUpdateWidget widget.isCorrect=${widget.isCorrect} oldWidget.isCorrect=${oldWidget.isCorrect} isAnimation=${_controller.isAnimating} status=${_controller.status}'); | |
| 86 | + if (widget.isCorrect != oldWidget.isCorrect) { | |
| 87 | + _controller.reset(); | |
| 88 | + if (widget.isCorrect == true) { | |
| 89 | + _controller.forward(); | |
| 90 | + } else if (widget.isCorrect == false) { | |
| 91 | + _controller.forward(); | |
| 92 | + } | |
| 93 | + } | |
| 94 | + } | |
| 95 | + | |
| 96 | + @override | |
| 97 | + void dispose() { | |
| 98 | + _controller.dispose(); | |
| 99 | + super.dispose(); | |
| 100 | + } | |
| 101 | + | |
| 102 | + @override | |
| 103 | + Widget build(BuildContext context) { | |
| 104 | + return AnimatedBuilder( | |
| 105 | + animation: _controller, | |
| 106 | + builder: (context, child) { | |
| 107 | + double offsetX = widget.isCorrect == false ? _shakeAnimation.value : 0; | |
| 108 | + double scale = widget.isCorrect == true ? _scaleAnimation.value : 1.0; | |
| 109 | + | |
| 110 | + return ThrottledGestureDetector( | |
| 111 | + onTap: widget.isClickable ? widget.onTap : null, | |
| 112 | + child: Transform.translate( | |
| 113 | + offset: Offset(offsetX, 0), | |
| 114 | + child: Transform.scale( | |
| 115 | + scale: scale, | |
| 116 | + child: Opacity( | |
| 117 | + opacity: widget.isClickable ? 1.0 : 0.5, | |
| 118 | + child: widget.child, | |
| 119 | + ), | |
| 120 | + ), | |
| 121 | + ), | |
| 122 | + ); | |
| 123 | + }, | |
| 124 | + ); | |
| 125 | + } | |
| 126 | +} | ... | ... |
lib/pages/practice/topic_picture_page.dart
| ... | ... | @@ -7,10 +7,9 @@ import 'package:wow_english/common/extension/string_extension.dart'; |
| 7 | 7 | import 'package:wow_english/common/widgets/ow_image_widget.dart'; |
| 8 | 8 | import 'package:wow_english/models/course_process_entity.dart'; |
| 9 | 9 | import 'package:wow_english/pages/practice/topic_type.dart'; |
| 10 | -import 'package:wow_english/pages/practice/widgets/shake_widget.dart'; | |
| 11 | 10 | import 'package:wow_english/route/route.dart'; |
| 12 | -import 'package:wow_english/utils/toast_util.dart'; | |
| 13 | 11 | |
| 12 | +import '../../common/widgets/option_widget.dart'; | |
| 14 | 13 | import '../../common/widgets/recorder_widget.dart'; |
| 15 | 14 | import '../../common/widgets/speaker_widget.dart'; |
| 16 | 15 | import 'bloc/topic_picture_bloc.dart'; |
| ... | ... | @@ -168,34 +167,30 @@ class _TopicPicturePage extends StatelessWidget { |
| 168 | 167 | final isAnswerOption = bloc.selectItem == index; |
| 169 | 168 | final answerCorrect = |
| 170 | 169 | isAnswerOption && bloc.checkAnswerRight(index) == true; |
| 171 | - final answerIncorrect = | |
| 172 | - isAnswerOption && bloc.checkAnswerRight(index) == false; | |
| 173 | 170 | return Container( |
| 174 | 171 | padding: EdgeInsets.symmetric(horizontal: 10.w), |
| 175 | - child: GestureDetector( | |
| 172 | + child: OptionWidget( | |
| 176 | 173 | onTap: () => bloc.add(SelectItemEvent(index)), |
| 177 | - child: ShakeWidget( | |
| 178 | - shouldShake: answerIncorrect, | |
| 174 | + isCorrect: isAnswerOption ? bloc.checkAnswerRight(index) : null, | |
| 175 | + child: Container( | |
| 176 | + padding: const EdgeInsets.all(4.5), | |
| 177 | + decoration: BoxDecoration( | |
| 178 | + color: isAnswerOption | |
| 179 | + ? getResultColor(answerCorrect) | |
| 180 | + : Colors.white, | |
| 181 | + borderRadius: BorderRadius.circular(15), | |
| 182 | + ), | |
| 183 | + height: 143.h, | |
| 184 | + width: 143.w, | |
| 179 | 185 | child: Container( |
| 180 | - padding: const EdgeInsets.all(4.5), | |
| 181 | 186 | decoration: BoxDecoration( |
| 182 | - color: isAnswerOption | |
| 183 | - ? getResultColor(answerCorrect) | |
| 184 | - : Colors.white, | |
| 185 | - borderRadius: BorderRadius.circular(15), | |
| 186 | - ), | |
| 187 | - height: 143.h, | |
| 188 | - width: 143.w, | |
| 189 | - child: Container( | |
| 190 | - decoration: BoxDecoration( | |
| 191 | - color: Colors.white, | |
| 192 | - borderRadius: BorderRadius.circular(15), | |
| 193 | - border: Border.all( | |
| 194 | - width: 1.0, color: const Color(0xFF140C10)), | |
| 195 | - image: DecorationImage( | |
| 196 | - fit: BoxFit.fitWidth, | |
| 197 | - image: NetworkImage(answerLis?.picUrl ?? ''))), | |
| 198 | - ), | |
| 187 | + color: Colors.white, | |
| 188 | + borderRadius: BorderRadius.circular(15), | |
| 189 | + border: Border.all( | |
| 190 | + width: 1.0, color: const Color(0xFF140C10)), | |
| 191 | + image: DecorationImage( | |
| 192 | + fit: BoxFit.fitWidth, | |
| 193 | + image: NetworkImage(answerLis?.picUrl ?? ''))), | |
| 199 | 194 | ), |
| 200 | 195 | ), |
| 201 | 196 | ), |
| ... | ... | @@ -240,53 +235,49 @@ class _TopicPicturePage extends StatelessWidget { |
| 240 | 235 | final isAnswerOption = bloc.selectItem == index; |
| 241 | 236 | final answerCorrect = |
| 242 | 237 | isAnswerOption && bloc.checkAnswerRight(index) == true; |
| 243 | - final answerIncorrect = | |
| 244 | - isAnswerOption && bloc.checkAnswerRight(index) == false; | |
| 245 | 238 | return Container( |
| 246 | 239 | padding: EdgeInsets.symmetric(horizontal: 10.w), |
| 247 | - child: GestureDetector( | |
| 240 | + child: OptionWidget( | |
| 248 | 241 | onTap: () => bloc.add(SelectItemEvent(index)), |
| 249 | - child: ShakeWidget( | |
| 250 | - shouldShake: answerIncorrect, | |
| 251 | - child: Container( | |
| 252 | - width: 143.w, | |
| 253 | - height: 143.h, | |
| 254 | - padding: EdgeInsets.only( | |
| 255 | - left: 13.w, right: 13.w, top: 13.h, bottom: 13.h), | |
| 256 | - decoration: BoxDecoration( | |
| 257 | - color: Colors.white, | |
| 258 | - borderRadius: BorderRadius.circular(15), | |
| 259 | - border: Border.all( | |
| 260 | - width: 1.0, color: const Color(0xFF140C10)), | |
| 261 | - ), | |
| 262 | - child: Column( | |
| 263 | - mainAxisAlignment: MainAxisAlignment.end, | |
| 264 | - children: [ | |
| 265 | - Expanded( | |
| 266 | - child: Container( | |
| 267 | - alignment: Alignment.center, | |
| 268 | - child: Text(answerLis?.word ?? '', | |
| 269 | - style: TextStyle( | |
| 270 | - fontSize: 20.sp, | |
| 271 | - color: const Color(0xFF333333))), | |
| 272 | - ), | |
| 273 | - ), | |
| 274 | - Container( | |
| 275 | - height: 30.h, | |
| 276 | - width: double.infinity, | |
| 277 | - decoration: BoxDecoration( | |
| 278 | - color: isAnswerOption | |
| 279 | - ? getResultColor(answerCorrect) | |
| 280 | - : Colors.white, | |
| 281 | - borderRadius: BorderRadius.circular(15.r), | |
| 282 | - border: Border.all( | |
| 283 | - width: 1.5, color: const Color(0xFF140C10)), | |
| 284 | - ), | |
| 242 | + isCorrect: isAnswerOption ? bloc.checkAnswerRight(index) : null, | |
| 243 | + child: Container( | |
| 244 | + width: 143.w, | |
| 245 | + height: 143.h, | |
| 246 | + padding: EdgeInsets.only( | |
| 247 | + left: 13.w, right: 13.w, top: 13.h, bottom: 13.h), | |
| 248 | + decoration: BoxDecoration( | |
| 249 | + color: Colors.white, | |
| 250 | + borderRadius: BorderRadius.circular(15), | |
| 251 | + border: | |
| 252 | + Border.all(width: 1.0, color: const Color(0xFF140C10)), | |
| 253 | + ), | |
| 254 | + child: Column( | |
| 255 | + mainAxisAlignment: MainAxisAlignment.end, | |
| 256 | + children: [ | |
| 257 | + Expanded( | |
| 258 | + child: Container( | |
| 285 | 259 | alignment: Alignment.center, |
| 286 | - child: Image.asset('choose'.assetPng), | |
| 287 | - ) | |
| 288 | - ], | |
| 289 | - ), | |
| 260 | + child: Text(answerLis?.word ?? '', | |
| 261 | + style: TextStyle( | |
| 262 | + fontSize: 20.sp, | |
| 263 | + color: const Color(0xFF333333))), | |
| 264 | + ), | |
| 265 | + ), | |
| 266 | + Container( | |
| 267 | + height: 30.h, | |
| 268 | + width: double.infinity, | |
| 269 | + decoration: BoxDecoration( | |
| 270 | + color: isAnswerOption | |
| 271 | + ? getResultColor(answerCorrect) | |
| 272 | + : Colors.white, | |
| 273 | + borderRadius: BorderRadius.circular(15.r), | |
| 274 | + border: Border.all( | |
| 275 | + width: 1.5, color: const Color(0xFF140C10)), | |
| 276 | + ), | |
| 277 | + alignment: Alignment.center, | |
| 278 | + child: Image.asset('choose'.assetPng), | |
| 279 | + ) | |
| 280 | + ], | |
| 290 | 281 | ), |
| 291 | 282 | ), |
| 292 | 283 | ), |
| ... | ... | @@ -349,32 +340,28 @@ class _TopicPicturePage extends StatelessWidget { |
| 349 | 340 | final isAnswerOption = bloc.selectItem == index; |
| 350 | 341 | final answerCorrect = |
| 351 | 342 | isAnswerOption && bloc.checkAnswerRight(index) == true; |
| 352 | - final answerIncorrect = | |
| 353 | - isAnswerOption && bloc.checkAnswerRight(index) == false; | |
| 354 | - return ShakeWidget( | |
| 355 | - shouldShake: answerIncorrect, | |
| 343 | + return OptionWidget( | |
| 344 | + onTap: () => bloc.add(SelectItemEvent(index)), | |
| 345 | + isCorrect: isAnswerOption ? bloc.checkAnswerRight(index) : null, | |
| 356 | 346 | child: Container( |
| 357 | 347 | padding: EdgeInsets.symmetric(horizontal: 20.w), |
| 358 | - child: GestureDetector( | |
| 359 | - onTap: () => bloc.add(SelectItemEvent(index)), | |
| 348 | + child: Container( | |
| 349 | + padding: const EdgeInsets.all(4.5), | |
| 350 | + decoration: BoxDecoration( | |
| 351 | + color: isAnswerOption | |
| 352 | + ? getResultColor(answerCorrect) | |
| 353 | + : Colors.white, | |
| 354 | + borderRadius: BorderRadius.circular(15), | |
| 355 | + ), | |
| 356 | + height: 143.h, | |
| 357 | + width: 163.w, | |
| 360 | 358 | child: Container( |
| 361 | - padding: const EdgeInsets.all(4.5), | |
| 362 | 359 | decoration: BoxDecoration( |
| 363 | - color: isAnswerOption | |
| 364 | - ? getResultColor(answerCorrect) | |
| 365 | - : Colors.white, | |
| 366 | - borderRadius: BorderRadius.circular(15), | |
| 367 | - ), | |
| 368 | - height: 143.h, | |
| 369 | - width: 163.w, | |
| 370 | - child: Container( | |
| 371 | - decoration: BoxDecoration( | |
| 372 | - color: Colors.white, | |
| 373 | - borderRadius: BorderRadius.circular(15), | |
| 374 | - image: DecorationImage( | |
| 375 | - fit: BoxFit.fill, | |
| 376 | - image: NetworkImage(answerList?.picUrl ?? ''))), | |
| 377 | - ), | |
| 360 | + color: Colors.white, | |
| 361 | + borderRadius: BorderRadius.circular(15), | |
| 362 | + image: DecorationImage( | |
| 363 | + fit: BoxFit.fill, | |
| 364 | + image: NetworkImage(answerList?.picUrl ?? ''))), | |
| 378 | 365 | ), |
| 379 | 366 | ), |
| 380 | 367 | ), |
| ... | ... | @@ -426,55 +413,51 @@ class _TopicPicturePage extends StatelessWidget { |
| 426 | 413 | final isAnswerOption = bloc.selectItem == index; |
| 427 | 414 | final answerCorrect = |
| 428 | 415 | isAnswerOption && bloc.checkAnswerRight(index) == true; |
| 429 | - final answerIncorrect = | |
| 430 | - isAnswerOption && bloc.checkAnswerRight(index) == false; | |
| 431 | - return GestureDetector( | |
| 416 | + return OptionWidget( | |
| 432 | 417 | onTap: () => bloc.add(SelectItemEvent(index)), |
| 433 | - child: ShakeWidget( | |
| 434 | - shouldShake: answerIncorrect, | |
| 418 | + isCorrect: isAnswerOption ? bloc.checkAnswerRight(index) : null, | |
| 419 | + child: Container( | |
| 420 | + width: 163.w, | |
| 421 | + height: 143.h, | |
| 422 | + padding: EdgeInsets.symmetric(horizontal: 10.w), | |
| 435 | 423 | child: Container( |
| 436 | - width: 163.w, | |
| 424 | + width: 143.w, | |
| 437 | 425 | height: 143.h, |
| 438 | - padding: EdgeInsets.symmetric(horizontal: 10.w), | |
| 439 | - child: Container( | |
| 440 | - width: 143.w, | |
| 441 | - height: 143.h, | |
| 442 | - padding: EdgeInsets.only( | |
| 443 | - left: 13.w, right: 13.w, top: 13.h, bottom: 13.h), | |
| 444 | - decoration: BoxDecoration( | |
| 445 | - color: Colors.white, | |
| 446 | - borderRadius: BorderRadius.circular(15), | |
| 447 | - border: Border.all( | |
| 448 | - width: 1.0, color: const Color(0xFF140C10)), | |
| 449 | - ), | |
| 450 | - child: Column( | |
| 451 | - mainAxisAlignment: MainAxisAlignment.end, | |
| 452 | - children: [ | |
| 453 | - Expanded( | |
| 454 | - child: Container( | |
| 455 | - alignment: Alignment.center, | |
| 456 | - child: Text(answerList.word ?? '', | |
| 457 | - style: TextStyle( | |
| 458 | - fontSize: 20.sp, | |
| 459 | - color: const Color(0xFF333333))), | |
| 460 | - ), | |
| 461 | - ), | |
| 462 | - Container( | |
| 463 | - height: 30.h, | |
| 464 | - width: double.infinity, | |
| 465 | - decoration: BoxDecoration( | |
| 466 | - color: isAnswerOption | |
| 467 | - ? getResultColor(answerCorrect) | |
| 468 | - : Colors.white, | |
| 469 | - borderRadius: BorderRadius.circular(15.r), | |
| 470 | - border: Border.all( | |
| 471 | - width: 1.5, color: const Color(0xFF140C10)), | |
| 472 | - ), | |
| 426 | + padding: EdgeInsets.only( | |
| 427 | + left: 13.w, right: 13.w, top: 13.h, bottom: 13.h), | |
| 428 | + decoration: BoxDecoration( | |
| 429 | + color: Colors.white, | |
| 430 | + borderRadius: BorderRadius.circular(15), | |
| 431 | + border: | |
| 432 | + Border.all(width: 1.0, color: const Color(0xFF140C10)), | |
| 433 | + ), | |
| 434 | + child: Column( | |
| 435 | + mainAxisAlignment: MainAxisAlignment.end, | |
| 436 | + children: [ | |
| 437 | + Expanded( | |
| 438 | + child: Container( | |
| 473 | 439 | alignment: Alignment.center, |
| 474 | - child: Image.asset('choose'.assetPng), | |
| 475 | - ) | |
| 476 | - ], | |
| 477 | - ), | |
| 440 | + child: Text(answerList.word ?? '', | |
| 441 | + style: TextStyle( | |
| 442 | + fontSize: 20.sp, | |
| 443 | + color: const Color(0xFF333333))), | |
| 444 | + ), | |
| 445 | + ), | |
| 446 | + Container( | |
| 447 | + height: 30.h, | |
| 448 | + width: double.infinity, | |
| 449 | + decoration: BoxDecoration( | |
| 450 | + color: isAnswerOption | |
| 451 | + ? getResultColor(answerCorrect) | |
| 452 | + : Colors.white, | |
| 453 | + borderRadius: BorderRadius.circular(15.r), | |
| 454 | + border: Border.all( | |
| 455 | + width: 1.5, color: const Color(0xFF140C10)), | |
| 456 | + ), | |
| 457 | + alignment: Alignment.center, | |
| 458 | + child: Image.asset('choose'.assetPng), | |
| 459 | + ) | |
| 460 | + ], | |
| 478 | 461 | ), |
| 479 | 462 | ), |
| 480 | 463 | ), | ... | ... |