#include #include "common/String.h" #include #include #include #include #include #include "Format.h" #include "graphics/Graphics.h" ByteString format::URLEncode(ByteString source) { char * src = (char *)source.c_str(); char * dst = new char[(source.length()*3)+2]; std::fill(dst, dst+(source.length()*3)+2, 0); char *d; for (d = dst; *d; d++) ; for (unsigned char *s = (unsigned char *)src; *s; s++) { if ((*s>='0' && *s<='9') || (*s>='a' && *s<='z') || (*s>='A' && *s<='Z')) *(d++) = *s; else { *(d++) = '%'; *(d++) = hex[*s>>4]; *(d++) = hex[*s&15]; } } *d = 0; ByteString finalString(dst); delete[] dst; return finalString; } ByteString format::UnixtimeToDate(time_t unixtime, ByteString dateFormat) { struct tm * timeData; char buffer[128]; timeData = localtime(&unixtime); strftime(buffer, 128, dateFormat.c_str(), timeData); return ByteString(buffer); } ByteString format::UnixtimeToDateMini(time_t unixtime) { time_t currentTime = time(NULL); struct tm currentTimeData = *localtime(¤tTime); struct tm timeData = *localtime(&unixtime); if(currentTimeData.tm_year != timeData.tm_year) { return UnixtimeToDate(unixtime, "%b %Y"); } else if(currentTimeData.tm_mon != timeData.tm_mon || currentTimeData.tm_mday != timeData.tm_mday) { return UnixtimeToDate(unixtime, "%d %B"); } else { return UnixtimeToDate(unixtime, "%H:%M:%S"); } } String format::CleanString(String dirtyString, bool ascii, bool color, bool newlines, bool numeric) { for (size_t i = 0; i < dirtyString.size(); i++) { switch(dirtyString[i]) { case '\b': if (color) { dirtyString.erase(i, 2); i--; } else i++; break; case '\x0E': if (color) { dirtyString.erase(i, 1); i--; } break; case '\x0F': if (color) { dirtyString.erase(i, 4); i--; } else i += 3; break; case '\r': case '\n': if (newlines) dirtyString[i] = ' '; break; default: if (numeric && (dirtyString[i] < '0' || dirtyString[i] > '9')) { dirtyString.erase(i, 1); i--; } // if less than ascii 20 or greater than ascii 126, delete else if (ascii && (dirtyString[i] < ' ' || dirtyString[i] > '~')) { dirtyString.erase(i, 1); i--; } break; } } return dirtyString; } std::vector format::VideoBufferToPTI(const VideoBuffer & vidBuf) { std::vector data; int dataSize = 0; char * buffer = (char*)Graphics::ptif_pack(vidBuf.Buffer, vidBuf.Width, vidBuf.Height, &dataSize); if(buffer) { data.insert(data.end(), buffer, buffer+dataSize); free(buffer); } return data; } VideoBuffer * format::PTIToVideoBuffer(std::vector & data) { int newWidth, newHeight; pixel * buffer = Graphics::ptif_unpack(&data[0], data.size(), &newWidth, &newHeight); if(buffer) { VideoBuffer * vb = new VideoBuffer(buffer, newWidth, newHeight); free(buffer); return vb; } return NULL; } std::vector format::VideoBufferToBMP(const VideoBuffer & vidBuf) { std::vector data; char buffer[54] = "BM"; int padding = 3 - (vidBuf.Width * 3 + 3) % 4; unsigned int fileSize = (vidBuf.Width * 3 + padding) * vidBuf.Height + 54; *(int *)(buffer + 2) = fileSize; *(short int *)(buffer + 6) = 0; // reserved; *(short int *)(buffer + 8) = 0; // reserved; *(int *)(buffer + 10) = 0x36; // 54 bytes from start to data *(int *)(buffer + 14) = 0x28; // 40 bytes in info header *(int *)(buffer + 18) = vidBuf.Width; *(int *)(buffer + 22) = vidBuf.Height; *(short int *)(buffer + 26) = 1; // 1 plane *(short int *)(buffer + 28) = 24; // 24 bits per pixel *(int *)(buffer + 30) = 0; // no compression *(int *)(buffer + 34) = fileSize - 54; // bitmap size *(int *)(buffer + 38) = 0xB13; // 72dpi *(int *)(buffer + 42) = 0xB13; // 72dpi *(int *)(buffer + 46) = 0; // all colors used *(int *)(buffer + 50) = 0; // all colors important data.insert(data.end(), buffer, buffer+54); char *currentRow = (char *)malloc((vidBuf.Width * 3) + padding); for(int y = vidBuf.Height - 1; y >= 0; y--) { int rowPos = 0; for(int x = 0; x < vidBuf.Width; x++) { currentRow[rowPos++] = PIXB(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXG(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXR(vidBuf.Buffer[(y*vidBuf.Width)+x]); } data.insert(data.end(), currentRow, currentRow+(vidBuf.Width*3)+padding); } free(currentRow); return data; } std::vector format::VideoBufferToPPM(const VideoBuffer & vidBuf) { std::vector data; char buffer[256]; sprintf(buffer, "P6\n%d %d\n255\n", vidBuf.Width, vidBuf.Height); data.insert(data.end(), buffer, buffer+strlen(buffer)); unsigned char * currentRow = new unsigned char[vidBuf.Width*3]; for(int y = 0; y < vidBuf.Height; y++) { int rowPos = 0; for(int x = 0; x < vidBuf.Width; x++) { currentRow[rowPos++] = PIXR(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXG(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXB(vidBuf.Buffer[(y*vidBuf.Width)+x]); } data.insert(data.end(), currentRow, currentRow+(vidBuf.Width*3)); } delete [] currentRow; return data; } struct PNGChunk { int Length; char Name[4]; char * Data; //char[4] CRC(); PNGChunk(int length, ByteString name) { if (name.length()!=4) throw std::runtime_error("Invalid chunk name"); std::copy(name.begin(), name.begin()+4, Name); Length = length; if (length) { Data = new char[length]; std::fill(Data, Data+length, 0); } else { Data = NULL; } } unsigned long CRC() { if (!Data) { return format::CalculateCRC((unsigned char*)Name, 4); } else { unsigned char * temp = new unsigned char[4+Length]; std::copy(Name, Name+4, temp); std::copy(Data, Data+Length, temp+4); unsigned long tempRet = format::CalculateCRC(temp, 4+Length); delete[] temp; return tempRet; } } ~PNGChunk() { delete[] Data; } }; std::vector format::VideoBufferToPNG(const VideoBuffer & vidBuf) { std::vector chunks; //Begin IHDR (Image header) chunk (Image size and depth) PNGChunk *IHDRChunk = new PNGChunk(13, "IHDR"); //Image Width IHDRChunk->Data[0] = (vidBuf.Width>>24)&0xFF; IHDRChunk->Data[1] = (vidBuf.Width>>16)&0xFF; IHDRChunk->Data[2] = (vidBuf.Width>>8)&0xFF; IHDRChunk->Data[3] = (vidBuf.Width)&0xFF; //Image Height IHDRChunk->Data[4] = (vidBuf.Height>>24)&0xFF; IHDRChunk->Data[5] = (vidBuf.Height>>16)&0xFF; IHDRChunk->Data[6] = (vidBuf.Height>>8)&0xFF; IHDRChunk->Data[7] = (vidBuf.Height)&0xFF; //Bit depth IHDRChunk->Data[8] = 8; //8bits per channel or 24bpp //Colour type IHDRChunk->Data[9] = 2; //RGB triple //Everything else is default chunks.push_back(IHDRChunk); //Begin image data, format is 8bit RGB (24bit pixel) int dataPos = 0; unsigned char * uncompressedData = new unsigned char[(vidBuf.Width*vidBuf.Height*3)+vidBuf.Height]; //Byte ordering and filtering unsigned char * previousRow = new unsigned char[vidBuf.Width*3]; std::fill(previousRow, previousRow+(vidBuf.Width*3), 0); unsigned char * currentRow = new unsigned char[vidBuf.Width*3]; for(int y = 0; y < vidBuf.Height; y++) { int rowPos = 0; for(int x = 0; x < vidBuf.Width; x++) { currentRow[rowPos++] = PIXR(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXG(vidBuf.Buffer[(y*vidBuf.Width)+x]); currentRow[rowPos++] = PIXB(vidBuf.Buffer[(y*vidBuf.Width)+x]); } uncompressedData[dataPos++] = 2; //Up Sub(x) filter for(int b = 0; b < rowPos; b++) { int filteredByte = (currentRow[b]-previousRow[b])&0xFF; uncompressedData[dataPos++] = filteredByte; } unsigned char * tempRow = previousRow; previousRow = currentRow; currentRow = tempRow; } delete[] currentRow; delete[] previousRow; //Compression int compressedBufferSize = (vidBuf.Width*vidBuf.Height*3)*2; unsigned char * compressedData = new unsigned char[compressedBufferSize]; int result; z_stream zipStream; zipStream.zalloc = Z_NULL; zipStream.zfree = Z_NULL; zipStream.opaque = Z_NULL; result = deflateInit2(&zipStream, 9, // level Z_DEFLATED, // method 10, // windowBits 1, // memLevel Z_DEFAULT_STRATEGY // strategy ); if (result != Z_OK) exit(result); zipStream.next_in = uncompressedData; zipStream.avail_in = dataPos; zipStream.next_out = compressedData; zipStream.avail_out = compressedBufferSize; result = deflate(&zipStream, Z_FINISH); if (result != Z_STREAM_END) exit(result); int compressedSize = compressedBufferSize-zipStream.avail_out; PNGChunk *IDATChunk = new PNGChunk(compressedSize, "IDAT"); std::copy(compressedData, compressedData+compressedSize, IDATChunk->Data); chunks.push_back(IDATChunk); deflateEnd(&zipStream); delete[] compressedData; delete[] uncompressedData; PNGChunk *IENDChunk = new PNGChunk(0, "IEND"); chunks.push_back(IENDChunk); //Write chunks to output buffer int finalDataSize = 8; for(std::vector::iterator iter = chunks.begin(), end = chunks.end(); iter != end; ++iter) { PNGChunk * cChunk = *iter; finalDataSize += 4 + 4 + 4; finalDataSize += cChunk->Length; } unsigned char * finalData = new unsigned char[finalDataSize]; int finalDataPos = 0; //PNG File header finalData[finalDataPos++] = 0x89; finalData[finalDataPos++] = 0x50; finalData[finalDataPos++] = 0x4E; finalData[finalDataPos++] = 0x47; finalData[finalDataPos++] = 0x0D; finalData[finalDataPos++] = 0x0A; finalData[finalDataPos++] = 0x1A; finalData[finalDataPos++] = 0x0A; for(std::vector::iterator iter = chunks.begin(), end = chunks.end(); iter != end; ++iter) { PNGChunk * cChunk = *iter; //Chunk length finalData[finalDataPos++] = (cChunk->Length>>24)&0xFF; finalData[finalDataPos++] = (cChunk->Length>>16)&0xFF; finalData[finalDataPos++] = (cChunk->Length>>8)&0xFF; finalData[finalDataPos++] = (cChunk->Length)&0xFF; //Chunk name std::copy(cChunk->Name, cChunk->Name+4, finalData+finalDataPos); finalDataPos += 4; //Chunk data if(cChunk->Data) { std::copy(cChunk->Data, cChunk->Data+cChunk->Length, finalData+finalDataPos); finalDataPos += cChunk->Length; } //Chunk CRC unsigned long tempCRC = cChunk->CRC(); finalData[finalDataPos++] = (tempCRC>>24)&0xFF; finalData[finalDataPos++] = (tempCRC>>16)&0xFF; finalData[finalDataPos++] = (tempCRC>>8)&0xFF; finalData[finalDataPos++] = (tempCRC)&0xFF; delete cChunk; } std::vector outputData(finalData, finalData+finalDataPos); delete[] finalData; return outputData; } //CRC functions, copypasta from W3 PNG spec. /* Table of CRCs of all 8-bit messages. */ unsigned long crc_table[256]; /* Flag: has the table been computed? Initially false. */ int crc_table_computed = 0; /* Make the table for a fast CRC. */ void make_crc_table(void) { unsigned long c; int n, k; for (n = 0; n < 256; n++) { c = (unsigned long) n; for (k = 0; k < 8; k++) { if (c & 1) c = 0xedb88320L ^ (c >> 1); else c = c >> 1; } crc_table[n] = c; } crc_table_computed = 1; } /* Update a running CRC with the bytes buf[0..len-1]--the CRC should be initialized to all 1's, and the transmitted value is the 1's complement of the final running CRC (see the crc() routine below)). */ unsigned long update_crc(unsigned long crc, unsigned char *buf, int len) { unsigned long c = crc; int n; if (!crc_table_computed) make_crc_table(); for (n = 0; n < len; n++) { c = crc_table[(c ^ buf[n]) & 0xff] ^ (c >> 8); } return c; } unsigned long format::CalculateCRC(unsigned char * data, int len) { return update_crc(0xffffffffL, data, len) ^ 0xffffffffL; }