Analysis of Flutter Framework

Analysis of the Flutter Framework (I) — Overview and Window

Analysis of the Flutter Framework (II) — Initialization

Analysis of the Flutter Framework (iii) — Widget, Element and RenderObject

Analysis of The Flutter Framework (IV) — The Operation of the Flutter Framework

Analysis of the Flutter Framework (5) — Animation

Analysis of the Flutter Framework (VI) — Layout

Analysis of the Flutter Framework (VII) — Drawing

preface

The previous two Flutter framework analysis articles introduced the rendering pipeline, window and frame initialization. This article continues to discuss the widgets, Elements and RenderObject systems that are important to Flutter app developers. The idea of Flutter is that everything is a Widget. Developers are mainly writing widgets when developing the Flutter app. So what’s the relationship between these three? How do they work? Let’s find out.

An overview of

This piece of content is more and a bit complex, in order not to let you get lost in the sea of source code, let’s take an example to briefly understand the system.

void main() {
  runApp(MyWidget());
}

class MyWidget extends StatelessWidget {
  final String _message = "Flutter frame Analysis";
  @override
  Widget build(BuildContext context) => ErrorWidget(_message);
}
Copy the code

This example uses the ErrorWidget that comes with Flutter to display our custom phrase “Flutter framework analysis”. That’s right, the ErrorWidget is the dreaded red and yellow message that appears on the screen when your code has a bug. I want to show you a screenshot.

It is used here because it is the simplest and least hierarchical Widget. To facilitate our understanding of the Flutter framework and avoid being dissuaded by the MaterialApp’s unfathomable Element tree and Render Tree.

Open the Flutter Inspector after running the above example:

MyWidget
ErrorWidget
RenderObjectToWidgetAdapter
RenderObjectToWidgetElement
StatelessElement
LeafRenderObjectElement
RenderObjectToWidgetElement
RenderView
MyWidget
StatelessElement
RenderObject
ErrorWidget
LeafRenderObjectElement
RenderObject
RenderView
RenderErrorBox

As you can see from the example above, widgets are used to describe the description or configuration of the corresponding Element. Element constitutes the Element tree. The main function of Element is to maintain the tree. The addition, deletion, update and traversal of nodes are all completed here. Elements are generated from widgets. Each Widget corresponds to an Element. But not every Widget/Element will have a RenderObject. The Widget will only have a RenderObject if it inherits from RenderObjectWidget.

In general, the following points:

  • WidgetIs theElementConfiguration or description of. The main job of the Developers of Flutter app is to build and build Flutter appsWidgetTo deal with. We don’t need to worry about tree maintenance updates, just focus on theWidgetState maintenance is fine, greatly reducing the developer’s burden.
  • ElementResponsible for maintaining Element Tree.ElementIt doesn’t care about the color, the font size, the display content, whatever the configuration or description of the UI is, it doesn’t care about the layout, it doesn’t care about the drawing, it just does its own tree.ElementMost of the work is in the build phase of the rendering pipeline.
  • RenderObjectBe responsible for laying things out, drawing those things. This is the layout and paint phases of the rendering pipeline.

Let’s take a look at the Widget, Element, and RenderObject using the source code.

Widget

The base class Widget is simple

@immutable
abstract class Widget extends DiagnosticableTree {

  const Widget({ this.key }); .@protected
  ElementcreateElement(); . }Copy the code

The createElement() method is responsible for instantiating the corresponding Element. Implemented by its subclasses. Let’s look at some of the more important subclasses:

StatelessWidget

abstract class StatelessWidget extends Widget {
  /// Initializes [key] for subclasses.
  const StatelessWidget({ Key key }) : super(key: key);
  
  @override
  StatelessElement createElement() => StatelessElement(this);

  @protected
  Widget build(BuildContext context);
}
Copy the code

The StatelessWidget will be familiar to Flutter developers. Its createElement method returns a StatelessElement instance.

StatelessWidget has no method to generate RenderObject. So StatelessWidget is just an intermediate layer that needs to implement the Build method to return child widgets.

StatefulWidget

abstract class StatefulWidget extends Widget {
  @override
  StatefulElement createElement() => StatefulElement(this);

  @protected
  State createState();
}
Copy the code

StatefulWidget will be familiar to Flutter developers. The createElement method returns a StatefulElement instance. The createState() method builds the State corresponding to the StatefulWidget.

StatefulWidget has no method to generate RenderObject. So StatefulWidget is an intermediate layer that requires the corresponding State implementation build method to return child widgets.

State

You can’t talk about StatefulWidget without talking about State.

abstract class State<T extends StatefulWidget> extends Diagnosticable {
  T get widget => _widget;
  T _widget;
  
  BuildContext get context => _element;
  StatefulElement _element;

  bool getmounted => _element ! =null;

  void initState() { }

  void didUpdateWidget(covariant T oldWidget) { }

  void setState(VoidCallback fn) {
    final dynamic result = fn() as dynamic;
    _element.markNeedsBuild();
  }

  void deactivate() { }
  
  void dispose() { }

  Widget build(BuildContext context);

  void didChangeDependencies() { }
}
Copy the code

As you can see from the source, State holds the corresponding Widget and Element. Notice that BuildContext get context => _element; . The build-in argument BuildContex actually returns Element.

Mounted: specifies whether the State is associated with an element in the Element tree. If State is not mounted == true, calling setState() will crash.

The initState() function is used to initialize State.

The function didUpdateWidget(coVariant T oldWidget) is called after the State has been changed to a new Widget. Yes, the Widget instances corresponding to State can be swapped as long as they are of the same type.

The setState() function is familiar. This function simply executes the incoming callback and calls _element.markNeedsbuild (). Will there be a problem if _element is empty? Mounted before calling setState(). Another point to note is that this function is also a point that triggers the rendering pipeline. I’ll pick up from this point in a future article to show you how the rendering pipeline works with widgets, Elements, and RenderObject architectures.

The function deactivate() is called after State’s corresponding Element has been removed from the tree, possibly temporarily.

The function dispose() is called after State’s corresponding Element has been removed permanently from the tree.

Build (BuildContext Context), you’re familiar with it, but I won’t go into that.

The didChangeDependencies() function is called when State’s dependencies change, but more on that later.

StatefullWidget and State are probably the most important things to deal with Flutter app developers. Some details need to be understood further with Element.

InheritedWidget

An InheritedWidget is neither a StatefullWidget nor a StatelessWidget. It’s used to pass data down. Child nodes under InheritedWidget can by calling BuildContext. InheritFromWidgetOfExactType () to obtain the InheritedWidget. Its createElement() function returns an InheritedElement.

abstract class InheritedWidget extends ProxyWidget {
  const InheritedWidget({ Key key, Widget child })
    : super(key: key, child: child);

  @override
  InheritedElement createElement() => InheritedElement(this);

  @protected
  bool updateShouldNotify(covariant InheritedWidget oldWidget);
}
Copy the code

RenderObjectWidget

RenderObjectWidget is used to configure the RenderObject. Its createElement() function returns RenderObjectElement. Implemented by its subclasses. In contrast to the other widgets mentioned above. There is a createRenderObject() method. To instantiate the RenderObject.

abstract class RenderObjectWidget extends Widget {

  const RenderObjectWidget({ Key key }) : super(key: key);

  @override
  RenderObjectElement createElement();

  @protected
  RenderObject createRenderObject(BuildContext context);

  @protected
  void updateRenderObject(BuildContext context, covariant RenderObject renderObject) { }

  @protected
  void didUnmountRenderObject(covariant RenderObject renderObject) { }
}
Copy the code

RenderObjectWidget is just a configuration. When the configuration changes that need to be applied to an existing RenderObject, the Flutter framework calls updateRenderObject() to set the new configuration to the corresponding RenderObject.

RenderObjectWidget has three important subclasses:

  • LeafRenderObjectWidgetthisWidgetThe configured node is at the bottom of the tree and has no children. The correspondingLeafRenderObjectElement.
  • SingleChildRenderObjectWidget, contains only one child. The correspondingSingleChildRenderObjectElement.
  • MultiChildRenderObjectWidget, have multiple children. The correspondingMultiChildRenderObjectElement.

Element

Element constitutes the Element Tree. The main thing this class is doing is maintaining the tree. As you can see from the Widget analysis above, it seems that each particular Widget has a corresponding Element. Especially for RenderObjectWidget. If I have an XXXRenderObjectWidget, its createElement() will usually return an XXXRenderObjectElement. For simplicity’s sake. Our analysis is limited to a few basic elements. Let’s start with the base class Element.

abstract class Element extends DiagnosticableTree implements BuildContext {
    Element _parent;
    Widget _widget;
    BuildOwner _owner;
    dynamic _slot;
    
    void visitChildren(ElementVisitor visitor) { }
    
    Element updateChild(Element child, Widget newWidget, dynamic newSlot) {
        
    }
    
    void mount(Element parent, dynamic newSlot) {
        
    }
    
    void unmount() {
         
    }
    
    void update(covariant Widget newWidget) {
        
    }
    
    @protected
    Element inflateWidget(Widget newWidget, dynamic newSlot) {
    ...
      final Element newChild = newWidget.createElement();
      newChild.mount(this, newSlot);
      return newChild;
    }
  
    void markNeedsBuild() {
      if (dirty)
        return;
      _dirty = true;
      owner.scheduleBuildFor(this);
    }
    
    void rebuild() {
      if(! _active || ! _dirty)return;
      performRebuild();
    }
  
    @protected
    void performRebuild();
}
Copy the code

Element holds the current Widget, a BuildOwner. This BuildOwner was instantiated earlier in the WidgetsBinding. Element is a tree structure that holds the _parent node. The _slot is set by the parent Element to tell where the current Element is on the parent node. Because Element’s base class does not know how subclasses will manage child nodes. So the function visitChildren() is implemented by subclasses to traverse the child node.

The important function updateChild() is to update a child node. There are four types of updates:

  • newWidgetIs empty, the oldWidgetAlso be empty. Do nothing.
  • newWidgetIs empty, the oldWidgetDon’t be empty. thisElementHas been removed.
  • newWidgetNot empty, oldWidgetIs empty. The callinflateWidget()With thisWigetInstantiate one for the configurationElement.
  • newWidgetNot empty, oldWidgetDon’t be empty. callupdate()Function updaterElement.update()Functions are implemented by subclasses.

When a new Element is instantiated, mount() is called to add it to the Element Tree. To remove, unmount() is called.

The function markNeedsBuild() marks Element as dirty. Indicates that the Element needs to be rebuilt for the next frame.

The rebuild() function is called during the build phase of the rendering pipeline. The concrete rebuild is in the function performRebuild(), which is implemented by the Element subclass.

Widgets have some important subclasses, and the corresponding Element has some important subclasses.

ComponentElement

ComponentElement indicates that the current Element is used to compose other elements.

abstract class ComponentElement extends Element {
  ComponentElement(Widget widget) : super(widget);

  Element _child;

  @override
  void performRebuild() {
    Widget built;
    built = build();
    _child = updateChild(_child, built, slot);
  }

  Widget build();
}
Copy the code

ComponentElement inherits from Element. Is an abstract class. _child is its child. Build () is called in the performRebuild() function to instantiate a Widget. The build() function is implemented by its subclasses.

StatelessElement

The Widget corresponding to StatelessElement is the familiar StatelessWidget.

class StatelessElement extends ComponentElement {

  @override
  Widget build() => widget.build(this);

  @override
  void update(StatelessWidget newWidget) {
    super.update(newWidget);
    _dirty = true; rebuild(); }}Copy the code

The build() function directly calls statelessWidget.build (). Now you know where your build() is called in the StatelessWidget. And, as you can see, the build() function takes this. We all know that the input to this function should be of type BuildContext. This entry is actually this StatelessElement.

StatefulElement

The Widget corresponding to the StatefulElement is the familiar StatefulWidget.

class StatefulElement extends ComponentElement {
  /// Creates an element that uses the given widget as its configuration.
  StatefulElement(StatefulWidget widget)
      : _state = widget.createState(),
        super(widget) {
    _state._element = this;
    _state._widget = widget;
  }

  @override
  Widget build() => state.build(this);
  
   @override
  void _firstBuild() {
    final dynamic debugCheckForReturnedFuture = _state.initState() 
    _state.didChangeDependencies();
    super._firstBuild();
  }

  @override
  void deactivate() {
    _state.deactivate();
    super.deactivate();
  }

  @override
  void unmount() {
    super.unmount();
    _state.dispose();
    _state._element = null;
    _state = null;
  }

  @override
  void didChangeDependencies() {
    super.didChangeDependencies(); _state.didChangeDependencies(); }}Copy the code

The createState() function corresponding to the StatefulWidget is called during StatefulElement construction. That is, State is instantiated when a StatefulElement is instantiated. And the State instance will be held by the StatefulElement instance. You can also see why the State of a StatefulWidget is managed by a separate State. A new StatefulWidget may be created on each refresh, but the State instance remains the same.

The build() function calls the familiar state.build (this), and now you know where the State build() function is called. And, as you can see, the build() function takes this. We all know that the input to this function should be of type BuildContext. This entry is actually the StatefulElement.

We all know that State is stateful, and the corresponding callback function is called when the State changes. These callbacks are actually called from the StatefulElement.

In function _firstBuild () will call in the State. The initState () and the State. The didChangeDependencies ().

State.deactivate() is called in the deactivate() function.

State-dispose () is called in the unmount() function.

In function didChangeDependencies () will be called the State. The didChangeDependencies ().

InheritedElement

The InheritedElement’s corresponding Widget is the InheritedWidget. The internal implementation maintains a Map of the child elements that depend on it, and calls the didChangeDependencies() callback if needed.

RenderObjectElement

The Widget corresponding to RenderObjectElement is RenderObjectWidget.

abstract class RenderObjectElement extends Element {
  RenderObject _renderObject;
  
  @override
  void mount(Element parent, dynamic newSlot) {
    super.mount(parent, newSlot);
    _renderObject = widget.createRenderObject(this);
    attachRenderObject(newSlot);
    _dirty = false;
  }
  
  @override
  void unmount() {
    super.unmount();
    widget.didUnmountRenderObject(renderObject);
  }
  
  @override
  void update(covariant RenderObjectWidget newWidget) {
    super.update(newWidget);
    widget.updateRenderObject(this, renderObject);
    _dirty = false;
  }
  
  @override
  void performRebuild() {
    widget.updateRenderObject(this, renderObject);
    _dirty = false;
  }
  
  @protected
  void insertChildRenderObject(covariant RenderObject child, covariant dynamic slot);

  @protected
  void moveChildRenderObject(covariant RenderObject child, covariant dynamic slot);

  @protected
  void removeChildRenderObject(covariant RenderObject child);

}
Copy the code

Functions are called mount () will call RenderObjectWidget. CreateRenderObject () to instantiate RenderObject.

Function update () and performRebuild () is invoked when invoked RenderObjectWidget. UpdateRenderObject ().

Function unmount to () is invoked when invoked RenderObjectWidget. DidUnmountRenderObject ().

RenderObject

RenderObject is responsible for the layout and paint phases of the rendering pipeline. Also maintain render Tree. The render Tree maintenance comes from the base class AbstractNode. Here we focus on some of the methods associated with the rendering pipeline.

abstract class RenderObject extends AbstractNode with DiagnosticableTreeMixin implements HitTestTarget {

  void markNeedsLayout() {
      ...
  }
  
  void markNeedsPaint() {
      ...
  }
  
  void layout(Constraints constraints, { bool parentUsesSize = false{...})if(sizedByParent) { performResize(); }... performLayout(); . }void performResize();
  
  void performLayout();
  
  void paint(PaintingContext context, Offset offset) { }
}
Copy the code

MarkNeedsLayout () marks that the RenderObject needs to be relaid. MarkNeedsPaint marks that the RenderObject needs to be redrawn. These two functions are just labeled. The Flutter framework schedules a frame after the tag, and the actual layout and drawing is not done until the next Vsync signal arrives.

The actual layout takes place in the function Layout (). This function is going to make a judgment if sizedByParent is true. PerformResize () is called. Indicates that the size of the RenderObject is determined only by its parent. PerformLayout () is then called to do the layout. Both performResize() and performLayout() require RenderObject subclasses to implement. `

conclusion

Widgets, Elements and RenderObject systems are at the heart of the Flutter framework. Element needs to be understood. The build phase of Flutter’s rendering pipeline is mainly about maintaining and updating element nodes in the Element Tree. Only by understanding Element and Element Tree can you truly master the Flutter framework. There are only static instructions in this article. In the next article I will try to analyze how the Flutter framework works from the perspective of the dynamic operation of the rendering pipeline.