diff --git a/Libraries/LibCore/IODevice.cpp b/Libraries/LibCore/IODevice.cpp index 4efda65ee40..453526e4083 100644 --- a/Libraries/LibCore/IODevice.cpp +++ b/Libraries/LibCore/IODevice.cpp @@ -315,4 +315,22 @@ bool IODevice::write(const StringView& v) return write((const u8*)v.characters_without_null_termination(), v.length()); } +LineIterator::LineIterator(IODevice& device, bool is_end) + : m_device(device) + , m_is_end(is_end) +{ + ++*this; +} + +bool LineIterator::at_end() const +{ + return m_device->eof(); +} + +LineIterator& LineIterator::operator++() +{ + m_buffer = m_device->read_line(); + return *this; +} + } diff --git a/Libraries/LibCore/IODevice.h b/Libraries/LibCore/IODevice.h index 6071a4c9126..70a75532a19 100644 --- a/Libraries/LibCore/IODevice.h +++ b/Libraries/LibCore/IODevice.h @@ -31,6 +31,28 @@ namespace Core { +// This is not necessarily a valid iterator in all contexts, +// if we had concepts, this would be InputIterator, not Copyable, Movable. +class LineIterator { + AK_MAKE_NONCOPYABLE(LineIterator); + +public: + explicit LineIterator(IODevice&, bool is_end = false); + + bool operator==(const LineIterator& other) const { return &other == this || (at_end() && other.is_end()) || (other.at_end() && is_end()); } + bool is_end() const { return m_is_end; } + bool at_end() const; + + LineIterator& operator++(); + + StringView operator*() const { return m_buffer; } + +private: + NonnullRefPtr m_device; + bool m_is_end { false }; + String m_buffer; +}; + class IODevice : public Object { C_OBJECT_ABSTRACT(IODevice) public: @@ -84,6 +106,9 @@ public: int printf(const char*, ...); + LineIterator line_begin() & { return LineIterator(*this); } + LineIterator line_end() { return LineIterator(*this, true); } + protected: explicit IODevice(Object* parent = nullptr);