-
Notifications
You must be signed in to change notification settings - Fork 0
/
ArchivoImagenPNG.cpp
119 lines (109 loc) · 3.69 KB
/
ArchivoImagenPNG.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#include "ArchivoImagenPNG.h"
bool ArchivoImagenPNG::guardarDatos (string cadena, const unsigned int offset){
// en el caso de jpg ese offset es relativo al offset donde empieza la informacion
// si offset = 0 , entonces el verdaderoOffset = offsetInicial + 0
// por ej ?0000000000000000000 (el primer byte es el separador)
string nombre = this->obtenerNombreCompleto();
unsigned int verdaderoOffset = this->offsetInicial + offset;
fstream f;
f.open(nombre.c_str(), ios::in | ios::out | ios::binary);
if(f.good()) {
f.seekp(verdaderoOffset, ios::beg);
f.write(cadena.c_str(), cadena.size());
return true;
}
return false;
}
string ArchivoImagenPNG::recuperarDatos (const unsigned int offset, const unsigned int bitsInformacion ){
string nombre = this->obtenerNombreCompleto();
char* buffer = new char[1];
string cadena;
//1- calcular el offset verdadero
unsigned int verdaderoOffset = this->offsetInicial + offset;
unsigned int bytesTotal = bitsInformacion/ BITS_BYTE;
// 2- abrir el archivo y hacer un seek a esa posicion
fstream f;
f.open(nombre.c_str(), ios::in | ios::out | ios::binary);
/*
3- leer secuencialmente los bytes del archivo hasta haber leido
(bitsInformacion/BITS_BYTE) bytes
BITS_BYTE es un define = 8
bits/byte
*/
if(f.good()) {
f.seekg(verdaderoOffset, ios::beg);
for(unsigned int i=0; i<bytesTotal; i++) {
if(!f.eof()) {
f.read(buffer, sizeof(buffer[0]));
cadena += buffer[0];
}
}
f.close();
delete[] buffer;
return cadena;
}
delete[] buffer;
return "";
}
unsigned int ArchivoImagenPNG::calcOffsetInicial() {
unsigned int offset = 0;
// si no fue calculada, obtenerla
if(this->offsetInicial == 0) {
// buscar de atras para adelante el caracter especial.
string nombre = this->obtenerNombreCompleto();
char* buffer = new char[1];
string cadena;
fstream f;
f.open(nombre.c_str(), ios::in | ios::binary);
if(f.good()) {
f.seekg(0, ios::end);
offset = f.tellg();
int i = offset;
while(i>0) {
f.read(buffer, sizeof(buffer[0]));
if(strcmp(buffer,"?")==0) {
offset=i;
i=0;
}
i--;
}
f.close();
}
delete[] buffer;
this->offsetInicial = offset;
}
return this->offsetInicial;
}
void ArchivoImagenPNG::preProcesar(){
ofstream salida;
char* relleno = new char[1];
relleno[0] = '0';
unsigned int tamanoOriginal = this->tamano;
//Apertura del png original
Image image;
string nombre = this->obtenerNombreCompleto();
image.read(nombre); //imagen original
//Cambio formato de Compresion
image.compressType(LZWCompression);
image.write(nombre);
unsigned int tamanoComprimida = this->calcularTamano(nombre);
// rellenar la imagen manteniendo el tamano original
int bytesRelleno = (tamanoOriginal - tamanoComprimida);
if (bytesRelleno > 0){
this->espacioDisponible = bytesRelleno;
this->bytesLibres = bytesRelleno;
this->offsetInicial= tamanoComprimida;
salida.open (nombre.c_str(), ios::out | ios::app | ios::binary);
if (salida.good()) {
for(int i=0; i<bytesRelleno; i++)
salida.write(reinterpret_cast<char *> (&relleno[0]), sizeof(relleno[0]));
salida.close();
}
}
else {
this->espacioDisponible = 0;
this->bytesLibres =0;
this->offsetInicial= 0;
}
delete[] relleno;
}