node.dart 4.17 KB
Newer Older
1 2 3 4
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

Adam Barth's avatar
Adam Barth committed
5 6 7 8
/// An abstract node in a tree
///
/// AbstractNode has as notion of depth, attachment, and parent, but does not
/// have a model for children.
Adam Barth's avatar
Adam Barth committed
9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
///
/// * When a subclass is changing the parent of a child, it should
///   call either parent.adoptChild(child) or parent.dropChild(child)
///   as appropriate. Subclasses should expose an API for
///   manipulating the tree if you want to (e.g. a setter for a
///   'child' property, or an 'add()' method to manipulate a list).
///
/// * You can see the current parent by querying 'parent'.
///
/// * You can see the current attachment state by querying
///   'attached'. The root of any tree that is to be considered
///   attached should be manually attached by calling 'attach()'.
///   Other than that, don't call 'attach()' or 'detach()'. This is
///   all managed automatically assuming you call the 'adoptChild()'
///   and 'dropChild()' methods appropriately.
///
/// * Subclasses that have children must override 'attach()' and
///   'detach()' as described below.
///
/// * Nodes always have a 'depth' greater than their ancestors'.
///   There's no guarantee regarding depth between siblings. The
///   depth of a node is used to ensure that nodes are processed in
///   depth order. The 'depth' of a child can be more than one
///   greater than the 'depth' of the parent, because the 'depth'
///   values are never decreased: all that matters is that it's
///   greater than the parent. Consider a tree with a root node A, a
///   child B, and a grandchild C. Initially, A will have 'depth' 0,
///   B 'depth' 1, and C 'depth' 2. If C is moved to be a child of A,
///   sibling of B, then the numbers won't change. C's 'depth' will
///   still be 2. This is all managed automatically assuming you call
///   'adoptChild()' and 'dropChild()' appropriately.
40 41 42 43 44 45
class AbstractNode {

  // AbstractNode represents a node in a tree.
  // The AbstractNode protocol is described in README.md.

  int _depth = 0;
Adam Barth's avatar
Adam Barth committed
46 47 48 49
  /// The depth of this node in the tree.
  ///
  /// The depth of nodes in a tree monotonically increases as you traverse down
  /// the trees.
50
  int get depth => _depth;
Adam Barth's avatar
Adam Barth committed
51 52 53

  /// Call only from overrides of [redepthChildren]
  void redepthChild(AbstractNode child) {
54
    assert(child.owner == owner);
55 56 57 58 59
    if (child._depth <= _depth) {
      child._depth = _depth + 1;
      child.redepthChildren();
    }
  }
Adam Barth's avatar
Adam Barth committed
60 61 62 63

  /// Override this function in subclasses with child nodes to call
  /// redepthChild(child) for each child. Do not call directly.
  void redepthChildren() { }
64

65 66 67 68
  Object _owner;
  /// The owner for this node (null if unattached).
  Object get owner => _owner;

Adam Barth's avatar
Adam Barth committed
69
  /// Whether this node is in a tree whose root is attached to something.
70
  bool get attached => _owner != null;
Adam Barth's avatar
Adam Barth committed
71

72
  /// Mark this node as attached to the given owner.
Adam Barth's avatar
Adam Barth committed
73
  ///
74 75
  /// Typically called only from the parent's attach(), and to mark the root of
  /// a tree attached.
76 77 78 79
  void attach(Object owner) {
    assert(owner != null);
    assert(_owner == null);
    _owner = owner;
80
  }
Adam Barth's avatar
Adam Barth committed
81 82 83

  /// Mark this node as detached.
  ///
84 85
  /// Typically called only from the parent's detach(), and to mark the root of
  /// a tree detached.
86
  void detach() {
87 88
    assert(_owner != null);
    _owner = null;
89
  }
Adam Barth's avatar
Adam Barth committed
90

91
  AbstractNode _parent;
Adam Barth's avatar
Adam Barth committed
92
  /// The parent of this node in the tree.
93
  AbstractNode get parent => _parent;
Adam Barth's avatar
Adam Barth committed
94 95 96

  /// Subclasses should call this function when they acquire a new child.
  void adoptChild(AbstractNode child) {
97 98
    assert(child != null);
    assert(child._parent == null);
99 100 101 102 103 104 105
    assert(() {
      AbstractNode node = this;
      while (node.parent != null)
        node = node.parent;
      assert(node != child); // indicates we are about to create a cycle
      return true;
    });
106 107
    child._parent = this;
    if (attached)
108
      child.attach(_owner);
109 110
    redepthChild(child);
  }
Adam Barth's avatar
Adam Barth committed
111 112 113

  /// Subclasses should call this function when they lose a child.
  void dropChild(AbstractNode child) {
114 115 116 117 118 119 120 121 122
    assert(child != null);
    assert(child._parent == this);
    assert(child.attached == attached);
    child._parent = null;
    if (attached)
      child.detach();
  }

}