porymap/src/ui/overlay.cpp

230 lines
6 KiB
C++
Raw Normal View History

#include "overlay.h"
2021-12-07 19:44:40 +00:00
#include "scripting.h"
#include "log.h"
void OverlayText::render(QPainter *painter, int x, int y) {
QFont font = painter->font();
font.setPixelSize(this->fontSize);
painter->setFont(font);
painter->setPen(this->color);
2022-10-11 04:47:42 +01:00
painter->drawStaticText(this->x + x, this->y + y, this->text);
}
void OverlayRect::render(QPainter *painter, int x, int y) {
QRectF rect(this->x + x, this->y + y, this->width, this->height);
painter->setPen(this->borderColor);
painter->drawRect(rect);
painter->fillRect(rect, this->fillColor);
}
void OverlayImage::render(QPainter *painter, int x, int y) {
painter->drawImage(this->x + x, this->y + y, this->image);
}
2022-10-11 15:34:09 +01:00
void OverlayPath::render(QPainter *painter, int x, int y) {
if (x != this->prevX || y != this->prevY) {
// Overlay has moved since the path was last drawn
path.translate(x - prevX, y - prevY);
}
this->prevX = x;
this->prevY = y;
painter->setPen(this->color);
painter->drawPath(this->path);
}
2021-11-30 17:58:39 +00:00
void Overlay::renderItems(QPainter *painter) {
2021-11-30 18:38:06 +00:00
if (this->hidden) return;
if (this->clippingRect) {
painter->setClipping(true);
painter->setClipRect(*this->clippingRect);
}
qreal oldOpacity = painter->opacity();
2021-12-31 21:12:04 +00:00
painter->setOpacity(this->opacity);
2021-11-30 18:38:06 +00:00
for (auto item : this->items)
item->render(painter, this->x, this->y);
painter->setOpacity(oldOpacity);
if (this->clippingRect) {
painter->setClipping(false);
}
2021-11-30 17:58:39 +00:00
}
void Overlay::clearItems() {
for (auto item : this->items) {
delete item;
}
this->items.clear();
}
QList<OverlayItem*> Overlay::getItems() {
return this->items;
}
bool Overlay::getHidden() {
return this->hidden;
}
2021-11-30 18:38:06 +00:00
void Overlay::setHidden(bool hidden) {
this->hidden = hidden;
}
2021-12-31 21:12:04 +00:00
int Overlay::getOpacity() {
return this->opacity * 100;
}
void Overlay::setOpacity(int opacity) {
if (opacity < 0 || opacity > 100) {
logError(QString("Invalid overlay opacity '%1'").arg(opacity));
return;
}
this->opacity = static_cast<qreal>(opacity) / 100;
}
int Overlay::getX() {
return this->x;
}
int Overlay::getY() {
return this->y;
}
void Overlay::setX(int x) {
this->x = x;
}
void Overlay::setY(int y) {
this->y = y;
}
void Overlay::setClippingRect(QRectF rect) {
if (this->clippingRect) {
delete this->clippingRect;
}
this->clippingRect = new QRectF(rect);
}
void Overlay::clearClippingRect() {
if (this->clippingRect) {
delete this->clippingRect;
}
this->clippingRect = nullptr;
}
void Overlay::setPosition(int x, int y) {
this->x = x;
this->y = y;
}
void Overlay::move(int deltaX, int deltaY) {
this->x += deltaX;
this->y += deltaY;
}
QColor Overlay::getColor(QString colorStr, bool * ok) {
if (colorStr.isEmpty())
colorStr = "transparent";
QColor color = QColor(colorStr);
if (!color.isValid()) {
logError(QString("\"%1\" is not a valid color. Colors can be in the format \"#RRGGBB\" or \"#AARRGGBB\"").arg(colorStr));
if (ok) *ok = false; // Not set to true in here, allow for repeat usage
}
return color;
}
bool Overlay::addText(const QString text, int x, int y, QString colorStr, int fontSize) {
bool ok = true;
QColor color = getColor(colorStr, &ok);
if (!ok) return false;
this->items.append(new OverlayText(text, x, y, color, fontSize));
return true;
}
bool Overlay::addRect(int x, int y, int width, int height, QString borderColorStr, QString fillColorStr) {
bool ok = true;
QColor borderColor = getColor(borderColorStr, &ok);
QColor fillColor = getColor(fillColorStr, &ok);
if (!ok) return false;
this->items.append(new OverlayRect(x, y, width, height, borderColor, fillColor));
return true;
}
bool Overlay::addPath(QList<int> x, QList<int> y, QString colorStr) {
bool ok = true;
QColor color = getColor(colorStr, &ok);
if (!ok) return false;
int numPoints = qMin(x.length(), y.length());
if (numPoints < 2) {
logError("Overlay path must have at least two points.");
return false;
}
QPainterPath path;
path.moveTo(x.at(0), y.at(0));
for (int i = 1; i < numPoints; i++)
path.lineTo(x.at(i), y.at(i));
this->items.append(new OverlayPath(path, color));
return true;
}
bool Overlay::addImage(int x, int y, QString filepath, bool useCache, int width, int height, int xOffset, int yOffset, qreal hScale, qreal vScale, QList<QRgb> palette, bool setTransparency) {
QImage image = useCache ? Scripting::getImage(filepath) : QImage(filepath);
if (image.isNull()) {
logError(QString("Failed to load image '%1'").arg(filepath));
return false;
}
int fullWidth = image.width();
int fullHeight = image.height();
// Negative values used as an indicator for "use full dimension"
if (width <= 0)
width = fullWidth;
if (height <= 0)
height = fullHeight;
if (xOffset < 0) xOffset = 0;
if (yOffset < 0) yOffset = 0;
if (width + xOffset > fullWidth || height + yOffset > fullHeight) {
logError(QString("%1x%2 image starting at (%3,%4) exceeds the image size for '%5'")
.arg(width)
.arg(height)
.arg(xOffset)
.arg(yOffset)
.arg(filepath));
return false;
}
// Get specified subset of image
if (width != fullWidth || height != fullHeight)
image = image.copy(xOffset, yOffset, width, height);
if (hScale != 1 || vScale != 1)
image = image.transformed(QTransform().scale(hScale, vScale));
2021-12-08 14:20:17 +00:00
for (int i = 0; i < palette.size(); i++)
image.setColor(i, palette.at(i));
if (setTransparency)
image.setColor(0, qRgba(0, 0, 0, 0));
this->items.append(new OverlayImage(x, y, image));
return true;
}
bool Overlay::addImage(int x, int y, QImage image) {
if (image.isNull()) {
logError(QString("Failed to load custom image"));
return false;
}
this->items.append(new OverlayImage(x, y, image));
return true;
}