porymap/src/ui/overlay.cpp
2022-10-15 11:22:55 -07:00

252 lines
6.4 KiB
C++

#include "overlay.h"
#include "scripting.h"
#include "log.h"
void OverlayText::render(QPainter *painter) {
QFont font = painter->font();
font.setPixelSize(this->fontSize);
painter->setFont(font);
painter->setPen(this->color);
painter->drawStaticText(this->x, this->y, this->text);
}
void OverlayPath::render(QPainter *painter) {
painter->fillPath(this->path, this->fillColor);
painter->setPen(this->borderColor);
painter->drawPath(this->path);
}
void OverlayImage::render(QPainter *painter) {
painter->drawImage(this->x, this->y, this->image);
}
void Overlay::renderItems(QPainter *painter) {
if (this->hidden) return;
painter->save();
QTransform transform = painter->transform();
transform.translate(this->x, this->y);
transform.rotate(this->angle);
transform.scale(this->hScale, this->vScale);
painter->setTransform(transform);
if (this->clippingRect) {
painter->setClipping(true);
painter->setClipRect(*this->clippingRect);
}
painter->setOpacity(this->opacity);
for (auto item : this->items)
item->render(painter);
painter->restore();
}
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;
}
void Overlay::setHidden(bool hidden) {
this->hidden = hidden;
}
int Overlay::getOpacity() {
return this->opacity * 100;
}
void Overlay::setOpacity(int opacity) {
if (opacity < 0 || opacity > 100) {
logError(QString("Invalid overlay opacity '%1', must be in range 0-100").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;
}
qreal Overlay::getHScale() {
return this->hScale;
}
qreal Overlay::getVScale() {
return this->vScale;
}
void Overlay::setHScale(qreal scale) {
this->hScale = scale;
}
void Overlay::setVScale(qreal scale) {
this->vScale = scale;
}
int Overlay::getRotation() {
return this->angle;
}
void Overlay::setRotation(int angle) {
this->angle = angle;
this->clampAngle();
}
void Overlay::rotate(int degrees) {
this->angle += degrees;
this->clampAngle();
}
void Overlay::clampAngle() {
// transform.rotate would handle this already, but we only
// want to report angles 0-359 for Overlay::getRotation
this->angle %= 360;
if (this->angle < 0)
this->angle += 360;
}
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) {
if (colorStr.isEmpty())
colorStr = "transparent";
QColor color = QColor(colorStr);
if (!color.isValid()) {
logWarn(QString("Invalid overlay color \"%1\". Colors can be in the format \"#RRGGBB\" or \"#AARRGGBB\"").arg(colorStr));
color = QColor("transparent");
}
return color;
}
void Overlay::addText(const QString text, int x, int y, QString colorStr, int fontSize) {
this->items.append(new OverlayText(text, x, y, getColor(colorStr), fontSize));
}
bool Overlay::addRect(int x, int y, int width, int height, QString borderColorStr, QString fillColorStr, int rounding) {
if (rounding < 0 || rounding > 100) {
logError(QString("Invalid rectangle rounding '%1', must be in range 0-100").arg(rounding));
return false;
}
QPainterPath path;
path.addRoundedRect(QRectF(x, y, width, height), rounding, rounding, Qt::RelativeSize);
this->items.append(new OverlayPath(path, getColor(borderColorStr), getColor(fillColorStr)));
return true;
}
bool Overlay::addPath(QList<int> xCoords, QList<int> yCoords, QString borderColorStr, QString fillColorStr) {
int numPoints = qMin(xCoords.length(), yCoords.length());
if (numPoints < 2) {
logError("Overlay path must have at least two points.");
return false;
}
QPainterPath path;
path.moveTo(xCoords.at(0), yCoords.at(0));
for (int i = 1; i < numPoints; i++)
path.lineTo(xCoords.at(i), yCoords.at(i));
this->items.append(new OverlayPath(path, getColor(borderColorStr), getColor(fillColorStr)));
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));
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;
}