# include "tools/cabana/util.h"
# include <QFontDatabase>
# include <QHelpEvent>
# include <QLocale>
# include <QPainter>
# include <QPixmapCache>
# include <QToolTip>
# include "selfdrive/ui/qt/util.h"
// SegmentTree
void SegmentTree : : build ( const QVector < QPointF > & arr ) {
size = arr . size ( ) ;
tree . resize ( 4 * size ) ; // size of the tree is 4 times the size of the array
if ( size > 0 ) {
build_tree ( arr , 1 , 0 , size - 1 ) ;
}
}
void SegmentTree : : build_tree ( const QVector < QPointF > & arr , int n , int left , int right ) {
if ( left = = right ) {
const double y = arr [ left ] . y ( ) ;
tree [ n ] = { y , y } ;
} else {
const int mid = ( left + right ) > > 1 ;
build_tree ( arr , 2 * n , left , mid ) ;
build_tree ( arr , 2 * n + 1 , mid + 1 , right ) ;
tree [ n ] = { std : : min ( tree [ 2 * n ] . first , tree [ 2 * n + 1 ] . first ) , std : : max ( tree [ 2 * n ] . second , tree [ 2 * n + 1 ] . second ) } ;
}
}
std : : pair < double , double > SegmentTree : : get_minmax ( int n , int left , int right , int range_left , int range_right ) const {
if ( range_left > right | | range_right < left )
return { std : : numeric_limits < double > : : max ( ) , std : : numeric_limits < double > : : lowest ( ) } ;
if ( range_left < = left & & range_right > = right )
return tree [ n ] ;
int mid = ( left + right ) > > 1 ;
auto l = get_minmax ( 2 * n , left , mid , range_left , range_right ) ;
auto r = get_minmax ( 2 * n + 1 , mid + 1 , right , range_left , range_right ) ;
return { std : : min ( l . first , r . first ) , std : : max ( l . second , r . second ) } ;
}
// MessageBytesDelegate
MessageBytesDelegate : : MessageBytesDelegate ( QObject * parent , bool multiple_lines ) : multiple_lines ( multiple_lines ) , QStyledItemDelegate ( parent ) {
fixed_font = QFontDatabase : : systemFont ( QFontDatabase : : FixedFont ) ;
byte_size = QFontMetrics ( fixed_font ) . size ( Qt : : TextSingleLine , " 00 " ) + QSize ( 0 , 2 ) ;
}
void MessageBytesDelegate : : setMultipleLines ( bool v ) {
if ( std : : exchange ( multiple_lines , v ) ! = multiple_lines ) {
std : : fill_n ( size_cache , std : : size ( size_cache ) , QSize { } ) ;
}
}
int MessageBytesDelegate : : widthForBytes ( int n ) const {
int h_margin = QApplication : : style ( ) - > pixelMetric ( QStyle : : PM_FocusFrameHMargin ) + 1 ;
return n * byte_size . width ( ) + h_margin * 2 ;
}
QSize MessageBytesDelegate : : sizeHint ( const QStyleOptionViewItem & option , const QModelIndex & index ) const {
int v_margin = QApplication : : style ( ) - > pixelMetric ( QStyle : : PM_FocusFrameVMargin ) + 1 ;
auto data = index . data ( BytesRole ) ;
if ( ! data . isValid ( ) ) {
return { 1 , byte_size . height ( ) + 2 * v_margin } ;
}
int n = data . toByteArray ( ) . size ( ) ;
assert ( n > = 0 & & n < = 64 ) ;
QSize size = size_cache [ n ] ;
if ( size . isEmpty ( ) ) {
if ( ! multiple_lines ) {
size . setWidth ( widthForBytes ( n ) ) ;
size . setHeight ( byte_size . height ( ) + 2 * v_margin ) ;
} else {
size . setWidth ( widthForBytes ( 8 ) ) ;
size . setHeight ( byte_size . height ( ) * std : : max ( 1 , n / 8 ) + 2 * v_margin ) ;
}
size_cache [ n ] = size ;
}
return size ;
}
void MessageBytesDelegate : : paint ( QPainter * painter , const QStyleOptionViewItem & option , const QModelIndex & index ) const {
auto data = index . data ( BytesRole ) ;
if ( ! data . isValid ( ) ) {
return QStyledItemDelegate : : paint ( painter , option , index ) ;
}
auto byte_list = data . toByteArray ( ) ;
auto colors = index . data ( ColorsRole ) . value < QVector < QColor > > ( ) ;
int v_margin = option . widget - > style ( ) - > pixelMetric ( QStyle : : PM_FocusFrameVMargin ) ;
int h_margin = option . widget - > style ( ) - > pixelMetric ( QStyle : : PM_FocusFrameHMargin ) ;
if ( option . state & QStyle : : State_Selected ) {
painter - > fillRect ( option . rect , option . palette . highlight ( ) ) ;
}
const QPoint pt { option . rect . left ( ) + h_margin , option . rect . top ( ) + v_margin } ;
QFont old_font = painter - > font ( ) ;
QPen old_pen = painter - > pen ( ) ;
painter - > setFont ( fixed_font ) ;
for ( int i = 0 ; i < byte_list . size ( ) ; + + i ) {
int row = ! multiple_lines ? 0 : i / 8 ;
int column = ! multiple_lines ? i : i % 8 ;
QRect r = QRect ( { pt . x ( ) + column * byte_size . width ( ) , pt . y ( ) + row * byte_size . height ( ) } , byte_size ) ;
if ( i < colors . size ( ) & & colors [ i ] . alpha ( ) > 0 ) {
if ( option . state & QStyle : : State_Selected ) {
painter - > setPen ( option . palette . color ( QPalette : : Text ) ) ;
painter - > fillRect ( r , option . palette . color ( QPalette : : Window ) ) ;
}
painter - > fillRect ( r , colors [ i ] ) ;
} else if ( option . state & QStyle : : State_Selected ) {
painter - > setPen ( option . palette . color ( QPalette : : HighlightedText ) ) ;
}
painter - > drawText ( r , Qt : : AlignCenter , toHex ( byte_list [ i ] ) ) ;
}
painter - > setFont ( old_font ) ;
painter - > setPen ( old_pen ) ;
}
// TabBar
int TabBar : : addTab ( const QString & text ) {
int index = QTabBar : : addTab ( text ) ;
QToolButton * btn = new ToolButton ( " x " , tr ( " Close Tab " ) ) ;
int width = style ( ) - > pixelMetric ( QStyle : : PM_TabCloseIndicatorWidth , nullptr , btn ) ;
int height = style ( ) - > pixelMetric ( QStyle : : PM_TabCloseIndicatorHeight , nullptr , btn ) ;
btn - > setFixedSize ( { width , height } ) ;
setTabButton ( index , QTabBar : : RightSide , btn ) ;
QObject : : connect ( btn , & QToolButton : : clicked , this , & TabBar : : closeTabClicked ) ;
return index ;
}
void TabBar : : closeTabClicked ( ) {
QObject * object = sender ( ) ;
for ( int i = 0 ; i < count ( ) ; + + i ) {
if ( tabButton ( i , QTabBar : : RightSide ) = = object ) {
emit tabCloseRequested ( i ) ;
break ;
}
}
}
QColor getColor ( const cabana : : Signal * sig ) {
float h = 19 * ( float ) sig - > lsb / 64.0 ;
h = fmod ( h , 1.0 ) ;
size_t hash = qHash ( sig - > name ) ;
float s = 0.25 + 0.25 * ( float ) ( hash & 0xff ) / 255.0 ;
float v = 0.75 + 0.25 * ( float ) ( ( hash > > 8 ) & 0xff ) / 255.0 ;
return QColor : : fromHsvF ( h , s , v ) ;
}
NameValidator : : NameValidator ( QObject * parent ) : QRegExpValidator ( QRegExp ( " ^( \\ w+) " ) , parent ) { }
QValidator : : State NameValidator : : validate ( QString & input , int & pos ) const {
input . replace ( ' ' , ' _ ' ) ;
return QRegExpValidator : : validate ( input , pos ) ;
}
DoubleValidator : : DoubleValidator ( QObject * parent ) : QDoubleValidator ( parent ) {
// Match locale of QString::toDouble() instead of system
QLocale locale ( QLocale : : C ) ;
locale . setNumberOptions ( QLocale : : RejectGroupSeparator ) ;
setLocale ( locale ) ;
}
namespace utils {
QPixmap icon ( const QString & id ) {
bool dark_theme = settings . theme = = DARK_THEME ;
QPixmap pm ;
QString key = " bootstrap_ " % id % ( dark_theme ? " 1 " : " 0 " ) ;
if ( ! QPixmapCache : : find ( key , & pm ) ) {
pm = bootstrapPixmap ( id ) ;
if ( dark_theme ) {
QPainter p ( & pm ) ;
p . setCompositionMode ( QPainter : : CompositionMode_SourceIn ) ;
p . fillRect ( pm . rect ( ) , QColor ( " #bbbbbb " ) ) ;
}
QPixmapCache : : insert ( key , pm ) ;
}
return pm ;
}
void setTheme ( int theme ) {
auto style = QApplication : : style ( ) ;
if ( ! style ) return ;
static int prev_theme = 0 ;
if ( theme ! = prev_theme ) {
prev_theme = theme ;
QPalette new_palette ;
if ( theme = = DARK_THEME ) {
// "Darcula" like dark theme
new_palette . setColor ( QPalette : : Window , QColor ( " #353535 " ) ) ;
new_palette . setColor ( QPalette : : WindowText , QColor ( " #bbbbbb " ) ) ;
new_palette . setColor ( QPalette : : Base , QColor ( " #3c3f41 " ) ) ;
new_palette . setColor ( QPalette : : AlternateBase , QColor ( " #3c3f41 " ) ) ;
new_palette . setColor ( QPalette : : ToolTipBase , QColor ( " #3c3f41 " ) ) ;
new_palette . setColor ( QPalette : : ToolTipText , QColor ( " #bbb " ) ) ;
new_palette . setColor ( QPalette : : Text , QColor ( " #bbbbbb " ) ) ;
new_palette . setColor ( QPalette : : Button , QColor ( " #3c3f41 " ) ) ;
new_palette . setColor ( QPalette : : ButtonText , QColor ( " #bbbbbb " ) ) ;
new_palette . setColor ( QPalette : : Highlight , QColor ( " #2f65ca " ) ) ;
new_palette . setColor ( QPalette : : HighlightedText , QColor ( " #bbbbbb " ) ) ;
new_palette . setColor ( QPalette : : BrightText , QColor ( " #f0f0f0 " ) ) ;
new_palette . setColor ( QPalette : : Disabled , QPalette : : ButtonText , QColor ( " #777777 " ) ) ;
new_palette . setColor ( QPalette : : Disabled , QPalette : : WindowText , QColor ( " #777777 " ) ) ;
new_palette . setColor ( QPalette : : Disabled , QPalette : : Text , QColor ( " #777777 " ) ) ; ;
new_palette . setColor ( QPalette : : Light , QColor ( " #777777 " ) ) ;
new_palette . setColor ( QPalette : : Dark , QColor ( " #353535 " ) ) ;
} else {
new_palette = style - > standardPalette ( ) ;
}
qApp - > setPalette ( new_palette ) ;
style - > polish ( qApp ) ;
for ( auto w : QApplication : : allWidgets ( ) ) {
w - > setPalette ( new_palette ) ;
}
}
}
} // namespace utils
QString toHex ( uint8_t byte ) {
static std : : array < QString , 256 > hex = [ ] ( ) {
std : : array < QString , 256 > ret ;
for ( int i = 0 ; i < 256 ; + + i ) ret [ i ] = QStringLiteral ( " %1 " ) . arg ( i , 2 , 16 , QLatin1Char ( ' 0 ' ) ) . toUpper ( ) ;
return ret ;
} ( ) ;
return hex [ byte ] ;
}
int num_decimals ( double num ) {
const QString string = QString : : number ( num ) ;
const QStringList split = string . split ( ' . ' ) ;
if ( split . size ( ) = = 1 ) {
return 0 ;
} else {
return split [ 1 ] . size ( ) ;
}
}