1.EBSåEBFFåºå«
2.Flutter 中使用 OverlayPortal 创建自定义下拉菜单
3.Android åçBottomSheet ä»ç»åå
4.WPF中 怎么禁用 Expander.Header中的源码ToggleButton 要源代码呀~~
5.Flutter(å)ä¹Flutterçå¸å±Widget
6.reactadmin?
EBSåEBFFåºå«
亲ï¼æ¨å¥½ï¼æå¯æ¯ä¸ªè±è¯ççåç§åï¼æ以ï¼è¿éé¢å¯¹äºææ¥è¯´ï¼é常ç®åï¼ä¸é¢ï¼æ为大家åå¤äºä¸ä¸ªè¡¨æ ¼ï¼å¤§å®¶å¯ä»¥ç²ç¥ççä¸ä¸ï¼å ·ä½ç¨æ³ï¼EBSæ¯è±æ缩åï¼å ¨ç§°ä¸ºElastic Block Storeï¼æ¯äºé©¬éAWSæä¾çä¸ç§åå¨æå¡ãè¿ç§æå¡å¯ä»¥æä¾é«å¯ç¨æ§ãé«å¯é æ§åé«æ§è½çååå¨èµæºãEBSå®é ä¸æ¯ä¸ä¸ªäºçï¼å¯ä»¥ä¸EC2å®ä¾è¿æ¥ï¼ç¨äºåå¨æ°æ®ï¼ç±»ä¼¼äºä¼ ç»ç硬çãèEBFFæ¯è±æ缩åï¼å ¨ç§°ä¸ºElastic Beanstalk File Formatï¼æ¯AWS Elastic Beanstalkä¸ç¨äºæå åé¨ç½²åºç¨ç¨åºçæä»¶æ ¼å¼ã
EBS主è¦ç¨äºåå¨æ°æ®ï¼å¨éè¦è¿è¡æä¹ ååå¨çåºæ¯ä¸ä½¿ç¨ãèEBFFåæ¯ç¨äºå°åºç¨ç¨åºæå æZIPæ件并ä¸ä¼ å°Elastic Beanstalkè¿è¡é¨ç½²ãå¨ä½¿ç¨Elastic Beanstalkè¿è¡åºç¨ç¨åºé¨ç½²æ¶ï¼æ们å¯ä»¥å°åºç¨ç¨åºçæºä»£ç æå æEBFFæ ¼å¼ï¼ç¶åå°å ¶ä¸ä¼ å°Elastic Beanstalkï¼Elastic Beanstalkä¼æ ¹æ®æ们çé ç½®èªå¨é¨ç½²ã管çåæ©å±åºç¨ç¨åºã
个人ç»éªï¼EBSå¯ä»¥ä½ä¸ºEC2å®ä¾çæä¹ ååå¨ï¼å¯ä»¥å¨EC2å®ä¾åæ¢æå ³æºåä»ç¶ä¿çæ°æ®ãèå¨ä½¿ç¨Elastic Beanstalké¨ç½²åºç¨ç¨åºæ¶ï¼EBFFå¯ä»¥å¸®å©æ们æ´æ¹ä¾¿å°æå åé¨ç½²åºç¨ç¨åºï¼æé«é¨ç½²æçãåæ¶ï¼Elastic Beanstalkè¿æä¾äºèªå¨æ©å±ãè´è½½åè¡¡çåè½ï¼å¯ä»¥å¸®å©æ们æ´å¥½å°ç®¡çåè¿è¡åºç¨ç¨åºã
é¢å¤æ©å±ï¼é¤äºEBSåEBFFï¼AWSè¿æä¾äºå¾å¤å ¶ä»çäºåå¨æå¡åé¨ç½²å·¥å ·ï¼å¦S3ãGlacierãCloudFrontãCodeDeployçï¼å¯ä»¥æ ¹æ®ä¸åçéæ±éæ©åéçæå¡ã
Flutter 中使用 OverlayPortal 创建自定义下拉菜单
在构建Flutter应用时,若需要实现具有展开动画、源码背景fade动画,源码且在切换时禁止收起的源码自定义下拉菜单,本文将提供详细的源码实现步骤。通过自定义一个OverlayEntry浮窗,源码table表格css源码结合CompositedTransformTarget与CompositedTransformFollower,源码可以轻松实现所需功能。源码
为了确保浮窗跟随特定按钮,源码需使用LayerLink关联二者。源码计算Offset位置,源码设置CompositedTransformFollower的源码值为offset:Offset(left, renderBox!.size.height),有效解决坐标问题。源码
在进行多个tab切换时,源码需先清除之前的源码OverlayEntry,避免叠加。当切换tab时,调用_overlayEntry的remove方法进行清空。setState()无法使OverlayEntry重新渲染,使用_overlayEntry.markNeedsBuild()可解决此问题。
当需要菜单高度自适应时,使用Container包裹菜单,代练app源码但未设置高度。此时,利用Tween动画的begin/end值实现高度自适应。结合AnimatedBuilder与SizeTransition组件,可以实现高度收起展开动画效果。存储一个_isExpanded变量记录收起展开状态,控制AnimatedBuilder动画触发时机。
调整动画方向,通过在SizeTransition内嵌套SlideTransition组件并设置position,实现更符合预期的收起展开效果。最终的动画效果如图所示。
添加透明遮照层,使用Stack组件包裹CompositedTransformFollower的child,并创建一个AnimatedBuilder作为透明遮照层,增强视觉效果。
总结来说,实现所需自定义下拉菜单的关键在于:菜单容器使用OverlayPortal、CompositedTransformTarget、CompositedTransformFollower三者结合;收起展开动画使用AnimatedBuilder、SizeTransition、SlideTransition三者结合;渐显动画直接使用FadeTransition。借条平台源码开发具体的代码实现和更多细节,请参阅源码地址。
Android åçBottomSheet ä»ç»åå
Android Support Library .2 æ¨åºä¹åï¼å¢å äºå 个åè½ï¼ä¾å¦æ¯æVector Drawables åAnimated Vector Drawablesï¼å¢å AppCompat DayNight 主é¢ï¼Design åºä¸å¢å Bottom Sheetsï¼RecyclerView æ¯æ auto-measurementï¼ä¹åçwrap_content ,match_parent é½å°å¯ä»¥åæ¥ä½ç¨çç
å ¬å¸çApp ä¹å使ç¨è¿ç¬¬ä¸æ¹ç[BottomSheet] ï¼ /BottomSheet ï¼ï¼ç°å¨Android æèªå·±çBottomSheet é£è¿ä¸èµ¶ç´§æ¢æåççãç¶è好äºå¤ç£¨ï¼Android åçBottomSheet èµæ太å°ï¼æ·±ç ä¸å»åç°BottomSheet å°±æ¯ä¸ªå¤§åï¼
BottomSheet 使ç¨éè¦CoordinatorLayoutä½ä¸ºç¶å¸å±ï¼BottomSheet çå¸å±ä½ä¸ºCoordinatorLayout çåå¸å±ï¼å¹¶ä¸BottomSheetBehaviorï¼æ¯å¦å ä¸app:layout_behavior=âandroid.support.design.widget.BottomSheetBehaviorâï¼
å®é 使ç¨è¿ç¨ä¸ä¸»è¦ä¾é BottomSheetBehavioræ¥æ§å¶BottomSheetçå±ç¤ºååè°ã
BottomSheetBehavior å ·æäºç§ç¶æï¼
设置ç¶æï¼
bottomSheetBehavior.setState(BottomSheetBehavior.STATE_EXPANDED);
åè°ï¼
强è°ï¼
BottomSheetBehaviorå°è½å¸®ä½ å®ç° 常驻bottom sheetï¼ persistent bottom sheetï¼ çåºæ¯, ä½è¿ä¸ªçæ¬è¿æä¾äºBottomSheetDialog å BottomSheetDialogFragment æ¥å®ç° modal bottom sheets çåºæ¯ãåªéè¦å°AppCompatDialog æè AppCompatDialogFragmentåå«æ¿æ¢æä¸è¿°ç两个æ§ä»¶ï¼ä½ å°±æ¥æäº bottom sheet é£æ ¼ç对è¯æ¡
ç¶èæ们å®é æ们éè¦BottomSheetDialog æ¯å±å¼çï¼èBottomSheetDialogåªå±ç¤ºä¸é¨å
åå ï¼BottomSheetDialogé»è®¤æ¯STATE_COLLAPSEDï¼ææBottomSheetDialog ä¾é peekHightæ¥è®¾ç½®é«åº¦ï¼ç³»ç»BottomSheetDialog é»è®¤é«åº¦ä¸ºdpï¼æ¥æºç å¾ç¥ï¼,é£æçæ¥è¯´æ们çBottomSheetDialog é«åº¦è¯¥æ¯dpï¼ä½æ¯æ们å®é åç°BottomSheetDialogé«åº¦ä¹ä¸çäºdpãæ们ç 究ä¸BottomSheetBehaviorçä¸æ§å¶BottomSheetDialogé«åº¦æºç :
éè¿æºç æ们å¯ä»¥å¾ç¥BottomSheetBehavioréè¿æ¹åchildçå移éèæ§å¶BottomSheetDialogçé«åº¦ï¼é»è®¤ç¶æ为STATE_COLLAPSED,childåä¸ç§»å¨mMaxOffseté«åº¦ï¼ä»èæ§å¶childæ¾ç¤ºé«åº¦ä¸ºmPeekHeight,è¿å°±éè¦childä¸parent 顶é¨å¯¹é½ï¼childçgetTop 为0ï¼
ç¶èæ们åå»æ¥çAndroidçBottomSheetDialog å ä¸å¸å±R.layout.design_bottom_sheet_dialog,åç°æ们èªå®ä¹ççBottomSheetDialog çcontentView æ¯æ¾ç½®å¨FrameLayout ä¸çï¼ç¶èFrameLayoutåºäºæäºåå 为åç´å± ä¸çï¼èä¸æ¯é¡¶é¨å¯¹é½ï¼ä»è导è´BottomSheetDialogå¨dpçåºç¡ä¸åä¸å移,åªå±ç¤ºä¸é¨åã
æ以æ们å¯ä»¥éè¿ä¸é¢æ¹æ³è§£å³BottomSheetDialog çæ¾ç¤ºé®é¢
解å³æ¹æ³å¦ä¸ï¼
å½æ们设置bottomSheetDialogæ¯æ¬¡ç¹å»åä¸new,èæ¯ç´æ¥showçè¯ï¼ç¶èå½æ们ä¼bottomSheetDialog å±å¼åï¼æ们å°BottomSheetDialogåä¸éèåï¼ åç¹å»å±ç¤ºBottomSheetDialogåï¼ä¼åç°é¡µé¢åªæ¯åæï¼BottomsheetDialogæªå±å¼ï¼è¿æ¯ç±äºä¹åæ们åä¸æ¶ç¼©éèBottomSheetDialogåï¼bottomSheetDialogBehaviorçç¶æ为éèï¼å次showä¹åï¼ç³»ç»æªæ¢å¤bottomSheetDialogBehaviorçç¶æï¼è¿æ¯éèï¼æ以å次ç¹å»å页é¢åªæ¯åæã
WPF中 怎么禁用 Expander.Header中的ToggleButton 要源代码呀~~
// exp 是 Expander 的名称
// HeaderSite 是 Windows 通用模板中 ToggleButton 的名称
var tb = exp.Template.FindName("HeaderSite", exp) as
System.Windows.Controls.Primitives.ToggleButton;
if (tb != null) tb.IsEnabled = false;
Flutter(å)ä¹Flutterçå¸å±Widget
ä¸.ååå¸å±ç»ä»¶
ååå¸å±ç»ä»¶çå«ä¹æ¯å ¶åªæä¸ä¸ªåç»ä»¶ï¼å¯ä»¥éè¿è®¾ç½®ä¸äºå±æ§è®¾ç½®è¯¥åç»ä»¶æå¨çä½ç½®ä¿¡æ¯çã
æ¯è¾å¸¸ç¨çååå¸å±ç»ä»¶æï¼AlignãCenterãPaddingãContainerã
1.1.Alignç»ä»¶1.1.1.Alignä»ç»çå°Alignè¿ä¸ªè¯ï¼æ们就ç¥éå®ææ们ç对é½æ¹å¼æå ³ã
å¨å ¶ä»ç«¯çå¼åä¸ï¼iOSãAndroidãå端ï¼Aligné常åªæ¯ä¸ä¸ªå±æ§èå·²ï¼ä½æ¯Flutterä¸Alignä¹æ¯ä¸ä¸ªç»ä»¶ã
æ们å¯ä»¥éè¿æºç æ¥çä¸ä¸Alignæåªäºå±æ§ï¼
constAlign({ Keykey,this.alignment:Alignment.center,//对é½æ¹å¼ï¼é»è®¤å± ä¸å¯¹é½this.widthFactor,//宽度å åï¼ä¸è®¾ç½®çæ åµï¼ä¼å°½å¯è½å¤§this.heightFactor,//é«åº¦å åï¼ä¸è®¾ç½®çæ åµï¼ä¼å°½å¯è½å¤§Widgetchild//è¦å¸å±çåWidget})è¿éæ们ç¹å«è§£éä¸ä¸widthFactoråheightFactorä½ç¨ï¼
å 为åç»ä»¶å¨ç¶ç»ä»¶ä¸ç对é½æ¹å¼å¿ é¡»æä¸ä¸ªåæï¼å°±æ¯ç¶ç»ä»¶å¾ç¥éèªå·±çèå´ï¼å®½åº¦åé«åº¦ï¼ï¼
å¦æwidthFactoråheightFactorä¸è®¾ç½®ï¼é£ä¹é»è®¤Alignä¼å°½å¯è½ç大ï¼å°½å¯è½å æ®èªå·±æå¨çç¶ç»ä»¶ï¼ï¼
æ们ä¹å¯ä»¥å¯¹ä»ä»¬è¿è¡è®¾ç½®ï¼æ¯å¦widthFactor设置为3ï¼é£ä¹ç¸å¯¹äºAlignç宽度æ¯åç»ä»¶è·¨åº¦ç3åï¼
1.1.2.Alignæ¼ç»æ们ç®åæ¼ç»ä¸ä¸Align:
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}1.2.Centerç»ä»¶1.2.1.Centerä»ç»Centerç»ä»¶æ们å¨åé¢å·²ç»ç¨è¿å¾å¤æ¬¡äºã
äºå®ä¸Centerç»ä»¶ç»§æ¿èªAlignï¼åªæ¯å°alignment设置为Alignment.centerã
æºç åæï¼
classCenterextendsAlign{ constCenter({ Keykey,doublewidthFactor,doubleheightFactor,Widgetchild}):super(key:key,widthFactor:widthFactor,heightFactor:heightFactor,child:child);}1.2.2.Centeræ¼ç»æ们å°ä¸é¢ç代ç Alignæ¢æCenter
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnCenter(child:Icon(Icons.pets,size:,color:Colors.red),widthFactor:3,heightFactor:3,);}}1.3.Paddingç»ä»¶1.3.1.Paddingä»ç»Paddingç»ä»¶å¨å ¶ä»ç«¯ä¹æ¯ä¸ä¸ªå±æ§èå·²ï¼ä½æ¯å¨Flutterä¸æ¯ä¸ä¸ªWidgetï¼ä½æ¯Flutterä¸æ²¡æMarginè¿æ ·ä¸ä¸ªWidgetï¼è¿æ¯å 为å¤è¾¹è·ä¹å¯ä»¥éè¿Paddingæ¥å®æã
Paddingé常ç¨äºè®¾ç½®åWidgetå°ç¶Widgetçè¾¹è·ï¼ä½ å¯ä»¥ç§°ä¹ä¸ºæ¯ç¶ç»ä»¶çå è¾¹è·æåWidgetçå¤è¾¹è·ï¼ã
æºç åæï¼
constPadding({ Keykey,@requiredthis.padding,//EdgeInsetsGeometryç±»åï¼æ½è±¡ç±»ï¼ï¼ä½¿ç¨EdgeInsetsWidgetchild,})1.3.2.Paddingæ¼ç»ä»£ç æ¼ç»ï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnPadding(padding:EdgeInsets.all(),child:Text("è«å¬ç©¿ææå¶å£°ï¼ä½å¦¨åå¸ä¸å¾è¡ã竹æèéè½»è马ï¼è°æï¼ä¸èçé¨ä»»å¹³çã",style:TextStyle(color:Colors.redAccent,fontSize:),),);}}1.4.Containerç»ä»¶Containerç»ä»¶ç±»ä¼¼äºå ¶ä»Androidä¸çViewï¼iOSä¸çUIViewã
å¦æä½ éè¦ä¸ä¸ªè§å¾ï¼æä¸ä¸ªèæ¯é¢è²ãå¾åãæåºå®ç尺寸ãéè¦ä¸ä¸ªè¾¹æ¡ãåè§çææï¼é£ä¹å°±å¯ä»¥ä½¿ç¨Containerç»ä»¶ã
.1.Containerä»ç»Containerå¨å¼åä¸è¢«ä½¿ç¨çé¢çæ¯é常é«çï¼ç¹å«æ¯æ们ç»å¸¸ä¼å°å ¶ä½ä¸ºå®¹å¨ç»ä»¶ã
ä¸é¢æ们æ¥çä¸ä¸Containeræåªäºå±æ§ï¼
Container({ this.alignment,this.padding,//容å¨å è¡¥ç½ï¼å±äºdecorationçè£ é¥°èå´Colorcolor,//èæ¯è²Decorationdecoration,//èæ¯è£ 饰DecorationforegroundDecoration,//åæ¯è£ 饰doublewidth,//容å¨ç宽度doubleheight,//容å¨çé«åº¦BoxConstraintsconstraints,//容å¨å¤§å°çéå¶æ¡ä»¶this.margin,//容å¨å¤è¡¥ç½ï¼ä¸å±äºdecorationçè£ é¥°èå´this.transform,//åæ¢this.child,})大å¤æ°å±æ§å¨ä»ç»å ¶å®å®¹å¨æ¶é½å·²ç»ä»ç»è¿äºï¼ä¸åèµè¿°ï¼ä½æ两ç¹éè¦è¯´æï¼
容å¨ç大å°å¯ä»¥éè¿widthãheightå±æ§æ¥æå®ï¼ä¹å¯ä»¥éè¿constraintsæ¥æå®ï¼å¦æåæ¶åå¨æ¶ï¼widthãheightä¼å ãå®é ä¸Containerå é¨ä¼æ ¹æ®widthãheightæ¥çæä¸ä¸ªconstraintsï¼
colorådecorationæ¯äºæ¥çï¼å®é ä¸ï¼å½æå®coloræ¶ï¼Containerå ä¼èªå¨å建ä¸ä¸ªdecorationï¼
decorationå±æ§ç¨åæ们详ç»å¦ä¹ ï¼
1.4.2.Containeræ¼ç»ç®åè¿è¡ä¸ä¸ªæ¼ç¤ºï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnCenter(child:Container(color:Color.fromRGBO(3,3,,.5),width:,height:,child:Icon(Icons.pets,size:,color:Colors.white),),);}}1.4.3.BoxDecorationContaineræä¸ä¸ªé常éè¦çå±æ§decorationï¼
ä»å¯¹åºçç±»åæ¯Decorationç±»åï¼ä½æ¯å®æ¯ä¸ä¸ªæ½è±¡ç±»ã
å¨å¼åä¸ï¼æ们ç»å¸¸ä½¿ç¨å®çå®ç°ç±»BoxDecorationæ¥è¿è¡å®ä¾åã
BoxDecoration常è§å±æ§ï¼
constBoxDecoration({ this.color,//é¢è²ï¼ä¼åContainerä¸çcolorå±æ§å²çªthis.image,//èæ¯å¾çthis.border,//è¾¹æ¡ï¼å¯¹åºç±»åæ¯Borderç±»åï¼éé¢æ¯ä¸ä¸ªè¾¹æ¡ä½¿ç¨BorderSidethis.borderRadius,//åè§ææthis.boxShadow,//é´å½±ææthis.gradient,//æ¸åææthis.backgroundBlendMode,//èæ¯æ··åthis.shape=BoxShape.rectangle,//å½¢å})é¨åæææ¼ç¤ºï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnCenter(child:Container(//color:Color.fromRGBO(3,3,,.5),width:,height:,child:Icon(Icons.pets,size:,color:Colors.white),decoration:BoxDecoration(color:Colors.amber,//èæ¯é¢è²border:Border.all(color:Colors.redAccent,width:3,style:BorderStyle.solid),//è¿éä¹å¯ä»¥ä½¿ç¨Border.allç»ä¸è®¾ç½®//top:BorderSide(//color:Colors.redAccent,//width:3,//style:BorderStyle.solid//),borderRadius:BorderRadius.circular(),//è¿éä¹å¯ä»¥ä½¿ç¨.onlyåå«è®¾ç½®boxShadow:[BoxShadow(offset:Offset(5,5),color:Colors.purple,blurRadius:5)],//shape:BoxShape.circle,//ä¼åborderRadiuså²çªgradient:LinearGradient(colors:[Colors.green,Colors.red])),),);}}1.4.4.å®ç°åè§å¾åä¸ä¸ä¸ªç« èæ们æå°å¯ä»¥éè¿Container+BoxDecorationæ¥å®ç°åè§å¾åã
å®ç°ä»£ç å¦ä¸ï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}0äº.å¤åå¸å±ç»ä»¶å¨å¼åä¸ï¼æ们ç»å¸¸éè¦å°å¤ä¸ªWidgetæ¾å¨ä¸èµ·è¿è¡å¸å±ï¼æ¯å¦æ°´å¹³æ¹åãåç´æ¹åæåï¼çè³ææ¶åéè¦ä»ä»¬è¿è¡å±å ï¼æ¯å¦å¾çä¸é¢æ¾ä¸æ®µæåçï¼
è¿ä¸ªæ¶åæ们éè¦ä½¿ç¨å¤åå¸å±ç»ä»¶ï¼Multi-childlayoutwidgetsï¼ã
æ¯è¾å¸¸ç¨çå¤åå¸å±ç»ä»¶æ¯RowãColumnãStackï¼æ们æ¥å¦ä¹ ä¸ä¸ä»ä»¬ç使ç¨ã
2.1.Flexç»ä»¶äºå®ä¸ï¼æ们å³å°å¦ä¹ çRowç»ä»¶åColumnç»ä»¶é½ç»§æ¿èªFlexç»ä»¶ã
Flexç»ä»¶åRowãColumnå±æ§ä¸»è¦çåºå«å°±æ¯å¤ä¸ä¸ªdirectionã
å½directionçå¼ä¸ºAxis.horizontalçæ¶åï¼åæ¯Rowã
å½directionçå¼ä¸ºAxis.verticalçæ¶åï¼åæ¯Columnã
å¨å¦ä¹ RowåColumnä¹åï¼æ们å å¦ä¹ 主轴å交åè½´çæ¦å¿µã
å 为Rowæ¯ä¸è¡æå¸ï¼Columnæ¯ä¸åæå¸ï¼é£ä¹å®ä»¬é½åå¨ä¸¤ä¸ªæ¹åï¼å¹¶ä¸ä¸¤ä¸ªWidgetæåçæ¹ååºè¯¥æ¯å¯¹ç«çã
å®ä»¬ä¹ä¸é½æ主轴ï¼MainAxisï¼å交åè½´ï¼CrossAxisï¼çæ¦å¿µï¼
对äºRowæ¥è¯´ï¼ä¸»è½´ï¼MainAxisï¼å交åè½´ï¼CrossAxisï¼åå«æ¯ä¸å¾
对äºColumnæ¥è¯´ï¼ä¸»è½´ï¼MainAxisï¼å交åè½´ï¼CrossAxisï¼åå«æ¯ä¸å¾
2.1.Rowç»ä»¶2.1.1.Rowä»ç»Rowç»ä»¶ç¨äºå°ææçåWidgetææä¸è¡ï¼å®é ä¸è¿ç§å¸å±åºè¯¥æ¯åé´äºWebçFlexå¸å±ã
å¦æçæFlexå¸å±ï¼ä¼åç°é常ç®åã
ä»æºç ä¸æ¥çRowçå±æ§ï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}1mainAxisSizeï¼
表示Rowå¨ä¸»è½´(æ°´å¹³)æ¹åå ç¨ç空é´ï¼é»è®¤æ¯MainAxisSize.maxï¼è¡¨ç¤ºå°½å¯è½å¤çå ç¨æ°´å¹³æ¹åç空é´ï¼æ¤æ¶æ 论åwidgetså®é å ç¨å¤å°æ°´å¹³ç©ºé´ï¼Rowç宽度å§ç»çäºæ°´å¹³æ¹åçæ大宽度
èMainAxisSize.min表示尽å¯è½å°çå ç¨æ°´å¹³ç©ºé´ï¼å½åwidgets没æå 满水平å©ä½ç©ºé´ï¼åRowçå®é 宽度çäºææåwidgetså ç¨çç水平空é´ï¼
mainAxisAlignmentï¼è¡¨ç¤ºåWidgetså¨Rowæå ç¨ç水平空é´å 对é½æ¹å¼
å¦æmainAxisSizeå¼ä¸ºMainAxisSize.minï¼åæ¤å±æ§æ æä¹ï¼å 为åwidgetsç宽度çäºRowç宽度
åªæå½mainAxisSizeçå¼ä¸ºMainAxisSize.maxæ¶ï¼æ¤å±æ§æææä¹
MainAxisAlignment.start表示沿textDirectionçåå§æ¹å对é½ï¼
å¦textDirectionåå¼ä¸ºTextDirection.ltræ¶ï¼åMainAxisAlignment.start表示左对é½ï¼textDirectionåå¼ä¸ºTextDirection.rtlæ¶è¡¨ç¤ºä»å³å¯¹é½ã
èMainAxisAlignment.endåMainAxisAlignment.startæ£å¥½ç¸åï¼
MainAxisAlignment.centerè¡¨ç¤ºå± ä¸å¯¹é½ã
crossAxisAlignmentï¼è¡¨ç¤ºåWidgetså¨çºµè½´æ¹åç对é½æ¹å¼
Rowçé«åº¦çäºåWidgetsä¸æé«çåå ç´ é«åº¦
å®çåå¼åMainAxisAlignmentä¸æ ·(å å«startãendãcenterä¸ä¸ªå¼)
ä¸åçæ¯crossAxisAlignmentçåèç³»æ¯verticalDirectionï¼å³verticalDirectionå¼ä¸ºVerticalDirection.downæ¶crossAxisAlignment.startæ顶é¨å¯¹é½ï¼verticalDirectionå¼ä¸ºVerticalDirection.upæ¶ï¼crossAxisAlignment.startæåºé¨å¯¹é½ï¼ècrossAxisAlignment.endåcrossAxisAlignment.startæ£å¥½ç¸åï¼
2.1.2.Rowæ¼ç»æ们æ¥å¯¹é¨åå±æ§è¿è¡ç®åç代ç æ¼ç»ï¼å ¶ä»ä¸äºå±æ§å¤§å®¶èªå·±å¦ä¹ ä¸ä¸
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}.1.3.mainAxisSizeé»è®¤æ åµä¸ï¼Rowä¼å°½å¯è½å æ®å¤ç宽度ï¼è®©åWidgetå¨å ¶ä¸è¿è¡æå¸ï¼è¿æ¯å 为mainAxisSizeå±æ§é»è®¤å¼æ¯MainAxisSize.maxã
æ们æ¥çä¸ä¸ï¼å¦æè¿ä¸ªå¼è¢«ä¿®æ¹ä¸ºMainAxisSize.maxä¼ä»ä¹ååï¼
2.1.4.TextBaselineå ³äºTextBaselineçåå¼è§£æ
2.1.5.Expandedå¦ææ们å¸æ红è²åé»è²çContainerWidgetä¸è¦è®¾ç½®åºå®ç宽度ï¼èæ¯å æ®å©ä½çé¨åï¼è¿ä¸ªæ¶ååºè¯¥å¦ä½å¤çå¢ï¼
è¿ä¸ªæ¶åæ们å¯ä»¥ä½¿ç¨Expandedæ¥å 裹ContainerWidgetï¼å¹¶ä¸å°å®ç宽度ä¸è®¾ç½®å¼ï¼
flexå±æ§ï¼å¼¹æ§ç³»æ°ï¼Rowä¼æ ¹æ®ä¸¤ä¸ªExpandedçå¼¹æ§ç³»æ°æ¥å³å®å®ä»¬å æ®å©ä¸ç©ºé´çæ¯ä¾
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}.2.Columnç»ä»¶Columnç»ä»¶ç¨äºå°ææçåWidgetææä¸åï¼å¦ä¼äºåé¢çRowåï¼Columnåªæ¯årowçæ¹åä¸åèå·²ã
2.2.1.Columnä»ç»æ们ç´æ¥çå®çæºç ï¼æ们åç°åRowå±æ§æ¯ä¸è´çï¼ä¸å解é
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}.2.2.Columnæ¼ç»æ们ç´æ¥å°Rowç代ç ä¸Rowæ¹ä¸ºColumnï¼æ¥ç代ç è¿è¡ææ
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}.3.Stackç»ä»¶å¨å¼åä¸ï¼æ们å¤ä¸ªç»ä»¶å¾æå¯è½éè¦éå æ¾ç¤ºï¼æ¯å¦å¨ä¸å¼ å¾çä¸æ¾ç¤ºæåæè ä¸ä¸ªæé®çã
å¨Androidä¸å¯ä»¥ä½¿ç¨Frameæ¥å®ç°ï¼å¨Web端å¯ä»¥ä½¿ç¨ç»å¯¹å®ä½ï¼å¨Flutterä¸æ们éè¦ä½¿ç¨å±å å¸å±Stackã
2.3.1.Stackä»ç»æ们è¿æ¯éè¿æºç æ¥çä¸ä¸Stackæåªäºå±æ§ï¼
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}6åæ°j解æï¼
alignmentï¼æ¤åæ°å³å®å¦ä½å»å¯¹é½æ²¡æå®ä½ï¼æ²¡æ使ç¨Positionedï¼æé¨åå®ä½çåwidgetãæè°é¨åå®ä½ï¼å¨è¿éç¹æ没æå¨æä¸ä¸ªè½´ä¸å®ä½ï¼leftãright为横轴ï¼topãbottom为纵轴ï¼åªè¦å å«æ个轴ä¸çä¸ä¸ªå®ä½å±æ§å°±ç®å¨è¯¥è½´ä¸æå®ä½ã
textDirectionï¼åRowãWrapçtextDirectionåè½ä¸æ ·ï¼é½ç¨äºå³å®alignment对é½çåèç³»å³ï¼textDirectionçå¼ä¸ºTextDirection.ltrï¼åalignmentçstart代表左ï¼end代表å³ï¼textDirectionçå¼ä¸ºTextDirection.rtlï¼åalignmentçstart代表å³ï¼end代表左ã
fitï¼æ¤åæ°ç¨äºå³å®æ²¡æå®ä½çåwidgetå¦ä½å»éåºStackç大å°ãStackFit.loose表示使ç¨åwidgetç大å°ï¼StackFit.expand表示æ©ä¼¸å°Stackç大å°ã
overflowï¼æ¤å±æ§å³å®å¦ä½æ¾ç¤ºè¶ åºStackæ¾ç¤ºç©ºé´çåwidgetï¼å¼ä¸ºOverflow.clipæ¶ï¼è¶ åºé¨åä¼è¢«åªè£ï¼éèï¼ï¼å¼ä¸ºOverflow.visibleæ¶åä¸ä¼ã
2.3.2.Stackæ¼ç»Stackä¼ç»å¸¸åPositionedä¸èµ·æ¥ä½¿ç¨ï¼Positionedå¯ä»¥å³å®ç»ä»¶å¨Stackä¸çä½ç½®ï¼ç¨äºå®ç°ç±»ä¼¼äºWebä¸çç»å¯¹å®ä½ææã
ä¸ä¸ªç®åçæ¼ç»ï¼
注æï¼Positionedç»ä»¶åªè½å¨Stackä¸ä½¿ç¨ã
classMyHomeBodyextendsStatelessWidget{ @overrideWidgetbuild(BuildContextcontext){ returnAlign(child:Icon(Icons.pets,size:,color:Colors.red),alignment:Alignment.bottomRight,widthFactor:3,heightFactor:3,);}}7\
åæï¼/post/reactadmin?
开箱即用的React前端框架——ReactAdmin
ReactAdmin是一个Github上免费开源的前端框架(不是组件库,也不是模板,它是一个框架),采用es6、React和MaterialDesign构建基于Rest/GraphQlAPI的Web应用程序。在React上star数超过8k。
ReactAdmin不是个UI组件库,它是一个前端框架,因此你基本上基本上只要按照官网的文档进行一些配置等然后在其基础上开发自己的应用程序即可,可谓开箱即用,意识就是都给你集成好了。
你可以直接使用以下命令进行安装(这是安装react-admin及所有的依赖)
下面我们看一下官网提供的一个最简单的示例,你可以在它的主仓库中获取
我们进入到simple中,大致看一下代码和目录结构
我们安装一些依赖然后启动
成功后打开浏览器,这是使用react-admin最简单的一个例子
一图了解
由于ReactAdmin是一个非常复杂的框架,你可以参考提供的文档,我这里就不详细介绍了,感兴趣的可以直接看文档,文档是xmindcrack.jar源码英文的,所有的说明都在文档中。
ReactAdmin它是一个集合,它将一些前端开发所需要的东西都集成了进来,然后做好,我们直接使用即可,不仅仅适合个人学习,也适合通过它来构建企业级的应用。我们不仅仅是拿过来直接用,我们可看一看别人是怎么实现这样的一个框架的,从源码去学习会更快的提升自己的水平,希望对你有所帮助!
react-admin一款基于reactjs后台解决方案
使用ES6,React和MaterialDesign在REST/GraphQLAPI之上构建在浏览器中运行的管理应用程序的一款前端框架。Githubstar8.7K+,MIT协议。由marmelab开源和维护。marmelab还有一款非常热门的angularjs后台解决方案ng-admin基于angularjs感兴趣的同学可以去了解下。react-admin官网示例截图如下:
该Resource组件是一个配置组件,它允许以限定子组件对于每个管理视图的:list,edit,和create。vscode调试react源码这些组件使用MaterialUI和react-admin中的自定义组件:
antdesign表格标题下面还有子标题一、如图展示表格如何展示下方嵌套的deptName
在这里插入描述
官网在这里插入描述
二、解决方案
Columnalign="center"title="部门"dataIndex={ ["dept","deptName"]}/
1
1
技术交流分享/面试总结
微信名片
打开CSDNAPP,看更多技术内容
最新发布保姆级教程:AntDesignVue中a-table嵌套子表格
AntDesignVue中a-table嵌套子表格及只打开一个嵌套表格的方法
继续访问
Antd(Ant-design),嵌套子表格(expandedRowRender)的异步获取数据
使用阿里的ant-design开源框架,要在表格里面嵌套子表格,需要在用户点击父表格的一行数据后,获取该行的key,然后去异步请求后台的数据用来填充子表格的内容。如果这样写(省略无关代码):expandedRowRender=(record)={ dispatch({ type:'flow/getPlanList',payload:{ contractId:record.contract_id,//该参数...
继续访问
react-antdesigntable表格多级可编辑表格
antd-react3X版本多级可编辑table
继续访问
react.jsant-design中table树结构三级嵌套时逻辑问题
实现三级嵌套树结构时,勾选三级里的子节点时候父节点也会自动勾选,当子节点大于一项时候取消勾选某一子结构时父节点不变;当子节点只有一项时父节点也会自动取消勾选importStandardTablePagefrom'@/components/StandardTablePage';//用户选中某一行userSelect(record,selected,selectedRows,nativeEvent){ let{ selectedRowKeys}=this.st.
继续访问
Reactantd的table表格之嵌套表格
Reactantd的table表格之嵌套表格最近做了几个pc端的后台管理需求,涉及了table中的嵌套表格,嵌套的子表格大体分为两种效果:效果1-----点击展开新的子表格,旧的子表格关闭效果2-----可同时展开多个子表格效果1:嵌套表格,实现点击展开按钮,展开子表格请求接口数据,点击新的子表格收起原来的子表格效果//设置一个State用来储存展开的行,控制属性const[expandedRowKeys,setExpandedRowKeys]=useStateanygt
继续访问
vue嵌套表格组件_支持嵌套对象、多级数组的Vue动态多级表单组件——vue-dynamic-form-component...
方便不想看完全篇文章的童鞋,简单总结一下,这是篇软广,主要是推广自己在业务中沉淀的一个开源组件vue-dynamic-form-component。基于element-ui实现的vue组件,只需编写类似async-validator的规则,自动生成对应的表单,支持常见输入类型的同时,支持嵌套对象、hashmap、多维数组等复杂类型。有需要的童鞋欢迎使用和贡献代码,顺便给个star(我...
继续访问
antdtable嵌套子表格后端动态获取数据rudex写法示例
有一个需求是可以使主表格里每一栏数据展开,在子table里显示与其相关的子数据项,展开的时候去向后台请求数据显示。用的组件库是Antd。首先我们看Antd官方文档的Table有嵌套子表格的功能,可见我们需要使用expandedRowRender参数,但是尝试在expandedRowRender函数中进行请求,会发现发出了连续的请求,所以我们把请求写在onExpand中,只在点击展开图表的时候发出一次。之后我们写onExpand函数,注意这里的参数要写上expanded,代表是展开还是合并,我之前
继续访问
React-Antd-表格-嵌套子表格
文档地址:链接.import{ Table,Badge,Menu,Dropdown,Space}from"antd";import{ DownOutlined}from"@ant-design/icons";import{ useEffect,useState}from"react";exportdefaultfunctionApp(){ constcolumns=[//定义外层表格头数据{ title:"姓名",dataInde
继续访问
ReactantdTable实现单元格点击表头斜线分组等功能
reactantd单元格添加点击事件自定义单元格
继续访问
进阶Ant-Design-Vue你知道table多级表头嵌套展开写法吗?
前言:在Ant-Design-Vue的前端项目中,我们会经常处理表格,表单这些组件元素,熟练运用并知道它们在使用过程中的联系与区别,这是一个前端必不可少的哟。本文我旨在解决两个问题:(1)如何便于更好的嵌套多级表头(2)如何通过a-checkbox控制全选,单选显示a-table对应的列元素类似于ElementUI,Ant-Design-vue中有很多相似点,但又不完全苟同,有很多自己独有的写法和思想。相信很多人都是先入手ElementUI,再入手Ant,这其实是对开发者比较友好的方式,如果
继续访问
前端面试题(react)
性能优化分为2个方面setState是修改其中的部分状态,相当于Object.assign,只是覆盖,不会减少原来的状态;replaceState是完全替换原来的状态,相当于赋值,将原来的state替换为另一个对象,如果新状态属性减少,那么state中就没有这个状态了接收旧的state和action,返回新的state受控组件就是可以被react状态控制的组件在react中,Inputtextarea等组件默认是非受控组件(输入框内部的值是用户控制,和React无关)
继续访问
Antd表格设置表头分组实现可编辑行
主要通过onCell方法修改children中cloumn的属性。
继续访问
antd能自适应吗_admin-antd-react是一个后台前端解决方案,它基于React、AntDesign和UmiJs实现。...
admin-antd-react是一个后台前端解决方案,它基于React、AntDesign和UmiJs实现。--::?阅读次#介绍[admin-antd-react]()是一个后台前端解决方案,它基于[React](...
继续访问
react-antd-Table相似表格不同字段处理
1、当两个表格字段相似时,但有一两个字段不同,我们可以将不同的字段单独以对象的形式抽出,根据情况push进去即可。2、代码参考constchange={ title:'操作',dataIndex:'operate',key:'operate',width:'%',align:'center',render:(text,record)={
继续访问
React中控制AntDesignTable列的显示与隐藏
React中控制AntDesignTable列的显示与隐藏
继续访问
热门推荐使用antd中Table组件某一列有多个变量值需要写入
当某一列需要有多个变量值写入时,我们就不可以再用dataIndex来定义一个ID,具体解决方案将在文中给出。
继续访问
reactant-designtable显示数据以及上传数据显示到另一列中
需求是:用table展示数据,并在每一行的最后一列中给一个上传附件的按钮,可上传多个附件,上传之后在另一列去显示数据,可以删除附件因为有别的组件用到columns,他不需要上传附件功能,我就把columns的公共部分提出去了,在这个组件需要的时候在push到column中。columns如下:/***处理上传附件列*/handleColumn=()={ letcArr=this.state.columns;
继续访问
antd表单一个label下多个内容
1.如何实现一个label旁并排多项内容比如这样,在other的label下,既有输入文本域,又有toggle开关。同时两者有序地并排在右侧,同一行。方法是再用一层Form.Item分别包裹右侧内容,并且设置noStyle属性示例代码:Form.Itemlabel="DateofBirth:"Form.Itemname="birthDate"noStyle
继续访问
React针对antDesignselect组件进行二次封装
React针对antDesign库select组件进行二次封装由于业务需要对select进行样式上的修改,部分select还需要使用原样式。这种情况可以通过两种方式来实现:1通过className进行样式覆盖2通过二次封装组件,相对于仅修改css样式来说更加的灵活本次介绍第二种方式对组件进行二次封装/**@Date/5/8*@Authorzuolinya*@Descriptionantdselect组件二次封装*1设置为圆角*/import
继续访问
React基于AntdDesign的RadioGroup按钮组控件封装
开源Vue后端UI开箱即用解决方案——vuestic-admin这是一个Vue的后端开箱即用UI项目框架,和之前的ReactAdmin类似,它是一个框架,也就意味着它帮你完成了很多公用的部分,你只需要在其基础上进行自己的项目扩展即可。大体上这是由Vue和bootstrap4构建的,其中还集成了很多其他的东西。
如果你使用yarn你也可以使用它安装
在安装好vuestic后,你就可以使用它进行项目创建了
创建好后大致看一下目录结构(可能不清晰,你可以直接搭建体验)
成功后打开localhost:
如果你想详细的学习,你可以查看官网文档
浏览器兼容性,很遗憾只支持到IE+和主流的Chrome、FireFox、Safari、Edge等
目前有很多开箱即用的解决方案,还有一些仅仅是UI模板,每个解决方案都有各自的优势,我们尽可能的减少不必要的开发时间浪费,在通用的基础上在扩展,最主要的还是学习看源码,希望对大家有所帮助,谢谢!
react中实现登录鉴权vue中会使用导航守卫判断token,
react中使用redirect方式
使用高阶组件定义PrivateRoute导出
在router中用
PrivateRoute?path="/admin/roles"?exact?component={ Roles}/PrivateRoute的方式判断需要token的路由
import?React,?{ ?Component?}?from?'react'
import?{ Redirect,?Route}?from?'react-router-dom'
function?PrivateRoute({ ?component:Children,?...rest?})?{
return?(
Route{ ...rest}
render={ ()?={
let?token?=?localStorage.getItem('token')if(token){
return?Children/
}else{
returnRedirect
to={
'/login'
}/
}
}
}
/);
}export?default?PrivateRoute