2009-04-25 01:35:41 +02:00
|
|
|
// Scintilla source code edit control
|
|
|
|
/** @file XPM.cxx
|
|
|
|
** Define a class that holds data in the X Pixmap (XPM) format.
|
|
|
|
**/
|
|
|
|
// Copyright 1998-2003 by Neil Hodgson <neilh@scintilla.org>
|
|
|
|
// The License.txt file describes the conditions under which this software may be distributed.
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2015-06-07 23:19:26 +02:00
|
|
|
#include <string.h>
|
2009-04-25 01:35:41 +02:00
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
|
2009-04-25 01:35:41 +02:00
|
|
|
#include "Platform.h"
|
|
|
|
|
|
|
|
#include "XPM.h"
|
|
|
|
|
|
|
|
#ifdef SCI_NAMESPACE
|
|
|
|
using namespace Scintilla;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static const char *NextField(const char *s) {
|
|
|
|
// In case there are leading spaces in the string
|
|
|
|
while (*s && *s == ' ') {
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
while (*s && *s != ' ') {
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
while (*s && *s == ' ') {
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Data lines in XPM can be terminated either with NUL or "
|
|
|
|
static size_t MeasureLength(const char *s) {
|
|
|
|
size_t i = 0;
|
|
|
|
while (s[i] && (s[i] != '\"'))
|
|
|
|
i++;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
ColourDesired XPM::ColourFromCode(int ch) const {
|
|
|
|
return colourCodeTable[ch];
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void XPM::FillRun(Surface *surface, int code, int startX, int y, int x) {
|
|
|
|
if ((code != codeTransparent) && (startX != x)) {
|
2015-06-07 23:19:26 +02:00
|
|
|
PRectangle rc = PRectangle::FromInts(startX, y, x, y + 1);
|
2009-04-25 01:35:41 +02:00
|
|
|
surface->FillRectangle(rc, ColourFromCode(code));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
XPM::XPM(const char *textForm) {
|
2009-04-25 01:35:41 +02:00
|
|
|
Init(textForm);
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
XPM::XPM(const char *const *linesForm) {
|
2009-04-25 01:35:41 +02:00
|
|
|
Init(linesForm);
|
|
|
|
}
|
|
|
|
|
|
|
|
XPM::~XPM() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void XPM::Init(const char *textForm) {
|
|
|
|
// Test done is two parts to avoid possibility of overstepping the memory
|
|
|
|
// if memcmp implemented strangely. Must be 4 bytes at least at destination.
|
|
|
|
if ((0 == memcmp(textForm, "/* X", 4)) && (0 == memcmp(textForm, "/* XPM */", 9))) {
|
|
|
|
// Build the lines form out of the text form
|
2013-08-28 02:44:27 +02:00
|
|
|
std::vector<const char *> linesForm = LinesFormFromTextForm(textForm);
|
|
|
|
if (!linesForm.empty()) {
|
|
|
|
Init(&linesForm[0]);
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// It is really in line form
|
|
|
|
Init(reinterpret_cast<const char * const *>(textForm));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-13 00:19:51 +02:00
|
|
|
void XPM::Init(const char *const *linesForm) {
|
2009-04-25 01:35:41 +02:00
|
|
|
height = 1;
|
|
|
|
width = 1;
|
|
|
|
nColours = 1;
|
2013-08-28 02:44:27 +02:00
|
|
|
pixels.clear();
|
2009-04-25 01:35:41 +02:00
|
|
|
codeTransparent = ' ';
|
|
|
|
if (!linesForm)
|
|
|
|
return;
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
std::fill(colourCodeTable, colourCodeTable+256, 0);
|
2009-04-25 01:35:41 +02:00
|
|
|
const char *line0 = linesForm[0];
|
|
|
|
width = atoi(line0);
|
|
|
|
line0 = NextField(line0);
|
|
|
|
height = atoi(line0);
|
2013-08-28 02:44:27 +02:00
|
|
|
pixels.resize(width*height);
|
2009-04-25 01:35:41 +02:00
|
|
|
line0 = NextField(line0);
|
|
|
|
nColours = atoi(line0);
|
|
|
|
line0 = NextField(line0);
|
|
|
|
if (atoi(line0) != 1) {
|
|
|
|
// Only one char per pixel is supported
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int c=0; c<nColours; c++) {
|
|
|
|
const char *colourDef = linesForm[c+1];
|
2013-08-28 02:44:27 +02:00
|
|
|
int code = static_cast<unsigned char>(colourDef[0]);
|
2009-04-25 01:35:41 +02:00
|
|
|
colourDef += 4;
|
2013-08-28 02:44:27 +02:00
|
|
|
ColourDesired colour(0xff, 0xff, 0xff);
|
2009-04-25 01:35:41 +02:00
|
|
|
if (*colourDef == '#') {
|
2013-08-28 02:44:27 +02:00
|
|
|
colour.Set(colourDef);
|
2009-04-25 01:35:41 +02:00
|
|
|
} else {
|
2015-06-07 23:19:26 +02:00
|
|
|
codeTransparent = static_cast<char>(code);
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
2013-08-28 02:44:27 +02:00
|
|
|
colourCodeTable[code] = colour;
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
for (int y=0; y<height; y++) {
|
|
|
|
const char *lform = linesForm[y+nColours+1];
|
|
|
|
size_t len = MeasureLength(lform);
|
2015-06-07 23:19:26 +02:00
|
|
|
for (size_t x = 0; x<len; x++)
|
2013-08-28 02:44:27 +02:00
|
|
|
pixels[y * width + x] = static_cast<unsigned char>(lform[x]);
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void XPM::Draw(Surface *surface, PRectangle &rc) {
|
2013-08-28 02:44:27 +02:00
|
|
|
if (pixels.empty()) {
|
2009-04-25 01:35:41 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Centre the pixmap
|
2015-06-07 23:19:26 +02:00
|
|
|
int startY = static_cast<int>(rc.top + (rc.Height() - height) / 2);
|
|
|
|
int startX = static_cast<int>(rc.left + (rc.Width() - width) / 2);
|
2010-07-13 00:19:51 +02:00
|
|
|
for (int y=0; y<height; y++) {
|
2009-04-25 01:35:41 +02:00
|
|
|
int prevCode = 0;
|
|
|
|
int xStartRun = 0;
|
|
|
|
for (int x=0; x<width; x++) {
|
2013-08-28 02:44:27 +02:00
|
|
|
int code = pixels[y * width + x];
|
2009-04-25 01:35:41 +02:00
|
|
|
if (code != prevCode) {
|
|
|
|
FillRun(surface, prevCode, startX + xStartRun, startY + y, startX + x);
|
|
|
|
xStartRun = x;
|
|
|
|
prevCode = code;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FillRun(surface, prevCode, startX + xStartRun, startY + y, startX + width);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
void XPM::PixelAt(int x, int y, ColourDesired &colour, bool &transparent) const {
|
|
|
|
if (pixels.empty() || (x<0) || (x >= width) || (y<0) || (y >= height)) {
|
|
|
|
colour = 0;
|
|
|
|
transparent = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int code = pixels[y * width + x];
|
|
|
|
transparent = code == codeTransparent;
|
|
|
|
if (transparent) {
|
|
|
|
colour = 0;
|
|
|
|
} else {
|
|
|
|
colour = ColourFromCode(code).AsLong();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<const char *> XPM::LinesFormFromTextForm(const char *textForm) {
|
2009-04-25 01:35:41 +02:00
|
|
|
// Build the lines form out of the text form
|
2013-08-28 02:44:27 +02:00
|
|
|
std::vector<const char *> linesForm;
|
2009-04-25 01:35:41 +02:00
|
|
|
int countQuotes = 0;
|
|
|
|
int strings=1;
|
|
|
|
int j=0;
|
|
|
|
for (; countQuotes < (2*strings) && textForm[j] != '\0'; j++) {
|
|
|
|
if (textForm[j] == '\"') {
|
|
|
|
if (countQuotes == 0) {
|
|
|
|
// First field: width, height, number of colors, chars per pixel
|
|
|
|
const char *line0 = textForm + j + 1;
|
|
|
|
// Skip width
|
|
|
|
line0 = NextField(line0);
|
|
|
|
// Add 1 line for each pixel of height
|
|
|
|
strings += atoi(line0);
|
|
|
|
line0 = NextField(line0);
|
|
|
|
// Add 1 line for each colour
|
|
|
|
strings += atoi(line0);
|
|
|
|
}
|
|
|
|
if (countQuotes / 2 >= strings) {
|
|
|
|
break; // Bad height or number of colors!
|
|
|
|
}
|
|
|
|
if ((countQuotes & 1) == 0) {
|
2013-08-28 02:44:27 +02:00
|
|
|
linesForm.push_back(textForm + j + 1);
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
countQuotes++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (textForm[j] == '\0' || countQuotes / 2 > strings) {
|
|
|
|
// Malformed XPM! Height + number of colors too high or too low
|
2013-08-28 02:44:27 +02:00
|
|
|
linesForm.clear();
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
return linesForm;
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
RGBAImage::RGBAImage(int width_, int height_, float scale_, const unsigned char *pixels_) :
|
|
|
|
height(height_), width(width_), scale(scale_) {
|
|
|
|
if (pixels_) {
|
|
|
|
pixelBytes.assign(pixels_, pixels_ + CountBytes());
|
|
|
|
} else {
|
|
|
|
pixelBytes.resize(CountBytes());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RGBAImage::RGBAImage(const XPM &xpm) {
|
|
|
|
height = xpm.GetHeight();
|
|
|
|
width = xpm.GetWidth();
|
|
|
|
scale = 1;
|
|
|
|
pixelBytes.resize(CountBytes());
|
|
|
|
for (int y=0; y<height; y++) {
|
|
|
|
for (int x=0; x<width; x++) {
|
|
|
|
ColourDesired colour;
|
|
|
|
bool transparent = false;
|
|
|
|
xpm.PixelAt(x, y, colour, transparent);
|
|
|
|
SetPixel(x, y, colour, transparent ? 0 : 255);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RGBAImage::~RGBAImage() {
|
|
|
|
}
|
|
|
|
|
|
|
|
int RGBAImage::CountBytes() const {
|
|
|
|
return width * height * 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
const unsigned char *RGBAImage::Pixels() const {
|
|
|
|
return &pixelBytes[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
void RGBAImage::SetPixel(int x, int y, ColourDesired colour, int alpha) {
|
|
|
|
unsigned char *pixel = &pixelBytes[0] + (y*width+x) * 4;
|
|
|
|
// RGBA
|
|
|
|
pixel[0] = static_cast<unsigned char>(colour.GetRed());
|
|
|
|
pixel[1] = static_cast<unsigned char>(colour.GetGreen());
|
|
|
|
pixel[2] = static_cast<unsigned char>(colour.GetBlue());
|
|
|
|
pixel[3] = static_cast<unsigned char>(alpha);
|
|
|
|
}
|
2009-04-25 01:35:41 +02:00
|
|
|
|
2015-06-07 23:19:26 +02:00
|
|
|
RGBAImageSet::RGBAImageSet() : height(-1), width(-1) {
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
RGBAImageSet::~RGBAImageSet() {
|
2009-04-25 01:35:41 +02:00
|
|
|
Clear();
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
/// Remove all images.
|
|
|
|
void RGBAImageSet::Clear() {
|
|
|
|
for (ImageMap::iterator it=images.begin(); it != images.end(); ++it) {
|
|
|
|
delete it->second;
|
|
|
|
it->second = 0;
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
2013-08-28 02:44:27 +02:00
|
|
|
images.clear();
|
2009-04-25 01:35:41 +02:00
|
|
|
height = -1;
|
|
|
|
width = -1;
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
/// Add an image.
|
|
|
|
void RGBAImageSet::Add(int ident, RGBAImage *image) {
|
|
|
|
ImageMap::iterator it=images.find(ident);
|
|
|
|
if (it == images.end()) {
|
|
|
|
images[ident] = image;
|
|
|
|
} else {
|
|
|
|
delete it->second;
|
|
|
|
it->second = image;
|
|
|
|
}
|
2009-04-25 01:35:41 +02:00
|
|
|
height = -1;
|
|
|
|
width = -1;
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
/// Get image by id.
|
|
|
|
RGBAImage *RGBAImageSet::Get(int ident) {
|
|
|
|
ImageMap::iterator it = images.find(ident);
|
|
|
|
if (it != images.end()) {
|
|
|
|
return it->second;
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
2013-08-28 02:44:27 +02:00
|
|
|
return NULL;
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
/// Give the largest height of the set.
|
|
|
|
int RGBAImageSet::GetHeight() const {
|
2009-04-25 01:35:41 +02:00
|
|
|
if (height < 0) {
|
2013-08-28 02:44:27 +02:00
|
|
|
for (ImageMap::const_iterator it=images.begin(); it != images.end(); ++it) {
|
|
|
|
if (height < it->second->GetHeight()) {
|
|
|
|
height = it->second->GetHeight();
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (height > 0) ? height : 0;
|
|
|
|
}
|
|
|
|
|
2013-08-28 02:44:27 +02:00
|
|
|
/// Give the largest width of the set.
|
|
|
|
int RGBAImageSet::GetWidth() const {
|
2009-04-25 01:35:41 +02:00
|
|
|
if (width < 0) {
|
2013-08-28 02:44:27 +02:00
|
|
|
for (ImageMap::const_iterator it=images.begin(); it != images.end(); ++it) {
|
|
|
|
if (width < it->second->GetWidth()) {
|
|
|
|
width = it->second->GetWidth();
|
2009-04-25 01:35:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (width > 0) ? width : 0;
|
|
|
|
}
|