| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | #include "displaytext.h" | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  | #include "mainwindow.h" | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | #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); | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   setUndoRedoEnabled (false); | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |   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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Q_EMIT selectCallsign(e->modifiers ()); | 
					
						
							|  |  |  |   QTextEdit::mouseDoubleClickEvent(e); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayText::insertLineSpacer(QString const& line) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   appendText (line, "#d3d3d3"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  | void DisplayText::appendText(QString const& text, QColor bg, QString const& call1, QString const& call2) | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | { | 
					
						
							|  |  |  |   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); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   QTextCharFormat format = cursor.charFormat(); | 
					
						
							|  |  |  |   format.clearBackground(); | 
					
						
							|  |  |  |   int text_index {0}; | 
					
						
							|  |  |  |   if (call1.size ()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       auto call_index = text.indexOf (call1); | 
					
						
							|  |  |  |       if (call_index != -1) // sanity check | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           auto pos = highlighted_calls_.find (call1); | 
					
						
							|  |  |  |           if (pos != highlighted_calls_.end ()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               cursor.insertText(text.left (call_index), format); | 
					
						
							|  |  |  |               if (pos.value ().first.isValid ()) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                   format.setBackground (pos.value ().first); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               if (pos.value ().second.isValid ()) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                   format.setForeground (pos.value ().second); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               cursor.insertText(text.mid (call_index, call1.size ()), format); | 
					
						
							|  |  |  |               text_index = call_index + call1.size (); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (call2.size ()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       auto call_index = text.indexOf (call2, text_index); | 
					
						
							|  |  |  |       if (call_index != -1) // sanity check | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           auto pos = highlighted_calls_.find (call2); | 
					
						
							|  |  |  |           if (pos != highlighted_calls_.end ()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               format.setBackground (bg); | 
					
						
							|  |  |  |               format.clearForeground (); | 
					
						
							|  |  |  |               cursor.insertText(text.mid (text_index, call_index - text_index), format); | 
					
						
							|  |  |  |               if (pos.value ().second.isValid ()) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                   format.setBackground (pos.value ().first); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               if (pos.value ().second.isValid ()) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                   format.setForeground (pos.value ().second); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               cursor.insertText(text.mid (call_index, call2.size ()), format); | 
					
						
							|  |  |  |               text_index = call_index + call2.size (); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   format.setBackground (bg); | 
					
						
							|  |  |  |   format.clearForeground (); | 
					
						
							|  |  |  |   cursor.insertText(text.mid (text_index), format); | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // 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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   int i1=countryName.indexOf(";"); | 
					
						
							|  |  |  |   if(m_bPrincipalPrefix) { | 
					
						
							|  |  |  |     int i2=countryName.lastIndexOf(";"); | 
					
						
							|  |  |  |     if(i1>0) countryName=countryName.mid(i1+2,i2-i1-2); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if(i1>0) countryName=countryName.mid(0,i1); | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |   // do some obvious abbreviations | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |     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"); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   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, | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |                                      QColor color_DXCC, QColor color_NewCall, bool ppfx, | 
					
						
							|  |  |  |                                      bool bCQonly) | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   m_bPrincipalPrefix=ppfx; | 
					
						
							|  |  |  |   QColor bg {Qt::transparent}; | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |   bool CQcall = false; | 
					
						
							|  |  |  |   if (decodedText.string ().contains (" CQ ") | 
					
						
							|  |  |  |       || decodedText.string ().contains (" CQDX ") | 
					
						
							|  |  |  |       || decodedText.string ().contains (" QRZ ")) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       CQcall = true; | 
					
						
							|  |  |  |       bg = color_CQ; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   if(bCQonly and !CQcall) return; | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |   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 (); | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   QString dxCall; | 
					
						
							|  |  |  |   QString dxGrid; | 
					
						
							|  |  |  |   decodedText.deCallAndGrid (dxCall, dxGrid); | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |   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); | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |   appendText (message.trimmed (), bg, decodedText.call (), dxCall); | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  |     } else if(modeTx.mid(0,6)=="FT8fox") { | 
					
						
							|  |  |  |       t = QDateTime::currentDateTimeUtc().toString("hhmmss") + \ | 
					
						
							|  |  |  |         " Tx" + modeTx.mid(7) + " " + text; | 
					
						
							| 
									
										
										
										
											2018-02-08 21:28:33 -05:00
										 |  |  |     } 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"); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-05 11:33:30 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void DisplayText::displayFoxToBeCalled(QString t, QColor bg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   appendText(t,bg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   void update_selection (QTextCursor& cursor, QColor const& bg, QColor const& fg) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (!cursor.isNull ()) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         QTextCharFormat format {cursor.charFormat ()}; | 
					
						
							|  |  |  |         if (bg.isValid ()) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             format.setBackground (bg); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             format.clearBackground (); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         if (fg.isValid ()) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             format.setForeground (fg); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             format.clearForeground (); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         cursor.mergeCharFormat (format); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void reset_selection (QTextCursor& cursor) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (!cursor.isNull ()) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // restore previous text format, we rely on the text | 
					
						
							|  |  |  |         // char format at he start of the selection being the | 
					
						
							|  |  |  |         // old one which should be the case | 
					
						
							|  |  |  |         auto c2 = cursor; | 
					
						
							|  |  |  |         c2.setPosition (c2.selectionStart ()); | 
					
						
							|  |  |  |         cursor.setCharFormat (c2.charFormat ()); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DisplayText::highlight_callsign (QString const& callsign, QColor const& bg, QColor const& fg, bool last_only) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   QTextCharFormat old_format {currentCharFormat ()}; | 
					
						
							|  |  |  |   QTextCursor cursor {document ()}; | 
					
						
							|  |  |  |   if (last_only) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       cursor.movePosition (QTextCursor::End); | 
					
						
							|  |  |  |       cursor = document ()->find (callsign, cursor | 
					
						
							|  |  |  |                                   , QTextDocument::FindBackward | QTextDocument::FindWholeWords); | 
					
						
							|  |  |  |       if (bg.isValid () || fg.isValid ()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           update_selection (cursor, bg, fg); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           reset_selection (cursor); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       auto pos = highlighted_calls_.find (callsign); | 
					
						
							|  |  |  |       if (bg.isValid () || fg.isValid ()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           auto colours = qMakePair (bg, fg); | 
					
						
							|  |  |  |           if (pos == highlighted_calls_.end ()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               pos = highlighted_calls_.insert (callsign.toUpper (), colours); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               pos.value () = colours; // update colours | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           while (!cursor.isNull ()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               cursor = document ()->find (callsign, cursor, QTextDocument::FindWholeWords); | 
					
						
							|  |  |  |               update_selection (cursor, bg, fg); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (pos != highlighted_calls_.end ()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           highlighted_calls_.erase (pos); | 
					
						
							|  |  |  |           QTextCursor cursor {document ()}; | 
					
						
							|  |  |  |           while (!cursor.isNull ()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               cursor = document ()->find (callsign, cursor, QTextDocument::FindWholeWords); | 
					
						
							|  |  |  |               reset_selection (cursor); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   setCurrentCharFormat (old_format); | 
					
						
							|  |  |  | } |