summaryrefslogtreecommitdiffstats
path: root/src/devices/base/device_group.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/devices/base/device_group.cpp')
-rw-r--r--src/devices/base/device_group.cpp362
1 files changed, 362 insertions, 0 deletions
diff --git a/src/devices/base/device_group.cpp b/src/devices/base/device_group.cpp
new file mode 100644
index 0000000..df230d1
--- /dev/null
+++ b/src/devices/base/device_group.cpp
@@ -0,0 +1,362 @@
+/***************************************************************************
+ * Copyright (C) 2005-2006 Nicolas Hadacek <hadacek@kde.org> *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU General Public License as published by *
+ * the Free Software Foundation; either version 2 of the License, or *
+ * (at your option) any later version. *
+ ***************************************************************************/
+#include "device_group.h"
+
+#if !defined(NO_KDE)
+# include <qpainter.h>
+# include <kglobal.h>
+
+QColor Device::statusColor(Status status)
+{
+ switch (status.type()) {
+ case Status::Future: return Qt::blue;
+ case Status::InProduction: return Qt::green;
+ case Status::Mature:
+ case Status::NotRecommended: return QColor("orange");
+ case Status::EOL: return Qt::red;
+ case Status::Unknown:
+ case Status::Nb_Types: break;
+ }
+ return Qt::black;
+}
+
+QString coloredString(const QString &text, QColor color)
+{
+ return QString("<font color=\"") + color.name() + "\">" + text + "</font>";
+}
+
+QString supportedString(bool supported)
+{
+ return coloredString(supported ? i18n("Supported") : i18n("Unsupported"),
+ supported ? Qt::green : Qt::red);
+}
+
+class Tick {
+public:
+ Tick() {}
+ Tick(double value, double oValue) {
+ s = KGlobal::locale()->formatNumber(value, 1);
+ min = oValue;
+ }
+ QString s;
+ double min;
+};
+
+class TickMap : public QMap<double, Tick>
+{
+public:
+ TickMap() {}
+ void add(double value, double oValue) {
+ insert(value, Tick(value, oValue), false);
+ (*this)[value].min = QMIN((*this)[value].min, oValue);
+ }
+};
+
+QPixmap drawGraph(const QValueVector<Device::RangeBox> &boxes)
+{
+ const uint w = 300, h = 200;
+ QPixmap pixmap(w, h);
+ pixmap.fill(Qt::white);
+ QPainter p(&pixmap);
+ QFontMetrics f(p.font());
+ TickMap xTicks, yTicks;
+ xTicks.add(0.0, 0.0);
+ yTicks.add(0.0, 0.0);
+ for (uint i=0; i<boxes.count(); i++) {
+// qDebug("box #%i: %f=[%f %f] %f=[%f %f]", i, boxes[i].start.x, boxes[i].start.yMin,
+// boxes[i].start.yMax, boxes[i].end.x, boxes[i].end.yMin, boxes[i].end.yMax);
+ xTicks.add(boxes[i].start.x, boxes[i].start.yMin);
+ xTicks.add(boxes[i].start.x, boxes[i].start.yMax);
+ xTicks.add(boxes[i].end.x, boxes[i].end.yMin);
+ xTicks.add(boxes[i].end.x, boxes[i].end.yMax);
+ yTicks.add(boxes[i].start.yMin, boxes[i].start.x);
+ yTicks.add(boxes[i].start.yMax, boxes[i].start.x);
+ yTicks.add(boxes[i].end.yMin, boxes[i].end.x);
+ yTicks.add(boxes[i].end.yMax, boxes[i].end.x);
+ }
+ double xMax = 0.0, yMax = 0.0;
+ int xStart = 0;
+ int yStart = h-1 - f.lineSpacing();
+ TickMap::const_iterator it = xTicks.begin();
+ for (; it!=xTicks.end(); ++it) {
+ xStart = QMAX(xStart, f.width(it.data().s));
+ xMax = QMAX(xMax, it.key());
+ }
+ for (it = yTicks.begin(); it!=yTicks.end(); ++it)
+ yMax = QMAX(yMax, it.key());
+ int xEnd = w-1 - f.width(xTicks[xMax].s)/2;
+ QRect rect = f.boundingRect(yTicks[yMax].s);
+ int yEnd = rect.height()/2;
+
+ // draw boxes
+ p.setPen(Qt::lightGray);
+ p.setBrush(Qt::lightGray);
+ for (uint i=0; i<boxes.count(); i++) {
+ double ax = double(xEnd - xStart)/xMax;
+ double ay = double(yEnd - yStart)/yMax;
+ QPointArray pa(4);
+ pa.setPoint(0, qRound(ax*boxes[i].start.x), qRound(ay*boxes[i].start.yMin));
+ pa.setPoint(1, qRound(ax*boxes[i].end.x), qRound(ay*boxes[i].end.yMin));
+ pa.setPoint(2, qRound(ax*boxes[i].end.x), qRound(ay*boxes[i].end.yMax));
+ pa.setPoint(3, qRound(ax*boxes[i].start.x), qRound(ay*boxes[i].start.yMax));
+ pa.translate(xStart, yStart);
+ p.drawPolygon(pa);
+ }
+
+ // draw axis
+ p.setPen(Qt::black);
+ p.drawLine(xStart, yStart, w-1, yStart);
+ p.drawLine(xStart, yStart, xStart, 0);
+
+ // draw ticks and lines
+ p.setPen(Qt::DotLine);
+ for (it = yTicks.begin(); it!=yTicks.end(); ++it) {
+ int y1 = yStart + qRound(it.key()*(yEnd-yStart)/yMax);
+ QRect rect = f.boundingRect(it.data().s);
+ p.drawText(xStart/2-rect.width()/2 , y1+rect.height()/2, it.data().s);
+ int xmin = xStart + qRound(it.data().min*(xEnd-xStart)/xMax);
+ p.drawLine(xStart, y1, xmin, y1);
+ }
+ for (it = xTicks.begin(); it!=xTicks.end(); ++it) {
+ int x1 = xStart + qRound(it.key()*(xEnd-xStart)/xMax);
+ QRect rect = f.boundingRect(it.data().s);
+ p.drawText(x1-rect.width()/2, h-1, it.data().s);
+ int ymin = yStart + qRound(it.data().min*(yEnd-yStart)/yMax);
+ p.drawLine(x1, yStart, x1, ymin);
+ }
+
+ return pixmap;
+}
+
+QPixmap Device::vddGraph(const QString &xLabel, const QString &yLabel,
+ const QValueVector<Device::RangeBox> &boxes)
+{
+ uint sp = 10;
+ QPixmap graph = drawGraph(boxes);
+ QPainter p;
+ QFontMetrics f(p.font());
+ QPixmap pixmap(graph.width() + sp + f.width(xLabel), graph.height() + sp + f.lineSpacing());
+ pixmap.fill(Qt::white);
+ copyBlt(&pixmap, 0, f.lineSpacing() + sp, &graph, 0, 0, graph.width(), graph.height());
+ p.begin(&pixmap);
+ p.setPen(Qt::black);
+ p.drawText(0, f.lineSpacing(), yLabel);
+ p.drawText(pixmap.width()-1-f.width(xLabel), pixmap.height()-1, xLabel);
+ return pixmap;
+}
+
+const Device::Package *Device::barPackage(const char *name, const Device::Data &data)
+{
+ for (uint i=0; i<data.packages().count(); i++)
+ for (uint k=0; k<data.packages()[i].types.count(); k++)
+ if ( Package::TYPE_DATA[data.packages()[i].types[k]].name==name ) return &data.packages()[i];
+ return 0;
+}
+
+QPixmap Device::pinsGraph(const Device::Package &package)
+{
+ QPixmap pixmap;
+ QPainter p;
+ QFontMetrics fm(p.font());
+ uint nb = package.pins.count();
+ const int hspacing = 3, wspacing = 3, wmark = 10, wpin = 4;
+ int theight = fm.ascent() + (fm.ascent()%2==0 ? 1 : 0);
+ int height = hspacing + (nb/2)*(hspacing + theight);
+ int wnumber = fm.width("1");
+ wnumber = QMAX(wnumber, fm.width(QString::number(nb/2)));
+ wnumber = QMAX(wnumber, fm.width(QString::number(nb/2+1)));
+ wnumber = QMAX(wnumber, fm.width(QString::number(nb)));
+ int bwidth = 4*wspacing + 2*wnumber + wmark;
+ int lwidth = 0, rwidth = 0;
+ for (uint k=0; k<nb/2; k++) {
+ lwidth = QMAX(lwidth, fm.width(package.pins[k]));
+ rwidth = QMAX(rwidth, fm.width(package.pins[nb-k-1]));
+ }
+ int bx = lwidth + wspacing + wpin;
+ int width = bx + bwidth + wpin + wspacing + rwidth;
+ pixmap.resize(width, height);
+ pixmap.fill(Qt::white);
+ p.begin(&pixmap);
+ p.setPen(QPen(Qt::black, 2));
+ p.drawRect(bx, 1, bwidth, height-1);
+ p.drawArc(bx+wspacing+wnumber+wspacing, -wmark/2+2, wmark, wmark, 0, -180*16);
+ for (uint k=0; k<nb/2; k++) {
+ int h = hspacing + theight/2 + k*(hspacing + theight);
+ p.drawLine(bx-wpin-1, h, bx, h);
+ p.drawLine(bx+bwidth, h, bx+bwidth+wpin, h);
+ h += theight/2;
+ QString label = package.pins[k];
+ p.drawText(bx-wpin-wspacing-fm.width(label), h, label);
+ p.drawText(bx+bwidth+wpin+wspacing, h, package.pins[nb-k-1]);
+ uint pin = (k+1);
+ if ( pin==1 || pin==(nb/2) ) {
+ p.drawText(bx+wspacing, h, QString::number(pin));
+ label = QString::number(nb-k);
+ p.drawText(bx+bwidth-wspacing-fm.width(label), h, label);
+ }
+ }
+ p.end();
+ return pixmap;
+}
+
+QString Device::htmlInfo(const Device::Data &data, const QString &deviceHref, const QString &documentHtml)
+{
+ QString doc;
+
+ // title
+ doc += "<h1>";
+ bool first = true;
+ FOR_EACH(Special, special) {
+ for (uint k=0; k<data.frequencyRanges().count(); k++) {
+ if ( data.frequencyRanges()[k].special!=special ) continue;
+ if (first) first = false;
+ else doc += " / ";
+ doc += data.fname(special);
+ break;
+ }
+ }
+ doc += "</h1>";
+
+ doc += "<table>";
+ QString status = coloredString(data.status().label(), statusColor(data.status()));
+ doc += htmlTableRow(i18n("Status"), status);
+ if ( data.alternatives().count() ) {
+ QString s;
+ for (uint i=0; i<data.alternatives().count(); i++) {
+ if ( i!=0 ) s += ", ";
+ if ( deviceHref.isEmpty() ) s += data.alternatives()[i].upper();
+ else {
+ QString href = deviceHref.arg(data.alternatives()[i].upper());
+ s += QString("<a href=\"%1\">%2</a>").arg(href).arg(data.alternatives()[i].upper());
+ }
+ }
+ doc += htmlTableRow(i18n("Alternatives"), s);
+ }
+ doc += documentHtml;
+ doc += "</table>";
+
+ doc += "<hr />";
+ doc += "<table>";
+ doc += data.group().informationHtml(data);
+ QString s;
+ for (uint i=0; i<data.packages().count(); i++)
+ for (uint k=0; k<data.packages()[i].types.count(); k++)
+ s += i18n(Package::TYPE_DATA[data.packages()[i].types[k]].label) + QString("[%1] ").arg(data.packages()[i].pins.count());
+ doc += htmlTableRow(i18n("Packaging"), s);
+ doc += "</table>";
+
+ return doc;
+}
+
+QString Device::htmlPinDiagrams(const Device::Data &data, const QString &imagePrefix, QMimeSourceFactory *msf)
+{
+ QString doc;
+ // pins
+ const Package *package = 0;
+ for (uint i=0; Package::TYPE_DATA[i].name; i++) {
+ if ( Package::TYPE_DATA[i].shape!=Package::Bar ) continue;
+ package = barPackage(Package::TYPE_DATA[i].name, data);
+ if (package) break;
+ }
+ if (package) {
+ QPixmap pix = pinsGraph(*package);
+ doc += "<table cellpadding=\"3\"><tr bgcolor=\"gray\"><th align=\"center\">";
+ for (uint k=0; k<package->types.count(); k++) {
+ if ( k!=0 ) doc += " ";
+ doc += i18n(Package::TYPE_DATA[package->types[k]].label);
+ doc += "(" + QString::number(package->pins.count()) + ")";
+ }
+ doc += "</th></tr><tr><td align=\"center\">";
+ QString label = data.name() + "_pins_graph.png";
+ doc += "<img src=\"" + imagePrefix + label + "\" />";
+ if (msf) msf->setPixmap(label, pix);
+ doc += "</td></tr></table>";
+ }
+ return doc;
+}
+
+QString Device::htmlVoltageFrequencyGraphs(const Device::Data &data, const QString &imagePrefix, QMimeSourceFactory *msf)
+{
+ QString doc;
+ FOR_EACH(Special, special) {
+ for (uint k=0; k<data.frequencyRanges().count(); k++) {
+ const Device::FrequencyRange &fr = data.frequencyRanges()[k];
+ if ( fr.special!=special ) continue;
+ doc += "<h3>" + data.fname(special) + " - " + i18n("Temperature range: ") + fr.operatingCondition.label() + "</h3>";
+ QString label = data.name() + "_" + data.fname(special) + "_"
+ + fr.operatingCondition.key() + ".png";
+ doc += "<img src=\"" + imagePrefix + label + "\" />";
+ if (msf) msf->setPixmap(label, Device::vddGraph(i18n("F (MHz)"), i18n("Vdd (V)"), fr.vdds));
+ }
+ }
+ return doc;
+}
+
+QPixmap Device::memoryGraph(const QValueList<MemoryGraphData> &r)
+{
+ QValueList<MemoryGraphData> ranges = r;
+ QPixmap pixmap;
+ QPainter p;
+ QFontMetrics fm(p.font());
+ // order
+ qHeapSort(ranges);
+ // add empty ranges
+ QValueList<MemoryGraphData>::iterator it;
+ for (it=ranges.begin(); it!=ranges.end(); ) {
+ QValueList<MemoryGraphData>::iterator prev = it;
+ ++it;
+ if ( it==ranges.end() ) break;
+ if ( (*prev).endAddress+1==(*it).startAddress ) continue;
+ MemoryGraphData data;
+ data.startAddress = (*prev).endAddress + 1;
+ data.endAddress = (*it).startAddress-1;
+ ranges.insert(it, data);
+ }
+ // compute widths and total height
+ int theight = fm.ascent() + (fm.ascent()%2==0 ? 1 : 0);
+ int hspacing = 5;
+ int height = 1;
+ int w1 = 0, w2 = 0;
+ for (it=ranges.begin(); it!=ranges.end(); ++it) {
+ w1 = QMAX(w1, fm.width((*it).start));
+ w1 = QMAX(w1, fm.width((*it).end));
+ w2 = QMAX(w2, fm.width((*it).label));
+ (*it).height = 2*hspacing + theight;
+ if ( (*it).startAddress!=(*it).endAddress ) (*it).height += 2*theight;
+ height += (*it).height;
+ }
+ int wspacing = 4;
+ int width = wspacing + w1 + wspacing + wspacing + w2;
+ pixmap.resize(width, height);
+ pixmap.fill(Qt::white);
+ p.begin(&pixmap);
+ int h = 0;
+ // draw ranges
+ for (it=ranges.begin(); it!=ranges.end(); ++it) {
+ p.setPen(QPen(Qt::black, 1, Qt::DotLine));
+ p.drawLine(0,h, width-1,h);
+ p.setPen(QPen(Qt::black, 1));
+ p.setBrush((*it).label.isEmpty() ? Qt::gray : Qt::white);
+ p.drawRect(0,h, wspacing+w1+wspacing,(*it).height+1);
+ int hmid = h+(*it).height/2+theight/2;
+ p.drawText(wspacing+w1+wspacing+wspacing,hmid, (*it).label);
+ if ( (*it).startAddress==(*it).endAddress ) p.drawText(wspacing,hmid, (*it).start);
+ else {
+ p.drawText(wspacing,h+theight, (*it).start);
+ p.drawText(wspacing,h+(*it).height-3, (*it).end);
+ }
+ h += (*it).height;
+ p.setPen(QPen(Qt::black, 1, Qt::DotLine));
+ p.drawLine(0,h, width-1,h);
+ }
+ p.end();
+ return pixmap;
+}
+
+#endif