com.raquo.laminar.inserters.InsertContext.scala Maven / Gradle / Ivy
package com.raquo.laminar.inserters
import com.raquo.ew.JsMap
import com.raquo.laminar.DomApi
import com.raquo.laminar.nodes.{ChildNode, CommentNode, ParentNode, ReactiveElement}
import org.scalajs.dom
import scala.collection.immutable
import scala.scalajs.js
// #TODO[Naming] This feels more like InserterState?
// "Extra nodes" are more like "content nodes"
// @Note only parentNode and sentinelNode are used by all Inserter-s.
// - Other fields may remain un-updated if they are not needed for a particular use case.
/**
* InsertContext represents the state of the DOM inside an inserter block like `child <-- ...`,
* `children <-- ...`, `child.text <-- ...`, etc. The data stored in this context is used
* by Laminar to efficiently update the DOM, to detect (and recover from) external changes
* to the DOM, and for other related tasks.
*
* InsertContext is a mutable data structure that is created once for each inserter, and the
* inserter updates it as it processes new data. However, in case of `onMountInsert`, only
* one context is created, and is then reused for all inserters created inside
* `onMountInsert`. This allows for intuitive preservation of DOM state if the element is
* unmounted and mounted again (for example, `onMountInsert(child <-- stream)` will
* keep the last emitted child in the DOM even if the element is unmounted and re-mounted).
*
* #Note: The params that describe `extraNodes` below can get out of sync with the real DOM.
*
* This can happen if an child element is removed from the DOM – either externally, or more
* likely because it was moved from this inserter into another one, and the addition to the
* other inserter was processed before the removal from this inserter is processed (the
* order of these operations depends on the propagation order of the observables feeding
* these two inserters). The Inserter code must account for this and not fail in such cases,
* and must correct the values accordingly on the next observable update.
*
* #Note: The params that describe `extraNodes` below must be kept consistent manually (#Perf)
*
* Inserter "steals" an element from this one just before the
* observable source of this inserter provides a new list of
* children (with the stolen element removed from the list).
*
* @param sentinelNode - A special invisible comment node that tells Laminar where to
* insert the dynamic children, and where to expect previously
* inserted dynamic children.
* @param strictMode - If true, Laminar guarantees that it will keep a dedicated
* sentinel node instead of using the extra node (content node)
* for that purpose. This is needed in order to allow users to
* move an element from one inserter to another, or to externally
* remove some of the elements previously added by an inserter.
* child.text does not need any of that, so for performance it
* does not use strict mode, it replaces the sentinel comment
* node with the subsequent text nodes. Inserters should be able
* to safely switch to their preferred mode when receiving
* context left by the previous inserter in onMountBind.
* @param extraNodeCount - Number of child nodes in addition to the sentinel node.
* Warning: can get out of sync with the real DOM!
* @param extraNodes - Ordered list of child nodes in addition to the sentinel node.
* Warning: can get out of sync with the real DOM!
* @param extraNodesMap - Map of child nodes, for more efficient search
* Warning: can get out of sync with the real DOM!
*/
final class InsertContext(
val parentNode: ReactiveElement.Base,
var sentinelNode: ChildNode.Base,
var strictMode: Boolean,
var extraNodeCount: Int, // This is separate from `extraNodesMap` for performance #TODO[Performance]: Check if this is still relevant with JsMap
var extraNodes: immutable.Seq[ChildNode.Base],
var extraNodesMap: JsMap[dom.Node, ChildNode.Base]
) {
/**
* This method converts the InsertContext from loose mode to strict mode.
* ChildrenInserter and ChildInserter call this when receiving a context from
* ChildTextInserter. This can happen when switching from `child.text <-- ...`
* to e.g. `children <-- ...` inside onMountInsert.
*
* Prerequisite: context must be in loose mode, and in valid state: no extra nodes allowed.
*/
def forceSetStrictMode(): Unit = {
if (strictMode || extraNodeCount != 0) {
// #Note: if extraNodeCount == 0, it is also assumed (but not tested) that extraNodes and extraNodesMap are empty.
throw new Exception(s"forceSetStrictMode invoked when not allowed, inside parent = ${DomApi.debugNodeOuterHtml(parentNode.ref)}")
}
if (extraNodesMap == null) {
// In loose mode, extraNodesMap is likely to be null, so we need to initialize it.
extraNodesMap = new JsMap()
}
if (sentinelNode.ref.isInstanceOf[dom.Comment]) {
// This means there are no content nodes.
// We assume that all extraNode fields are properly zeroed, so there is nothing to do.
} else {
// In loose mode, child content nodes are written to sentinelNode field,
// so there are no extraNodes.
// So, if we find a content node in sentinelNode, we need to reclassify
// it as such for the strict mode, and insert a new sentinel node into the DOM.
val contentNode = sentinelNode
val newSentinelNode = new CommentNode("")
DomApi.insertBefore(
parent = parentNode.ref,
newChild = newSentinelNode.ref,
referenceChild = contentNode.ref
)
// Convert loose mode context values to strict mode context values
sentinelNode = newSentinelNode
extraNodeCount = 1
extraNodes = contentNode :: Nil
extraNodesMap.set(contentNode.ref, contentNode) // we initialized the map above
}
strictMode = true
}
/** #Note: this does NOT update the context to match the DOM. */
def removeOldChildNodesFromDOM(after: ChildNode.Base): Unit = {
var remainingOldExtraNodeCount = extraNodeCount
while (remainingOldExtraNodeCount > 0) {
val prevChildRef = after.ref.nextSibling
if (prevChildRef == null) {
// We expected more previous children to be in the DOM, but we reached the end of the DOM.
// Those children must have been removed from the DOM manually, or moved to a different inserter.
// So, the DOM state is now correct, albeit "for the wrong reasons". All is good. End the loop.
remainingOldExtraNodeCount = 0
} else {
val maybePrevChild = extraNodesMap.get(prevChildRef)
if (maybePrevChild.isEmpty) {
// Similar to the prevChildRef == null case above, we've exhausted the DOM,
// except we stumbled on some unrelated element instead. We only allow external
// removals from the DOM, not external additions in the middle of dynamic children list,
// so this unrelated element is good evidence that there are no more old child nodes
// to be found.
remainingOldExtraNodeCount = 0
} else {
maybePrevChild.foreach { prevChild =>
// @Note: DOM update
ParentNode.removeChild(parent = parentNode, child = prevChild)
remainingOldExtraNodeCount -= 1
}
}
}
}
}
}
object InsertContext {
/** Reserve the spot for when we actually insert real nodes later */
def reserveSpotContext(
parentNode: ReactiveElement.Base,
strictMode: Boolean,
hooks: js.UndefOr[InserterHooks]
): InsertContext = {
val sentinelNode = new CommentNode("")
ParentNode.appendChild(parent = parentNode, child = sentinelNode, hooks)
unsafeMakeReservedSpotContext(
parentNode = parentNode,
sentinelNode = sentinelNode,
strictMode = strictMode
)
}
/** Reserve the spot for when we actually insert real nodes later.
*
* Unsafe: you must make sure yourself that sentinelNode is already
* a child of parentNode in the real DOM.
*
* This method is exposed to help third parties make hydration helpers.
*/
def unsafeMakeReservedSpotContext(
parentNode: ReactiveElement.Base,
sentinelNode: ChildNode.Base,
strictMode: Boolean
): InsertContext = {
// #Warning[Fragile] - We avoid instantiating a JsMap in loose mode, for performance.
// The JsMap is initialized if/when needed, in forceSetStrictMode.
new InsertContext(
parentNode = parentNode,
sentinelNode = sentinelNode,
strictMode = strictMode,
extraNodeCount = 0,
extraNodes = Nil,
extraNodesMap = if (strictMode) new JsMap() else null
)
}
private[laminar] def nodesToMap(nodes: immutable.Seq[ChildNode.Base]): JsMap[dom.Node, ChildNode.Base] = {
val acc = new JsMap[dom.Node, ChildNode.Base]()
nodes.foreach { node =>
acc.set(node.ref, node)
}
acc
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy