192 lines
6.4 KiB
C++
192 lines
6.4 KiB
C++
/***********************************************************************
|
|
* ftermbuffer.cpp - Buffer for virtual terminal strings *
|
|
* *
|
|
* This file is part of the FINAL CUT widget toolkit *
|
|
* *
|
|
* Copyright 2017-2021 Markus Gans *
|
|
* *
|
|
* FINAL CUT is free software; you can redistribute it and/or modify *
|
|
* it under the terms of the GNU Lesser General Public License as *
|
|
* published by the Free Software Foundation; either version 3 of *
|
|
* the License, or (at your option) any later version. *
|
|
* *
|
|
* FINAL CUT is distributed in the hope that it will be useful, but *
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU Lesser General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU Lesser General Public *
|
|
* License along with this program. If not, see *
|
|
* <http://www.gnu.org/licenses/>. *
|
|
***********************************************************************/
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "final/fc.h"
|
|
#include "final/fcolorpair.h"
|
|
#include "final/fstring.h"
|
|
#include "final/fstyle.h"
|
|
#include "final/ftermbuffer.h"
|
|
#include "final/fvterm.h"
|
|
#include "final/ftypes.h"
|
|
|
|
namespace finalcut
|
|
{
|
|
|
|
//----------------------------------------------------------------------
|
|
// class FTermBuffer
|
|
//----------------------------------------------------------------------
|
|
FTermBuffer::~FTermBuffer() noexcept = default; // destructor
|
|
|
|
|
|
// public methods of FTermBuffer
|
|
//----------------------------------------------------------------------
|
|
FString FTermBuffer::toString() const
|
|
{
|
|
std::wstring wide_string{};
|
|
wide_string.reserve(data.size());
|
|
std::for_each ( data.begin()
|
|
, data.end()
|
|
, [&wide_string] (const FChar& fchar)
|
|
{
|
|
for (auto&& ch : fchar.ch)
|
|
{
|
|
if ( ch == L'\0' )
|
|
return;
|
|
else
|
|
wide_string.push_back(ch);
|
|
}
|
|
}
|
|
);
|
|
return wide_string;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
int FTermBuffer::write (const FString& string)
|
|
{
|
|
assert ( ! string.isNull() );
|
|
data.reserve(data.size() + string.getLength());
|
|
const auto last = string.end();
|
|
auto begin = string.begin();
|
|
auto iter = begin;
|
|
int char_width{0};
|
|
|
|
for (auto&& ch : string)
|
|
{
|
|
auto width = getColumnWidth(ch);
|
|
auto wspace = std::iswspace(wint_t(ch));
|
|
|
|
if ( width == 0 && ! wspace ) // zero-width character
|
|
{
|
|
if ( iter == begin)
|
|
++begin;
|
|
|
|
++iter;
|
|
}
|
|
else if ( iter != begin )
|
|
add(begin, iter, char_width);
|
|
|
|
if ( iter == begin && (width > 0 || is7bit(ch)) ) // 1st char
|
|
++iter;
|
|
|
|
if ( width > 0 )
|
|
char_width += width;
|
|
|
|
if ( wspace )
|
|
add(begin, iter, char_width);
|
|
}
|
|
|
|
if ( iter == last )
|
|
add(begin, iter, char_width);
|
|
|
|
return int(string.getLength());
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
int FTermBuffer::write (wchar_t ch)
|
|
{
|
|
FChar nc{FVTerm::getAttribute()}; // next character
|
|
nc.ch[0] = ch;
|
|
addColumnWidth(nc); // add column width
|
|
nc.attr.bit.no_changes = false;
|
|
nc.attr.bit.printed = false;
|
|
data.emplace_back(nc);
|
|
return 1;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FTermBuffer::write (const FStyle& style) const
|
|
{
|
|
Style attr = style.getStyle();
|
|
|
|
if ( attr == Style::None )
|
|
FVTerm::setNormal();
|
|
else
|
|
{
|
|
if ( (attr & Style::Bold) != Style::None ) FVTerm::setBold();
|
|
if ( (attr & Style::Dim) != Style::None ) FVTerm::setDim();
|
|
if ( (attr & Style::Italic) != Style::None ) FVTerm::setItalic();
|
|
if ( (attr & Style::Underline) != Style::None ) FVTerm::setUnderline();
|
|
if ( (attr & Style::Blink) != Style::None ) FVTerm::setBlink();
|
|
if ( (attr & Style::Reverse) != Style::None ) FVTerm::setReverse();
|
|
if ( (attr & Style::Standout) != Style::None ) FVTerm::setStandout();
|
|
if ( (attr & Style::Invisible) != Style::None ) FVTerm::setInvisible();
|
|
if ( (attr & Style::Protected) != Style::None ) FVTerm::setProtected();
|
|
if ( (attr & Style::CrossedOut) != Style::None ) FVTerm::setCrossedOut();
|
|
if ( (attr & Style::DoubleUnderline) != Style::None ) FVTerm::setDoubleUnderline();
|
|
if ( (attr & Style::Transparent) != Style::None ) FVTerm::setTransparent();
|
|
if ( (attr & Style::ColorOverlay) != Style::None ) FVTerm::setColorOverlay();
|
|
if ( (attr & Style::InheritBackground) != Style::None ) FVTerm::setInheritBackground();
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FTermBuffer::write (const FColorPair& pair) const
|
|
{
|
|
FVTerm::setColor(pair.getForegroundColor(), pair.getBackgroundColor());
|
|
}
|
|
|
|
|
|
// private methods of FTermBuffer
|
|
//----------------------------------------------------------------------
|
|
void FTermBuffer::add ( FString::const_iterator& begin
|
|
, FString::const_iterator& end
|
|
, int& char_width )
|
|
{
|
|
if ( begin == end )
|
|
return;
|
|
|
|
FChar nc{FVTerm::getAttribute()}; // next character
|
|
nc.attr.byte[2] = 0;
|
|
nc.attr.byte[3] = 0;
|
|
|
|
if ( char_width == 2 && FTerm::getEncoding() != Encoding::UTF8 )
|
|
{
|
|
nc.ch[0] = '.';
|
|
nc.attr.bit.char_width = 1;
|
|
}
|
|
else
|
|
nc.attr.bit.char_width = char_width & 0x03;
|
|
|
|
std::copy(begin, std::min(end, begin + UNICODE_MAX), nc.ch.begin());
|
|
data.emplace_back(nc);
|
|
begin = end;
|
|
char_width = 0;
|
|
}
|
|
|
|
|
|
// FTermBuffer non-member operators
|
|
//----------------------------------------------------------------------
|
|
FTermBuffer::FCharVector& operator << ( FTermBuffer::FCharVector& term_string
|
|
, const FTermBuffer& buf )
|
|
{
|
|
if ( ! buf.data.empty() )
|
|
term_string.assign(buf.data.begin(), buf.data.end());
|
|
|
|
return term_string;
|
|
}
|
|
|
|
} // namespace finalcut
|