@@ -2137,16 +2137,18 @@ def save(self, fp, format=None, **params):
21372137
21382138 if format .upper () not in SAVE :
21392139 init ()
2140- if params .pop (' save_all' , False ):
2140+ if params .pop (" save_all" , False ):
21412141 save_handler = SAVE_ALL [format .upper ()]
21422142 else :
21432143 save_handler = SAVE [format .upper ()]
21442144
2145- if params .get (' convert_mode' ):
2145+ if params .get (" convert_mode" ):
21462146 plugin = sys .modules [save_handler .__module__ ]
2147- converted_im = self ._convert_mode (plugin , params )
2148- if converted_im :
2149- return converted_im .save (fp , format , ** params )
2147+ if hasattr (plugin , "_supported_modes" ):
2148+ modes = plugin ._supported_modes ()
2149+ converted_im = self ._convert_mode (modes , params )
2150+ if converted_im :
2151+ return converted_im .save (fp , format , ** params )
21502152
21512153 self .encoderinfo = params
21522154 self .encoderconfig = ()
@@ -2166,32 +2168,57 @@ def save(self, fp, format=None, **params):
21662168 if open_fp :
21672169 fp .close ()
21682170
2169- def _convert_mode (self , plugin , params ):
2170- if not hasattr ( plugin , '_convert_mode' ) :
2171+ def _convert_mode (self , modes , params = {} ):
2172+ if not modes or self . mode in modes :
21712173 return
2172- new_mode = plugin ._convert_mode (self )
2173- if self .mode == 'LA' and new_mode == 'P' :
2174- alpha = self .getchannel ('A' )
2174+ if self .mode == "P" :
2175+ preferred_modes = []
2176+ if "A" in self .im .getpalettemode ():
2177+ preferred_modes .append ("RGBA" )
2178+ preferred_modes .append ("RGB" )
2179+ else :
2180+ preferred_modes = {
2181+ "CMYK" : ["RGB" ],
2182+ "RGB" : ["CMYK" ],
2183+ "RGBX" : ["RGB" ],
2184+ "RGBa" : ["RGBA" , "RGB" ],
2185+ "RGBA" : ["RGB" ],
2186+ "LA" : ["RGBA" , "P" , "L" ],
2187+ "La" : ["LA" , "L" ],
2188+ "L" : ["RGB" ],
2189+ "F" : ["I" ],
2190+ "I" : ["L" , "RGB" ],
2191+ "1" : ["L" ],
2192+ "YCbCr" : ["RGB" ],
2193+ "LAB" : ["RGB" ],
2194+ "HSV" : ["RGB" ],
2195+ }.get (self .mode , [])
2196+ for new_mode in preferred_modes :
2197+ if new_mode in modes :
2198+ break
2199+ else :
2200+ new_mode = modes [0 ]
2201+ if self .mode == "LA" and new_mode == "P" :
2202+ alpha = self .getchannel ("A" )
21752203 # Convert the image into P mode but only use 255 colors
21762204 # in the palette out of 256.
2177- im = self .convert ('L' ) \
2178- .convert ('P' , palette = ADAPTIVE , colors = 255 )
2205+ im = self .convert ("L" ).convert ("P" , palette = ADAPTIVE , colors = 255 )
21792206 # Set all pixel values below 128 to 255, and the rest to 0.
21802207 mask = eval (alpha , lambda px : 255 if px < 128 else 0 )
21812208 # Paste the color of index 255 and use alpha as a mask.
21822209 im .paste (255 , mask )
21832210 # The transparency index is 255.
2184- im .info [' transparency' ] = 255
2211+ im .info [" transparency" ] = 255
21852212 return im
21862213
2187- elif self .mode == 'I' :
2188- im = self .point ([i // 256 for i in range (65536 )], 'L' )
2189- return im .convert (new_mode ) if new_mode != 'L' else im
2214+ elif self .mode == "I" :
2215+ im = self .point ([i // 256 for i in range (65536 )], "L" )
2216+ return im .convert (new_mode ) if new_mode != "L" else im
21902217
2191- elif self .mode in (' RGBA' , 'LA' ) and new_mode in (' RGB' , 'L' ):
2192- fill_color = params .get (' fill_color' , ' white' )
2218+ elif self .mode in (" RGBA" , "LA" ) and new_mode in (" RGB" , "L" ):
2219+ fill_color = params .get (" fill_color" , " white" )
21932220 background = new (new_mode , self .size , fill_color )
2194- background .paste (self , self .getchannel ('A' ))
2221+ background .paste (self , self .getchannel ("A" ))
21952222 return background
21962223
21972224 elif new_mode :
0 commit comments