242 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			242 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | #include "displaytext.h" | ||
|  | 
 | ||
|  | #include <QMouseEvent> | ||
|  | #include <QDateTime> | ||
|  | #include <QTextCharFormat> | ||
|  | #include <QTextCursor> | ||
|  | #include <QTextBlock> | ||
|  | #include <QMenu> | ||
|  | #include <QAction> | ||
|  | 
 | ||
|  | #include "qt_helpers.hpp" | ||
|  | 
 | ||
|  | #include "moc_displaytext.cpp" | ||
|  | 
 | ||
|  | DisplayText::DisplayText(QWidget *parent) | ||
|  |   : QTextEdit(parent) | ||
|  |   , erase_action_ {new QAction {tr ("&Erase"), this}} | ||
|  | { | ||
|  |   setReadOnly (true); | ||
|  |   viewport ()->setCursor (Qt::ArrowCursor); | ||
|  |   setWordWrapMode (QTextOption::NoWrap); | ||
|  | 
 | ||
|  |   // max lines to limit heap usage | ||
|  |   document ()->setMaximumBlockCount (5000); | ||
|  | 
 | ||
|  |   // context menu erase action | ||
|  |   setContextMenuPolicy (Qt::CustomContextMenu); | ||
|  |   connect (this, &DisplayText::customContextMenuRequested, [this] (QPoint const& position) { | ||
|  |       auto * menu = createStandardContextMenu (position); | ||
|  |       menu->addAction (erase_action_); | ||
|  |       menu->exec (mapToGlobal (position)); | ||
|  |       delete menu; | ||
|  |     }); | ||
|  |   connect (erase_action_, &QAction::triggered, this, &DisplayText::erase); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::erase () | ||
|  | { | ||
|  |   clear (); | ||
|  |   Q_EMIT erased (); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::setContentFont(QFont const& font) | ||
|  | { | ||
|  |   char_font_ = font; | ||
|  |   selectAll (); | ||
|  |   auto cursor = textCursor (); | ||
|  |   cursor.beginEditBlock (); | ||
|  |   auto char_format = cursor.charFormat (); | ||
|  |   char_format.setFont (char_font_); | ||
|  |   cursor.mergeCharFormat (char_format); | ||
|  |   cursor.clearSelection (); | ||
|  |   cursor.movePosition (QTextCursor::End); | ||
|  | 
 | ||
|  |   // position so viewport scrolled to left | ||
|  |   cursor.movePosition (QTextCursor::Up); | ||
|  |   cursor.movePosition (QTextCursor::StartOfLine); | ||
|  |   cursor.endEditBlock (); | ||
|  | 
 | ||
|  |   setTextCursor (cursor); | ||
|  |   ensureCursorVisible (); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::mouseDoubleClickEvent(QMouseEvent *e) | ||
|  | { | ||
|  |   bool ctrl = (e->modifiers() & Qt::ControlModifier); | ||
|  |   bool alt = (e->modifiers() & Qt::AltModifier); | ||
|  |   emit(selectCallsign(alt,ctrl)); | ||
|  |   QTextEdit::mouseDoubleClickEvent(e); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::insertLineSpacer(QString const& line) | ||
|  | { | ||
|  |   appendText (line, "#d3d3d3"); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::appendText(QString const& text, QColor bg) | ||
|  | { | ||
|  |   auto cursor = textCursor (); | ||
|  |   cursor.movePosition (QTextCursor::End); | ||
|  |   auto block_format = cursor.blockFormat (); | ||
|  |   block_format.setBackground (bg); | ||
|  |   if (0 == cursor.position ()) | ||
|  |     { | ||
|  |       cursor.setBlockFormat (block_format); | ||
|  |       auto char_format = cursor.charFormat (); | ||
|  |       char_format.setFont (char_font_); | ||
|  |       cursor.setCharFormat (char_format); | ||
|  |     } | ||
|  |   else | ||
|  |     { | ||
|  |       cursor.insertBlock (block_format); | ||
|  |     } | ||
|  |   cursor.insertText (text); | ||
|  | 
 | ||
|  |   // position so viewport scrolled to left | ||
|  |   cursor.movePosition (QTextCursor::StartOfLine); | ||
|  |   setTextCursor (cursor); | ||
|  |   ensureCursorVisible (); | ||
|  |   document ()->setMaximumBlockCount (document ()->maximumBlockCount ()); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | QString DisplayText::appendDXCCWorkedB4(QString message, QString const& callsign, QColor * bg, | ||
|  |           LogBook const& logBook, QColor color_CQ, | ||
|  |           QColor color_DXCC, | ||
|  |           QColor color_NewCall) | ||
|  | { | ||
|  |   // allow for seconds | ||
|  |   int padding {message.indexOf (" ") > 4 ? 2 : 0}; | ||
|  |   QString call = callsign; | ||
|  |   QString countryName; | ||
|  |   bool callWorkedBefore; | ||
|  |   bool countryWorkedBefore; | ||
|  | 
 | ||
|  |   if(call.length()==2) { | ||
|  |     int i0=message.indexOf("CQ "+call); | ||
|  |     call=message.mid(i0+6,-1); | ||
|  |     i0=call.indexOf(" "); | ||
|  |     call=call.mid(0,i0); | ||
|  |   } | ||
|  |   if(call.length()<3) return message; | ||
|  |   if(!call.contains(QRegExp("[0-9]|[A-Z]"))) return message; | ||
|  | 
 | ||
|  |   logBook.match(/*in*/call,/*out*/countryName,callWorkedBefore,countryWorkedBefore); | ||
|  | 
 | ||
|  |   message = message.trimmed (); | ||
|  |   QString appendage; | ||
|  |   if (!countryWorkedBefore) // therefore not worked call either | ||
|  |     { | ||
|  |       appendage += "!"; | ||
|  |       *bg = color_DXCC; | ||
|  |     } | ||
|  |   else | ||
|  |     { | ||
|  |       if (!callWorkedBefore) // but have worked the country | ||
|  |         { | ||
|  |           appendage += "~"; | ||
|  |           *bg = color_NewCall; | ||
|  |         } | ||
|  |       else | ||
|  |         { | ||
|  |           appendage += " ";  // have worked this call before | ||
|  |           *bg = color_CQ; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |   // do some obvious abbreviations | ||
|  |   countryName.replace ("Islands", "Is."); | ||
|  |   countryName.replace ("Island", "Is."); | ||
|  |   countryName.replace ("North ", "N. "); | ||
|  |   countryName.replace ("Northern ", "N. "); | ||
|  |   countryName.replace ("South ", "S. "); | ||
|  |   countryName.replace ("East ", "E. "); | ||
|  |   countryName.replace ("Eastern ", "E. "); | ||
|  |   countryName.replace ("West ", "W. "); | ||
|  |   countryName.replace ("Western ", "W. "); | ||
|  |   countryName.replace ("Central ", "C. "); | ||
|  |   countryName.replace (" and ", " & "); | ||
|  |   countryName.replace ("Republic", "Rep."); | ||
|  |   countryName.replace ("United States", "U.S.A."); | ||
|  |   countryName.replace ("Fed. Rep. of ", ""); | ||
|  |   countryName.replace ("French ", "Fr."); | ||
|  |   countryName.replace ("Asiatic", "AS"); | ||
|  |   countryName.replace ("European", "EU"); | ||
|  |   countryName.replace ("African", "AF"); | ||
|  | 
 | ||
|  |   appendage += countryName; | ||
|  | 
 | ||
|  |   // use a nbsp to save the start of appended text so we can find | ||
|  |   // it again later, align appended data at a fixed column if | ||
|  |   // there is space otherwise let it float to the right | ||
|  |   int space_count {40 + padding - message.size ()}; | ||
|  |   if (space_count > 0) | ||
|  |     { | ||
|  |       message += QString {space_count, QChar {' '}}; | ||
|  |     } | ||
|  |   message += QChar::Nbsp + appendage; | ||
|  | 
 | ||
|  |   return message; | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::displayDecodedText(DecodedText const& decodedText, QString const& myCall, | ||
|  |                                      bool displayDXCCEntity, LogBook const& logBook, | ||
|  |                                      QColor color_CQ, QColor color_MyCall, | ||
|  |                                      QColor color_DXCC, QColor color_NewCall) | ||
|  | { | ||
|  |   QColor bg {Qt::white}; | ||
|  |   bool CQcall = false; | ||
|  |   if (decodedText.string ().contains (" CQ ") | ||
|  |       || decodedText.string ().contains (" CQDX ") | ||
|  |       || decodedText.string ().contains (" QRZ ")) | ||
|  |     { | ||
|  |       CQcall = true; | ||
|  |       bg = color_CQ; | ||
|  |     } | ||
|  |   if (myCall != "" and ( | ||
|  |                         decodedText.indexOf (" " + myCall + " ") >= 0 | ||
|  |                         or decodedText.indexOf (" " + myCall + "/") >= 0 | ||
|  |                         or decodedText.indexOf ("/" + myCall + " ") >= 0 | ||
|  |                         or decodedText.indexOf ("<" + myCall + " ") >= 0 | ||
|  |                         or decodedText.indexOf (" " + myCall + ">") >= 0)) { | ||
|  |     bg = color_MyCall; | ||
|  |   } | ||
|  |   auto message = decodedText.string (); | ||
|  |   message = message.left (message.indexOf (QChar::Nbsp)); // strip appended info | ||
|  |   if (displayDXCCEntity && CQcall) | ||
|  |     // if enabled add the DXCC entity and B4 status to the end of the | ||
|  |     // preformated text line t1 | ||
|  |     message = appendDXCCWorkedB4 (message, decodedText.CQersCall (), &bg, logBook, color_CQ, | ||
|  |                                   color_DXCC, color_NewCall); | ||
|  |   appendText (message.trimmed (), bg); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | void DisplayText::displayTransmittedText(QString text, QString modeTx, qint32 txFreq, | ||
|  |                                          QColor color_TxMsg, bool bFastMode) | ||
|  | { | ||
|  |     QString t1=" @  "; | ||
|  |     if(modeTx=="FT8") t1=" ~  "; | ||
|  |     if(modeTx=="JT4") t1=" $  "; | ||
|  |     if(modeTx=="JT65") t1=" #  "; | ||
|  |     if(modeTx=="MSK144") t1=" &  "; | ||
|  |     QString t2; | ||
|  |     t2.sprintf("%4d",txFreq); | ||
|  |     QString t; | ||
|  |     if(bFastMode or modeTx=="FT8") { | ||
|  |       t = QDateTime::currentDateTimeUtc().toString("hhmmss") + \ | ||
|  |         "  Tx      " + t2 + t1 + text; | ||
|  |     } else { | ||
|  |       t = QDateTime::currentDateTimeUtc().toString("hhmm") + \ | ||
|  |         "  Tx      " + t2 + t1 + text; | ||
|  |     } | ||
|  |     appendText (t, color_TxMsg); | ||
|  | } | ||
|  | 
 | ||
|  | void DisplayText::displayQSY(QString text) | ||
|  | { | ||
|  |   QString t = QDateTime::currentDateTimeUtc().toString("hhmmss") + "            " + text; | ||
|  |   appendText (t, "hotpink"); | ||
|  | } |