@@ -2129,16 +2129,18 @@ def save(self, fp, format=None, **params):
21292129
21302130 if format .upper () not in SAVE :
21312131 init ()
2132- if params .pop (' save_all' , False ):
2132+ if params .pop (" save_all" , False ):
21332133 save_handler = SAVE_ALL [format .upper ()]
21342134 else :
21352135 save_handler = SAVE [format .upper ()]
21362136
2137- if params .get (' convert_mode' ):
2137+ if params .get (" convert_mode" ):
21382138 plugin = sys .modules [save_handler .__module__ ]
2139- converted_im = self ._convert_mode (plugin , params )
2140- if converted_im :
2141- return converted_im .save (fp , format , ** params )
2139+ if hasattr (plugin , "_supported_modes" ):
2140+ modes = plugin ._supported_modes ()
2141+ converted_im = self ._convert_mode (modes , params )
2142+ if converted_im :
2143+ return converted_im .save (fp , format , ** params )
21422144
21432145 self .encoderinfo = params
21442146 self .encoderconfig = ()
@@ -2158,32 +2160,57 @@ def save(self, fp, format=None, **params):
21582160 if open_fp :
21592161 fp .close ()
21602162
2161- def _convert_mode (self , plugin , params ):
2162- if not hasattr ( plugin , '_convert_mode' ) :
2163+ def _convert_mode (self , modes , params = {} ):
2164+ if not modes or self . mode in modes :
21632165 return
2164- new_mode = plugin ._convert_mode (self )
2165- if self .mode == 'LA' and new_mode == 'P' :
2166- alpha = self .getchannel ('A' )
2166+ if self .mode == "P" :
2167+ preferred_modes = []
2168+ if "A" in self .im .getpalettemode ():
2169+ preferred_modes .append ("RGBA" )
2170+ preferred_modes .append ("RGB" )
2171+ else :
2172+ preferred_modes = {
2173+ "CMYK" : ["RGB" ],
2174+ "RGB" : ["CMYK" ],
2175+ "RGBX" : ["RGB" ],
2176+ "RGBa" : ["RGBA" , "RGB" ],
2177+ "RGBA" : ["RGB" ],
2178+ "LA" : ["RGBA" , "P" , "L" ],
2179+ "La" : ["LA" , "L" ],
2180+ "L" : ["RGB" ],
2181+ "F" : ["I" ],
2182+ "I" : ["L" , "RGB" ],
2183+ "1" : ["L" ],
2184+ "YCbCr" : ["RGB" ],
2185+ "LAB" : ["RGB" ],
2186+ "HSV" : ["RGB" ],
2187+ }.get (self .mode , [])
2188+ for new_mode in preferred_modes :
2189+ if new_mode in modes :
2190+ break
2191+ else :
2192+ new_mode = modes [0 ]
2193+ if self .mode == "LA" and new_mode == "P" :
2194+ alpha = self .getchannel ("A" )
21672195 # Convert the image into P mode but only use 255 colors
21682196 # in the palette out of 256.
2169- im = self .convert ('L' ) \
2170- .convert ('P' , palette = ADAPTIVE , colors = 255 )
2197+ im = self .convert ("L" ).convert ("P" , palette = ADAPTIVE , colors = 255 )
21712198 # Set all pixel values below 128 to 255, and the rest to 0.
21722199 mask = eval (alpha , lambda px : 255 if px < 128 else 0 )
21732200 # Paste the color of index 255 and use alpha as a mask.
21742201 im .paste (255 , mask )
21752202 # The transparency index is 255.
2176- im .info [' transparency' ] = 255
2203+ im .info [" transparency" ] = 255
21772204 return im
21782205
2179- elif self .mode == 'I' :
2180- im = self .point ([i // 256 for i in range (65536 )], 'L' )
2181- return im .convert (new_mode ) if new_mode != 'L' else im
2206+ elif self .mode == "I" :
2207+ im = self .point ([i // 256 for i in range (65536 )], "L" )
2208+ return im .convert (new_mode ) if new_mode != "L" else im
21822209
2183- elif self .mode in (' RGBA' , 'LA' ) and new_mode in (' RGB' , 'L' ):
2184- fill_color = params .get (' fill_color' , ' white' )
2210+ elif self .mode in (" RGBA" , "LA" ) and new_mode in (" RGB" , "L" ):
2211+ fill_color = params .get (" fill_color" , " white" )
21852212 background = new (new_mode , self .size , fill_color )
2186- background .paste (self , self .getchannel ('A' ))
2213+ background .paste (self , self .getchannel ("A" ))
21872214 return background
21882215
21892216 elif new_mode :
0 commit comments