Skip to content

Latest commit

 

History

History
1759 lines (1514 loc) · 76 KB

README_legacy.md

File metadata and controls

1759 lines (1514 loc) · 76 KB

BadFL

Pub Version

A flutter package, including components, implementations, helper functions and extensions. Designed to provide a simple way to develop with flutter.

Overview

The bad_fl expresses component properties in a form closer to CSS representation, making it easier for web developers to get started.

NOTE: This library completely abandons semantics! (For example, Container+GestureDetector is used to express a Button rather than build-in ones) People who have related needs or care about this should use it with caution.

Usage

Extension

Extension methods of built-in types, named in the form of <Type>Ext.

IterableExt & IterableWidgetExt

source code

👉 slotted: build a new element from each element of the iterable and insert slot elements between every two elements.

class Example1 extends StatelessWidget {
  const Example1({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        // mark as `List<Widget>` to use `slotted` with default builder (asIs)
        children: const <Widget>[
          Text('item1'),
          Text('item2'),
          Text('item3'),
          Text('item4'),
        ].slotted(slot: const Divider()),
      ),
    );
  }
}

class Example2 extends StatelessWidget {
  const Example2({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        // custom builder for widget creation
        children: const [1, 2, 3, 4].slotted(
          builder: (int num) => Text('item$num'),
          slot: const Divider(),
        ),
      ),
    );
  }
}

Helper

BadDebouncer

source code

void main() async {
  final debouncer = BadDebouncer(
    delay: const Duration(seconds: 1),
    defaultAction: () {
      print('default action');
    },
  );

  // 1. call the default action
  debouncer();

  // 2. call with a custom action
  await Future.delayed(const Duration(seconds: 2));
  debouncer.call(() {
    print('wrong action');
  });

  // 3. cancel the current delayed call
  debouncer.cancel();
  debouncer(() {
    print('correct action');
  });
}

// Output:
// default action
// correct action

BadThrottler

source code

void main() async {
  final throttler = BadThrottler(defaultAction: () async {
    await Future.delayed(const Duration(milliseconds: 500));
    print('default action');
  });

  // call the default action
  throttler();

  // this call will be ignored
  throttler(() async {
    print('hello');
  });

  // call with a custom action
  await Future.delayed(const Duration(seconds: 1));
  throttler(() async {
    print('hello again');
  });
}

// Output:
// default action
// hello again

Impl

These classes are fully annotated, please refer to source code for more details.

Some classes need to be initialized before use. Due to privacy issues, some initialization have to be done after the user has accepted the privacy policy.

  • prepare: can be called as soon as possible
  • extend: may only be called after the privacy policy has been accepted
Class prepare extend
FileCacheImpl
ClipboardImpl
EvCenterImpl
ExternalLinkImpl
ImageOPImpl
KVStorageImpl
MetaImpl
RequestImpl

Layout

BadExpandable

source code

Property Type Default Description
initialOpen bool true whether the panel is open initially
headerHeight double 28 height of the header
gap double 0 gap between the header and the child
title Widget - title widget
emptyIcon Widget? - icon to be displayed when the panel is empty
openedIcon Widget Icon(Icons.arrow_drop_up, size: 24) icon to be displayed when the panel is opened
closedIcon Widget Icon(Icons.arrow_drop_down, size: 24) icon to be displayed when the panel is closed
child Widget? - child widget to be displayed when the panel is open

BadExpandable has two constructors:

  • the default constructor is when there is a child (child cannot be null)
  • BadExpandable.empty means when there is no child (emptyIcon cannot be null)

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: [
          BadExpandable(
            title: const Text('block1'),
            child: Container(
              height: 200,
              color: Colors.red,
            ),
          ),
          BadExpandable(
            title: const Text('block2'),
            child: Container(
              height: 200,
              color: Colors.green,
            ),
          ),
          const BadExpandable.empty(
            title: Text('empty block'),
            emptyIcon: Icon(Icons.hourglass_empty, size: 16),
          ),
          BadExpandable(
            title: const Text('block3'),
            child: Container(
              height: 200,
              color: Colors.orange,
            ),
          ),
        ],
      ),
    );
  }
}

BadPanel

source code

👉 BadPanel

Property Type Default Description
options BadPanelOptions BadPanelOptions() options of the panel
title Widget - title widget, null if no title
items List<BadPanelItem> - items in panel

👉 BadPanelOptions

Property Type Default Description
fill Color Colors.white background color of panel
borderRadius double 8 border radius of panel
divider Widget Divider(height: 0.5, thickness: 0.5, color: Colors.grey) divider between items
itemHeight double 54 height of each item

👉 BadPanelItem

Property Type Default Description
label Widget - label widget
body Widget? - body widget, null if no body
suffix Widget? - suffix widget, null if no suffix
onTap VoidCallback? - callback when item is clicked, null if not clickable

BadPanelItem layout logic:

  • If there is a suffix, prioritize the space for the suffix
  • If there is a body, the label and body share the remaining space
  • If there is no body, the label occupies the remaining space exclusively

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      backgroundColor: Colors.grey[200],
      body: ListView(
        padding: const EdgeInsets.all(12),
        children: [
          BadPanel(
            title: const Padding(
              padding: EdgeInsets.only(bottom: 8),
              child: Text('Profile'),
            ),
            items: const [
              BadPanelItem(
                label: Text('Avatar'),
                suffix: CircleAvatar(
                  radius: 18,
                  backgroundImage: NetworkImage('https://picsum.photos/200'),
                ),
              ),
              BadPanelItem(
                label: Text('Nickname'),
                body: Text('nickname', textAlign: TextAlign.end),
                suffix: Icon(Icons.edit_note),
              ),
              BadPanelItem(
                label: Text('HomePage'),
                body: Text('https://example.com', textAlign: TextAlign.end),
              ),
              BadPanelItem(
                label: Text('About'),
                body: Text(
                  'lorem ipsum dolor sit amet, consectetur adipiscing elit',
                  maxLines: 2,
                  overflow: TextOverflow.ellipsis,
                ),
              ),
            ],
          ),
        ],
      ),
    );
  }
}

BadScrollAnchorScope

source code

This is a wrapper on top of SingleChildScrollView, allowing its child elements to act as anchors (children are wrapped with Column):

  • can be used as scroll target
  • can listen to its show/hide state changes

👉 BadScrollAnchorScope<AsKey extends Object>

Property Type Default Description
scrollDirection Axis Axis.vertical passed into the inner SingleChildScrollView
padding EdgeInsetsGeometry? - passed into the inner SingleChildScrollView
physics ScrollPhysics? - passed into the inner SingleChildScrollView
scrollController ScrollController? - passed into the inner SingleChildScrollView (if not provided, a internal ScrollController will be maintained)
crossAxisAlignment CrossAxisAlignment CrossAxisAlignment.center passed to the inner Column
controller BadScrollAnchorController<AsKey> - controller for the scope
children List<Widget> - widgets to be placed inside the inner SingleChildScrollView (wrapped in a Column)

👉 BadScrollAnchor<AsKey extends Object>

Property Type Default Description
asKey AsKey - value to be used as key
child Widget - child widget

👉 BadScrollAnchorController<AsKey extends Object>

Property Type Default Description
scrollDirection Axis Axis.vertical the scrollDirection of the inner ScrollView
Property / Method Type / Signature Description
anchorCount int number of anchor in the scope
anchors Iterable<AsKey> all keys of the anchors
jumpToAnchor void Function(AsKey key, [double offset]) jump to the anchor point related to the key with an optional offset
animateToAnchor FutureOr<void> Function(AsKey key, {Duration duration, Curve curve, double offset}) animate to the anchor point related to the key with an optional offset
updateAnchorPos void Function() update the position of the anchor points in the scope so that the observers can be notified accurately
addObserver void Function(void Function(AsKey key, AnchorStatus status) observer) add an observer
removeObserver void Function(void Function(AsKey key, AnchorStatus status) observer) remove an observer

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  int ptr = 0;
  final sc = ScrollController();
  final sac = BadScrollAnchorController<int>();

  @override
  void initState() {
    super.initState();

    sac.addObserver((key, status) {
      if (status == AnchorStatus.show) {
        setState(() {
          ptr = key;
        });
      } else {
        setState(() {
          ptr = key + 1;
        });
      }
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('top anchor: $ptr')),
      body: Column(
        children: [
          Padding(
            padding: const EdgeInsets.all(16),
            child: Row(
              children: [
                Expanded(
                  child: ElevatedButton(
                    onPressed: () => sac.jumpToAnchor(2),
                    child: const Text('jump to 2'),
                  ),
                ),
                const SizedBox(width: 16),
                Expanded(
                  child: ElevatedButton(
                    onPressed: () => sac.animateToAnchor(4),
                    child: const Text('animate to 4'),
                  ),
                ),
              ],
            ),
          ),
          Expanded(
            child: BadScrollAnchorScope(
              controller: sac,
              scrollController: sc,
              padding: const EdgeInsets.all(16),
              children: [
                BadScrollAnchor(
                  asKey: 0,
                  child: Container(
                    width: double.infinity,
                    height: 300,
                    color: Colors.blue,
                    child: const Text('anchor 0'),
                  ),
                ),
                BadScrollAnchor(
                  asKey: 1,
                  child: Container(
                    width: double.infinity,
                    height: 400,
                    color: Colors.grey,
                    padding: const EdgeInsets.all(50),
                    child: Column(
                      children: [
                        const Text('anchor 1'),
                        const SizedBox(height: 50),
                        BadScrollAnchor(
                          asKey: 2,
                          child: Container(
                            width: double.infinity,
                            height: 100,
                            color: Colors.cyan,
                            child: const Text(
                                'anchor 2\nnested anchor works too!'),
                          ),
                        ),
                      ],
                    ),
                  ),
                ),
                BadScrollAnchor(
                  asKey: 3,
                  child: Container(
                    width: double.infinity,
                    height: 200,
                    color: Colors.green,
                    child: const Text('anchor 3'),
                  ),
                ),
                BadScrollAnchor(
                  asKey: 4,
                  child: Container(
                    width: double.infinity,
                    height: 500,
                    color: Colors.amber,
                    child: const Text('anchor 4'),
                  ),
                ),
                BadScrollAnchor(
                  asKey: 5,
                  child: Container(
                    width: double.infinity,
                    height: 700,
                    color: Colors.teal,
                    child: const Text('anchor 5'),
                  ),
                ),
              ],
            ),
          ),
        ],
      ),
    );
  }
}

BadTree

source code

Property Type Default Description
controller BadTreeController<TreeNodeData>? - controller for tree state management
tree TreeNodeData - tree data
childrenProvider List<TreeNodeData>? Function(TreeNodeData node) - function to provide children of a node, return null if the node is a leaf
nodeBuilder Widget Function(TreeNode<TreeNodeData> node) - builder for each node

It provides a very flexible way to build and control your tree widgets, which gives you as much control over the tree as possible (node rendering, expanded and collapsed states, etc.).

When you modify a node, it will re-render the smallest subtree (that is, the subtree with the modified node as the root node).

It internally maintains the mapping from TreeNodeData to TreeNode:

  • 😊 You can use BadTreeController to get control of the corresponding TreeNode from the original data ( via BadTreeController.getTreeNodeByData)
  • ⚠️ But this also requires that the childrenProvider you provide is idempotent.

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  final controller = BadTreeController<Map<String, dynamic>>();

  Set<Map<String, dynamic>> selected = {};

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        actions: [
          IconButton(
            // rerender the whole tree
            onPressed: () => controller.rerender(),
            icon: const Icon(Icons.refresh),
          ),
          IconButton(
            // collapse all nodes
            onPressed: () => controller.collapseAll(),
            icon: const Icon(Icons.unfold_less),
          ),
          IconButton(
            // expand all nodes
            onPressed: () => controller.expandAll(),
            icon: const Icon(Icons.unfold_more),
          ),
        ],
      ),
      body: SingleChildScrollView(
        child: BadTree<Map<String, dynamic>>(
          controller: controller,
          tree: const {
            'name': 'root',
            'children': [
              {
                'name': 'child1',
                'children': [
                  {'name': 'child1.1'},
                  {'name': 'child1.2'},
                ],
              },
              {
                'name': 'child2',
                'children': [
                  {'name': 'child2.1'},
                  {'name': 'child2.2'},
                ],
              }
            ],
          },
          childrenProvider: (node) => node['children'],
          nodeBuilder: (node) {
            return Padding(
              padding: EdgeInsets.only(left: 16.0 * node.depth),
              child: Container(
                width: double.infinity,
                color: selected.contains(node.data)
                    ? Colors.blue.withOpacity(0.5)
                    : null,
                child: Row(
                  children: [
                    IconButton(
                      onPressed: node.toggleExpanded,
                      icon: node.expanded
                          ? const Icon(Icons.arrow_drop_down)
                          : const Icon(Icons.arrow_right),
                    ),
                    Expanded(
                      child: GestureDetector(
                        onTap: () {
                          selected.contains(node.data)
                              ? selected.remove(node.data)
                              : selected.add(node.data);

                          // rerender subtree of the selected node
                          node.rerender();
                        },
                        child: Text(node.data['name']),
                      ),
                    ),
                  ],
                ),
              ),
            );
          },
        ),
      ),
    );
  }
}

Mixin

BadDisposeMixin

source code

Mixin on GetxController, used to execute the dispose method in the onClose lifecycle.

👉 BadDisposeScrollMixin: provide a ScrollController instance named sc
👉 BadDisposeTextEditingMixin: provide a TextEditingController instance named tec

BadSearchMixin

source code

Mixin on GetxController, provides search-related functions (pagination, refresh, load more, etc.)

Field Type Override Description
sc ScrollController this should be attached to the list container the data is displayed in, whether it's a ListView, GridView, CustomScrollView, etc
pending RxBool indicate if the search is pending
pageSize int the page size to search, default to 20
target String the target string to search (read-only)
pageNo int the next page number to search (read-only)
resultList List<T> result container
onSearchEvent * void Function(SearchEvent event) override this method to handle search event (do some log, toast, etc)
fetcher Future<Iterable<ListItemType>?> Function(String target, int pageNo) specific implementation of data request. It should return an Iterable<T> (success) or null (failed).
NOTE: this should never be called directly, use nextPage, reloadPage or searchPage when needed
nextPage ** Future<void> Function() search the next page
reloadPage ** Future<void> Function() reload the first page
searchPage ** Future<void> Function(String newTarget) search the first page with a new target
  • *: This is optional, you can override this method to handle search events. by default, it will do nothing.
  • **: You can call these methods to trigger the corresponding actions. refer to the example below for usage, source code for more details.

class _ExampleController extends GetxController with BadSearchMixin<String> {
  @override
  int get pageSize => 5;

  @override
  Future<Iterable<String>?> fetcher(String target, int pageNo) async {
    // wait for 1 second to simulate the network delay
    await Future.delayed(const Duration(seconds: 1));

    // mock the search result
    int base = (pageNo - 1) * pageSize;
    return List.generate(pageSize, (index) => 'Item ${base + index}');
  }

  @override
  void onReady() {
    super.onReady();

    nextPage();
  }
}

class Example extends GetView<_ExampleController> {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    Get.put(_ExampleController());

    return Scaffold(
      appBar: AppBar(
        title: Obx(() {
          return controller.pending.isTrue
              ? const Text('status: pending')
              : const Text('status: idle');
        }),
      ),
      body: Column(
        children: [
          Padding(
            padding: const EdgeInsets.all(8),
            child: Row(
              children: [
                Expanded(
                  child: ElevatedButton(
                    onPressed: controller.reloadPage,
                    child: const Text('Refresh'),
                  ),
                ),
                const SizedBox(width: 8),
                Expanded(
                  child: ElevatedButton(
                    onPressed: controller.nextPage,
                    child: const Text('More'),
                  ),
                ),
              ],
            ),
          ),
          const Divider(height: 1, thickness: 1),
          Expanded(
            child: Obx(
                  () =>
                  ListView.builder(
                    controller: controller.sc,
                    itemCount: controller.resultList.length,
                    itemBuilder: (_, index) {
                      return ListTile(
                        title: Text(controller.resultList[index]),
                      );
                    },
                  ),
            ),
          ),
        ],
      ),
    );
  }
}

Prefab

BadButton

source code

Property Type Default Description
width double? - width of the button
height double - height of the button
margin EdgeInsets? - margin of the button
padding EdgeInsets? - padding of the button
border Border? - border of the button
borderRadius double 0 border radius of the button
fill Color? - background color of the button
child Widget - child widget of the button
onClick VoidCallback - click callback of the button

BadCheckbox

source code

Property Type Default Description
size double - size of the checkbox
icon Widget? - icon of the checkbox (Available when constructed using BadCheckBox.icon)
iconBuilder Widget Function(bool) - icon builder of the checkbox (Available when constructed using BadCheckBox.builder)
iconSize double size size of the icon
border Border? - border of the checkbox
rounded bool true whether the checkbox is rounded
fill Color? - background color of the checkbox when unchecked
fillChecked Color? fill background color of the checkbox when checked
checked bool - whether the checkbox is checked
onTap VoidCallback - tap callback of the checkbox

There are two ways to construct a BadCheckbox:

  • BadCheckbox.icon: Use a fixed icon, the icon will be displayed when the checkbox is checked. This is the regular checkbox usage.
  • BadCheckbox.builder: Use a custom icon builder. The builder will be called every time the check state changes and get the icon that should be displayed currently (if the result is null, the icon will not be displayed).

You can easily create a checkbox with multiple states using the BadCheckbox.builder (a checkbox with three states is demonstrated in Example 3)

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  bool _v1 = false;
  bool _v2 = false;
  int _count = 0;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: [
          // Example 1: Regular Checkbox
          BadCheckBox.icon(
            size: 24,
            icon: const Icon(Icons.check, size: 18),
            checked: _v1,
            border: Border.all(),
            onTap: () {
              setState(() {
                _v1 = !_v1;
              });
            },
          ),
          // Example 2: Custom Icon Builder
          BadCheckBox.iconBuilder(
            size: 24,
            iconBuilder: (c) {
              return c
                  ? const Icon(
                Icons.check,
                color: Colors.green,
              )
                  : const Icon(
                Icons.close,
                color: Colors.red,
              );
            },
            checked: _v2,
            onTap: () {
              setState(() {
                _v2 = !_v2;
              });
            },
          ),
          // Example 3: Multiple States (3 states)
          BadCheckBox.iconBuilder(
            size: 24,
            iconBuilder: (c) {
              return c
                  ? const Icon(
                Icons.check,
                color: Colors.green,
              )
                  : _count % 3 == 1
                  ? const Icon(
                Icons.close,
                color: Colors.red,
              )
                  : const Icon(
                Icons.remove,
                color: Colors.blue,
              );
            },
            checked: _count % 3 == 0,
            onTap: () {
              setState(() {
                _count += 1;
              });
            },
          ),
        ],
      ),
    );
  }
}

BadKatex

source code

Property Type Default Description
raw String - raw string containing the formula (wrapped by $ or $$)
prefixes List<InlineSpan>? - prefix widgets of the paragraph (e.g. icon)
style TextStyle? - style of the paragraph
formulaStyle TextStyle? - style of the formula (will be merged with style)
maxLines int? - maximum number of lines
overflow TextOverflow? - null if maxLines is null
- TextOverflow.ellipsis if maxLines is set
overflow style of the paragraph

It is a wrapper of flutter_math_fork. You can directly pass in the paragraph containing the formula as a raw string, and the formula will be automatically found and rendered.

  • Use $ to wrap inline formulas in the paragraph
  • Use $$ to wrap block formulas in the paragraph
  • The \unicode{<code>} directive has been converted (not supported by flutter_math_fork)

class Example extends StatelessWidget {
  final String raw = r'''
When $a \ne 0$, there are two solutions to $ax^2 + bx + c = 0$ and they are
$$x = {-b \pm \sqrt{b^2-4ac} \over 2a}.$$
''';

  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        children: [
          BadKatex(
            raw: raw,
            prefixes: const [
              WidgetSpan(
                child: Icon(Icons.question_answer, color: Colors.green),
              ),
            ],
            formulaStyle: const TextStyle(color: Colors.orange),
          ),
        ],
      ),
    );
  }
}

BadOTPInput

source code

Property Type Default Description
width double? - width of the input
height double - height of the input
placeholder String? - placeholder of the input
onChanged ValueChanged<String>? - callback when the value changes
onSubmitted ValueSetter<String>? - callback when the user submits the input
inputType TextInputType TextInputType.text type of input (mainly affects keyboard layout on mobile)
textInputAction TextInputAction TextInputAction.done action button on mobile keyboard
formatters List<TextInputFormatter>? - formatters to restrict input
style TextStyle? - style of the input
placeholderStyle TextStyle? - style of the placeholder
padding double 8 space between prefixWidget/sendWidget and outside of the input field
space double 8 space between prefixWidget/sendWidget and text of the input field
fill Color? - background color of the input
border Border? - border of the input
borderRadius double 0 border radius of the input
prefixWidget Widget? - widget to display before the input field
sendWidget Widget Icon(Icons.send, size: 16, color: Colors.blue) widget to display as role of send button
onSendTapped VoidCallback - callback when the sendWidget is tapped

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        children: [
          BadOTPInput(
            width: 120,
            height: 32,
            border: Border.all(color: Colors.grey),
            borderRadius: 8,
            onSendTapped: () {
              print('send tapped');
            },
            onSubmitted: (code) {
              print('submitted: $code');
            },
          ),
        ],
      ),
    );
  }
}

BadPasswordInput

source code

Property Type Default Description
width double? - width of the input
height double - height of the input
initialVisibility bool false initial visibility of the password
placeholder String? - placeholder of the input
onVisibilityChanged ValueChanged<bool>? - visibility change callback
onChanged ValueChanged<String>? - callback when the value changes
onSubmitted ValueSetter<String>? - callback when the user submits the input
textInputAction TextInputAction TextInputAction.done action button on mobile keyboard
formatters List<TextInputFormatter>? - formatters to restrict input
style TextStyle? - style of the input
placeholderStyle TextStyle? - style of the placeholder
padding double 8 space between prefixWidget/sendWidget and outside of the input field
space double 8 space between prefixWidget/sendWidget and text of the input field
fill Color? - background color of the input
border Border? - border of the input
borderRadius double 0 border radius of the input
prefixWidget Widget? - widget to display before the input field
suffixWidget Widget - widget to display after the input field
visibleWidget Widget Icon(Icons.visibility_outlined, size: 16, color: Colors.grey) widget to display when the password is visible
hiddenWidget Widget Icon(Icons.visibility_off_outlined, size: 16, color: Colors.grey) widget to display when the password is hidden

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        children: [
          BadPasswordInput(
            width: 120,
            height: 32,
            border: Border.all(color: Colors.grey),
            borderRadius: 8,
            onVisibilityChanged: (to) {
              print('visibility: $to');
            },
            onSubmitted: (code) {
              print('submitted: $code');
            },
          ),
        ],
      ),
    );
  }
}

BadRadio

source code

Property Type Default Description
activeIndex int - index of the active item
onTap ValueChanged<int> - callback when the item is tapped (won't be called if the active item is tapped)
width double? - width of the radio group
height double - height of the radio group
padding EdgeInsets - space between content and outside of the radio group
border Border? - border of the radio group
borderRadius double 0 border radius of the radio group
fill Color? - background color of the radio group
gradient Gradient? - background gradient of the radio group (fill will be ignored if this is specified)
activeFill Color? - background color of the active item
activeGradient Gradient? - background gradient of the active item (activeFill will be ignored if this is specified)
values List<Item> (Item is a generic type) - values of the radio group
childBuilder Widget Function(Item) - builder for each item
activeChildBuilder Widget Function(Item)? - builder for the active item, if not specified, childBuilder will be used

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  int activeIndex = 0;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        children: [
          BadRadio(
            activeIndex: activeIndex,
            onTap: (int to) {
              setState(() {
                activeIndex = to;
              });
              print('tapped $to');
            },
            height: 32,
            borderRadius: 16,
            fill: Colors.grey[200],
            activeFill: Colors.blue[200],
            values: const ['Item1', 'Item2', 'Item3'],
            childBuilder: (label) => BadText(label),
            activeChildBuilder: (label) {
              return BadText('$label!', color: Colors.white);
            },
          ),
        ],
      ),
    );
  }
}

BadSwitch

source code

Property Type Default Description
active bool - whether the switch is active
onTap VoidCallback - callback when the switch is tapped
width double - width of the switch
height double - height of the switch
gap double 1 gap between the track and the handle, in [0, height/2)
handleColor Color Colors.white color of the handle
handleColorActive Color Colors.white color of the handle when active
trackColor Color Colors.grey color of the track
trackColorActive Color Colors.blue color of the track when active

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  bool active = false;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Column(
        children: [
          BadSwitch(
            width: 64,
            height: 32,
            gap: 2,
            active: active,
            handleColorActive: Colors.orange,
            trackColor: Colors.grey[200]!,
            trackColorActive: Colors.orange[200]!,
            onTap: () {
              setState(() {
                active = !active;
              });
              print('active: $active');
            },
          ),
        ],
      ),
    );
  }
}

BadText

source code

Property Type Default Description
text String - text content
fontFamily String? - font family
selectable bool - false for BadText
- true for BadText.selectable
whether the text is selectable
color Color? - text color
fontSize double 16 font size
fontWeight FontWeight FontWeight.w400 font weight
lineHeight double fontSize * 1.2 line height
underline bool false whether the text is underlined
italic bool false whether the text is italic
shadows List<Shadow>? - shadows of the text
textAlign TextAlign TextAlign.start text alignment
textDirection TextDirection TextDirection.ltr text direction
overflow TextOverflow? - text overflow style (does not work with BadText.selectable)
maxLines int? - maximum number of lines

const lorem =
    'Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.';

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: const [
          BadText('BadText', color: Colors.grey),
          BadText(lorem),
          Divider(),
          BadText('BadText with maxLine 1', color: Colors.grey),
          BadText(lorem, maxLines: 1),
          Divider(),
          BadText('BadText.selectable', color: Colors.grey),
          BadText.selectable(lorem),
          Divider(),
          BadText('BadText.selectable with maxLine 1', color: Colors.grey),
          BadText.selectable(lorem, maxLines: 1),
        ],
      ),
    );
  }
}

BadTextField

source code

Property Type Default Description
controller TextEditingController? - controller of the text field
width double? - width of the text field
height double - height of the text field
initialValue String? - initial value of the text field
placeholder String? - placeholder text
onChanged ValueChanged<String>? - callback when the value changes
onCleared VoidCallback? - callback when the clearWidget is tapped
onSubmitted ValueSetter<String>? - callback when the user submits (e.g. press enter)
textInputAction TextInputAction TextInputAction.newline action button on mobile keyboard
formatters List<TextInputFormatter>? - formatters to restrict text field
maxLength int? - maximum length of the text field
style TextStyle? - style of the text field
placeholderStyle TextStyle? - style of the placeholder
countStyle TextStyle? - style of the count text
padding EdgeInsets EdgeInsets.all(8) space between content and outside of the text field
space double 4 space between text and clearWidget/countText
fill Color? - background color of the text field
border Border? - border of the text field
borderRadius double 0 border radius of the text field
clearWidget Widget Icon(Icons.clear, size: 20, color: Colors.grey) widget to click to clear the text field

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: [
          const Text('BadTextField with clear'),
          BadTextField(
            height: 108,
            borderRadius: 8,
            fill: Colors.grey[200]!,
            clearWidget: const Text('clear all!'),
          ),
          const Divider(),
          const Text('BadTextField with counter'),
          BadTextField(
            height: 108,
            borderRadius: 8,
            fill: Colors.grey[200]!,
            maxLength: 200,
          ),
        ],
      ),
    );
  }
}

BadTextInput

source code

Property Type Default Description
controller TextEditingController? - controller of the input field
width double? - width of the input field
height double - height of the input field
initialValue String? - initial value of the input field
placeholder String? - placeholder text
onChanged ValueChanged<String>? - callback when the value changes
onCleared VoidCallback? - callback when the clearWidget is tapped
onSubmitted ValueSetter<String>? - callback when the user submits (e.g. press enter)
inputType TextInputType TextInputType.text type of input (mainly affects keyboard layout on mobile)
textInputAction TextInputAction TextInputAction.done action button on mobile keyboard
formatters List<TextInputFormatter>? - formatters to restrict input field
style TextStyle? - style of the input field
placeholderStyle TextStyle? - style of the placeholder
padding double 8 space between prefixWidget/suffixWidget and outside of the input field
space double 8 space between prefixWidget/suffixWidget and text of the input field
fill Color? - background color of the input field
border Border? - border of the input field
borderRadius double 0 border radius of the input field
prefixWidget Widget? - widget to display before the input field
suffixWidget Widget? - widget to display after the input field
clearWidget Widget Icon(Icons.clear, size: 16, color: Colors.grey) widget to click to clear the input field

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: [
          const Text('BadTextInput with background color'),
          BadTextInput(
            height: 36,
            borderRadius: 4,
            fill: Colors.grey[200],
          ),
          const Divider(),
          const Text('BadTextInput with border'),
          BadTextInput(
            height: 36,
            border: Border.all(),
            borderRadius: 4,
          ),
          const Divider(),
          const Text('BadTextInput with underline'),
          BadTextInput(
            height: 36,
            border: const Border(bottom: BorderSide()),
          ),
          const Divider(),
          const Text('BadTextInput with prefix / suffix / ...'),
          BadTextInput(
            height: 36,
            placeholder: 'Search something ...',
            placeholderStyle: TextStyle(
              fontSize: 14,
              color: Colors.grey[400],
              fontWeight: FontWeight.w400,
            ),
            border: const Border(bottom: BorderSide()),
            prefixWidget: const Icon(Icons.search),
            suffixWidget: TextButton(
              onPressed: () {
                print('confirm');
              },
              child: const Text('Confirm'),
            ),
          ),
        ],
      ),
    );
  }
}

BadWebview

source code

// TODO: finish documentation after the implementation is completed

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<StatefulWidget> createState() => ExampleState();
}

class ExampleState extends State<Example> {
  final refresher = Refresher();
  double progress = 0.0;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const BadText('Example'),
        actions: [
          IconButton(
            icon: const Icon(Icons.refresh),
            onPressed: refresher.refresh,
          ),
        ],
      ),
      body: Column(
        children: [
          LinearProgressIndicator(value: progress),
          BadWebview.remote(
            onProgress: (v) => setState(() => progress = v),
            uri: Uri.parse('https://example.com/'),
          )
        ],
      ),
    );
  }
}

Wrapper

Non-visual components that wrap other components to provide additional functionality.

BadBackToTop

source code

Implement the back-to-top logic (show and hide, scroll to the top) according to the passed ScrollController

There are two implementations, BadBackToTop will jump directly to the top when clicked, and BadBackToTop.animated will animate to the top when clicked

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  final controller = ScrollController();

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('Example')),
      floatingActionButton: Column(
        mainAxisSize: MainAxisSize.min,
        children: [
          BadBackToTop(
            scrollController: controller,
            child: const Text('jump to top'),
          ),
          BadBackToTop.animated(
            scrollController: controller,
            child: const Text('animate to top'),
          ),
        ],
      ),
      body: ListView(
        controller: controller,
        padding: const EdgeInsets.all(16),
        children: [
          Container(height: 100, color: Colors.orange),
          Container(height: 300, color: Colors.blue),
          Container(height: 400, color: Colors.grey),
          Container(height: 200, color: Colors.green),
          Container(height: 500, color: Colors.amber),
          Container(height: 700, color: Colors.teal),
        ],
      ),
    );
  }
}

BadClickable

source code

Add click event listener for the widget.

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: BadClickable(
        onClick: () => print('clicked'),
        child: Container(width: 100, height: 100, color: Colors.blue),
      ),
    );
  }
}

BadFloating

source code

A wrapper that allows the components inside it to be dragged and placed anywhere on the screen.

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('BadFloating')),
      body: LayoutBuilder(builder: (_, constraint) {
        return Stack(
          children: [
            Container(
              width: constraint.maxWidth,
              height: constraint.maxHeight,
              color: Colors.grey[200],
            ),
            BadFloating(
              containerSize: Size(constraint.maxWidth, constraint.maxHeight),
              floatingSize: const Size(50, 50),
              initialPosition: const BadFloatingPosition.br(50, 50),
              child: Container(
                width: 50,
                height: 50,
                decoration: BoxDecoration(
                  borderRadius: BorderRadius.circular(8),
                  color: Colors.green[300],
                ),
                // child: Text('xxx'),
              ),
            ),
          ],
        );
      }),
    );
  }
}

BadHeroPreviewer

source code

Provide a preview view for the component, with hero animation when opening the preview view.

  • image preview:

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: GridView.builder(
        padding: const EdgeInsets.all(8),
        gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount(
          mainAxisSpacing: 8,
          crossAxisSpacing: 8,
          crossAxisCount: 3,
        ),
        itemCount: 15,
        itemBuilder: (_, int index) {
          return BadHeroPreviewer(
            displayWidget: Image.network(
              'https://picsum.photos/seed/num-$index/200',
              width: 50,
            ),
          );
        },
      ),
    );
  }
}
  • custom widget:

class Example extends StatelessWidget {
  const Example({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Center(
        child: BadHeroPreviewer(
          displayWidget: Container(
            color: Colors.green,
            child: const Text('Click Me!'),
          ),
          previewWidget: Container(
            color: Colors.orange,
            alignment: Alignment.center,
            child: const Text(
              'Wow!\nYou clicked me!',
              style: TextStyle(
                color: Colors.white,
                fontSize: 32,
                fontWeight: FontWeight.w800,
              ),
            ),
          ),
        ),
      ),
    );
  }
}

BadRefreshable

source code

A simple encapsulation of easy_refresh, providing pull-down refresh and pull-up loading.

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  int _count = 10;

  Future<void> refresh() async {
    await Future<void>.delayed(const Duration(seconds: 1));
    setState(() {
      _count = 10;
    });
  }

  Future<void> fetch() async {
    await Future<void>.delayed(const Duration(seconds: 1));
    setState(() {
      _count += 10;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      floatingActionButton: Text('Total: $_count'),
      body: BadRefreshable(
        onRefresh: refresh,
        onLoadMore: fetch,
        child: ListView.separated(
          itemCount: _count,
          itemBuilder: (_, int index) {
            return ListTile(title: Text('Item $index'));
          },
          separatorBuilder: (_, __) => const Divider(),
        ),
      ),
    );
  }
}

BadSnapshot

source code

A wrapper for capturing snapshot of its child widget.

@JS('saveImage')
external JSVoid saveImage(JSUint8Array bytes);

class Example extends StatefulWidget {
  const Example({super.key});

  @override
  State<Example> createState() => _ExampleState();
}

class _ExampleState extends State<Example> {
  final controller = BadSnapshotController();

  void takeSnapshot() async {
    final image = await controller.captureAsPngBuffer();
    saveImage(image.toJS);
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Text('Example'),
        actions: [
          IconButton(
            icon: const Icon(Icons.camera_alt),
            onPressed: takeSnapshot,
          ),
        ],
      ),
      backgroundColor: Colors.grey[200],
      body: ListView(
        padding: const EdgeInsets.all(16),
        children: [
          BadSnapshot(
            controller: controller,
            child: BadPanel(
              title: const Padding(
                padding: EdgeInsets.only(bottom: 8),
                child: Text('Profile'),
              ),
              items: const [
                BadPanelItem(
                  label: Text('Avatar'),
                  suffix: CircleAvatar(
                    radius: 18,
                    backgroundImage: NetworkImage('https://picsum.photos/200'),
                  ),
                ),
                BadPanelItem(
                  label: Text('Nickname'),
                  body: Text('nickname', textAlign: TextAlign.end),
                  suffix: Icon(Icons.edit_note),
                ),
                BadPanelItem(
                  label: Text('HomePage'),
                  body: Text('https://example.com', textAlign: TextAlign.end),
                ),
                BadPanelItem(
                  label: Text('About'),
                  body: Text(
                    'lorem ipsum dolor sit amet, consectetur adipiscing elit',
                    maxLines: 2,
                    overflow: TextOverflow.ellipsis,
                  ),
                ),
              ],
            ),
          ),
        ],
      ),
    );
  }
}
/**
 * @param bytes {Uint8Array}
 */
function saveImage(bytes) {
    const blob = new Blob([bytes], {type: 'image/png'});
    const url = URL.createObjectURL(blob);

    // open in new tab
    window.open(url, '_blank')

    // or download directly
    // const a = document.createElement('a');
    // a.href = url;
    // a.download = 'image.png';
    // a.click();
}