summaryrefslogtreecommitdiffstats
path: root/sc/inc/fstalgorithm.hxx
diff options
context:
space:
mode:
Diffstat (limited to 'sc/inc/fstalgorithm.hxx')
-rw-r--r--sc/inc/fstalgorithm.hxx131
1 files changed, 131 insertions, 0 deletions
diff --git a/sc/inc/fstalgorithm.hxx b/sc/inc/fstalgorithm.hxx
new file mode 100644
index 000000000..f3f189184
--- /dev/null
+++ b/sc/inc/fstalgorithm.hxx
@@ -0,0 +1,131 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ */
+
+#pragma once
+
+#include <mdds/flat_segment_tree.hpp>
+#include <vector>
+
+namespace sc {
+
+template<typename Key, typename Span>
+void buildSpan(
+ std::vector<Span>& rSpans,
+ typename mdds::flat_segment_tree<Key,bool>::const_iterator it,
+ typename mdds::flat_segment_tree<Key,bool>::const_iterator itEnd, const Key* pStart )
+{
+ Key nLastPos = it->first;
+ bool bLastVal = it->second;
+ for (++it; it != itEnd; ++it)
+ {
+ Key nThisPos = it->first;
+ bool bThisVal = it->second;
+
+ if (bLastVal)
+ {
+ Key nIndex1 = nLastPos;
+ Key nIndex2 = nThisPos-1;
+
+ if (!pStart || *pStart < nIndex1)
+ rSpans.push_back(Span(nIndex1, nIndex2));
+ else if (*pStart <= nIndex2)
+ rSpans.push_back(Span(*pStart, nIndex2));
+ }
+
+ nLastPos = nThisPos;
+ bLastVal = bThisVal;
+ }
+}
+
+template<typename Key, typename Val, typename Span>
+void buildSpanWithValue(
+ std::vector<Span>& rSpans,
+ typename mdds::flat_segment_tree<Key,Val>::const_iterator it,
+ typename mdds::flat_segment_tree<Key,Val>::const_iterator itEnd )
+{
+ Key nLastPos = it->first;
+ Val nLastVal = it->second;
+ for (++it; it != itEnd; ++it)
+ {
+ Key nThisPos = it->first;
+ Val nThisVal = it->second;
+
+ if (nLastVal)
+ {
+ Key nIndex1 = nLastPos;
+ Key nIndex2 = nThisPos-1;
+ rSpans.push_back(Span(nIndex1, nIndex2, nLastVal));
+ }
+
+ nLastPos = nThisPos;
+ nLastVal = nThisVal;
+ }
+}
+
+/**
+ * Convert a flat_segment_tree structure whose value type is boolean, into
+ * an array of ranges that corresponds with the segments that have a 'true'
+ * value.
+ */
+template<typename Key, typename Span>
+std::vector<Span> toSpanArray( const mdds::flat_segment_tree<Key,bool>& rTree )
+{
+ typedef mdds::flat_segment_tree<Key,bool> FstType;
+
+ std::vector<Span> aSpans;
+
+ typename FstType::const_iterator it = rTree.begin(), itEnd = rTree.end();
+ buildSpan<Key,Span>(aSpans, it, itEnd, nullptr);
+ return aSpans;
+}
+
+/**
+ * Convert a flat_segment_tree structure into an array of ranges with
+ * values. Only those ranges whose value is evaluated to be true will be
+ * included. The value type must be something that supports bool operator.
+ * The span type must support a constructor that takes a start key, an end
+ * key and a value in this order.
+ */
+template<typename Key, typename Val, typename Span>
+std::vector<Span> toSpanArrayWithValue( const mdds::flat_segment_tree<Key,Val>& rTree )
+{
+ typedef mdds::flat_segment_tree<Key,Val> FstType;
+
+ std::vector<Span> aSpans;
+
+ typename FstType::const_iterator it = rTree.begin(), itEnd = rTree.end();
+ buildSpanWithValue<Key,Val,Span>(aSpans, it, itEnd);
+ return aSpans;
+}
+
+template<typename Key, typename Span>
+std::vector<Span> toSpanArray( const mdds::flat_segment_tree<Key,bool>& rTree, Key nStartPos )
+{
+ typedef mdds::flat_segment_tree<Key,bool> FstType;
+
+ std::vector<Span> aSpans;
+ if (!rTree.is_tree_valid())
+ return aSpans;
+
+ bool bThisVal = false;
+ std::pair<typename FstType::const_iterator, bool> r =
+ rTree.search_tree(nStartPos, bThisVal);
+
+ if (!r.second)
+ // Tree search failed.
+ return aSpans;
+
+ typename FstType::const_iterator it = r.first, itEnd = rTree.end();
+ buildSpan<Key,Span>(aSpans, it, itEnd, &nStartPos);
+ return aSpans;
+}
+
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */