2016-06-14 03:30:03 +08:00
|
|
|
// Aseprite Document Library
|
2018-03-16 19:59:34 +08:00
|
|
|
// Copyright (c) 2016-2018 David Capello
|
2016-06-14 03:30:03 +08:00
|
|
|
//
|
|
|
|
// This file is released under the terms of the MIT license.
|
|
|
|
// Read LICENSE.txt for more information.
|
|
|
|
|
|
|
|
#ifndef DOC_SELECTED_FRAMES_H_INCLUDED
|
|
|
|
#define DOC_SELECTED_FRAMES_H_INCLUDED
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "doc/frame_range.h"
|
|
|
|
|
2018-06-09 01:40:02 +08:00
|
|
|
#include <iosfwd>
|
2016-06-14 03:30:03 +08:00
|
|
|
#include <iterator>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace doc {
|
|
|
|
|
|
|
|
class SelectedFrames {
|
|
|
|
typedef std::vector<FrameRange> Ranges;
|
|
|
|
|
|
|
|
public:
|
|
|
|
class const_iterator : public std::iterator<std::forward_iterator_tag, frame_t> {
|
2018-03-16 19:59:34 +08:00
|
|
|
static const int kNullFrame = -2;
|
2016-06-14 03:30:03 +08:00
|
|
|
public:
|
|
|
|
const_iterator(const Ranges::const_iterator& it)
|
2018-03-16 19:59:34 +08:00
|
|
|
: m_it(it), m_frame(kNullFrame) {
|
2016-06-14 03:30:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const_iterator& operator++() {
|
2018-03-16 19:59:34 +08:00
|
|
|
if (m_frame == kNullFrame)
|
2016-06-14 03:30:03 +08:00
|
|
|
m_frame = m_it->fromFrame;
|
|
|
|
|
2018-03-16 19:59:34 +08:00
|
|
|
if (m_it->fromFrame <= m_it->toFrame) {
|
|
|
|
if (m_frame < m_it->toFrame)
|
|
|
|
++m_frame;
|
|
|
|
else {
|
|
|
|
m_frame = kNullFrame;
|
|
|
|
++m_it;
|
|
|
|
}
|
|
|
|
}
|
2016-06-14 03:30:03 +08:00
|
|
|
else {
|
2018-03-16 19:59:34 +08:00
|
|
|
if (m_frame > m_it->toFrame)
|
|
|
|
--m_frame;
|
|
|
|
else {
|
|
|
|
m_frame = kNullFrame;
|
|
|
|
++m_it;
|
|
|
|
}
|
2016-06-14 03:30:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
frame_t operator*() const {
|
2018-03-16 19:59:34 +08:00
|
|
|
if (m_frame == kNullFrame)
|
2016-06-14 03:30:03 +08:00
|
|
|
m_frame = m_it->fromFrame;
|
|
|
|
return m_frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const const_iterator& o) const {
|
|
|
|
return (m_it == o.m_it && m_frame == o.m_frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const const_iterator& it) const {
|
|
|
|
return !operator==(it);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
mutable Ranges::const_iterator m_it;
|
|
|
|
mutable frame_t m_frame;
|
|
|
|
};
|
|
|
|
|
2016-06-14 04:47:42 +08:00
|
|
|
class const_reverse_iterator : public std::iterator<std::forward_iterator_tag, frame_t> {
|
|
|
|
public:
|
|
|
|
const_reverse_iterator(const Ranges::const_reverse_iterator& it)
|
|
|
|
: m_it(it), m_frame(-1) {
|
|
|
|
}
|
|
|
|
|
|
|
|
const_reverse_iterator& operator++() {
|
|
|
|
if (m_frame < 0)
|
|
|
|
m_frame = m_it->toFrame;
|
|
|
|
|
|
|
|
if (m_frame > m_it->fromFrame)
|
|
|
|
--m_frame;
|
|
|
|
else {
|
|
|
|
m_frame = -1;
|
|
|
|
++m_it;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
frame_t operator*() const {
|
|
|
|
if (m_frame < 0)
|
|
|
|
m_frame = m_it->toFrame;
|
|
|
|
return m_frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const const_reverse_iterator& o) const {
|
|
|
|
return (m_it == o.m_it && m_frame == o.m_frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const const_reverse_iterator& it) const {
|
|
|
|
return !operator==(it);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
mutable Ranges::const_reverse_iterator m_it;
|
|
|
|
mutable frame_t m_frame;
|
|
|
|
};
|
|
|
|
|
2016-06-14 22:14:24 +08:00
|
|
|
class Reversed {
|
|
|
|
public:
|
|
|
|
typedef const_reverse_iterator const_iterator;
|
|
|
|
|
|
|
|
const_iterator begin() const { return m_selectedFrames.rbegin(); }
|
|
|
|
const_iterator end() const { return m_selectedFrames.rend(); }
|
|
|
|
|
|
|
|
Reversed(const SelectedFrames& selectedFrames)
|
|
|
|
: m_selectedFrames(selectedFrames) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const SelectedFrames& m_selectedFrames;
|
|
|
|
};
|
|
|
|
|
2016-06-14 03:30:03 +08:00
|
|
|
const_iterator begin() const { return const_iterator(m_ranges.begin()); }
|
|
|
|
const_iterator end() const { return const_iterator(m_ranges.end()); }
|
2016-06-14 04:47:42 +08:00
|
|
|
const_reverse_iterator rbegin() const { return const_reverse_iterator(m_ranges.rbegin()); }
|
|
|
|
const_reverse_iterator rend() const { return const_reverse_iterator(m_ranges.rend()); }
|
2016-06-14 03:30:03 +08:00
|
|
|
|
2016-06-14 04:47:17 +08:00
|
|
|
std::size_t size() const;
|
2016-06-14 22:05:03 +08:00
|
|
|
std::size_t ranges() const { return m_ranges.size(); }
|
2016-06-14 04:47:17 +08:00
|
|
|
bool empty() const { return m_ranges.empty(); }
|
|
|
|
|
2016-08-12 00:45:50 +08:00
|
|
|
void clear();
|
2016-06-14 03:30:03 +08:00
|
|
|
void insert(frame_t frame);
|
|
|
|
void insert(frame_t fromFrame, frame_t toFrame);
|
2018-03-16 19:59:34 +08:00
|
|
|
SelectedFrames filter(frame_t fromFrame, frame_t toFrame) const;
|
2016-06-14 03:30:03 +08:00
|
|
|
|
|
|
|
bool contains(frame_t frame) const;
|
|
|
|
|
2016-06-14 22:05:03 +08:00
|
|
|
frame_t firstFrame() const { return (!m_ranges.empty() ? m_ranges.front().fromFrame: -1); }
|
|
|
|
frame_t lastFrame() const { return (!m_ranges.empty() ? m_ranges.back().toFrame: -1); }
|
|
|
|
|
2016-08-13 00:08:22 +08:00
|
|
|
void displace(frame_t frameDelta);
|
2016-06-14 22:14:24 +08:00
|
|
|
Reversed reversed() const { return Reversed(*this); }
|
|
|
|
|
2018-03-16 19:59:34 +08:00
|
|
|
SelectedFrames makeReverse() const;
|
|
|
|
SelectedFrames makePingPong() const;
|
|
|
|
|
2016-08-12 02:39:57 +08:00
|
|
|
bool operator==(const SelectedFrames& o) const {
|
|
|
|
return m_ranges == o.m_ranges;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const SelectedFrames& o) const {
|
|
|
|
return !operator==(o);
|
|
|
|
}
|
|
|
|
|
2018-06-09 01:40:02 +08:00
|
|
|
bool write(std::ostream& os) const;
|
|
|
|
bool read(std::istream& is);
|
|
|
|
|
2016-06-14 03:30:03 +08:00
|
|
|
private:
|
|
|
|
Ranges m_ranges;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace doc
|
|
|
|
|
2016-06-14 22:08:17 +08:00
|
|
|
#endif // DOC_SELECTED_FRAMES_H_INCLUDED
|