@@ -2190,16 +2190,18 @@ def save(self, fp, format=None, **params):
21902190
21912191 if format .upper () not in SAVE :
21922192 init ()
2193- if params .pop (' save_all' , False ):
2193+ if params .pop (" save_all" , False ):
21942194 save_handler = SAVE_ALL [format .upper ()]
21952195 else :
21962196 save_handler = SAVE [format .upper ()]
21972197
2198- if params .get (' convert_mode' ):
2198+ if params .get (" convert_mode" ):
21992199 plugin = sys .modules [save_handler .__module__ ]
2200- converted_im = self ._convert_mode (plugin , params )
2201- if converted_im :
2202- return converted_im .save (fp , format , ** params )
2200+ if hasattr (plugin , "_supported_modes" ):
2201+ modes = plugin ._supported_modes ()
2202+ converted_im = self ._convert_mode (modes , params )
2203+ if converted_im :
2204+ return converted_im .save (fp , format , ** params )
22032205
22042206 self .encoderinfo = params
22052207 self .encoderconfig = ()
@@ -2219,32 +2221,57 @@ def save(self, fp, format=None, **params):
22192221 if open_fp :
22202222 fp .close ()
22212223
2222- def _convert_mode (self , plugin , params ):
2223- if not hasattr ( plugin , '_convert_mode' ) :
2224+ def _convert_mode (self , modes , params = {} ):
2225+ if not modes or self . mode in modes :
22242226 return
2225- new_mode = plugin ._convert_mode (self )
2226- if self .mode == 'LA' and new_mode == 'P' :
2227- alpha = self .getchannel ('A' )
2227+ if self .mode == "P" :
2228+ preferred_modes = []
2229+ if "A" in self .im .getpalettemode ():
2230+ preferred_modes .append ("RGBA" )
2231+ preferred_modes .append ("RGB" )
2232+ else :
2233+ preferred_modes = {
2234+ "CMYK" : ["RGB" ],
2235+ "RGB" : ["CMYK" ],
2236+ "RGBX" : ["RGB" ],
2237+ "RGBa" : ["RGBA" , "RGB" ],
2238+ "RGBA" : ["RGB" ],
2239+ "LA" : ["RGBA" , "P" , "L" ],
2240+ "La" : ["LA" , "L" ],
2241+ "L" : ["RGB" ],
2242+ "F" : ["I" ],
2243+ "I" : ["L" , "RGB" ],
2244+ "1" : ["L" ],
2245+ "YCbCr" : ["RGB" ],
2246+ "LAB" : ["RGB" ],
2247+ "HSV" : ["RGB" ],
2248+ }.get (self .mode , [])
2249+ for new_mode in preferred_modes :
2250+ if new_mode in modes :
2251+ break
2252+ else :
2253+ new_mode = modes [0 ]
2254+ if self .mode == "LA" and new_mode == "P" :
2255+ alpha = self .getchannel ("A" )
22282256 # Convert the image into P mode but only use 255 colors
22292257 # in the palette out of 256.
2230- im = self .convert ('L' ) \
2231- .convert ('P' , palette = ADAPTIVE , colors = 255 )
2258+ im = self .convert ("L" ).convert ("P" , palette = ADAPTIVE , colors = 255 )
22322259 # Set all pixel values below 128 to 255, and the rest to 0.
22332260 mask = eval (alpha , lambda px : 255 if px < 128 else 0 )
22342261 # Paste the color of index 255 and use alpha as a mask.
22352262 im .paste (255 , mask )
22362263 # The transparency index is 255.
2237- im .info [' transparency' ] = 255
2264+ im .info [" transparency" ] = 255
22382265 return im
22392266
2240- elif self .mode == 'I' :
2241- im = self .point ([i // 256 for i in range (65536 )], 'L' )
2242- return im .convert (new_mode ) if new_mode != 'L' else im
2267+ elif self .mode == "I" :
2268+ im = self .point ([i // 256 for i in range (65536 )], "L" )
2269+ return im .convert (new_mode ) if new_mode != "L" else im
22432270
2244- elif self .mode in (' RGBA' , 'LA' ) and new_mode in (' RGB' , 'L' ):
2245- fill_color = params .get (' fill_color' , ' white' )
2271+ elif self .mode in (" RGBA" , "LA" ) and new_mode in (" RGB" , "L" ):
2272+ fill_color = params .get (" fill_color" , " white" )
22462273 background = new (new_mode , self .size , fill_color )
2247- background .paste (self , self .getchannel ('A' ))
2274+ background .paste (self , self .getchannel ("A" ))
22482275 return background
22492276
22502277 elif new_mode :
0 commit comments