Original address: medium.com/nerd-for-te…

Author: achraf-feydi.medium.com/

Published: May 24, 2021-8 minutes to read

If you have read the previous article implementing a desktop GUI with Flutter WEB (Part 1: Introduction), I am trying to explain some of the main difficulties I encountered when implementing FlutterGUI with Flutter WEB.


In this section, we’ll try to implement a Dock similar to the one I implemented on the FlutterGUI project.

I’ll focus on explaining the mechanics behind it and illustrate it with some examples. I’ll save it for you to build a beautiful one with fancy animations.

This code is different from the code in my FlutterGUI repo. This is a cleaner, simpler version.

The dock implementation will run on the Flutter WEB and desktop.

By the end of this article, you’ll be able to build all of these docks and more.

Table of contents.

  1. Set up a basic Dock
  2. Detect mouse hovers and turn them into valuable data.
  3. Refresh your math
  4. Realization of animation
  5. Context menu. Right click!
  6. More examples

So, let’s dance it! 🔥

Set up a basic Dock

In the code below, I just use the stack and location widget to position the Dock at the bottom of the screen.

@override
Widget build(BuildContext context) {

  return Scaffold(
    appBar: AppBar(
      title: Text(widget.title),
    ),
    body: _getBody(),
  );
}

Widget _getBody() {
  return Stack(
    children: [
      Positioned(
          bottom: 40,
          child: _getDock())
    ],
  );
  
}

Widget _getDock() {
  // Our dock will go here
}
// Default DockItem size (width & hight)
final _dockItemDefaultSize = 40.0;
// Dock item size + Padding.
final _dockItemDefaultSizeWithSpacing = 100.0;
// List of items I'll be showing in the Dock.
List<int> items = [for(var i=0; i<6; i+=1) i];
Copy the code

Now we’ll implement the Dock itself.

This code is very simple and self-explanatory.

Note: arguments in bold are the most important.

Widget _getBody() {
  return Stack(
    children: [
      Positioned(
          bottom: 40,
          right: 0,
          left: 0,
          child: _getDock())
    ],
  );

}

Widget _getDock() {
  return Center(
    child: Container(
      color: Colors.black38,
      width: _dockItemDefaultSizeWithSpacing*values.length,
      child: Row(
        mainAxisAlignment: MainAxisAlignment.spaceEvenly,
        children: _getChildren(),
      ),
    ),
  );
}

List<Widget> _getChildren() {

  List<Widget> items = [];
  for(var i = 0; i< values.length; i++){ items.add(_generateItem(i)); }return items;
}

Widget _generateItem(int index) {

  return Container(
    width: _dockItemDefaultSizeWithSpacing,
    child: Center(
      child: Card(
        child: Container(
          height: _dockItemDefaultSize,
          width: _dockItemDefaultSize,
          child: Center(child: Text(values[index].toString()),),
        ),
      ),
    ),
  );
  
}
Copy the code

Detect mouse hovers and turn them into valuable data

We’ll start by declaring two variables.

  • _offset: Will keep the mouse on the Dock.
  • _currentIndex: will keep the dock-item being hovered.

(The use of these variables is explained below)

To detect mouse position, we can tilt our Dock in the MouseRegion widget.

var _offset = 0.0;
var _currentIndex = - 1;
Widget _getDock() {
  return Center(
    child: MouseRegion(
      onHover: (event){
        setState(() {
          _offset = event.localPosition.dx;
          _currentIndex =  (_getOffset()).toInt();
        });
      },
      onExit: (event){
        setState(() {
          _offset = 0 ;
          _currentIndex = - 1;
        });
      },
      child: Container(
        color: Colors.black38,
        width: _dockItemDefaultSizeWithSpacing*values.length,
        child: Row(
          mainAxisAlignment: MainAxisAlignment.spaceEvenly,
          children: _getChildren(),
        ),
      ),
    ),
  );
}
double _getOffset(){
  return _offset/_dockItemDefaultSize;
}
Copy the code

The _getOffset() method converts _offset(which requires a value between 0 and width_of_Dock) to a value between 0 and 6 (the number of items). It’s simple math!

To get _currentIndex, we simply divide the fractional part of _getOffset().

Refresh your math

Let’s say we’re hovering over project 3.

Let’s assume that we want our generic animation to be between 0 and 1. So we want project 3 to have an animation of 1, his neighbors 2 and 4 to have an animation of 0.5, and the other neighbors not to be animated.

This behavior can easily be expressed in terms of the equation of a circle.

The equation of a circle with center x0,y0 and radius r is.

We want our circle to be centered around the position of the mouse on the X-axis, and we want it to be centered around 0 on the Y-axis.

Therefore, x0 will take the value of _getOffset() and y0 will be equal to 0.

We can easily derive the value of y from this equation.

The equation above describes the upper half of a circle with center (x0,0) and radius r.

In this case, y is evaluated between 0 and r depending on x (where the Dock Item is) and x0 (where the mouse is).

Since we need a value between 0 and 1 to animate, we can use y over r.

Or we could just do it.

Now back to coding.

double _getVariation(int x, double x0,double radius){
  if(_offset==0) return 0 ;
  var z = radius - (x - x0)*(x - x0);
  if(z<0) return 0 ;
  return sqrt(z/radius);
}
Copy the code

Realization of animation

Now let’s do a simple animation to test our code.

We will add Margin at the bottom of each item using the animation values we calculated with the _getVariation equation.

We get to

  • X: index + 0.5 = Center of Dock project.
  • x0 : _getOffset= Cursor position.
  • R: equal to 3 = We want the animation to affect three items centered on X0.
Widget _generateItem(int index) {

  double dx = _getVariation(index + 0.5 ,_getOffset(),3);
  return Container(
    width: _dockItemDefaultSizeWithSpacing,
    margin: EdgeInsets.only(bottom: dx * 20 ),
    child: Center(
      child: Card(
        color: index==_currentIndex? Colors.blue:Colors.white,
        child: Container(

          height: _dockItemDefaultSize,
          width: _dockItemDefaultSize,
          child: Center(child: Text(values[index].toString()),),
        ),
      ),
    ),
  );
Copy the code

Here we are.

Context menu. Right click!

If you use Flutter Web, add this line of code to the main method to disable the default right click in the browser.

import 'dart:html';

void main() {
  window.document.onContextMenu.listen((evt) => evt.preventDefault());
  // ...
}
Copy the code

Stackoverflow.com/questions/6…

Now, we will pop our generated item in the Listener widget to detect right-click events.

List<Widget> _getChildren() {

  List<Widget> items = [];
  for(var i = 0; i< values.length; i++){ items.add(Listener( onPointerDown: (event){ _onPointerDown(event,_currentIndex); }, child: _generateItem(i))); }return items;
}
Future<void> _onPointerDown(PointerDownEvent event,int currentIndex) async {


  List<PopupMenuEntry<int>> menuItems;


    menuItems = [
      PopupMenuItem(child: Text('apply +1'), value: 1),
      PopupMenuItem(child: Text('apply -1'), value: 2),
      PopupMenuItem(child: Text('set to 0'), value: 3)];if (event.kind == PointerDeviceKind.mouse &&
      event.buttons == kSecondaryMouseButton) {
    final overlay =
    Overlay.of(context).context.findRenderObject() as RenderBox;
    final menuItem = await showMenu<int>(
        context: context,
        items: menuItems,
        position: RelativeRect.fromSize(
            event.position & Size(48.0.48.0), overlay.size));
    
    switch (menuItem) {
      case 0:
      // open;
        break;
      case 1:
      // add 1;
        values[currentIndex] = values[currentIndex] +1  ;
        break;
      case 2:
      // minus 1 ;
        values[currentIndex] = values[currentIndex]  - 1  ;
        break;
      case 3:
      // set to 0;
        values[currentIndex] = 0  ;
        break;
      default: } setState(() {}); }}Copy the code

Results:

More examples

Rotating animation

Widget _generateItemWithRotate(int index) {
  double dx = _getVariation(index + 0.5, _getOffset(), 1);
  print(dx);
  return Container(
    width: _dockItemDefaultSizeWithSpacing,
    child: Center(
      child: Transform.rotate(
        angle: dx * pi / 2,
        child: Card(
          color: index == _currentIndex ? Colors.blue : Colors.white,
          child: Stack(
            children: [
              Positioned.fill(
                child: Transform.rotate(
                    angle: -pi / 2, child: Center(child: Text("Click!"))),
              ),
              Opacity(
                opacity: 1 - dx,
                child: Container(
                  color: Colors.white,
                  height: _dockItemDefaultSize,
                  width: _dockItemDefaultSize,
                  child: Center(
                    child: Text(values[index].toString()),
                  ),
                ),
              ),
            ],
          ),
        ),
      ),
    ),
  );
}
Copy the code

Flip the animation

Widget _generateItemWithFlip(int index) {
  double dx = _getVariation(index + 0.5, _getOffset(), 1);
  print(dx);
  returnContainer( width: _dockItemDefaultSizeWithSpacing, child: Center( child: Transform( alignment: FractionalOffset.center, transform: Matrix4.identity() .. setEntry(3.2.0.002)
        ..rotateX(pi * dx),
      child: Card(
        color: index == _currentIndex ? Colors.blue : Colors.white,
        child: Stack(
          children: [
            Positioned.fill(
              child: Transform.rotate(
                  angle: pi, child: Center(child: Text("😎"))),
            ),
            Opacity(
              opacity: 1 - dx,
              child: Container(
                color: Colors.white,
                height: _dockItemDefaultSize,
                width: _dockItemDefaultSize,
                child: Center(
                  child: Text(values[index].toString()),
                ),
              ),
            ),
          ],
        ),
      ),
    )),
  );
}
Copy the code

Color animation

Widget _generateItemColored(int index) {
  double dx =
      _getVariation(index + 0.5, _getOffset(), values.length.toDouble());

  return Container(
    width: _dockItemDefaultSizeWithSpacing,
    child: Center(
      child: Card(
        color: Color.lerp(Colors.deepPurple, Colors.blue, dx),
        child: Container(
          height: _dockItemDefaultSize,
          width: _dockItemDefaultSize,
          child: Center(
            child: Text(
              values[index].toString(),
              style: TextStyle(
                  color: Color.lerp(Colors.deepPurple, Colors.white, dx)),
            ),
          ),
        ),
      ),
    ),
  );
}
Copy the code

Github.com/achreffaidi…

What’s next?

In this series of articles, I’ll explain how I implemented some of the complex widgets in this project. I hope it helps other developers make cool, useful projects.

You can find all the links below. I will update the articles as soon as they are ready.

  • Implementing the Desktop GUI using the Flutter WEB (Part 1: Introduction)
  • Implementing a desktop GUI using the Flutter WEB (Part 2: Dock)
  • Desktop GUI implementation using the Flutter WEB (Part 3: Draggable and adjustable Windows)
  • Implementing a desktop GUI using the Flutter WEB (Part 4: Windows XP Crash)
  • Desktop GRAPHICAL User Interface using Flutter WEB (Part 5: Full screen Animation)
  • Desktop GUI implementation using the Flutter WEB (Part 6: Github pages and custom urls)
  • Implement the desktop GUI using the Flutter WEB (Part 2: Dock)

www.deepl.com translation