loro-internal 1.12.0

Loro internal library. Do not use it directly as it's not stable.
Documentation
use std::collections::BTreeMap;

use loro_common::{ContainerID, ID};

use crate::{container::idx::ContainerIdx, event::InternalDiff, OpLog};

use super::{DiffCalcVersionInfo, DiffCalculatorTrait, DiffMode};

#[derive(Debug)]
pub(crate) struct CounterDiffCalculator {
    ops: BTreeMap<ID, f64>,
}

impl CounterDiffCalculator {
    pub(crate) fn new(_idx: ContainerIdx) -> Self {
        Self {
            ops: BTreeMap::new(),
        }
    }
}

impl DiffCalculatorTrait for CounterDiffCalculator {
    fn start_tracking(&mut self, _oplog: &OpLog, _vv: &crate::VersionVector, _mode: DiffMode) {}

    fn apply_change(
        &mut self,
        _oplog: &OpLog,
        op: crate::op::RichOp,
        _vv: Option<&crate::VersionVector>,
    ) {
        let id = op.id();
        self.ops.insert(
            id,
            *op.op().content.as_future().unwrap().as_counter().unwrap(),
        );
    }

    fn finish_this_round(&mut self) {}

    fn calculate_diff(
        &mut self,
        _idx: ContainerIdx,
        _oplog: &OpLog,
        info: DiffCalcVersionInfo,
        _on_new_container: impl FnMut(&ContainerID),
    ) -> (InternalDiff, DiffMode) {
        let mut diff = 0.;
        let (b, a) = info.from_vv.diff_iter(info.to_vv);

        for sub in b {
            for (_, c) in self.ops.range(sub.norm_id_start()..sub.norm_id_end()) {
                diff -= c;
            }
        }
        for sub in a {
            for (_, c) in self.ops.range(sub.norm_id_start()..sub.norm_id_end()) {
                diff += c;
            }
        }

        (InternalDiff::Counter(diff), DiffMode::Linear)
    }
}