@@ -2040,16 +2040,18 @@ def save(self, fp, format=None, **params):
20402040
20412041 if format .upper () not in SAVE :
20422042 init ()
2043- if params .pop (' save_all' , False ):
2043+ if params .pop (" save_all" , False ):
20442044 save_handler = SAVE_ALL [format .upper ()]
20452045 else :
20462046 save_handler = SAVE [format .upper ()]
20472047
2048- if params .get (' convert_mode' ):
2048+ if params .get (" convert_mode" ):
20492049 plugin = sys .modules [save_handler .__module__ ]
2050- converted_im = self ._convert_mode (plugin , params )
2051- if converted_im :
2052- return converted_im .save (fp , format , ** params )
2050+ if hasattr (plugin , "_supported_modes" ):
2051+ modes = plugin ._supported_modes ()
2052+ converted_im = self ._convert_mode (modes , params )
2053+ if converted_im :
2054+ return converted_im .save (fp , format , ** params )
20532055
20542056 self .encoderinfo = params
20552057 self .encoderconfig = ()
@@ -2069,32 +2071,57 @@ def save(self, fp, format=None, **params):
20692071 if open_fp :
20702072 fp .close ()
20712073
2072- def _convert_mode (self , plugin , params ):
2073- if not hasattr ( plugin , '_convert_mode' ) :
2074+ def _convert_mode (self , modes , params = {} ):
2075+ if not modes or self . mode in modes :
20742076 return
2075- new_mode = plugin ._convert_mode (self )
2076- if self .mode == 'LA' and new_mode == 'P' :
2077- alpha = self .getchannel ('A' )
2077+ if self .mode == "P" :
2078+ preferred_modes = []
2079+ if "A" in self .im .getpalettemode ():
2080+ preferred_modes .append ("RGBA" )
2081+ preferred_modes .append ("RGB" )
2082+ else :
2083+ preferred_modes = {
2084+ "CMYK" : ["RGB" ],
2085+ "RGB" : ["CMYK" ],
2086+ "RGBX" : ["RGB" ],
2087+ "RGBa" : ["RGBA" , "RGB" ],
2088+ "RGBA" : ["RGB" ],
2089+ "LA" : ["RGBA" , "P" , "L" ],
2090+ "La" : ["LA" , "L" ],
2091+ "L" : ["RGB" ],
2092+ "F" : ["I" ],
2093+ "I" : ["L" , "RGB" ],
2094+ "1" : ["L" ],
2095+ "YCbCr" : ["RGB" ],
2096+ "LAB" : ["RGB" ],
2097+ "HSV" : ["RGB" ],
2098+ }.get (self .mode , [])
2099+ for new_mode in preferred_modes :
2100+ if new_mode in modes :
2101+ break
2102+ else :
2103+ new_mode = modes [0 ]
2104+ if self .mode == "LA" and new_mode == "P" :
2105+ alpha = self .getchannel ("A" )
20782106 # Convert the image into P mode but only use 255 colors
20792107 # in the palette out of 256.
2080- im = self .convert ('L' ) \
2081- .convert ('P' , palette = ADAPTIVE , colors = 255 )
2108+ im = self .convert ("L" ).convert ("P" , palette = ADAPTIVE , colors = 255 )
20822109 # Set all pixel values below 128 to 255, and the rest to 0.
20832110 mask = eval (alpha , lambda px : 255 if px < 128 else 0 )
20842111 # Paste the color of index 255 and use alpha as a mask.
20852112 im .paste (255 , mask )
20862113 # The transparency index is 255.
2087- im .info [' transparency' ] = 255
2114+ im .info [" transparency" ] = 255
20882115 return im
20892116
2090- elif self .mode == 'I' :
2091- im = self .point ([i // 256 for i in range (65536 )], 'L' )
2092- return im .convert (new_mode ) if new_mode != 'L' else im
2117+ elif self .mode == "I" :
2118+ im = self .point ([i // 256 for i in range (65536 )], "L" )
2119+ return im .convert (new_mode ) if new_mode != "L" else im
20932120
2094- elif self .mode in (' RGBA' , 'LA' ) and new_mode in (' RGB' , 'L' ):
2095- fill_color = params .get (' fill_color' , ' white' )
2121+ elif self .mode in (" RGBA" , "LA" ) and new_mode in (" RGB" , "L" ):
2122+ fill_color = params .get (" fill_color" , " white" )
20962123 background = new (new_mode , self .size , fill_color )
2097- background .paste (self , self .getchannel ('A' ))
2124+ background .paste (self , self .getchannel ("A" ))
20982125 return background
20992126
21002127 elif new_mode :
0 commit comments