MythTV  master
xmlparsebase.cpp
Go to the documentation of this file.
1 
2 // Own header
3 #include "xmlparsebase.h"
4 
5 // C++/C headers
6 #include <typeinfo>
7 
8 // QT headers
9 #include <QFile>
10 #include <QDomDocument>
11 #include <QString>
12 #include <QBrush>
13 #include <QLinearGradient>
14 #include <QRadialGradient>
15 
16 // libmythbase headers
17 #include "libmythbase/mythconfig.h"
19 
20 // Mythui headers
21 #include "mythmainwindow.h"
22 #include "mythuihelper.h"
23 
24 /* ui type includes */
25 #include "mythscreentype.h"
26 #include "mythuiimage.h"
27 #include "mythuiprocedural.h"
28 #include "mythuitext.h"
29 #include "mythuitextedit.h"
30 #include "mythuiclock.h"
31 #include "mythuibuttonlist.h"
32 #include "mythuibutton.h"
33 #include "mythuispinbox.h"
34 #include "mythuicheckbox.h"
35 #include "mythuiprogressbar.h"
36 #include "mythuiscrollbar.h"
37 #include "mythuigroup.h"
38 
39 #if CONFIG_QTWEBKIT
40 #include "mythuiwebbrowser.h"
41 #endif
42 
43 #include "mythuiguidegrid.h"
44 #include "mythuishape.h"
45 #include "mythuibuttontree.h"
46 #include "mythuivideo.h"
47 #include "mythuieditbar.h"
48 #include "mythfontproperties.h"
49 
50 #define LOC QString("XMLParseBase: ")
51 
52 QString XMLParseBase::getFirstText(QDomElement &element)
53 {
54  for (QDomNode dname = element.firstChild(); !dname.isNull();
55  dname = dname.nextSibling())
56  {
57  QDomText t = dname.toText();
58  if (!t.isNull())
59  return t.data();
60  }
61  return {};
62 }
63 
64 bool XMLParseBase::parseBool(const QString &text)
65 {
66  QString s = text.toLower();
67  return (s == "yes" || s == "true" || (s.toInt() != 0));
68 }
69 
70 bool XMLParseBase::parseBool(QDomElement &element)
71 {
72  return parseBool(getFirstText(element));
73 }
74 
75 MythPoint XMLParseBase::parsePoint(const QString &text, bool normalize)
76 {
77  MythPoint retval;
78  QStringList values = text.split(',', Qt::SkipEmptyParts);
79  if (values.size() == 2)
80  retval = MythPoint(values[0], values[1]);
81 
82  if (normalize)
83  retval.NormPoint();
84 
85  return retval;
86 }
87 
88 MythPoint XMLParseBase::parsePoint(QDomElement &element, bool normalize)
89 {
90  return parsePoint(getFirstText(element), normalize);
91 }
92 
93 QSize XMLParseBase::parseSize(const QString &text, bool normalize)
94 {
95  int x = 0;
96  int y = 0;
97  QSize retval;
98 
99  QStringList tmp = text.split(",");
100  bool x_ok = false;
101  bool y_ok = false;
102  if (tmp.size() >= 2)
103  {
104  x = tmp[0].toInt(&x_ok);
105  y = tmp[1].toInt(&y_ok);
106  }
107 
108  if (x_ok && y_ok)
109  {
110  if (x == -1 || y == -1)
111  {
112  QRect uiSize = GetMythMainWindow()->GetUIScreenRect();
113  x = uiSize.width();
114  y = uiSize.height();
115  normalize = false;
116  }
117 
118  retval = QSize(x, y);
119  }
120 
121  if (normalize)
122  retval = GetMythMainWindow()->NormSize(retval);
123 
124  return retval;
125 }
126 
127 QSize XMLParseBase::parseSize(QDomElement &element, bool normalize)
128 {
129  return parseSize(getFirstText(element), normalize);
130 }
131 
132 MythRect XMLParseBase::parseRect(const QString &text, bool normalize)
133 {
134  MythRect retval;
135  QStringList values = text.split(',', Qt::SkipEmptyParts);
136  if (values.size() == 4)
137  retval = MythRect(values[0], values[1], values[2], values[3]);
138  if (values.size() == 5)
139  retval = MythRect(values[0], values[1], values[2], values[3],
140  values[4]);
141 
142  if (normalize)
143  retval.NormRect();
144 
145  return retval;
146 }
147 
148 MythRect XMLParseBase::parseRect(QDomElement &element, bool normalize)
149 {
150  return parseRect(getFirstText(element), normalize);
151 }
152 
153 int XMLParseBase::parseAlignment(const QString &text)
154 {
155  int alignment = Qt::AlignLeft | Qt::AlignTop;
156 
157  QStringList values = text.split(',');
158 
159  QStringList::Iterator it;
160  for ( it = values.begin(); it != values.end(); ++it )
161  {
162 
163  QString align = *it;
164  align = align.trimmed();
165  align = align.toLower();
166 
167  if (align == "center" || align == "allcenter")
168  {
169  alignment &= ~(Qt::AlignHorizontal_Mask | Qt::AlignVertical_Mask);
170  alignment |= Qt::AlignCenter;
171  break;
172  }
173  if (align == "justify")
174  {
175  alignment &= ~Qt::AlignHorizontal_Mask;
176  alignment |= Qt::AlignJustify;
177  }
178  else if (align == "left")
179  {
180  alignment &= ~Qt::AlignHorizontal_Mask;
181  alignment |= Qt::AlignLeft;
182  }
183  else if (align == "hcenter")
184  {
185  alignment &= ~Qt::AlignHorizontal_Mask;
186  alignment |= Qt::AlignHCenter;
187  }
188  else if (align == "right")
189  {
190  alignment &= ~Qt::AlignHorizontal_Mask;
191  alignment |= Qt::AlignRight;
192  }
193  else if (align == "top")
194  {
195  alignment &= ~Qt::AlignVertical_Mask;
196  alignment |= Qt::AlignTop;
197  }
198  else if (align == "vcenter")
199  {
200  alignment &= ~Qt::AlignVertical_Mask;
201  alignment |= Qt::AlignVCenter;
202  }
203  else if (align == "bottom")
204  {
205  alignment &= ~Qt::AlignVertical_Mask;
206  alignment |= Qt::AlignBottom;
207  }
208  }
209 
210  return alignment;
211 }
212 
213 int XMLParseBase::parseAlignment(QDomElement &element)
214 {
215  return parseAlignment(getFirstText(element));
216 }
217 
218 QBrush XMLParseBase::parseGradient(const QDomElement &element)
219 {
220  QBrush brush;
221  QString gradientStart = element.attribute("start", "");
222  QString gradientEnd = element.attribute("end", "");
223  int gradientAlpha = element.attribute("alpha", "255").toInt();
224  QString direction = element.attribute("direction", "vertical");
225 
226  QGradientStops stops;
227 
228  if (!gradientStart.isEmpty())
229  {
230  auto startColor = QColor(gradientStart);
231  startColor.setAlpha(gradientAlpha);
232  QGradientStop stop(0.0, startColor);
233  stops.append(stop);
234  }
235 
236  for (QDomNode child = element.firstChild(); !child.isNull();
237  child = child.nextSibling())
238  {
239  QDomElement childElem = child.toElement();
240  if (childElem.tagName() == "stop")
241  {
242  double position = childElem.attribute("position", "0").toDouble();
243  QString color = childElem.attribute("color", "");
244  int alpha = childElem.attribute("alpha", "-1").toInt();
245  if (alpha < 0)
246  alpha = gradientAlpha;
247  auto stopColor = QColor(color);
248  stopColor.setAlpha(alpha);
249  QGradientStop stop((position / 100), stopColor);
250  stops.append(stop);
251  }
252  }
253 
254  if (!gradientEnd.isEmpty())
255  {
256  auto endColor = QColor(gradientEnd);
257  endColor.setAlpha(gradientAlpha);
258  QGradientStop stop(1.0, endColor);
259  stops.append(stop);
260  }
261 
262  if (direction == "radial")
263  {
264  QRadialGradient gradient;
265  gradient.setCoordinateMode(QGradient::ObjectBoundingMode);
266  double x1 = 0.5;
267  double y1 = 0.5;
268  double radius = 0.5;
269  gradient.setCenter(x1,y1);
270  gradient.setFocalPoint(x1,y1);
271  gradient.setRadius(radius);
272  gradient.setStops(stops);
273  brush = QBrush(gradient);
274  }
275  else // Linear
276  {
277  QLinearGradient gradient;
278  gradient.setCoordinateMode(QGradient::ObjectBoundingMode);
279  double x1 = 0.0;
280  double y1 = 0.0;
281  double x2 = 0.0;
282  double y2 = 0.0;
283  if (direction == "vertical")
284  {
285  x1 = 0.5;
286  x2 = 0.5;
287  y1 = 0.0;
288  y2 = 1.0;
289  }
290  else if (direction == "diagonal")
291  {
292  x1 = 0.0;
293  x2 = 1.0;
294  y1 = 0.0;
295  y2 = 1.0;
296  }
297  else // Horizontal
298  {
299  x1 = 0.0;
300  x2 = 1.0;
301  y1 = 0.5;
302  y2 = 0.5;
303  }
304 
305  gradient.setStart(x1, y1);
306  gradient.setFinalStop(x2, y2);
307  gradient.setStops(stops);
308  brush = QBrush(gradient);
309  }
310 
311 
312  return brush;
313 }
314 
315 QString XMLParseBase::parseText(QDomElement &element)
316 {
317  QString text = getFirstText(element);
318 
319  // Escape xml-escaped newline
320  text.replace("\\n", QString("<newline>"));
321 
322  // Remove newline whitespace added by
323  // xml formatting
324  QStringList lines = text.split('\n');
325  QStringList::iterator lineIt;
326 
327  for (lineIt = lines.begin(); lineIt != lines.end(); ++lineIt)
328  {
329  (*lineIt) = (*lineIt).trimmed();
330  }
331 
332  text = lines.join(" ");
333 
334  text.replace(QString("<newline>"), QString("\n"));
335 
336  return text;
337 }
338 
339 static MythUIType *globalObjectStore = nullptr;
340 static QStringList loadedBaseFiles;
341 
343 {
344  if (!globalObjectStore)
345  globalObjectStore = new MythUIType(nullptr, "global store");
346  return globalObjectStore;
347 }
348 
350 {
351  delete globalObjectStore;
352  globalObjectStore = nullptr;
354 
355  // clear any loaded base xml files which will force a reload the next time they are used
356  loadedBaseFiles.clear();
357 }
358 
360  QDomElement &element,
361  MythUIType *parent,
362  bool showWarnings)
363 {
364  if (!parent)
365  {
366  LOG(VB_GENERAL, LOG_ERR, LOC + "Parent is NULL");
367  return;
368  }
369 
370  QMap<QString, QString> dependsMap;
371  for (QDomNode child = element.firstChild(); !child.isNull();
372  child = child.nextSibling())
373  {
374  QDomElement info = child.toElement();
375  if (!info.isNull())
376  {
377  QString type = info.tagName();
378  if (type == "fontdef")
379  {
380  bool global = (GetGlobalObjectStore() == parent);
382  filename, info, parent, global, showWarnings);
383 
384  if (!global && font)
385  {
386  QString name = info.attribute("name");
387  parent->AddFont(name, font);
388  }
389 
390  delete font;
391  }
392  else if (type == "imagetype" ||
393  type == "procedural" ||
394  type == "textarea" ||
395  type == "group" ||
396  type == "textedit" ||
397  type == "button" ||
398  type == "buttonlist" ||
399  type == "buttonlist2" ||
400  type == "buttontree" ||
401  type == "spinbox" ||
402  type == "checkbox" ||
403  type == "statetype" ||
404  type == "clock" ||
405  type == "progressbar" ||
406  type == "scrollbar" ||
407  type == "webbrowser" ||
408  type == "guidegrid" ||
409  type == "shape" ||
410  type == "editbar" ||
411  type == "video")
412  {
413  ParseUIType(filename, info, type, parent, nullptr, showWarnings, dependsMap);
414  }
415  else
416  {
417  // This will print an error if there is no match.
418  parent->ParseElement(filename, info, showWarnings);
419  }
420  }
421  }
422  parent->SetDependsMap(dependsMap);
423  parent->ConnectDependants(true);
424  parent->Finalize();
425 }
426 
428  const QString &filename,
429  QDomElement &element, const QString &type,
430  MythUIType *parent,
431  MythScreenType *screen,
432  bool showWarnings,
433  QMap<QString, QString> &parentDependsMap)
434 {
435  QString name = element.attribute("name", "");
436  if (name.isEmpty())
437  {
438  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
439  "This element requires a name");
440  return nullptr;
441  }
442 
443  MythUIType *olduitype = nullptr;
444 
445  // check for name in immediate parent as siblings cannot share names
446  if (parent && parent->GetChild(name))
447  {
448  // if we're the global object store, assume it's just a theme overriding
449  // the defaults..
450  if (parent == GetGlobalObjectStore())
451  return nullptr;
452 
453  // Reuse the existing child and reparse
454  olduitype = parent->GetChild(name);
455  }
456 
457  MythUIType *uitype = nullptr;
458  MythUIType *base = nullptr;
459 
460  QString inherits = element.attribute("from", "");
461  if (!inherits.isEmpty())
462  {
463  if (parent)
464  base = parent->GetChild(inherits);
465 
466  // might remove this
467  if (screen && !base)
468  base = screen->GetChild(inherits);
469 
470  if (!base)
471  base = GetGlobalObjectStore()->GetChild(inherits);
472 
473  if (!base)
474  {
475  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
476  QString("Couldn't find object '%1' to inherit '%2' from")
477  .arg(inherits, name));
478  return nullptr;
479  }
480  }
481 
482  if (type == "imagetype")
483  uitype = new MythUIImage(parent, name);
484  else if (type == "procedural")
485  uitype = new MythUIProcedural(parent, name);
486  else if (type == "textarea")
487  uitype = new MythUIText(parent, name);
488  else if (type == "group")
489  uitype = new MythUIGroup(parent, name);
490  else if (type == "textedit")
491  uitype = new MythUITextEdit(parent, name);
492  else if (type == "button")
493  uitype = new MythUIButton(parent, name);
494  else if (type == "buttonlist2" || type == "buttonlist")
495  uitype = new MythUIButtonList(parent, name);
496  else if (type == "buttontree")
497  uitype = new MythUIButtonTree(parent, name);
498  else if (type == "spinbox")
499  uitype = new MythUISpinBox(parent, name);
500  else if (type == "checkbox")
501  uitype = new MythUICheckBox(parent, name);
502  else if (type == "statetype")
503  uitype = new MythUIStateType(parent, name);
504  else if (type == "clock")
505  uitype = new MythUIClock(parent, name);
506  else if (type == "progressbar")
507  uitype = new MythUIProgressBar(parent, name);
508  else if (type == "scrollbar") {
509  uitype = new MythUIScrollBar(parent, name);
510 #if CONFIG_QTWEBKIT
511  } else if (type == "webbrowser") {
512  uitype = new MythUIWebBrowser(parent, name);
513 #endif
514  } else if (type == "guidegrid") {
515  uitype = new MythUIGuideGrid(parent, name);
516  } else if (type == "shape") {
517  uitype = new MythUIShape(parent, name);
518  } else if (type == "editbar") {
519  uitype = new MythUIEditBar(parent, name);
520  } else if (type == "video") {
521  uitype = new MythUIVideo(parent, name);
522  } else if (type == "window" && parent == GetGlobalObjectStore()) {
523  uitype = new MythScreenType(parent, name);
524  } else {
525  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
526  "Unknown widget type.");
527  return nullptr;
528  }
529 
530  if (!uitype)
531  {
532  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
533  "Failed to instantiate widget type.");
534  return nullptr;
535  }
536 
537  if (olduitype && parent)
538  {
539  if (typeid(*olduitype) != typeid(*uitype))
540  {
541  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
542  QString("Duplicate name: '%1' in parent '%2'")
543  .arg(name, parent->objectName()));
544  parent->DeleteChild(olduitype);
545  }
546  else
547  {
548  parent->DeleteChild(uitype);
549  uitype = olduitype;
550  }
551  }
552 
553  if (base)
554  {
555  if (typeid(*base) != typeid(*uitype))
556  {
557  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, element,
558  QString("Type of new widget '%1' doesn't match old '%2'")
559  .arg(name, inherits));
560  if (parent)
561  parent->DeleteChild(uitype);
562  return nullptr;
563  }
564  uitype->CopyFrom(base);
565 
566  }
567 
568  QString dependee = element.attribute("depends", "");
569  if (!dependee.isEmpty())
570  parentDependsMap.insert(name, dependee);
571 
572  QFileInfo fi(filename);
573  uitype->SetXMLName(name);
574  uitype->SetXMLLocation(fi.fileName(), element.lineNumber());
575 
576  // If this was copied from another uitype then it already has a depends
577  // map so we want to append to that one
578  QMap<QString, QString> dependsMap = uitype->GetDependsMap();
579  for (QDomNode child = element.firstChild(); !child.isNull();
580  child = child.nextSibling())
581  {
582  QDomElement info = child.toElement();
583  if (!info.isNull())
584  {
585  if (info.tagName() == "fontdef")
586  {
587  bool global = (GetGlobalObjectStore() == parent);
589  filename, info, parent, global, showWarnings);
590 
591  if (!global && font)
592  {
593  QString name2 = info.attribute("name");
594  uitype->AddFont(name2, font);
595  }
596 
597  delete font;
598  }
599  else if (info.tagName() == "imagetype" ||
600  info.tagName() == "procedural" ||
601  info.tagName() == "textarea" ||
602  info.tagName() == "group" ||
603  info.tagName() == "textedit" ||
604  info.tagName() == "button" ||
605  info.tagName() == "buttonlist" ||
606  info.tagName() == "buttonlist2" ||
607  info.tagName() == "buttontree" ||
608  info.tagName() == "spinbox" ||
609  info.tagName() == "checkbox" ||
610  info.tagName() == "statetype" ||
611  info.tagName() == "clock" ||
612  info.tagName() == "progressbar" ||
613  info.tagName() == "scrollbar" ||
614  info.tagName() == "webbrowser" ||
615  info.tagName() == "guidegrid" ||
616  info.tagName() == "shape" ||
617  info.tagName() == "editbar" ||
618  info.tagName() == "video")
619  {
620  ParseUIType(filename, info, info.tagName(),
621  uitype, screen, showWarnings, dependsMap);
622  }
623  else
624  {
625  // This will print an error if there is no match.
626  uitype->ParseElement(filename, info, showWarnings);
627  }
628  }
629  }
630  uitype->SetDependsMap(dependsMap);
631 
632  uitype->Finalize();
633  return uitype;
634 }
635 
636 bool XMLParseBase::WindowExists(const QString &xmlfile,
637  const QString &windowname)
638 {
639  const QStringList searchpath = GetMythUI()->GetThemeSearchPath();
640  for (const auto & dir : std::as_const(searchpath))
641  {
642  QString themefile = dir + xmlfile;
643  QFile f(themefile);
644 
645  if (!f.open(QIODevice::ReadOnly))
646  continue;
647 
648  QDomDocument doc;
649 #if QT_VERSION < QT_VERSION_CHECK(6,5,0)
650  QString errorMsg;
651  int errorLine = 0;
652  int errorColumn = 0;
653 
654  if (!doc.setContent(&f, false, &errorMsg, &errorLine, &errorColumn))
655  {
656  LOG(VB_GENERAL, LOG_ERR, LOC +
657  QString("Location: '%1' @ %2 column: %3"
658  "\n\t\t\tError: %4")
659  .arg(qPrintable(themefile)).arg(errorLine).arg(errorColumn)
660  .arg(qPrintable(errorMsg)));
661  f.close();
662  continue;
663  }
664 #else
665  auto parseResult = doc.setContent(&f);
666  if (!parseResult)
667  {
668  LOG(VB_GENERAL, LOG_ERR, LOC +
669  QString("Location: '%1' @ %2 column: %3"
670  "\n\t\t\tError: %4")
671  .arg(qPrintable(themefile)).arg(parseResult.errorLine)
672  .arg(parseResult.errorColumn)
673  .arg(qPrintable(parseResult.errorMessage)));
674  f.close();
675  continue;
676  }
677 #endif
678  f.close();
679 
680  QDomElement docElem = doc.documentElement();
681  QDomNode n = docElem.firstChild();
682  while (!n.isNull())
683  {
684  QDomElement e = n.toElement();
685  if (!e.isNull())
686  {
687  if (e.tagName() == "window")
688  {
689  QString name = e.attribute("name", "");
690  if (name == windowname)
691  return true;
692  }
693  }
694  n = n.nextSibling();
695  }
696  }
697 
698  return false;
699 }
700 
701 bool XMLParseBase::LoadWindowFromXML(const QString &xmlfile,
702  const QString &windowname,
703  MythUIType *parent)
704 {
705  bool onlyLoadWindows = true;
706  bool showWarnings = true;
707 
708  const QStringList searchpath = GetMythUI()->GetThemeSearchPath();
709  for (const auto & dir : std::as_const(searchpath))
710  {
711  QString themefile = dir + xmlfile;
712  LOG(VB_GUI, LOG_INFO, LOC + QString("Loading window %1 from %2").arg(windowname, themefile));
713  if (doLoad(windowname, parent, themefile,
714  onlyLoadWindows, showWarnings))
715  {
716  return true;
717  }
718  LOG(VB_FILE, LOG_ERR, LOC + "No theme file " + themefile);
719  }
720 
721  LOG(VB_GENERAL, LOG_ERR, LOC +
722  QString("Unable to load window '%1' from '%2'")
723  .arg(windowname, xmlfile));
724 
725  return false;
726 }
727 
728 bool XMLParseBase::doLoad(const QString &windowname,
729  MythUIType *parent,
730  const QString &filename,
731  bool onlyLoadWindows,
732  bool showWarnings)
733 {
734  QDomDocument doc;
735  QFile f(filename);
736 
737  if (!f.open(QIODevice::ReadOnly))
738  return false;
739 
740 #if QT_VERSION < QT_VERSION_CHECK(6,5,0)
741  QString errorMsg;
742  int errorLine = 0;
743  int errorColumn = 0;
744 
745  if (!doc.setContent(&f, false, &errorMsg, &errorLine, &errorColumn))
746  {
747  LOG(VB_GENERAL, LOG_ERR, LOC +
748  QString("Location: '%1' @ %2 column: %3"
749  "\n\t\t\tError: %4")
750  .arg(qPrintable(filename)).arg(errorLine).arg(errorColumn)
751  .arg(qPrintable(errorMsg)));
752  f.close();
753  return false;
754  }
755 #else
756  auto parseResult = doc.setContent(&f);
757  if (!parseResult)
758  {
759  LOG(VB_GENERAL, LOG_ERR, LOC +
760  QString("Location: '%1' @ %2 column: %3"
761  "\n\t\t\tError: %4")
762  .arg(qPrintable(filename)).arg(parseResult.errorLine)
763  .arg(parseResult.errorColumn)
764  .arg(qPrintable(parseResult.errorMessage)));
765  f.close();
766  return false;
767  }
768 #endif
769 
770  f.close();
771 
772  QDomElement docElem = doc.documentElement();
773  QDomNode n = docElem.firstChild();
774  while (!n.isNull())
775  {
776  QDomElement e = n.toElement();
777  if (!e.isNull())
778  {
779  if (e.tagName() == "include")
780  {
781  QString include = getFirstText(e);
782 
783  if (!include.isEmpty())
784  LoadBaseTheme(include);
785  }
786 
787  if (onlyLoadWindows && e.tagName() == "window")
788  {
789  QString name = e.attribute("name", "");
790  QString include = e.attribute("include", "");
791  if (name.isEmpty())
792  {
793  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, e,
794  "Window needs a name");
795  return false;
796  }
797 
798  if (!include.isEmpty())
799  LoadBaseTheme(include);
800 
801  if (name == windowname)
802  {
803  ParseChildren(filename, e, parent, showWarnings);
804  return true;
805  }
806  }
807 
808  if (!onlyLoadWindows)
809  {
810  QString type = e.tagName();
811  if (type == "font" || type == "fontdef")
812  {
813  bool global = (GetGlobalObjectStore() == parent);
815  filename, e, parent, global, showWarnings);
816 
817  if (!global && font)
818  {
819  QString name = e.attribute("name");
820  parent->AddFont(name, font);
821  }
822  delete font;
823  }
824  else if (type == "imagetype" ||
825  type == "procedural" ||
826  type == "textarea" ||
827  type == "group" ||
828  type == "textedit" ||
829  type == "button" ||
830  type == "buttonlist" ||
831  type == "buttonlist2" ||
832  type == "buttontree" ||
833  type == "spinbox" ||
834  type == "checkbox" ||
835  type == "statetype" ||
836  type == "window" ||
837  type == "clock" ||
838  type == "progressbar" ||
839  type == "scrollbar" ||
840  type == "webbrowser" ||
841  type == "guidegrid" ||
842  type == "shape" ||
843  type == "editbar" ||
844  type == "video")
845  {
846 
847  // We don't want widgets in base.xml
848  // depending on each other so ignore dependsMap
849  QMap<QString, QString> dependsMap;
850  MythUIType *uitype = nullptr;
851  uitype = ParseUIType(filename, e, type, parent,
852  nullptr, showWarnings, dependsMap);
853  if (uitype)
854  uitype->ConnectDependants(true);
855  }
856  else
857  {
858  VERBOSE_XML(VB_GENERAL, LOG_ERR, filename, e,
859  "Unknown widget type");
860  }
861  }
862  }
863  n = n.nextSibling();
864  }
865  return !onlyLoadWindows;
866 }
867 
869 {
870  bool ok = false;
871  bool loadOnlyWindows = false;
872  bool showWarnings = true;
873 
874  const QStringList searchpath = GetMythUI()->GetThemeSearchPath();
875  for (const auto & dir : std::as_const(searchpath))
876  {
877  QString themefile = dir + "base.xml";
878  if (doLoad(QString(), GetGlobalObjectStore(), themefile,
879  loadOnlyWindows, showWarnings))
880  {
881  LOG(VB_GUI, LOG_INFO, LOC +
882  QString("Loaded base theme from '%1'").arg(themefile));
883  // Don't complain about duplicate definitions after first
884  // successful load (set showWarnings to false).
885  showWarnings = false;
886  ok = true;
887  }
888  else
889  {
890  LOG(VB_GUI | VB_FILE, LOG_WARNING, LOC +
891  QString("No theme file '%1'").arg(themefile));
892  }
893  }
894 
895  return ok;
896 }
897 
898 bool XMLParseBase::LoadBaseTheme(const QString &baseTheme)
899 {
900  LOG(VB_GUI, LOG_INFO, LOC +
901  QString("Asked to load base file from '%1'").arg(baseTheme));
902 
903  if (loadedBaseFiles.contains(baseTheme))
904  {
905  LOG(VB_GUI, LOG_INFO, LOC +
906  QString("Base file already loaded '%1'").arg(baseTheme));
907  return true;
908  }
909 
910  bool ok = false;
911  bool loadOnlyWindows = false;
912  bool showWarnings = true;
913 
914  const QStringList searchpath = GetMythUI()->GetThemeSearchPath();
915  for (const auto & dir : std::as_const(searchpath))
916  {
917  QString themefile = dir + baseTheme;
918  if (doLoad(QString(), GetGlobalObjectStore(), themefile,
919  loadOnlyWindows, showWarnings))
920  {
921  LOG(VB_GUI, LOG_INFO, LOC +
922  QString("Loaded base theme from '%1'").arg(themefile));
923  // Don't complain about duplicate definitions after first
924  // successful load (set showWarnings to false).
925  showWarnings = false;
926  ok = true;
927  }
928  else
929  {
930  LOG(VB_GUI | VB_FILE, LOG_WARNING, LOC +
931  QString("No theme file '%1'").arg(themefile));
932  }
933  }
934 
935  if (ok)
936  loadedBaseFiles.append(baseTheme);
937 
938  return ok;
939 }
940 
941 bool XMLParseBase::CopyWindowFromBase(const QString &windowname,
942  MythScreenType *win)
943 {
944  MythUIType *ui = GetGlobalObjectStore()->GetChild(windowname);
945  if (!ui)
946  {
947  LOG(VB_GENERAL, LOG_ERR, LOC +
948  QString("Unable to load window '%1' from base") .arg(windowname));
949  return false;
950  }
951 
952  auto *st = dynamic_cast<MythScreenType *>(ui);
953  if (!st)
954  {
955  LOG(VB_GENERAL, LOG_ERR, LOC +
956  QString("UI Object '%1' is not a ScreenType") .arg(windowname));
957  return false;
958  }
959 
960  win->CopyFrom(st);
961  return true;
962 }
MythUIType::SetXMLLocation
void SetXMLLocation(const QString &filename, int where)
Definition: mythuitype.h:178
mythuibuttontree.h
XMLParseBase::LoadBaseTheme
static bool LoadBaseTheme(void)
Definition: xmlparsebase.cpp:868
mythuitext.h
mythuiprogressbar.h
XMLParseBase::parseAlignment
static int parseAlignment(const QString &text)
Definition: xmlparsebase.cpp:153
MythUIImage
Image widget, displays a single image or multiple images in sequence.
Definition: mythuiimage.h:97
XMLParseBase::parsePoint
static MythPoint parsePoint(const QString &text, bool normalize=true)
Definition: xmlparsebase.cpp:75
MythUIType::DeleteChild
void DeleteChild(const QString &name)
Delete a named child of this UIType.
Definition: mythuitype.cpp:153
MythUIScreenBounds::GetUIScreenRect
QRect GetUIScreenRect()
Definition: mythuiscreenbounds.cpp:198
MythUIType::ConnectDependants
void ConnectDependants(bool recurse=false)
Definition: mythuitype.cpp:1438
mythuivideo.h
x2
static int x2
Definition: mythsocket.cpp:51
MythUIType::GetChild
MythUIType * GetChild(const QString &name) const
Get a named child of this UIType.
Definition: mythuitype.cpp:138
MythUIEditBar
A narrow purpose widget used to represent cut positions and regions when editing a video.
Definition: mythuieditbar.h:16
XMLParseBase::ClearGlobalObjectStore
static void ClearGlobalObjectStore(void)
Definition: xmlparsebase.cpp:349
mythuiscrollbar.h
MythUITextEdit
A text entry and edit widget.
Definition: mythuitextedit.h:34
MythUIType::AddFont
bool AddFont(const QString &text, MythFontProperties *fontProp)
Definition: mythuitype.cpp:1326
MythUIGroup
Create a group of widgets.
Definition: mythuigroup.h:11
LOG
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
Definition: mythlogging.h:39
MythScreenType
Screen in which all other widgets are contained and rendered.
Definition: mythscreentype.h:45
XMLParseBase::GetGlobalObjectStore
static MythUIType * GetGlobalObjectStore(void)
Definition: xmlparsebase.cpp:342
MythUIGuideGrid
A narrow purpose widget used to show television programs and the timeslots they occupy on channels....
Definition: mythuiguidegrid.h:43
MythUIClock
A simple text clock widget.
Definition: mythuiclock.h:25
MythRect
Wrapper around QRect allowing us to handle percentage and other relative values for areas in mythui.
Definition: mythrect.h:17
mythuieditbar.h
MythScreenType::CopyFrom
void CopyFrom(MythUIType *base) override
Copy this widgets state from another.
Definition: mythscreentype.cpp:520
mythuibuttonlist.h
mythuiimage.h
mythuiclock.h
mythuiguidegrid.h
XMLParseBase::parseText
static QString parseText(QDomElement &element)
Definition: xmlparsebase.cpp:315
tmp
static guint32 * tmp
Definition: goom_core.cpp:26
loadedBaseFiles
static QStringList loadedBaseFiles
Definition: xmlparsebase.cpp:340
MythUIType::GetDependsMap
QMap< QString, QString > GetDependsMap() const
Definition: mythuitype.h:202
mythfontproperties.h
mythlogging.h
MythUIProgressBar
Progress bar widget.
Definition: mythuiprogressbar.h:12
hardwareprofile.i18n.t
t
Definition: i18n.py:36
VERBOSE_XML
#define VERBOSE_XML(type, level, filename, element, msg)
Definition: xmlparsebase.h:15
MythFontProperties
Definition: mythfontproperties.h:13
x1
static int x1
Definition: mythsocket.cpp:50
XMLParseBase::CopyWindowFromBase
static bool CopyWindowFromBase(const QString &windowname, MythScreenType *win)
Definition: xmlparsebase.cpp:941
XMLParseBase::getFirstText
static QString getFirstText(QDomElement &element)
Definition: xmlparsebase.cpp:52
MythUIButton
A single button widget.
Definition: mythuibutton.h:21
MythUIButtonTree
A tree widget for displaying and navigating a MythGenericTree()
Definition: mythuibuttontree.h:16
MythUIType::CopyFrom
virtual void CopyFrom(MythUIType *base)
Copy this widgets state from another.
Definition: mythuitype.cpp:1171
xmlparsebase.h
MythUIThemeHelper::GetThemeSearchPath
QStringList GetThemeSearchPath()
Definition: mythuithemehelper.cpp:151
XMLParseBase::parseGradient
static QBrush parseGradient(const QDomElement &element)
Definition: xmlparsebase.cpp:218
MythPoint::NormPoint
void NormPoint(void)
Definition: mythrect.cpp:471
XMLParseBase::parseRect
static MythRect parseRect(const QString &text, bool normalize=true)
Definition: xmlparsebase.cpp:132
MythUIScrollBar
Scroll bar widget.
Definition: mythuiscrollbar.h:15
MythUIProcedural
Definition: mythuiprocedural.h:12
XMLParseBase::WindowExists
static bool WindowExists(const QString &xmlfile, const QString &windowname)
Definition: xmlparsebase.cpp:636
MythUICheckBox
A checkbox widget supporting three check states - on,off,half and two conditions - selected and unsel...
Definition: mythuicheckbox.h:15
mythuigroup.h
MythUIType
The base class on which all widgets and screens are based.
Definition: mythuitype.h:85
MythUIShape
A widget for rendering primitive shapes and lines.
Definition: mythuishape.h:21
MythUIType::SetDependsMap
void SetDependsMap(QMap< QString, QString > dependsMap)
Definition: mythuitype.cpp:1428
mythuispinbox.h
mythuihelper.h
MythRect::NormRect
void NormRect(void)
Definition: mythrect.cpp:109
MythUIText
All purpose text widget, displays a text string.
Definition: mythuitext.h:28
MythUIVideo
Video widget, displays raw image data.
Definition: mythuivideo.h:14
XMLParseBase::ParseUIType
static MythUIType * ParseUIType(const QString &filename, QDomElement &element, const QString &type, MythUIType *parent, MythScreenType *screen, bool showWarnings, QMap< QString, QString > &parentDependsMap)
Definition: xmlparsebase.cpp:427
XMLParseBase::doLoad
static bool doLoad(const QString &windowname, MythUIType *parent, const QString &filename, bool onlyLoadWindows, bool showWarnings)
Definition: xmlparsebase.cpp:728
mythuitextedit.h
XMLParseBase::LoadWindowFromXML
static bool LoadWindowFromXML(const QString &xmlfile, const QString &windowname, MythUIType *parent)
Definition: xmlparsebase.cpp:701
MythUIType::SetXMLName
void SetXMLName(const QString &name)
Definition: mythuitype.h:182
GetMythMainWindow
MythMainWindow * GetMythMainWindow(void)
Definition: mythmainwindow.cpp:104
MythFontProperties::ParseFromXml
static MythFontProperties * ParseFromXml(const QString &filename, const QDomElement &element, MythUIType *parent=nullptr, bool addToGlobal=false, bool showWarnings=true)
Definition: mythfontproperties.cpp:173
MythUISpinBox
A widget for offering a range of numerical values where only the the bounding values and interval are...
Definition: mythuispinbox.h:16
XMLParseBase::ParseChildren
static void ParseChildren(const QString &filename, QDomElement &element, MythUIType *parent, bool showWarnings)
Definition: xmlparsebase.cpp:359
mythuibutton.h
azlyrics.info
dictionary info
Definition: azlyrics.py:7
XMLParseBase::parseSize
static QSize parseSize(const QString &text, bool normalize=true)
Definition: xmlparsebase.cpp:93
MythUIWebBrowser
Web browsing widget.
Definition: mythuiwebbrowser.h:132
mythuicheckbox.h
globalObjectStore
static MythUIType * globalObjectStore
Definition: xmlparsebase.cpp:339
MythUIType::ParseElement
virtual bool ParseElement(const QString &filename, QDomElement &element, bool showWarnings)
Parse the xml definition of this widget setting the state of the object accordingly.
Definition: mythuitype.cpp:1237
GetMythUI
MythUIHelper * GetMythUI()
Definition: mythuihelper.cpp:66
MythUIButtonList
List widget, displays list items in a variety of themeable arrangements and can trigger signals when ...
Definition: mythuibuttonlist.h:191
mythuiwebbrowser.h
build_compdb.filename
filename
Definition: build_compdb.py:21
mythmainwindow.h
XMLParseBase::parseBool
static bool parseBool(const QString &text)
Definition: xmlparsebase.cpp:64
LOC
#define LOC
Definition: xmlparsebase.cpp:50
mythuishape.h
MythUIStateType
This widget is used for grouping other widgets for display when a particular named state is called....
Definition: mythuistatetype.h:22
MythUIType::Finalize
virtual void Finalize(void)
Perform any post-xml parsing initialisation tasks.
Definition: mythuitype.cpp:1312
mythscreentype.h
MythUIScreenBounds::NormSize
QSize NormSize(QSize Size) const
Definition: mythuiscreenbounds.cpp:219
MythPoint
Wrapper around QPoint allowing us to handle percentage and other relative values for positioning in m...
Definition: mythrect.h:88
mythuiprocedural.h