mirror of https://github.com/fltk/fltk
Redo Fluid_Image class to use Fl_Shared_Image instead of its own subclasses
to read files... git-svn-id: file:///fltk/svn/fltk/branches/branch-1.1@1730 ea41ed52-d2ee-0310-a9c1-e6b18d33e121
This commit is contained in:
parent
c147aca054
commit
2b826b0f31
3
CHANGES
3
CHANGES
|
@ -67,6 +67,9 @@ CHANGES IN FLTK 1.1.0b6
|
|||
printf's in some of the newer widgets.
|
||||
- The default behavior of Fl::error() is now to display
|
||||
an error but not to exit; Fl::fatal() still exits.
|
||||
- FLUID now uses the Fl_Shared_Image class, so FLUID-
|
||||
generated GUIs can embed any of the supported image
|
||||
file formats.
|
||||
|
||||
|
||||
CHANGES IN FLTK 1.1.0b5
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
//
|
||||
// "$Id: Fluid_Image.cxx,v 1.7.2.9.2.4 2001/11/22 15:35:01 easysw Exp $"
|
||||
// "$Id: Fluid_Image.cxx,v 1.7.2.9.2.5 2001/11/25 22:51:34 easysw Exp $"
|
||||
//
|
||||
// Pixmap label support for the Fast Light Tool Kit (FLTK).
|
||||
//
|
||||
|
@ -37,319 +37,98 @@
|
|||
extern void goto_source_dir(); // in fluid.C
|
||||
extern void leave_source_dir(); // in fluid.C
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
#include <FL/Fl_Pixmap.H>
|
||||
|
||||
class pixmap_image : public Fluid_Image {
|
||||
protected:
|
||||
Fl_Pixmap *p;
|
||||
int numlines;
|
||||
int *linelength;
|
||||
public:
|
||||
pixmap_image(const char *name, FILE *);
|
||||
~pixmap_image();
|
||||
virtual void image(Fl_Widget *); // set the image of this widget
|
||||
virtual void deimage(Fl_Widget *); // set the deimage of this widget
|
||||
virtual void write_static();
|
||||
virtual void write_code(int inactive = 0);
|
||||
static int test_file(char *buffer);
|
||||
};
|
||||
|
||||
int pixmap_image::test_file(char *buffer) {
|
||||
return (strstr(buffer,"/* XPM") != 0);
|
||||
void Fluid_Image::image(Fl_Widget *o) {
|
||||
o->image(img);
|
||||
}
|
||||
|
||||
void pixmap_image::image(Fl_Widget *o) {
|
||||
o->image(p);
|
||||
void Fluid_Image::deimage(Fl_Widget *o) {
|
||||
o->deimage(img);
|
||||
}
|
||||
|
||||
void pixmap_image::deimage(Fl_Widget *o) {
|
||||
o->deimage(p);
|
||||
}
|
||||
static int pixmap_header_written = 0;
|
||||
static int bitmap_header_written = 0;
|
||||
static int image_header_written = 0;
|
||||
|
||||
static int pixmap_header_written;
|
||||
void Fluid_Image::write_static() {
|
||||
if (!img) return;
|
||||
if (img->count() > 1) {
|
||||
// Write Pixmap data...
|
||||
write_c("\n");
|
||||
if (pixmap_header_written != write_number) {
|
||||
write_c("#include <FL/Fl_Pixmap.H>\n");
|
||||
pixmap_header_written = write_number;
|
||||
}
|
||||
write_c("static const char *%s[] = {\n",
|
||||
unique_id(this, "idata", filename_name(name()), 0));
|
||||
write_cstring(img->data()[0], strlen(img->data()[0]));
|
||||
|
||||
void pixmap_image::write_static() {
|
||||
if (!p) return;
|
||||
write_c("\n");
|
||||
if (pixmap_header_written != write_number) {
|
||||
write_c("#include <FL/Fl_Pixmap.H>\n");
|
||||
pixmap_header_written = write_number;
|
||||
int i;
|
||||
int ncolors, chars_per_color;
|
||||
sscanf(img->data()[0], "%*d%*d%d%d", &ncolors, &chars_per_color);
|
||||
|
||||
if (ncolors < 0) {
|
||||
write_c(",\n");
|
||||
write_cstring(img->data()[1], ncolors * -4);
|
||||
i = 2;
|
||||
} else {
|
||||
for (i = 1; i <= ncolors; i ++) {
|
||||
write_c(",\n");
|
||||
write_cstring(img->data()[i], strlen(img->data()[i]));
|
||||
}
|
||||
}
|
||||
for (; i < img->count(); i ++) {
|
||||
write_c(",\n");
|
||||
write_cstring(img->data()[i], img->w() * chars_per_color);
|
||||
}
|
||||
write_c("\n};\n");
|
||||
write_c("static Fl_Pixmap %s(%s);\n",
|
||||
unique_id(this, "image", filename_name(name()), 0),
|
||||
unique_id(this, "idata", filename_name(name()), 0));
|
||||
} else if (img->d() == 0) {
|
||||
// Write Bitmap data...
|
||||
write_c("\n");
|
||||
if (bitmap_header_written != write_number) {
|
||||
write_c("#include <FL/Fl_Bitmap.H>\n");
|
||||
bitmap_header_written = write_number;
|
||||
}
|
||||
write_c("static unsigned char %s[] =\n",
|
||||
unique_id(this, "idata", filename_name(name()), 0));
|
||||
write_cstring(img->data()[0], ((img->w() + 7) / 8) * img->h());
|
||||
write_c(";\n");
|
||||
write_c("static Fl_Bitmap %s(%s, %d, %d);\n",
|
||||
unique_id(this, "image", filename_name(name()), 0),
|
||||
unique_id(this, "idata", filename_name(name()), 0),
|
||||
img->w(), img->h());
|
||||
} else {
|
||||
// Write image data...
|
||||
write_c("\n");
|
||||
if (image_header_written != write_number) {
|
||||
write_c("#include <FL/Fl_Image.H>\n");
|
||||
image_header_written = write_number;
|
||||
}
|
||||
write_c("static unsigned char %s[] =\n",
|
||||
unique_id(this, "idata", filename_name(name()), 0));
|
||||
write_cstring(img->data()[0], (img->w() * img->d() + img->ld()) * img->h());
|
||||
write_c(";\n");
|
||||
write_c("static Fl_RGB_Image %s(%s, %d, %d, %d, %d);\n",
|
||||
unique_id(this, "image", filename_name(name()), 0),
|
||||
unique_id(this, "idata", filename_name(name()), 0),
|
||||
img->w(), img->h(), img->d(), img->ld());
|
||||
}
|
||||
write_c("static const char *%s[] = {\n",
|
||||
unique_id(this, "image", filename_name(name()), 0));
|
||||
int l;
|
||||
for (l = 0; l < numlines; l++) {
|
||||
if (l) write_c(",\n");
|
||||
write_cstring(p->data()[l],linelength[l]);
|
||||
}
|
||||
write_c("\n};\n");
|
||||
write_c("static Fl_Pixmap %s(%s);\n",
|
||||
unique_id(this, "pixmap", filename_name(name()), 0),
|
||||
}
|
||||
|
||||
void Fluid_Image::write_code(int inactive) {
|
||||
if (!img) return;
|
||||
write_c("%so->%s(%s);\n", indent(), inactive ? "deimage" : "image",
|
||||
unique_id(this, "image", filename_name(name()), 0));
|
||||
}
|
||||
|
||||
void pixmap_image::write_code(int inactive) {
|
||||
if (!p) return;
|
||||
write_c("%so->%s(%s);\n", indent(), inactive ? "deimage" : "image",
|
||||
unique_id(this, "pixmap", filename_name(name()), 0));
|
||||
}
|
||||
|
||||
static int hexdigit(int x) {
|
||||
if (isdigit(x)) return x-'0';
|
||||
if (isupper(x)) return x-'A'+10;
|
||||
if (islower(x)) return x-'a'+10;
|
||||
return 20;
|
||||
}
|
||||
|
||||
#define MAXSIZE 2048
|
||||
#define INITIALLINES 1024
|
||||
|
||||
pixmap_image::pixmap_image(const char *name, FILE *f) : Fluid_Image(name) {
|
||||
p = 0;
|
||||
numlines = 0;
|
||||
linelength = 0;
|
||||
|
||||
if (!f) return; // for subclasses
|
||||
// read all the c-strings out of the file:
|
||||
char* local_data[INITIALLINES];
|
||||
char** data = local_data;
|
||||
int local_length[INITIALLINES];
|
||||
int* length = local_length;
|
||||
int malloc_size = INITIALLINES;
|
||||
char buffer[MAXSIZE+20];
|
||||
int i = 0;
|
||||
while (fgets(buffer,MAXSIZE+20,f)) {
|
||||
if (buffer[0] != '\"') continue;
|
||||
char *myp = buffer;
|
||||
char *q = buffer+1;
|
||||
while (*q != '\"' && myp < buffer+MAXSIZE) {
|
||||
if (*q == '\\') switch (*++q) {
|
||||
case '\n':
|
||||
fgets(q,(buffer+MAXSIZE+20)-q,f); break;
|
||||
case 0:
|
||||
break;
|
||||
case 'x': {
|
||||
q++;
|
||||
int n = 0;
|
||||
for (int x = 0; x < 3; x++) {
|
||||
int d = hexdigit(*q);
|
||||
if (d > 15) break;
|
||||
n = (n<<4)+d;
|
||||
q++;
|
||||
}
|
||||
*myp++ = n;
|
||||
} break;
|
||||
default: {
|
||||
int c = *q++;
|
||||
if (c>='0' && c<='7') {
|
||||
c -= '0';
|
||||
for (int x=0; x<2; x++) {
|
||||
int d = hexdigit(*q);
|
||||
if (d>7) break;
|
||||
c = (c<<3)+d;
|
||||
q++;
|
||||
}
|
||||
}
|
||||
*myp++ = c;
|
||||
} break;
|
||||
} else {
|
||||
*myp++ = *q++;
|
||||
}
|
||||
}
|
||||
*myp++ = 0;
|
||||
if (i >= malloc_size) {
|
||||
malloc_size = 2*malloc_size;
|
||||
if (data == local_data) {
|
||||
data = (char**)malloc(malloc_size*sizeof(char*));
|
||||
memcpy(data, local_data, i*sizeof(char*));
|
||||
length = (int*)malloc(malloc_size*sizeof(int));
|
||||
memcpy(length, local_length, i*sizeof(int));
|
||||
} else {
|
||||
data = (char**)realloc(data, malloc_size*sizeof(char*));
|
||||
length = (int*)realloc(length, malloc_size*sizeof(int));
|
||||
}
|
||||
}
|
||||
data[i] = new char[myp-buffer];
|
||||
memcpy(data[i], buffer,myp-buffer);
|
||||
length[i] = myp-buffer-1;
|
||||
i++;
|
||||
}
|
||||
|
||||
if (data == local_data) {
|
||||
data = (char**)malloc(i*sizeof(char*));
|
||||
memcpy(data, local_data, i*sizeof(char*));
|
||||
length = (int*)malloc(i*sizeof(int));
|
||||
memcpy(length, local_length, i*sizeof(int));
|
||||
}
|
||||
numlines = i;
|
||||
linelength = length;
|
||||
p = new Fl_Pixmap(data);
|
||||
}
|
||||
|
||||
pixmap_image::~pixmap_image() {
|
||||
if (p && p->data()) {
|
||||
char** real_data = (char**)(p->data());
|
||||
for (int i = 0; i < numlines; i++) delete[] real_data[i];
|
||||
free((void*)real_data);
|
||||
}
|
||||
free((void*)linelength);
|
||||
delete p;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
|
||||
class gif_image : public pixmap_image {
|
||||
public:
|
||||
gif_image(const char *name, FILE *);
|
||||
~gif_image();
|
||||
static int test_file(char *buffer);
|
||||
};
|
||||
|
||||
int gif_image::test_file(char *buffer) {
|
||||
return !strncmp(buffer,"GIF",3);
|
||||
}
|
||||
|
||||
// function in gif.C:
|
||||
int gif2xpm(
|
||||
const char *infname,// filename for error messages
|
||||
FILE *GifFile, // file to read
|
||||
char*** datap, // return xpm data here
|
||||
int** lengthp, // return line lengths here
|
||||
int inumber // which image in movie (0 = first)
|
||||
);
|
||||
|
||||
gif_image::gif_image(const char *name, FILE *f) : pixmap_image(name,0) {
|
||||
char** datap;
|
||||
numlines = gif2xpm(name,f,&datap,&linelength,0);
|
||||
if (numlines) p = new Fl_Pixmap(datap);
|
||||
else p = 0;
|
||||
}
|
||||
|
||||
gif_image::~gif_image() {
|
||||
if (p && p->data()) {
|
||||
char** real_data = (char**)(p->data());
|
||||
for (int i = 0; i < 3; i++) delete[] real_data[i];
|
||||
delete[] real_data;
|
||||
// p->data(0,0);
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
#include <FL/Fl_Bitmap.H>
|
||||
|
||||
class bitmap_image : public Fluid_Image {
|
||||
Fl_Bitmap *p;
|
||||
public:
|
||||
~bitmap_image();
|
||||
bitmap_image(const char *name, FILE *);
|
||||
virtual void image(Fl_Widget *); // set the image of this widget
|
||||
virtual void deimage(Fl_Widget *); // set the deimage of this widget
|
||||
virtual void write_static();
|
||||
virtual void write_code(int inactive = 0);
|
||||
static int test_file(char *buffer);
|
||||
};
|
||||
|
||||
// bad test, always do this last!
|
||||
int bitmap_image::test_file(char *buffer) {
|
||||
return (strstr(buffer,"#define ") != 0);
|
||||
}
|
||||
|
||||
void bitmap_image::image(Fl_Widget *o) {
|
||||
o->image(p);
|
||||
}
|
||||
|
||||
void bitmap_image::deimage(Fl_Widget *o) {
|
||||
o->deimage(p);
|
||||
}
|
||||
|
||||
static int bitmap_header_written;
|
||||
|
||||
void bitmap_image::write_static() {
|
||||
if (!p) return;
|
||||
write_c("\n");
|
||||
if (bitmap_header_written != write_number) {
|
||||
write_c("#include <FL/Fl_Bitmap.H>\n");
|
||||
bitmap_header_written = write_number;
|
||||
}
|
||||
#if 0 // older one
|
||||
write_c("static unsigned char %s[] = { \n",
|
||||
unique_id(this, "bits", filename_name(name()), 0));
|
||||
int n = ((p->w+7)/8)*p->h;
|
||||
int linelength = 0;
|
||||
for (int i = 0; i < n; i++) {
|
||||
if (i) {write_c(","); linelength++;}
|
||||
if (linelength > 75) {write_c("\n"); linelength=0;}
|
||||
int v = p->array[i];
|
||||
write_c("%d",v);
|
||||
linelength++; if (v>9) linelength++; if (v>99) linelength++;
|
||||
}
|
||||
write_c("\n};\n");
|
||||
#else // this seems to produce slightly shorter c++ files
|
||||
write_c("static unsigned char %s[] =\n",
|
||||
unique_id(this, "bits", filename_name(name()), 0));
|
||||
int n = ((p->w()+7)/8)*p->h();
|
||||
write_cstring((const char*)(p->array), n);
|
||||
write_c(";\n");
|
||||
#endif
|
||||
write_c("static Fl_Bitmap %s(%s, %d, %d);\n",
|
||||
unique_id(this, "bitmap", filename_name(name()), 0),
|
||||
unique_id(this, "bits", filename_name(name()), 0),
|
||||
p->w(), p->h());
|
||||
}
|
||||
|
||||
void bitmap_image::write_code(int inactive) {
|
||||
if (!p) return;
|
||||
write_c("%so->%s(%s);\n", indent(), inactive ? "deimage" : "image",
|
||||
unique_id(this, "bitmap", filename_name(name()), 0));
|
||||
}
|
||||
|
||||
bitmap_image::bitmap_image(const char *name, FILE *f) : Fluid_Image(name) {
|
||||
p = 0; // if any problems with parse we exit with this zero
|
||||
char buffer[1024];
|
||||
char junk[1024];
|
||||
int wh[2]; // width and height
|
||||
int i;
|
||||
for (i = 0; i<2; i++) {
|
||||
for (;;) {
|
||||
if (!fgets(buffer,1024,f)) return;
|
||||
int r = sscanf(buffer,"#define %s %d",junk,&wh[i]);
|
||||
if (r >= 2) break;
|
||||
}
|
||||
}
|
||||
// skip to data array:
|
||||
for (;;) {
|
||||
if (!fgets(buffer,1024,f)) return;
|
||||
if (!strncmp(buffer,"static ",7)) break;
|
||||
}
|
||||
int n = ((wh[0]+7)/8)*wh[1];
|
||||
uchar *data = new uchar[n];
|
||||
// read the data:
|
||||
i = 0;
|
||||
for (;i<n;) {
|
||||
if (!fgets(buffer,1024,f)) return;
|
||||
const char *a = buffer;
|
||||
while (*a && i<n) {
|
||||
int t;
|
||||
if (sscanf(a," 0x%x",&t)>0) data[i++] = t;
|
||||
while (*a && *a++ != ',');
|
||||
}
|
||||
}
|
||||
p = new Fl_Bitmap(data,wh[0],wh[1]);
|
||||
}
|
||||
|
||||
bitmap_image::~bitmap_image() {
|
||||
if (p) {
|
||||
delete[] (uchar*)(p->array);
|
||||
delete p;
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
|
||||
static Fluid_Image** images; // sorted list
|
||||
static int numimages;
|
||||
static int tablesize;
|
||||
static Fluid_Image** images = 0; // sorted list
|
||||
static int numimages = 0;
|
||||
static int tablesize = 0;
|
||||
|
||||
Fluid_Image* Fluid_Image::find(const char *name) {
|
||||
if (!name || !*name) return 0;
|
||||
|
@ -374,28 +153,15 @@ Fluid_Image* Fluid_Image::find(const char *name) {
|
|||
leave_source_dir();
|
||||
return 0;
|
||||
}
|
||||
fclose(f);
|
||||
|
||||
Fluid_Image *ret;
|
||||
Fluid_Image *ret = new Fluid_Image(name);
|
||||
|
||||
// now see if we can identify the type, by reading in some data
|
||||
// and asking all the types we know about:
|
||||
|
||||
char buffer[1025];
|
||||
fread(buffer, 1, 1024, f);
|
||||
rewind(f);
|
||||
buffer[1024] = 0; // null-terminate so strstr() works
|
||||
|
||||
if (pixmap_image::test_file(buffer)) {
|
||||
ret = new pixmap_image(name,f);
|
||||
} else if (gif_image::test_file(buffer)) {
|
||||
ret = new gif_image(name,f);
|
||||
} else if (bitmap_image::test_file(buffer)) {
|
||||
ret = new bitmap_image(name,f);
|
||||
} else {
|
||||
if (!ret->img->w() || !ret->img->h()) {
|
||||
delete ret;
|
||||
ret = 0;
|
||||
read_error("%s : unrecognized image format", name);
|
||||
}
|
||||
fclose(f);
|
||||
leave_source_dir();
|
||||
if (!ret) return 0;
|
||||
|
||||
|
@ -403,7 +169,8 @@ Fluid_Image* Fluid_Image::find(const char *name) {
|
|||
numimages++;
|
||||
if (numimages > tablesize) {
|
||||
tablesize = tablesize ? 2*tablesize : 16;
|
||||
images = (Fluid_Image**)realloc(images, tablesize*sizeof(Fluid_Image*));
|
||||
if (images) images = (Fluid_Image**)realloc(images, tablesize*sizeof(Fluid_Image*));
|
||||
else images = (Fluid_Image**)malloc(tablesize*sizeof(Fluid_Image*));
|
||||
}
|
||||
for (b = numimages-1; b > a; b--) images[b] = images[b-1];
|
||||
images[a] = ret;
|
||||
|
@ -415,6 +182,7 @@ Fluid_Image::Fluid_Image(const char *name) {
|
|||
name_ = strdup(name);
|
||||
written = 0;
|
||||
refcount = 0;
|
||||
img = Fl_Shared_Image::get(name);
|
||||
}
|
||||
|
||||
void Fluid_Image::increment() {
|
||||
|
@ -432,6 +200,7 @@ Fluid_Image::~Fluid_Image() {
|
|||
for (a = 0;; a++) if (images[a] == this) break;
|
||||
numimages--;
|
||||
for (; a < numimages; a++) images[a] = images[a+1];
|
||||
img->release();
|
||||
free((void*)name_);
|
||||
}
|
||||
|
||||
|
@ -442,13 +211,14 @@ Fluid_Image::~Fluid_Image() {
|
|||
const char *ui_find_image_name;
|
||||
Fluid_Image *ui_find_image(const char *oldname) {
|
||||
goto_source_dir();
|
||||
const char *name = fl_file_chooser("Image","*.{bm|xbm|xpm|gif}",oldname);
|
||||
const char *name = fl_file_chooser("Image?","*.{bm|gif|jpg|pbm|pgm|png|ppm|xbm|xpm}",oldname);
|
||||
ui_find_image_name = name;
|
||||
Fluid_Image *ret = (name && *name) ? Fluid_Image::find(name) : 0;
|
||||
leave_source_dir();
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
||||
//
|
||||
// End of "$Id: Fluid_Image.cxx,v 1.7.2.9.2.4 2001/11/22 15:35:01 easysw Exp $".
|
||||
// End of "$Id: Fluid_Image.cxx,v 1.7.2.9.2.5 2001/11/25 22:51:34 easysw Exp $".
|
||||
//
|
||||
|
|
|
@ -1,13 +1,11 @@
|
|||
//
|
||||
// "$Id: Fluid_Image.h,v 1.3.2.4.2.1 2001/09/29 06:20:15 easysw Exp $"
|
||||
// "$Id: Fluid_Image.h,v 1.3.2.4.2.2 2001/11/25 22:51:34 easysw Exp $"
|
||||
//
|
||||
// Pixmap image header file for the Fast Light Tool Kit (FLTK).
|
||||
// Image header file for the Fast Light Tool Kit (FLTK).
|
||||
//
|
||||
// This class stores the image labels for widgets in fluid. This is
|
||||
// not a class in fltk itself, and this will produce different types of
|
||||
// code depending on what the image type is. There are private subclasses
|
||||
// in Fluid_Image.C for each type of image format. Right now only xpm
|
||||
// files are supported.
|
||||
// not a class in FLTK itself, and will produce different types of
|
||||
// code depending on what the image type is.
|
||||
//
|
||||
// Copyright 1998-2001 by Bill Spitzak and others.
|
||||
//
|
||||
|
@ -30,23 +28,27 @@
|
|||
//
|
||||
|
||||
#ifndef FLUID_IMAGE_H
|
||||
#define FLUID_IMAGE_H
|
||||
# define FLUID_IMAGE_H
|
||||
|
||||
# include <FL/Fl_Shared_Image.H>
|
||||
|
||||
|
||||
class Fluid_Image {
|
||||
const char *name_;
|
||||
int refcount;
|
||||
Fl_Shared_Image *img;
|
||||
protected:
|
||||
Fluid_Image(const char *name); // no public constructor
|
||||
virtual ~Fluid_Image(); // no public destructor
|
||||
~Fluid_Image(); // no public destructor
|
||||
public:
|
||||
int written;
|
||||
static Fluid_Image* find(const char *);
|
||||
void decrement(); // reference counting & automatic free
|
||||
void increment();
|
||||
virtual void image(Fl_Widget *) = 0; // set the image of this widget
|
||||
virtual void deimage(Fl_Widget *) = 0; // set the deimage of this widget
|
||||
virtual void write_static() = 0;
|
||||
virtual void write_code(int inactive = 0) = 0;
|
||||
void image(Fl_Widget *); // set the image of this widget
|
||||
void deimage(Fl_Widget *); // set the deimage of this widget
|
||||
void write_static();
|
||||
void write_code(int inactive = 0);
|
||||
const char *name() const {return name_;}
|
||||
};
|
||||
|
||||
|
@ -58,5 +60,5 @@ extern const char *ui_find_image_name;
|
|||
#endif
|
||||
|
||||
//
|
||||
// End of "$Id: Fluid_Image.h,v 1.3.2.4.2.1 2001/09/29 06:20:15 easysw Exp $".
|
||||
// End of "$Id: Fluid_Image.h,v 1.3.2.4.2.2 2001/11/25 22:51:34 easysw Exp $".
|
||||
//
|
||||
|
|
Loading…
Reference in New Issue