render.rb

  1#!/bin/env ruby
  2
  3require 'xml'
  4
  5resolutions = {
  6	'mdpi' => 1,
  7	'hdpi' => 1.5,
  8	'xhdpi' => 2,
  9	'xxhdpi' => 3,
 10	'xxxhdpi' => 4,
 11	}
 12
 13images = {
 14	'main_logo.svg' => ['conversations/main_logo', 200],
 15    'quicksy_main_logo.svg' => ['quicksy/main_logo', 200],
 16    'splash_logo.svg' => ['conversations/splash_logo', 144],
 17    'quicksy_splash_logo.svg' => ['quicksy/splash_logo', 144],
 18    'ic_search_black.svg' => ['ic_search_background_black', 144],
 19    'ic_search_white.svg' => ['ic_search_background_white', 144],
 20    'ic_no_results_white.svg' => ['ic_no_results_background_white', 144],
 21    'ic_no_results_black.svg' => ['ic_no_results_background_black', 144],
 22	'play_video_white.svg' => ['play_video_white', 128],
 23	'play_gif_white.svg' => ['play_gif_white', 128],
 24    'play_video_black.svg' => ['play_video_black', 128],
 25    'play_gif_black.svg' => ['play_gif_black', 128],
 26    'open_pdf_black.svg' => ['open_pdf_black', 128],
 27    'open_pdf_white.svg' => ['open_pdf_white', 128],
 28	'conversations_mono.svg' => ['conversations/ic_notification', 24],
 29    'quicksy_mono.svg' => ['quicksy/ic_notification', 24],
 30    'flip_camera_android-black-24dp.svg' => ['ic_flip_camera_android_black_24dp', 24],
 31    'ic_missed_call_notification.svg' => ['ic_missed_call_notification', 24],
 32	'ic_send_text_offline.svg' => ['ic_send_text_offline', 36],
 33	'ic_send_text_offline_white.svg' => ['ic_send_text_offline_white', 36],
 34	'ic_send_text_online.svg' => ['ic_send_text_online', 36],
 35	'ic_send_text_away.svg' => ['ic_send_text_away', 36],
 36	'ic_send_text_dnd.svg' => ['ic_send_text_dnd', 36],
 37	'ic_send_photo_online.svg' => ['ic_send_photo_online', 36],
 38	'ic_send_photo_offline.svg' => ['ic_send_photo_offline', 36],
 39	'ic_send_photo_offline_white.svg' => ['ic_send_photo_offline_white', 36],
 40	'ic_send_photo_away.svg' => ['ic_send_photo_away', 36],
 41	'ic_send_photo_dnd.svg' => ['ic_send_photo_dnd', 36],
 42	'ic_send_location_online.svg' => ['ic_send_location_online', 36],
 43	'ic_send_location_offline.svg' => ['ic_send_location_offline', 36],
 44	'ic_send_location_offline_white.svg' => ['ic_send_location_offline_white', 36],
 45	'ic_send_location_away.svg' => ['ic_send_location_away', 36],
 46	'ic_send_location_dnd.svg' => ['ic_send_location_dnd', 36],
 47	'ic_send_voice_online.svg' => ['ic_send_voice_online', 36],
 48	'ic_send_voice_offline.svg' => ['ic_send_voice_offline', 36],
 49	'ic_send_voice_offline_white.svg' => ['ic_send_voice_offline_white', 36],
 50	'ic_send_voice_away.svg' => ['ic_send_voice_away', 36],
 51	'ic_send_voice_dnd.svg' => ['ic_send_voice_dnd', 36],
 52	'ic_send_cancel_online.svg' => ['ic_send_cancel_online', 36],
 53	'ic_send_cancel_offline.svg' => ['ic_send_cancel_offline', 36],
 54	'ic_send_cancel_offline_white.svg' => ['ic_send_cancel_offline_white', 36],
 55	'ic_send_cancel_away.svg' => ['ic_send_cancel_away', 36],
 56	'ic_send_cancel_dnd.svg' => ['ic_send_cancel_dnd', 36],
 57	'ic_send_picture_online.svg' => ['ic_send_picture_online', 36],
 58	'ic_send_picture_offline.svg' => ['ic_send_picture_offline', 36],
 59	'ic_send_picture_offline_white.svg' => ['ic_send_picture_offline_white', 36],
 60	'ic_send_picture_away.svg' => ['ic_send_picture_away', 36],
 61	'ic_send_picture_dnd.svg' => ['ic_send_picture_dnd', 36],
 62	'ic_send_videocam_online.svg' => ['ic_send_videocam_online', 36],
 63	'ic_send_videocam_offline.svg' => ['ic_send_videocam_offline', 36],
 64	'ic_send_videocam_offline_white.svg' => ['ic_send_videocam_offline_white', 36],
 65	'ic_send_videocam_away.svg' => ['ic_send_videocam_away', 36],
 66	'ic_send_videocam_dnd.svg' => ['ic_send_videocam_dnd', 36],
 67	'ic_notifications_none_white80.svg' => ['ic_notifications_none_white80', 24],
 68	'ic_notifications_off_white80.svg' => ['ic_notifications_off_white80', 24],
 69	'ic_notifications_paused_white80.svg' => ['ic_notifications_paused_white80', 24],
 70	'ic_notifications_white80.svg' => ['ic_notifications_white80', 24],
 71	'ic_verified_fingerprint.svg' => ['ic_verified_fingerprint', 36],
 72    'qrcode-scan.svg' => ['ic_qr_code_scan_white_24dp', 24],
 73	'message_bubble_received.svg' => ['message_bubble_received.9', 0],
 74	'message_bubble_received_grey.svg' => ['message_bubble_received_grey.9', 0],
 75	'message_bubble_received_dark.svg' => ['message_bubble_received_dark.9', 0],
 76	'message_bubble_received_warning.svg' => ['message_bubble_received_warning.9', 0],
 77	'message_bubble_received_white.svg' => ['message_bubble_received_white.9', 0],
 78	'message_bubble_sent.svg' => ['message_bubble_sent.9', 0],
 79	'message_bubble_sent_grey.svg' => ['message_bubble_sent_grey.9', 0],
 80	'date_bubble_white.svg' => ['date_bubble_white.9', 0],
 81	'date_bubble_grey.svg' => ['date_bubble_grey.9', 0],
 82	'marker.svg' => ['marker', 0]
 83	}
 84
 85# Executable paths for Mac OSX
 86# "/Applications/Inkscape.app/Contents/Resources/bin/inkscape"
 87
 88inkscape = "inkscape"
 89imagemagick = "magick"
 90
 91def execute_cmd(cmd)
 92	puts cmd
 93	system cmd
 94end
 95
 96images.each do |source_filename, settings|
 97	svg_content = File.read(source_filename)
 98
 99	svg = XML::Document.string(svg_content)
100	base_width = svg.root["width"].to_i
101	base_height = svg.root["height"].to_i
102
103	guides = svg.find(".//sodipodi:guide","sodipodi:http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd")
104
105	resolutions.each do |resolution, factor|
106		output_filename, base_size = settings
107
108		if base_size > 0
109			width = factor * base_size
110			height = factor * base_size
111		else
112			width = factor * base_width
113			height = factor * base_height
114		end
115
116        output_parts = output_filename.split('/')
117
118        if output_parts.count != 2
119    		path = "../src/main/res/drawable-#{resolution}/#{output_filename}.png"
120        else
121            path = "../src/#{output_parts[0]}/res/drawable-#{resolution}/#{output_parts[1]}.png"
122        end
123		execute_cmd "#{inkscape} #{source_filename} -C -w #{width} -h #{height} -e #{path}"
124
125		top = []
126		right = []
127		bottom = []
128		left = []
129
130		guides.each do |guide|
131			orientation = guide["orientation"]
132			x, y = guide["position"].split(",")
133			x, y = x.to_i, y.to_i
134
135			if orientation == "1,0" and y == base_height
136				top.push(x * factor)
137			end
138
139			if orientation == "0,1" and x == base_width
140				right.push((base_height - y) * factor)
141			end
142
143			if orientation == "1,0" and y == 0
144				bottom.push(x * factor)
145			end
146
147			if orientation == "0,1" and x == 0
148				left.push((base_height - y) * factor)
149			end
150		end
151
152		next if top.length != 2
153		next if right.length != 2
154		next if bottom.length != 2
155		next if left.length != 2
156
157		execute_cmd "#{imagemagick} -background none PNG32:#{path} -gravity center -extent #{width+2}x#{height+2} PNG32:#{path}"
158
159		draw_format = "-draw \"line %d,%d %d,%d\""
160		top_line = draw_format % [top.min + 1, 0, top.max, 0]
161		right_line = draw_format % [width + 1, right.min + 1, width + 1, right.max]
162		bottom_line = draw_format % [bottom.min + 1, height + 1, bottom.max, height + 1]
163		left_line = draw_format % [0, left.min + 1, 0, left.max]
164		draws = "#{top_line} #{right_line} #{bottom_line} #{left_line}"
165
166		execute_cmd "#{imagemagick} -background none PNG32:#{path} -fill black -stroke none #{draws} PNG32:#{path}"
167	end
168end