mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-26 03:11:59 +00:00
* Cure the crash that became a memory leak properly.
* Enable the loading of XPM files with crappy color strings. Print out a nice friendly message on what's gone wrong and how to resolve it properly. git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@3599 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
parent
53c6d464f6
commit
aa70718501
@ -1,3 +1,15 @@
|
|||||||
|
2002-02-27 Angus Leeming <a.leeming@ic.ac.uk>
|
||||||
|
|
||||||
|
* GraphicsCache.C: improve commentary to graphicsInit and where it
|
||||||
|
should really go.
|
||||||
|
|
||||||
|
* GraphicsImageXPM.C (~Data, free_color_table): resolve the crash
|
||||||
|
that became a memory leak properly. (Let the shared_c_ptr free the
|
||||||
|
color table.)
|
||||||
|
(reset, mapcolor): tidy up and introduce a work around for XPM files
|
||||||
|
with crappy color entries. Print out a nice friendly message on what's
|
||||||
|
gone wrong and how to resolve it properly.
|
||||||
|
|
||||||
2002-02-27 Angus Leeming <a.leeming@ic.ac.uk>
|
2002-02-27 Angus Leeming <a.leeming@ic.ac.uk>
|
||||||
|
|
||||||
* GraphicsImageXPM.[Ch]: more rigorous use of types (signed/unsigned).
|
* GraphicsImageXPM.[Ch]: more rigorous use of types (signed/unsigned).
|
||||||
|
@ -19,8 +19,10 @@
|
|||||||
#include "GraphicsParams.h"
|
#include "GraphicsParams.h"
|
||||||
#include "insets/insetgraphics.h"
|
#include "insets/insetgraphics.h"
|
||||||
|
|
||||||
// I think that graphicsInit should become Dialogs::graphicsInit.
|
|
||||||
// These #includes would then be moved there.
|
// I think that graphicsInit should become a new Dialogs::graphicsInit
|
||||||
|
// static method.
|
||||||
|
// These #includes would then be moved to Dialogs.C.
|
||||||
// Angus 25 Feb 2002
|
// Angus 25 Feb 2002
|
||||||
#include "GraphicsImageXPM.h"
|
#include "GraphicsImageXPM.h"
|
||||||
//#include "xformsGraphicsImage.h"
|
//#include "xformsGraphicsImage.h"
|
||||||
|
@ -384,9 +384,9 @@ string const unique_color_string(XpmImage const & image);
|
|||||||
// create a copy (using malloc and strcpy). If (!in) return 0;
|
// create a copy (using malloc and strcpy). If (!in) return 0;
|
||||||
char * clone_c_string(char const * in);
|
char * clone_c_string(char const * in);
|
||||||
|
|
||||||
// Given a string of the form #ff0571 create a string for the appropriate
|
// Given a string of the form #ff0571 create appropriate grayscale and
|
||||||
// grayscale or monochrome color.
|
// monochrome colors.
|
||||||
char * mapcolor(char * color, bool toGray);
|
void mapcolor(char const * c_color, char ** g_color_ptr, char ** m_color_ptr);
|
||||||
|
|
||||||
} // namespace anon
|
} // namespace anon
|
||||||
|
|
||||||
@ -401,9 +401,8 @@ GImageXPM::Data::Data()
|
|||||||
|
|
||||||
GImageXPM::Data::~Data()
|
GImageXPM::Data::~Data()
|
||||||
{
|
{
|
||||||
// Introduce temporary memory leak to fix crash.
|
if (colorTable_.unique())
|
||||||
// if (colorTable_.unique())
|
free_color_table(colorTable_.get(), ncolors_);
|
||||||
// free_color_table(colorTable_.get(), ncolors_);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -465,17 +464,40 @@ void GImageXPM::Data::reset(XpmImage & image)
|
|||||||
|
|
||||||
// 2. Ensure that the color table has g_color and m_color entries
|
// 2. Ensure that the color table has g_color and m_color entries
|
||||||
XpmColor * table = colorTable_.get();
|
XpmColor * table = colorTable_.get();
|
||||||
|
string buggy_color;
|
||||||
|
|
||||||
for (size_t i = 0; i < ncolors_; ++i) {
|
for (size_t i = 0; i < ncolors_; ++i) {
|
||||||
// If the c_color is defined and the equivalent
|
XpmColor & entry = table[i];
|
||||||
// grayscale one is not, then define it.
|
if (!entry.c_color)
|
||||||
if (table[i].c_color && !table[i].g_color)
|
continue;
|
||||||
table[i].g_color = mapcolor(table[i].c_color, true);
|
|
||||||
|
// A work-around for buggy XPM files that may be created by
|
||||||
|
// ImageMagick's convert.
|
||||||
|
string c_color = entry.c_color;
|
||||||
|
if (c_color[0] == '#' && c_color.size() > 7) {
|
||||||
|
if (buggy_color.empty())
|
||||||
|
buggy_color = c_color;
|
||||||
|
|
||||||
|
c_color = c_color.substr(0, 7);
|
||||||
|
free(entry.c_color);
|
||||||
|
entry.c_color = clone_c_string(c_color.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
// If the c_color is defined and the equivalent
|
// If the c_color is defined and the equivalent
|
||||||
// monochrome one is not, then define it.
|
// grayscale or monochrome ones are not, then define them.
|
||||||
if (table[i].c_color && !table[i].m_color)
|
mapcolor(entry.c_color, &entry.g_color, &entry.m_color);
|
||||||
table[i].m_color = mapcolor(table[i].c_color, false);
|
}
|
||||||
|
|
||||||
|
if (!buggy_color.empty()) {
|
||||||
|
lyxerr << "The XPM file contains silly colors, "
|
||||||
|
<< "an example being \""
|
||||||
|
<< buggy_color << "\".\n"
|
||||||
|
<< "This was cropped to \""
|
||||||
|
<< buggy_color.substr(0, 7)
|
||||||
|
<< "\" so you can see something!\n"
|
||||||
|
<< "If this file was created by ImageMagick's convert,\n"
|
||||||
|
<< "then upgrading may cure the problem."
|
||||||
|
<< std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -529,19 +551,27 @@ unsigned int GImageXPM::Data::color_none_id() const
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
// Given a string of the form #ff0571 create a string for the appropriate
|
// Given a string of the form #ff0571 create appropriate grayscale and
|
||||||
// grayscale or monochrome color.
|
// monochrome colors.
|
||||||
char * mapcolor(char * color, bool toGray)
|
void mapcolor(char const * c_color, char ** g_color_ptr, char ** m_color_ptr)
|
||||||
{
|
{
|
||||||
if (!color)
|
if (!c_color)
|
||||||
return 0;
|
return;
|
||||||
|
|
||||||
|
char * g_color = *g_color_ptr;
|
||||||
|
char * m_color = *m_color_ptr;
|
||||||
|
|
||||||
|
if (g_color && m_color)
|
||||||
|
// Already filled.
|
||||||
|
return;
|
||||||
|
|
||||||
Display * display = GUIRunTime::x11Display();
|
Display * display = GUIRunTime::x11Display();
|
||||||
Colormap cmap = GUIRunTime::x11Colormap();
|
Colormap cmap = GUIRunTime::x11Colormap();
|
||||||
XColor xcol;
|
XColor xcol;
|
||||||
XColor ccol;
|
XColor ccol;
|
||||||
if (XLookupColor(display, cmap, color, &xcol, &ccol) == 0)
|
if (XLookupColor(display, cmap, c_color, &xcol, &ccol) == 0)
|
||||||
return 0;
|
// Unable to parse c_color.
|
||||||
|
return;
|
||||||
|
|
||||||
// Note that X stores the RGB values in the range 0 - 65535
|
// Note that X stores the RGB values in the range 0 - 65535
|
||||||
// whilst we require them in the range 0 - 255.
|
// whilst we require them in the range 0 - 255.
|
||||||
@ -551,20 +581,27 @@ char * mapcolor(char * color, bool toGray)
|
|||||||
|
|
||||||
// This gives a good match to a human's RGB to luminance conversion.
|
// This gives a good match to a human's RGB to luminance conversion.
|
||||||
// (From xv's Postscript code --- Mike Ressler.)
|
// (From xv's Postscript code --- Mike Ressler.)
|
||||||
int mapped_color = int((0.32 * r) + (0.5 * g) + (0.18 * b));
|
int const gray = int((0.32 * r) + (0.5 * g) + (0.18 * b));
|
||||||
if (!toGray) // monochrome
|
|
||||||
mapped_color = (mapped_color < 128) ? 0 : 255;
|
|
||||||
|
|
||||||
ostringstream ostr;
|
ostringstream gray_stream;
|
||||||
|
gray_stream << "#" << std::setbase(16) << std::setfill('0')
|
||||||
|
<< std::setw(2) << gray
|
||||||
|
<< std::setw(2) << gray
|
||||||
|
<< std::setw(2) << gray;
|
||||||
|
|
||||||
ostr << "#" << std::setbase(16) << std::setfill('0')
|
int const mono = (gray < 128) ? 0 : 255;
|
||||||
<< std::setw(2) << mapped_color
|
ostringstream mono_stream;
|
||||||
<< std::setw(2) << mapped_color
|
mono_stream << "#" << std::setbase(16) << std::setfill('0')
|
||||||
<< std::setw(2) << mapped_color;
|
<< std::setw(2) << mono
|
||||||
|
<< std::setw(2) << mono
|
||||||
|
<< std::setw(2) << mono;
|
||||||
|
|
||||||
// This string is going into an XpmImage struct, so create a copy that
|
// This string is going into an XpmImage struct, so create copies that
|
||||||
// libXPM can free successfully.
|
// libXPM can free successfully.
|
||||||
return clone_c_string(ostr.str().c_str());
|
if (!g_color)
|
||||||
|
*g_color_ptr = clone_c_string(gray_stream.str().c_str());
|
||||||
|
if (!m_color)
|
||||||
|
*m_color_ptr = clone_c_string(mono_stream.str().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -591,7 +628,8 @@ void free_color_table(XpmColor * table, size_t size)
|
|||||||
free(table[i].g4_color);
|
free(table[i].g4_color);
|
||||||
free(table[i].c_color);
|
free(table[i].c_color);
|
||||||
}
|
}
|
||||||
free(table);
|
// Don't free the table itself. Let the shared_c_ptr do that.
|
||||||
|
// free(table);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -619,7 +657,7 @@ bool contains_color_none(XpmImage const & image)
|
|||||||
|
|
||||||
string const unique_color_string(XpmImage const & image)
|
string const unique_color_string(XpmImage const & image)
|
||||||
{
|
{
|
||||||
string id(image.cpp, 'A');
|
string id(image.cpp, ' ');
|
||||||
|
|
||||||
for(;;) {
|
for(;;) {
|
||||||
bool found_it = false;
|
bool found_it = false;
|
||||||
@ -631,25 +669,35 @@ string const unique_color_string(XpmImage const & image)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!found_it)
|
if (!found_it) {
|
||||||
|
std::cerr << "unique_color_string: \"" << id
|
||||||
|
<< "\"" << std::endl;
|
||||||
return id;
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
// A base 57 counter!
|
// Loop over the printable characters in the ASCII table.
|
||||||
// eg AAAz+1 = AABA, AABz+1 = AACA, AAzz+1 = ABAA
|
// Ie, count from char 32 (' ') to char 126 ('~')
|
||||||
|
// A base 94 counter!
|
||||||
string::size_type current_index = id.size() - 1;
|
string::size_type current_index = id.size() - 1;
|
||||||
bool continue_loop = true;
|
bool continue_loop = true;
|
||||||
while(continue_loop) {
|
while(continue_loop) {
|
||||||
continue_loop = false;
|
continue_loop = false;
|
||||||
|
|
||||||
if (id[current_index] == 'z') {
|
|
||||||
continue_loop = true;
|
|
||||||
if (current_index == 0) // failed!
|
|
||||||
break;
|
|
||||||
|
|
||||||
id[current_index] = 'A';
|
if (id[current_index] == 126) {
|
||||||
|
continue_loop = true;
|
||||||
|
if (current_index == 0)
|
||||||
|
// Unable to find a unique string
|
||||||
|
return image.colorTable[0].string;
|
||||||
|
|
||||||
|
id[current_index] = 32;
|
||||||
current_index -= 1;
|
current_index -= 1;
|
||||||
} else {
|
} else {
|
||||||
id[current_index] += 1;
|
id[current_index] += 1;
|
||||||
|
// Note that '"' is an illegal char in this
|
||||||
|
// context
|
||||||
|
if (id[current_index] == '"')
|
||||||
|
id[current_index] += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (continue_loop)
|
if (continue_loop)
|
||||||
|
Loading…
Reference in New Issue
Block a user