mirror of https://github.com/procxx/kepka.git
				
				
				
			codegen_style completely replaces style gen with MetaStyle.
codegen_numbers project started to replace MetaStyle completely.
This commit is contained in:
		
							parent
							
								
									edd26b3224
								
							
						
					
					
						commit
						4fe70c3a12
					
				
							
								
								
									
										14
									
								
								Telegram.sln
								
								
								
								
							
							
						
						
									
										14
									
								
								Telegram.sln
								
								
								
								
							|  | @ -9,6 +9,7 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Telegram", "Telegram\Telegr | ||||||
| 		{E4DF8176-4DEF-4859-962F-B497E3E7A323} = {E4DF8176-4DEF-4859-962F-B497E3E7A323} | 		{E4DF8176-4DEF-4859-962F-B497E3E7A323} = {E4DF8176-4DEF-4859-962F-B497E3E7A323} | ||||||
| 		{E417CAA4-259B-4C99-88E3-805F1300E8EB} = {E417CAA4-259B-4C99-88E3-805F1300E8EB} | 		{E417CAA4-259B-4C99-88E3-805F1300E8EB} = {E417CAA4-259B-4C99-88E3-805F1300E8EB} | ||||||
| 		{EB7D16AC-EACF-4577-B05A-F28E5F356794} = {EB7D16AC-EACF-4577-B05A-F28E5F356794} | 		{EB7D16AC-EACF-4577-B05A-F28E5F356794} = {EB7D16AC-EACF-4577-B05A-F28E5F356794} | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6} = {7C25BFBD-7930-4DE2-AF33-27CE1CC521E6} | ||||||
| 	EndProjectSection | 	EndProjectSection | ||||||
| EndProject | EndProject | ||||||
| Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "MetaStyle", "Telegram\MetaStyle.vcxproj", "{6F483617-7C84-4E7E-91D8-1FF28A4CE3A0}" | Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "MetaStyle", "Telegram\MetaStyle.vcxproj", "{6F483617-7C84-4E7E-91D8-1FF28A4CE3A0}" | ||||||
|  | @ -25,6 +26,8 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Packer", "Telegram\Packer.v | ||||||
| EndProject | EndProject | ||||||
| Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "codegen_style", "Telegram\build\vc\codegen_style\codegen_style.vcxproj", "{E4DF8176-4DEF-4859-962F-B497E3E7A323}" | Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "codegen_style", "Telegram\build\vc\codegen_style\codegen_style.vcxproj", "{E4DF8176-4DEF-4859-962F-B497E3E7A323}" | ||||||
| EndProject | EndProject | ||||||
|  | Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "codegen_numbers", "Telegram\build\vc\codegen_numbers\codegen_numbers.vcxproj", "{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}" | ||||||
|  | EndProject | ||||||
| Global | Global | ||||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||||
| 		Debug|Win32 = Debug|Win32 | 		Debug|Win32 = Debug|Win32 | ||||||
|  | @ -95,11 +98,22 @@ Global | ||||||
| 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|Win32.ActiveCfg = Release|Win32 | 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|Win32.ActiveCfg = Release|Win32 | ||||||
| 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|Win32.Build.0 = Release|Win32 | 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|Win32.Build.0 = Release|Win32 | ||||||
| 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|x64.ActiveCfg = Release|Win32 | 		{E4DF8176-4DEF-4859-962F-B497E3E7A323}.Release|x64.ActiveCfg = Release|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Debug|Win32.ActiveCfg = Debug|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Debug|Win32.Build.0 = Debug|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Debug|x64.ActiveCfg = Debug|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Deploy|Win32.ActiveCfg = Debug|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Deploy|Win32.Build.0 = Debug|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Deploy|x64.ActiveCfg = Release|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Deploy|x64.Build.0 = Release|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Release|Win32.ActiveCfg = Release|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Release|Win32.Build.0 = Release|Win32 | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}.Release|x64.ActiveCfg = Release|Win32 | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
| 	GlobalSection(SolutionProperties) = preSolution | 	GlobalSection(SolutionProperties) = preSolution | ||||||
| 		HideSolutionNode = FALSE | 		HideSolutionNode = FALSE | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
| 	GlobalSection(NestedProjects) = preSolution | 	GlobalSection(NestedProjects) = preSolution | ||||||
| 		{E4DF8176-4DEF-4859-962F-B497E3E7A323} = {2F863EAD-33C9-4014-A573-93F085BA9CB1} | 		{E4DF8176-4DEF-4859-962F-B497E3E7A323} = {2F863EAD-33C9-4014-A573-93F085BA9CB1} | ||||||
|  | 		{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6} = {2F863EAD-33C9-4014-A573-93F085BA9CB1} | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
| EndGlobal | EndGlobal | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| 
 | 
 | ||||||
| // Legacy styles | // Legacy styles | ||||||
| using "Resources/style_classes.txt"; | using "Resources/basic_types.style"; | ||||||
| using "Resources/style.txt"; | using "Resources/basic.style"; | ||||||
| 
 | 
 | ||||||
| //using "overview/overview.style"; | //using "overview/overview.style"; | ||||||
|  |  | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,413 @@ | ||||||
|  | /* | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | textStyle { | ||||||
|  | 	linkFlags: font; | ||||||
|  | 	linkFlagsOver: font; | ||||||
|  | 	linkFg: color; | ||||||
|  | 	linkFgDown: color; | ||||||
|  | 	monoFg: color; | ||||||
|  | 	selectBg: color; | ||||||
|  | 	selectOverlay: color; | ||||||
|  | 	lineHeight: pixels; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | linkButton { | ||||||
|  | 	color: color; | ||||||
|  | 	overColor: color; | ||||||
|  | 	downColor: color; | ||||||
|  | 	font: font; | ||||||
|  | 	overFont: font; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | sysButton { | ||||||
|  | 	size: size; | ||||||
|  | 	img: sprite; | ||||||
|  | 	color: color; | ||||||
|  | 	overColor: color; | ||||||
|  | 	duration: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatButton { | ||||||
|  | 	color: color; | ||||||
|  | 	overColor: color; | ||||||
|  | 	downColor: color; | ||||||
|  | 
 | ||||||
|  | 	bgColor: color; | ||||||
|  | 	overBgColor: color; | ||||||
|  | 	downBgColor: color; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	textTop: pixels; | ||||||
|  | 	overTextTop: pixels; | ||||||
|  | 	downTextTop: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 	overFont: font; | ||||||
|  | 	duration: int; | ||||||
|  | 	cursor: cursor; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | iconedButton { | ||||||
|  | 	icon: sprite; | ||||||
|  | 	iconPos: point; | ||||||
|  | 	downIcon: sprite; | ||||||
|  | 	downIconPos: point; | ||||||
|  | 
 | ||||||
|  | 	color: color; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	overBgColor: color; | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 	font: font; | ||||||
|  | 
 | ||||||
|  | 	opacity: double; | ||||||
|  | 	overOpacity: double; | ||||||
|  | 
 | ||||||
|  | 	textPos: point; | ||||||
|  | 	downTextPos: point; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 	cursor: cursor; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatCheckbox { | ||||||
|  | 	textColor: color; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	disColor: color; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 	textTop: pixels; | ||||||
|  | 	textLeft: pixels; | ||||||
|  | 	font: font; | ||||||
|  | 	duration: int; | ||||||
|  | 	bgFunc: transition; | ||||||
|  | 	cursor: cursor; | ||||||
|  | 
 | ||||||
|  | 	disabledCursor: cursor; | ||||||
|  | 
 | ||||||
|  | 	imageRect: sprite; | ||||||
|  | 	chkImageRect: sprite; | ||||||
|  | 	overImageRect: sprite; | ||||||
|  | 	chkOverImageRect: sprite; | ||||||
|  | 	disImageRect: sprite; | ||||||
|  | 	chkDisImageRect: sprite; | ||||||
|  | 
 | ||||||
|  | 	imagePos: point; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatInput { | ||||||
|  | 	textColor: color; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	bgActive: color; | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 	textMrg: margins; | ||||||
|  | 	align: align; | ||||||
|  | 	font: font; | ||||||
|  | 	cursor: cursor; | ||||||
|  | 
 | ||||||
|  | 	imgRect: sprite; | ||||||
|  | 	imgPos: point; | ||||||
|  | 
 | ||||||
|  | 	borderWidth: pixels; | ||||||
|  | 	borderColor: color; | ||||||
|  | 	borderActive: color; | ||||||
|  | 	borderError: color; | ||||||
|  | 
 | ||||||
|  | 	phColor: color; | ||||||
|  | 	phFocusColor: color; | ||||||
|  | 	phPos: point; | ||||||
|  | 	phAlign: align; | ||||||
|  | 	phShift: pixels; | ||||||
|  | 	phDuration: int; | ||||||
|  | 	phLeftFunc: transition; | ||||||
|  | 	phAlphaFunc: transition; | ||||||
|  | 	phColorFunc: transition; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatTextarea { | ||||||
|  | 	textColor: color; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	width: pixels; | ||||||
|  | 	textMrg: margins; | ||||||
|  | 	align: align; | ||||||
|  | 	font: font; | ||||||
|  | 	cursor: cursor; | ||||||
|  | 
 | ||||||
|  | 	phColor: color; | ||||||
|  | 	phFocusColor: color; | ||||||
|  | 	phPos: point; | ||||||
|  | 	phAlign: align; | ||||||
|  | 	phShift: pixels; | ||||||
|  | 	phDuration: int; | ||||||
|  | 	phLeftFunc: transition; | ||||||
|  | 	phAlphaFunc: transition; | ||||||
|  | 	phColorFunc: transition; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatScroll { | ||||||
|  | 	barColor: color; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	barOverColor: color; | ||||||
|  | 	bgOverColor: color; | ||||||
|  | 
 | ||||||
|  | 	round: pixels; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	minHeight: pixels; | ||||||
|  | 	deltax: pixels; | ||||||
|  | 	deltat: pixels; | ||||||
|  | 	deltab: pixels; | ||||||
|  | 
 | ||||||
|  | 	topsh: pixels; | ||||||
|  | 	bottomsh: pixels; | ||||||
|  | 	shColor: color; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 	hiding: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | countryInput { | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 	top: pixels; | ||||||
|  | 	bgColor: color; | ||||||
|  | 	ptrSize: size; | ||||||
|  | 	textMrg: margins; | ||||||
|  | 	font: font; | ||||||
|  | 	align: align; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | slider { | ||||||
|  | 	color: color; | ||||||
|  | 	thikness: pixels; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	bar: sprite; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | flatLabel { | ||||||
|  | 	font: font; | ||||||
|  | 	minWidth: pixels; | ||||||
|  | 	width: pixels; | ||||||
|  | 	align: align; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | switcher { | ||||||
|  | 	border: pixels; | ||||||
|  | 	borderColor: color; | ||||||
|  | 
 | ||||||
|  | 	bgColor: color; | ||||||
|  | 	bgHovered: color; | ||||||
|  | 	bgActive: color; | ||||||
|  | 
 | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 	textColor: color; | ||||||
|  | 	activeColor: color; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | dropdown { | ||||||
|  | 	border: pixels; | ||||||
|  | 	borderColor: color; | ||||||
|  | 
 | ||||||
|  | 	padding: margins; | ||||||
|  | 	shadow: sprite; | ||||||
|  | 	shadowShift: pixels; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 	width: pixels; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | PopupMenu { | ||||||
|  | 	skip: pixels; | ||||||
|  | 
 | ||||||
|  | 	shadow: sprite; | ||||||
|  | 	shadowShift: pixels; | ||||||
|  | 
 | ||||||
|  | 	itemBg: color; | ||||||
|  | 	itemBgOver: color; | ||||||
|  | 	itemFg: color; | ||||||
|  | 	itemFgOver: color; | ||||||
|  | 	itemFgDisabled: color; | ||||||
|  | 	itemFgShortcut: color; | ||||||
|  | 	itemFgShortcutOver: color; | ||||||
|  | 	itemFgShortcutDisabled: color; | ||||||
|  | 	itemPadding: margins; | ||||||
|  | 	itemFont: font; | ||||||
|  | 
 | ||||||
|  | 	separatorPadding: margins; | ||||||
|  | 	separatorWidth: pixels; | ||||||
|  | 	separatorFg: color; | ||||||
|  | 
 | ||||||
|  | 	arrow: sprite; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 
 | ||||||
|  | 	widthMin: pixels; | ||||||
|  | 	widthMax: pixels; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Tooltip { | ||||||
|  | 	textBg: color; | ||||||
|  | 	textFg: color; | ||||||
|  | 	textFont: font; | ||||||
|  | 	textBorder: color; | ||||||
|  | 	textPadding: margins; | ||||||
|  | 
 | ||||||
|  | 	shift: point; | ||||||
|  | 	skip: pixels; | ||||||
|  | 
 | ||||||
|  | 	widthMax: pixels; | ||||||
|  | 	linesMax: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | botKeyboardButton { | ||||||
|  | 	margin: pixels; | ||||||
|  | 	padding: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 	textTop: pixels; | ||||||
|  | 	downTextTop: pixels; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | BoxButton { | ||||||
|  | 	textFg: color; | ||||||
|  | 	textFgOver: color; | ||||||
|  | 	textBg: color; // rect of textBg with rounded rect of textBgOver upon it | ||||||
|  | 	textBgOver: color; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	textTop: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 	duration: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Checkbox { | ||||||
|  | 	textFg: color; | ||||||
|  | 	textBg: color; | ||||||
|  | 
 | ||||||
|  | 	checkFg: color; | ||||||
|  | 	checkFgOver: color; | ||||||
|  | 	checkFgActive: color; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	textPosition: point; | ||||||
|  | 	diameter: pixels; | ||||||
|  | 	thickness: pixels; | ||||||
|  | 	checkIcon: sprite; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 	duration: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Radiobutton { | ||||||
|  | 	textFg: color; | ||||||
|  | 	textBg: color; | ||||||
|  | 
 | ||||||
|  | 	checkFg: color; | ||||||
|  | 	checkFgOver: color; | ||||||
|  | 	checkFgActive: color; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	textPosition: point; | ||||||
|  | 	diameter: pixels; | ||||||
|  | 	thickness: pixels; | ||||||
|  | 	checkSkip: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 	duration: int; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | InputArea { | ||||||
|  | 	textFg: color; | ||||||
|  | 	textMargins: margins; | ||||||
|  | 
 | ||||||
|  | 	placeholderFg: color; | ||||||
|  | 	placeholderFgActive: color; | ||||||
|  | 	placeholderMargins: margins; | ||||||
|  | 	placeholderAlign: align; | ||||||
|  | 	placeholderShift: pixels; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 
 | ||||||
|  | 	borderFg: color; | ||||||
|  | 	borderFgActive: color; | ||||||
|  | 	borderFgError: color; | ||||||
|  | 
 | ||||||
|  | 	border: pixels; | ||||||
|  | 	borderActive: pixels; | ||||||
|  | 	borderError: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	heightMin: pixels; | ||||||
|  | 	heightMax: pixels; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | InputField { | ||||||
|  | 	textFg: color; | ||||||
|  | 	textMargins: margins; | ||||||
|  | 	textAlign: align; | ||||||
|  | 
 | ||||||
|  | 	placeholderFg: color; | ||||||
|  | 	placeholderFgActive: color; | ||||||
|  | 	placeholderMargins: margins; | ||||||
|  | 	placeholderAlign: align; | ||||||
|  | 	placeholderShift: pixels; | ||||||
|  | 
 | ||||||
|  | 	duration: int; | ||||||
|  | 
 | ||||||
|  | 	borderFg: color; | ||||||
|  | 	borderFgActive: color; | ||||||
|  | 	borderFgError: color; | ||||||
|  | 
 | ||||||
|  | 	border: pixels; | ||||||
|  | 	borderActive: pixels; | ||||||
|  | 	borderError: pixels; | ||||||
|  | 
 | ||||||
|  | 	font: font; | ||||||
|  | 
 | ||||||
|  | 	width: pixels; | ||||||
|  | 	height: pixels; | ||||||
|  | 
 | ||||||
|  | 	iconSprite: sprite; | ||||||
|  | 	iconPosition: point; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | PeerAvatarButton { | ||||||
|  | 	size: pixels; | ||||||
|  | 	photoSize: pixels; | ||||||
|  | } | ||||||
|  | @ -18,10 +18,8 @@ to link the code of portions of this program with the OpenSSL library. | ||||||
| Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE | ||||||
| Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| using "Resources/style_classes.txt"; | defaultFontFamily: 'Open Sans'; | ||||||
| 
 | semibold: 'Open Sans Semibold'; | ||||||
| defaultFontFamily: "Open Sans"; |  | ||||||
| semibold: "Open Sans Semibold"; |  | ||||||
| 
 | 
 | ||||||
| fsize: 13px; | fsize: 13px; | ||||||
| normalFont: font(fsize); | normalFont: font(fsize); | ||||||
|  | @ -385,7 +383,7 @@ btnDefIconed: iconedButton { | ||||||
| 	font: font(fsize); | 	font: font(fsize); | ||||||
| 
 | 
 | ||||||
| 	opacity: 0.78; | 	opacity: 0.78; | ||||||
| 	overOpacity: 1.; | 	overOpacity: 1; | ||||||
| 
 | 
 | ||||||
| 	textPos: point(0px, 0px); | 	textPos: point(0px, 0px); | ||||||
| 	downTextPos: point(0px, 0px); | 	downTextPos: point(0px, 0px); | ||||||
|  | @ -433,7 +431,7 @@ titleBackButton: iconedButton(btnDefIconed) { | ||||||
| 	width: -30px; | 	width: -30px; | ||||||
| 	height: 39px; | 	height: 39px; | ||||||
| 
 | 
 | ||||||
| 	opacity: 1.; | 	opacity: 1; | ||||||
| 	cursor: cursor(default); | 	cursor: cursor(default); | ||||||
| 
 | 
 | ||||||
| 	textPos: point(23px, 10px); | 	textPos: point(23px, 10px); | ||||||
|  | @ -1117,8 +1115,8 @@ msgLinkColor: #2a6dc2; | ||||||
| msgPressedLinkColor: #004bad; | msgPressedLinkColor: #004bad; | ||||||
| msgSkip: 40px; | msgSkip: 40px; | ||||||
| msgPtr: 8px; | msgPtr: 8px; | ||||||
| msgBG: ":/gui/art/bg.jpg"; | msgBG: ':/gui/art/bg.jpg'; | ||||||
| msgBG0: ":/gui/art/bg0.png"; | msgBG0: ':/gui/art/bg0.png'; | ||||||
| 
 | 
 | ||||||
| msgCheckPos: point(3px, 1px); | msgCheckPos: point(3px, 1px); | ||||||
| msgSendingImg: sprite(260px, 20px, 20px, 20px); | msgSendingImg: sprite(260px, 20px, 20px, 20px); | ||||||
|  | @ -1549,7 +1547,7 @@ reportSpamButton: flatButton(reportSpamHide) { | ||||||
| reportSpamSeparator: 30px; | reportSpamSeparator: 30px; | ||||||
| reportSpamBg: #fffffff0; | reportSpamBg: #fffffff0; | ||||||
| 
 | 
 | ||||||
| newMsgSound: ":/gui/art/newmsg.wav"; | newMsgSound: ':/gui/art/newmsg.wav'; | ||||||
| 
 | 
 | ||||||
| unreadBarHeight: 32px; | unreadBarHeight: 32px; | ||||||
| unreadBarMargin: 8px; | unreadBarMargin: 8px; | ||||||
|  | @ -2189,7 +2187,7 @@ mvDropdown: dropdown(dropdownDef) { | ||||||
| 	shadow: sprite(0px, 0px, 0px, 0px); | 	shadow: sprite(0px, 0px, 0px, 0px); | ||||||
| 	padding: margins(11px, 12px, 11px, 12px); | 	padding: margins(11px, 12px, 11px, 12px); | ||||||
| 
 | 
 | ||||||
| 	border: 0px; | 	border: 0; | ||||||
| 	width: 182px; | 	width: 182px; | ||||||
| } | } | ||||||
| mvButton: iconedButton(btnDefIconed) { | mvButton: iconedButton(btnDefIconed) { | ||||||
|  | @ -2197,8 +2195,8 @@ mvButton: iconedButton(btnDefIconed) { | ||||||
| 	overBgColor: #505050; | 	overBgColor: #505050; | ||||||
| 	font: font(fsize); | 	font: font(fsize); | ||||||
| 
 | 
 | ||||||
| 	opacity: 1.; | 	opacity: 1; | ||||||
| 	overOpacity: 1.; | 	overOpacity: 1; | ||||||
| 
 | 
 | ||||||
| 	width: -32px; | 	width: -32px; | ||||||
| 	height: 36px; | 	height: 36px; | ||||||
|  | @ -2298,8 +2296,8 @@ overviewFileExtFont: font(18px semibold); | ||||||
| 
 | 
 | ||||||
| // Mac specific | // Mac specific | ||||||
| 
 | 
 | ||||||
| macAccessoryWidth: 450.; | macAccessoryWidth: 450; | ||||||
| macAccessoryHeight: 90.; | macAccessoryHeight: 90; | ||||||
| macEnableFilterAdd: 2; | macEnableFilterAdd: 2; | ||||||
| macEnableFilterTop: 5; | macEnableFilterTop: 5; | ||||||
| macSelectorTop: 6; | macSelectorTop: 6; | ||||||
|  | @ -2312,8 +2310,8 @@ btnContext: iconedButton(btnDefIconed) { | ||||||
| 	overBgColor: btnWhiteHover; | 	overBgColor: btnWhiteHover; | ||||||
| 	font: font(14px); | 	font: font(14px); | ||||||
| 
 | 
 | ||||||
| 	opacity: 1.; | 	opacity: 1; | ||||||
| 	overOpacity: 1.; | 	overOpacity: 1; | ||||||
| 
 | 
 | ||||||
| 	width: -32px; | 	width: -32px; | ||||||
| 	height: 36px; | 	height: 36px; | ||||||
|  | @ -2400,8 +2398,8 @@ passcodeSkip: 31px; | ||||||
| 
 | 
 | ||||||
| mentionHeight: 40px; | mentionHeight: 40px; | ||||||
| mentionScroll: flatScroll(scrollDef) { | mentionScroll: flatScroll(scrollDef) { | ||||||
| 	topsh: 0px; | 	topsh: 0; | ||||||
| 	bottomsh: 0px; | 	bottomsh: 0; | ||||||
| } | } | ||||||
| mentionPadding: margins(8px, 5px, 8px, 5px); | mentionPadding: margins(8px, 5px, 8px, 5px); | ||||||
| mentionTop: 11px; | mentionTop: 11px; | ||||||
|  | @ -2421,7 +2419,7 @@ sessionsHeight: 440px; | ||||||
| sessionHeight: 70px; | sessionHeight: 70px; | ||||||
| sessionCurrentPadding: margins(0px, 7px, 0px, 4px); | sessionCurrentPadding: margins(0px, 7px, 0px, 4px); | ||||||
| sessionCurrentHeight: 118px; | sessionCurrentHeight: 118px; | ||||||
| sessionPadding: margins(21px, 10px, 21px, 0px); | sessionPadding: margins(21px, 10px, 21px, 0); | ||||||
| sessionNameFont: msgNameFont; | sessionNameFont: msgNameFont; | ||||||
| sessionActiveFont: msgDateFont; | sessionActiveFont: msgDateFont; | ||||||
| sessionActiveColor: #aaa; | sessionActiveColor: #aaa; | ||||||
|  |  | ||||||
|  | @ -26,7 +26,7 @@ textStyle { | ||||||
| 	monoFg: color; | 	monoFg: color; | ||||||
| 	selectBg: color; | 	selectBg: color; | ||||||
| 	selectOverlay: color; | 	selectOverlay: color; | ||||||
| 	lineHeight: pixels; | 	lineHeight: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| linkButton { | linkButton { | ||||||
|  | @ -42,7 +42,7 @@ sysButton { | ||||||
| 	img: sprite; | 	img: sprite; | ||||||
| 	color: color; | 	color: color; | ||||||
| 	overColor: color; | 	overColor: color; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| flatButton { | flatButton { | ||||||
|  | @ -54,16 +54,16 @@ flatButton { | ||||||
| 	overBgColor: color; | 	overBgColor: color; | ||||||
| 	downBgColor: color; | 	downBgColor: color; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	textTop: pixels; | 	textTop: number; | ||||||
| 	overTextTop: pixels; | 	overTextTop: number; | ||||||
| 	downTextTop: pixels; | 	downTextTop: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 	overFont: font; | 	overFont: font; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 	cursor: cursor; | 	cursor: cursor; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -76,17 +76,17 @@ iconedButton { | ||||||
| 	color: color; | 	color: color; | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	overBgColor: color; | 	overBgColor: color; | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 	font: font; | 	font: font; | ||||||
| 
 | 
 | ||||||
| 	opacity: double; | 	opacity: number; | ||||||
| 	overOpacity: double; | 	overOpacity: number; | ||||||
| 
 | 
 | ||||||
| 	textPos: point; | 	textPos: point; | ||||||
| 	downTextPos: point; | 	downTextPos: point; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 	cursor: cursor; | 	cursor: cursor; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -95,12 +95,12 @@ flatCheckbox { | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	disColor: color; | 	disColor: color; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 	textTop: pixels; | 	textTop: number; | ||||||
| 	textLeft: pixels; | 	textLeft: number; | ||||||
| 	font: font; | 	font: font; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 	bgFunc: transition; | 	bgFunc: transition; | ||||||
| 	cursor: cursor; | 	cursor: cursor; | ||||||
| 
 | 
 | ||||||
|  | @ -120,8 +120,8 @@ flatInput { | ||||||
| 	textColor: color; | 	textColor: color; | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	bgActive: color; | 	bgActive: color; | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 	textMrg: margins; | 	textMrg: margins; | ||||||
| 	align: align; | 	align: align; | ||||||
| 	font: font; | 	font: font; | ||||||
|  | @ -130,7 +130,7 @@ flatInput { | ||||||
| 	imgRect: sprite; | 	imgRect: sprite; | ||||||
| 	imgPos: point; | 	imgPos: point; | ||||||
| 
 | 
 | ||||||
| 	borderWidth: pixels; | 	borderWidth: number; | ||||||
| 	borderColor: color; | 	borderColor: color; | ||||||
| 	borderActive: color; | 	borderActive: color; | ||||||
| 	borderError: color; | 	borderError: color; | ||||||
|  | @ -139,8 +139,8 @@ flatInput { | ||||||
| 	phFocusColor: color; | 	phFocusColor: color; | ||||||
| 	phPos: point; | 	phPos: point; | ||||||
| 	phAlign: align; | 	phAlign: align; | ||||||
| 	phShift: pixels; | 	phShift: number; | ||||||
| 	phDuration: int; | 	phDuration: number; | ||||||
| 	phLeftFunc: transition; | 	phLeftFunc: transition; | ||||||
| 	phAlphaFunc: transition; | 	phAlphaFunc: transition; | ||||||
| 	phColorFunc: transition; | 	phColorFunc: transition; | ||||||
|  | @ -149,7 +149,7 @@ flatInput { | ||||||
| flatTextarea { | flatTextarea { | ||||||
| 	textColor: color; | 	textColor: color; | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	textMrg: margins; | 	textMrg: margins; | ||||||
| 	align: align; | 	align: align; | ||||||
| 	font: font; | 	font: font; | ||||||
|  | @ -159,8 +159,8 @@ flatTextarea { | ||||||
| 	phFocusColor: color; | 	phFocusColor: color; | ||||||
| 	phPos: point; | 	phPos: point; | ||||||
| 	phAlign: align; | 	phAlign: align; | ||||||
| 	phShift: pixels; | 	phShift: number; | ||||||
| 	phDuration: int; | 	phDuration: number; | ||||||
| 	phLeftFunc: transition; | 	phLeftFunc: transition; | ||||||
| 	phAlphaFunc: transition; | 	phAlphaFunc: transition; | ||||||
| 	phColorFunc: transition; | 	phColorFunc: transition; | ||||||
|  | @ -172,26 +172,26 @@ flatScroll { | ||||||
| 	barOverColor: color; | 	barOverColor: color; | ||||||
| 	bgOverColor: color; | 	bgOverColor: color; | ||||||
| 
 | 
 | ||||||
| 	round: pixels; | 	round: number; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	minHeight: pixels; | 	minHeight: number; | ||||||
| 	deltax: pixels; | 	deltax: number; | ||||||
| 	deltat: pixels; | 	deltat: number; | ||||||
| 	deltab: pixels; | 	deltab: number; | ||||||
| 
 | 
 | ||||||
| 	topsh: pixels; | 	topsh: number; | ||||||
| 	bottomsh: pixels; | 	bottomsh: number; | ||||||
| 	shColor: color; | 	shColor: color; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 	hiding: int; | 	hiding: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| countryInput { | countryInput { | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 	top: pixels; | 	top: number; | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	ptrSize: size; | 	ptrSize: size; | ||||||
| 	textMrg: margins; | 	textMrg: margins; | ||||||
|  | @ -201,53 +201,53 @@ countryInput { | ||||||
| 
 | 
 | ||||||
| slider { | slider { | ||||||
| 	color: color; | 	color: color; | ||||||
| 	thikness: pixels; | 	thikness: number; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	bar: sprite; | 	bar: sprite; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| flatLabel { | flatLabel { | ||||||
| 	font: font; | 	font: font; | ||||||
| 	minWidth: pixels; | 	minWidth: number; | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	align: align; | 	align: align; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| switcher { | switcher { | ||||||
| 	border: pixels; | 	border: number; | ||||||
| 	borderColor: color; | 	borderColor: color; | ||||||
| 
 | 
 | ||||||
| 	bgColor: color; | 	bgColor: color; | ||||||
| 	bgHovered: color; | 	bgHovered: color; | ||||||
| 	bgActive: color; | 	bgActive: color; | ||||||
| 
 | 
 | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 	textColor: color; | 	textColor: color; | ||||||
| 	activeColor: color; | 	activeColor: color; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| dropdown { | dropdown { | ||||||
| 	border: pixels; | 	border: number; | ||||||
| 	borderColor: color; | 	borderColor: color; | ||||||
| 
 | 
 | ||||||
| 	padding: margins; | 	padding: margins; | ||||||
| 	shadow: sprite; | 	shadow: sprite; | ||||||
| 	shadowShift: pixels; | 	shadowShift: number; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 	width: pixels; | 	width: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PopupMenu { | PopupMenu { | ||||||
| 	skip: pixels; | 	skip: number; | ||||||
| 
 | 
 | ||||||
| 	shadow: sprite; | 	shadow: sprite; | ||||||
| 	shadowShift: pixels; | 	shadowShift: number; | ||||||
| 
 | 
 | ||||||
| 	itemBg: color; | 	itemBg: color; | ||||||
| 	itemBgOver: color; | 	itemBgOver: color; | ||||||
|  | @ -261,15 +261,15 @@ PopupMenu { | ||||||
| 	itemFont: font; | 	itemFont: font; | ||||||
| 
 | 
 | ||||||
| 	separatorPadding: margins; | 	separatorPadding: margins; | ||||||
| 	separatorWidth: pixels; | 	separatorWidth: number; | ||||||
| 	separatorFg: color; | 	separatorFg: color; | ||||||
| 
 | 
 | ||||||
| 	arrow: sprite; | 	arrow: sprite; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 
 | 
 | ||||||
| 	widthMin: pixels; | 	widthMin: number; | ||||||
| 	widthMax: pixels; | 	widthMax: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Tooltip { | Tooltip { | ||||||
|  | @ -280,18 +280,18 @@ Tooltip { | ||||||
| 	textPadding: margins; | 	textPadding: margins; | ||||||
| 
 | 
 | ||||||
| 	shift: point; | 	shift: point; | ||||||
| 	skip: pixels; | 	skip: number; | ||||||
| 
 | 
 | ||||||
| 	widthMax: pixels; | 	widthMax: number; | ||||||
| 	linesMax: int; | 	linesMax: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| botKeyboardButton { | botKeyboardButton { | ||||||
| 	margin: pixels; | 	margin: number; | ||||||
| 	padding: pixels; | 	padding: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 	textTop: pixels; | 	textTop: number; | ||||||
| 	downTextTop: pixels; | 	downTextTop: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| BoxButton { | BoxButton { | ||||||
|  | @ -300,13 +300,13 @@ BoxButton { | ||||||
| 	textBg: color; // rect of textBg with rounded rect of textBgOver upon it | 	textBg: color; // rect of textBg with rounded rect of textBgOver upon it | ||||||
| 	textBgOver: color; | 	textBgOver: color; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	textTop: pixels; | 	textTop: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Checkbox { | Checkbox { | ||||||
|  | @ -317,16 +317,16 @@ Checkbox { | ||||||
| 	checkFgOver: color; | 	checkFgOver: color; | ||||||
| 	checkFgActive: color; | 	checkFgActive: color; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	textPosition: point; | 	textPosition: point; | ||||||
| 	diameter: pixels; | 	diameter: number; | ||||||
| 	thickness: pixels; | 	thickness: number; | ||||||
| 	checkIcon: sprite; | 	checkIcon: sprite; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Radiobutton { | Radiobutton { | ||||||
|  | @ -337,16 +337,16 @@ Radiobutton { | ||||||
| 	checkFgOver: color; | 	checkFgOver: color; | ||||||
| 	checkFgActive: color; | 	checkFgActive: color; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	textPosition: point; | 	textPosition: point; | ||||||
| 	diameter: pixels; | 	diameter: number; | ||||||
| 	thickness: pixels; | 	thickness: number; | ||||||
| 	checkSkip: pixels; | 	checkSkip: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 	duration: int; | 	duration: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| InputArea { | InputArea { | ||||||
|  | @ -357,23 +357,23 @@ InputArea { | ||||||
| 	placeholderFgActive: color; | 	placeholderFgActive: color; | ||||||
| 	placeholderMargins: margins; | 	placeholderMargins: margins; | ||||||
| 	placeholderAlign: align; | 	placeholderAlign: align; | ||||||
| 	placeholderShift: pixels; | 	placeholderShift: number; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 
 | 
 | ||||||
| 	borderFg: color; | 	borderFg: color; | ||||||
| 	borderFgActive: color; | 	borderFgActive: color; | ||||||
| 	borderFgError: color; | 	borderFgError: color; | ||||||
| 
 | 
 | ||||||
| 	border: pixels; | 	border: number; | ||||||
| 	borderActive: pixels; | 	borderActive: number; | ||||||
| 	borderError: pixels; | 	borderError: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	heightMin: pixels; | 	heightMin: number; | ||||||
| 	heightMax: pixels; | 	heightMax: number; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| InputField { | InputField { | ||||||
|  | @ -385,28 +385,28 @@ InputField { | ||||||
| 	placeholderFgActive: color; | 	placeholderFgActive: color; | ||||||
| 	placeholderMargins: margins; | 	placeholderMargins: margins; | ||||||
| 	placeholderAlign: align; | 	placeholderAlign: align; | ||||||
| 	placeholderShift: pixels; | 	placeholderShift: number; | ||||||
| 
 | 
 | ||||||
| 	duration: int; | 	duration: number; | ||||||
| 
 | 
 | ||||||
| 	borderFg: color; | 	borderFg: color; | ||||||
| 	borderFgActive: color; | 	borderFgActive: color; | ||||||
| 	borderFgError: color; | 	borderFgError: color; | ||||||
| 
 | 
 | ||||||
| 	border: pixels; | 	border: number; | ||||||
| 	borderActive: pixels; | 	borderActive: number; | ||||||
| 	borderError: pixels; | 	borderError: number; | ||||||
| 
 | 
 | ||||||
| 	font: font; | 	font: font; | ||||||
| 
 | 
 | ||||||
| 	width: pixels; | 	width: number; | ||||||
| 	height: pixels; | 	height: number; | ||||||
| 
 | 
 | ||||||
| 	iconSprite: sprite; | 	iconSprite: sprite; | ||||||
| 	iconPosition: point; | 	iconPosition: point; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PeerAvatarButton { | PeerAvatarButton { | ||||||
| 	size: pixels; | 	size: number; | ||||||
| 	photoSize: pixels; | 	photoSize: number; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -82,7 +82,7 @@ namespace { | ||||||
| 
 | 
 | ||||||
| 	HistoryItem *hoveredItem = 0, *pressedItem = 0, *hoveredLinkItem = 0, *pressedLinkItem = 0, *contextItem = 0, *mousedItem = 0; | 	HistoryItem *hoveredItem = 0, *pressedItem = 0, *hoveredLinkItem = 0, *pressedLinkItem = 0, *contextItem = 0, *mousedItem = 0; | ||||||
| 
 | 
 | ||||||
| 	QPixmap *sprite = 0, *emoji = 0, *emojiLarge = 0; | 	QPixmap *emoji = 0, *emojiLarge = 0; | ||||||
| 	style::font monofont; | 	style::font monofont; | ||||||
| 
 | 
 | ||||||
| 	struct CornersPixmaps { | 	struct CornersPixmaps { | ||||||
|  | @ -1993,23 +1993,6 @@ namespace { | ||||||
| 			if (family.isEmpty()) family = QFontDatabase::systemFont(QFontDatabase::FixedFont).family(); | 			if (family.isEmpty()) family = QFontDatabase::systemFont(QFontDatabase::FixedFont).family(); | ||||||
| 			::monofont = style::font(st::normalFont->f.pixelSize(), 0, family); | 			::monofont = style::font(st::normalFont->f.pixelSize(), 0, family); | ||||||
| 		} | 		} | ||||||
| 		if (!::sprite) { |  | ||||||
| 			QString spriteFilePostfix; |  | ||||||
| 			if (cRetina() || cScale() == dbisTwo) { |  | ||||||
| 				spriteFilePostfix = qsl("_200x"); |  | ||||||
| 			} else if (cScale() == dbisOneAndQuarter) { |  | ||||||
| 				spriteFilePostfix = qsl("_125x"); |  | ||||||
| 			} else if (cScale() == dbisOneAndHalf) { |  | ||||||
| 				spriteFilePostfix = qsl("_150x"); |  | ||||||
| 			} |  | ||||||
| 			QString spriteFile = qsl(":/gui/art/sprite") + spriteFilePostfix + qsl(".png"); |  | ||||||
| 			if (rtl()) { |  | ||||||
| 				::sprite = new QPixmap(QPixmap::fromImage(QImage(spriteFile).mirrored(true, false))); |  | ||||||
| 			} else { |  | ||||||
| 				::sprite = new QPixmap(spriteFile); |  | ||||||
| 			} |  | ||||||
|             if (cRetina()) ::sprite->setDevicePixelRatio(cRetinaFactor()); |  | ||||||
| 		} |  | ||||||
| 		emojiInit(); | 		emojiInit(); | ||||||
| 		if (!::emoji) { | 		if (!::emoji) { | ||||||
| 			::emoji = new QPixmap(QLatin1String(EName)); | 			::emoji = new QPixmap(QLatin1String(EName)); | ||||||
|  | @ -2070,8 +2053,6 @@ namespace { | ||||||
| 	void deinitMedia() { | 	void deinitMedia() { | ||||||
| 		audioFinish(); | 		audioFinish(); | ||||||
| 
 | 
 | ||||||
| 		delete ::sprite; |  | ||||||
| 		::sprite = 0; |  | ||||||
| 		delete ::emoji; | 		delete ::emoji; | ||||||
| 		::emoji = 0; | 		::emoji = 0; | ||||||
| 		delete ::emojiLarge; | 		delete ::emojiLarge; | ||||||
|  | @ -2149,7 +2130,7 @@ namespace { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	const QPixmap &sprite() { | 	const QPixmap &sprite() { | ||||||
| 		return *::sprite; | 		return style::spritePixmap(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	const QPixmap &emoji() { | 	const QPixmap &emoji() { | ||||||
|  |  | ||||||
|  | @ -36,7 +36,7 @@ class LogStream; | ||||||
| // Interface for reading a cleaned from comments file by basic tokens.
 | // Interface for reading a cleaned from comments file by basic tokens.
 | ||||||
| class BasicTokenizedFile { | class BasicTokenizedFile { | ||||||
| public: | public: | ||||||
| 	BasicTokenizedFile(const QString &filepath); | 	explicit BasicTokenizedFile(const QString &filepath); | ||||||
| 	BasicTokenizedFile(const BasicTokenizedFile &other) = delete; | 	BasicTokenizedFile(const BasicTokenizedFile &other) = delete; | ||||||
| 	BasicTokenizedFile &operator=(const BasicTokenizedFile &other) = delete; | 	BasicTokenizedFile &operator=(const BasicTokenizedFile &other) = delete; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -36,9 +36,9 @@ public: | ||||||
| 	CheckedUtf8String(const CheckedUtf8String &other) = default; | 	CheckedUtf8String(const CheckedUtf8String &other) = default; | ||||||
| 	CheckedUtf8String &operator=(const CheckedUtf8String &other) = default; | 	CheckedUtf8String &operator=(const CheckedUtf8String &other) = default; | ||||||
| 
 | 
 | ||||||
| 	CheckedUtf8String(const char *string, int size = -1); | 	explicit CheckedUtf8String(const char *string, int size = -1); | ||||||
| 	CheckedUtf8String(const QByteArray &string); | 	explicit CheckedUtf8String(const QByteArray &string); | ||||||
| 	CheckedUtf8String(const ConstUtf8String &string); | 	explicit CheckedUtf8String(const ConstUtf8String &string); | ||||||
| 
 | 
 | ||||||
| 	bool isValid() const { | 	bool isValid() const { | ||||||
| 		return valid_; | 		return valid_; | ||||||
|  |  | ||||||
|  | @ -32,7 +32,7 @@ namespace common { | ||||||
| // Reads a file removing all C-style comments.
 | // Reads a file removing all C-style comments.
 | ||||||
| class CleanFile { | class CleanFile { | ||||||
| public: | public: | ||||||
| 	CleanFile(const QString &filepath); | 	explicit CleanFile(const QString &filepath); | ||||||
| 	CleanFile(const CleanFile &other) = delete; | 	CleanFile(const CleanFile &other) = delete; | ||||||
| 	CleanFile &operator=(const CleanFile &other) = delete; | 	CleanFile &operator=(const CleanFile &other) = delete; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -31,7 +31,7 @@ namespace common { | ||||||
| // Not null-terminated! It does not hold any ownership.
 | // Not null-terminated! It does not hold any ownership.
 | ||||||
| class ConstUtf8String { | class ConstUtf8String { | ||||||
| public: | public: | ||||||
| 	ConstUtf8String(const char *string, int size = -1) : string_(string) { | 	explicit ConstUtf8String(const char *string, int size = -1) : string_(string) { | ||||||
| 		if (size < 0) { | 		if (size < 0) { | ||||||
| 			size = strlen(string); | 			size = strlen(string); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | @ -0,0 +1,135 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/common/cpp_file.h" | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QFileInfo> | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace common { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | void writeLicense(QTextStream &stream, const ProjectInfo &project) { | ||||||
|  | 	stream << "\
 | ||||||
|  | /*\n\
 | ||||||
|  | WARNING! All changes made in this file will be lost!\n\ | ||||||
|  | Created from '" << project.source << "' by '" << project.name << "'\n\ | ||||||
|  | \n\ | ||||||
|  | This file is part of Telegram Desktop,\n\ | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org\n\ | ||||||
|  | \n\ | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify\n\ | ||||||
|  | it under the terms of the GNU General Public License as published by\n\ | ||||||
|  | the Free Software Foundation, either version 3 of the License, or\n\ | ||||||
|  | (at your option) any later version.\n\ | ||||||
|  | \n\ | ||||||
|  | It is distributed in the hope that it will be useful,\n\ | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of\n\ | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n\ | ||||||
|  | GNU General Public License for more details.\n\ | ||||||
|  | \n\ | ||||||
|  | In addition, as a special exception, the copyright holders give permission\n\ | ||||||
|  | to link the code of portions of this program with the OpenSSL library.\n\ | ||||||
|  | \n\ | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE\n\ | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org\n\ | ||||||
|  | */\n"; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | CppFile::CppFile(const QString &path, const ProjectInfo &project) | ||||||
|  | : stream_(&content_) | ||||||
|  | , forceReGenerate_(project.forceReGenerate) { | ||||||
|  | 	bool cpp = path.toLower().endsWith(".cpp"); | ||||||
|  | 
 | ||||||
|  | 	QFileInfo info(path); | ||||||
|  | 	info.dir().mkpath("."); | ||||||
|  | 	filepath_ = info.absoluteFilePath(); | ||||||
|  | 
 | ||||||
|  | 	writeLicense(stream_, project); | ||||||
|  | 	if (cpp) { | ||||||
|  | 		if (!project.precompiledHeader.isEmpty()) { | ||||||
|  | 			include(project.precompiledHeader); | ||||||
|  | 		} | ||||||
|  | 		include(info.baseName() + ".h").newline(); | ||||||
|  | 	} else { | ||||||
|  | 		stream() << "#pragma once"; | ||||||
|  | 		newline().newline(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | CppFile &CppFile::include(const QString &header) { | ||||||
|  | 	stream() << "#include \"" << header << "\""; | ||||||
|  | 	return newline(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | CppFile &CppFile::pushNamespace(const QString &name) { | ||||||
|  | 	namespaces_.push_back(name); | ||||||
|  | 
 | ||||||
|  | 	stream() << "namespace"; | ||||||
|  | 	if (!name.isEmpty()) { | ||||||
|  | 		stream() << ' ' << name; | ||||||
|  | 	} | ||||||
|  | 	stream() << " {"; | ||||||
|  | 	return newline(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | CppFile &CppFile::popNamespace() { | ||||||
|  | 	if (namespaces_.isEmpty()) { | ||||||
|  | 		return *this; | ||||||
|  | 	} | ||||||
|  | 	auto name = namespaces_.back(); | ||||||
|  | 	namespaces_.pop_back(); | ||||||
|  | 
 | ||||||
|  | 	stream() << "} // namespace"; | ||||||
|  | 	if (!name.isEmpty()) { | ||||||
|  | 		stream() << ' ' << name; | ||||||
|  | 	} | ||||||
|  | 	return newline(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool CppFile::finalize() { | ||||||
|  | 	while (!namespaces_.isEmpty()) { | ||||||
|  | 		popNamespace(); | ||||||
|  | 	} | ||||||
|  | 	stream_.flush(); | ||||||
|  | 
 | ||||||
|  | 	QFile file(filepath_); | ||||||
|  | 	if (!forceReGenerate_ && file.open(QIODevice::ReadOnly)) { | ||||||
|  | 		if (file.readAll() == content_) { | ||||||
|  | 			file.close(); | ||||||
|  | 			return true; | ||||||
|  | 		} | ||||||
|  | 		file.close(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!file.open(QIODevice::WriteOnly)) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (file.write(content_) != content_.size()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace common
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,70 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include <QtCore/QVector> | ||||||
|  | #include <QtCore/QTextStream> | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace common { | ||||||
|  | 
 | ||||||
|  | struct ProjectInfo { | ||||||
|  | 	QString name; | ||||||
|  | 	QString source; | ||||||
|  | 	QString precompiledHeader; | ||||||
|  | 	bool forceReGenerate; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Creates a file with license header and codegen warning.
 | ||||||
|  | class CppFile { | ||||||
|  | public: | ||||||
|  | 	// If "basepath" is empty the folder containing "path" will be chosen.
 | ||||||
|  | 	// File ending with .cpp will be treated as source, otherwise like header.
 | ||||||
|  | 	CppFile(const QString &path, const ProjectInfo &project); | ||||||
|  | 
 | ||||||
|  | 	QTextStream &stream() { | ||||||
|  | 		return stream_; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	CppFile &newline() { | ||||||
|  | 		stream() << "\n"; | ||||||
|  | 		return *this; | ||||||
|  | 	} | ||||||
|  | 	CppFile &include(const QString &header); | ||||||
|  | 
 | ||||||
|  | 	// Empty name adds anonymous namespace.
 | ||||||
|  | 	CppFile &pushNamespace(const QString &name = QString()); | ||||||
|  | 	CppFile &popNamespace(); | ||||||
|  | 
 | ||||||
|  | 	bool finalize(); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 	QString filepath_; | ||||||
|  | 	QByteArray content_; | ||||||
|  | 	QTextStream stream_; | ||||||
|  | 	QVector<QString> namespaces_; | ||||||
|  | 	bool forceReGenerate_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace common
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,54 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/numbers/generator.h" | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QSet> | ||||||
|  | #include <functional> | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | Generator::Generator(const Rules &rules, const QString &destBasePath, const common::ProjectInfo &project) | ||||||
|  | : rules_(rules) | ||||||
|  | , basePath_(destBasePath) | ||||||
|  | , project_(project) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeHeader() { | ||||||
|  | 	header_ = std::make_unique<common::CppFile>(basePath_ + ".h", project_); | ||||||
|  | 
 | ||||||
|  | 	header_->stream() << "QVector<int> phoneNumberParse(const QString &number);\n"; | ||||||
|  | 
 | ||||||
|  | 	return header_->finalize(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeSource() { | ||||||
|  | 	source_ = std::make_unique<common::CppFile>(basePath_ + ".cpp", project_); | ||||||
|  | 
 | ||||||
|  | 	return source_->finalize(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <memory> | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include <QtCore/QSet> | ||||||
|  | #include "codegen/common/cpp_file.h" | ||||||
|  | #include "codegen/numbers/parsed_file.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | 
 | ||||||
|  | class Generator { | ||||||
|  | public: | ||||||
|  | 	Generator(const Rules &rules, const QString &destBasePath, const common::ProjectInfo &project); | ||||||
|  | 	Generator(const Generator &other) = delete; | ||||||
|  | 	Generator &operator=(const Generator &other) = delete; | ||||||
|  | 
 | ||||||
|  | 	bool writeHeader(); | ||||||
|  | 	bool writeSource(); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 	const Rules &rules_; | ||||||
|  | 	QString basePath_; | ||||||
|  | 	const common::ProjectInfo &project_; | ||||||
|  | 	std::unique_ptr<common::CppFile> source_, header_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,36 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include <QtCore/QCoreApplication> | ||||||
|  | 
 | ||||||
|  | #include "codegen/numbers/options.h" | ||||||
|  | #include "codegen/numbers/processor.h" | ||||||
|  | 
 | ||||||
|  | int main(int argc, char *argv[]) { | ||||||
|  | 	QCoreApplication app(argc, argv); | ||||||
|  | 
 | ||||||
|  | 	auto options = codegen::numbers::parseOptions(); | ||||||
|  | 	if (options.inputPath.isEmpty()) { | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	codegen::numbers::Processor processor(options); | ||||||
|  | 	return processor.launch(); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,74 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/numbers/options.h" | ||||||
|  | 
 | ||||||
|  | #include <ostream> | ||||||
|  | #include <QtCore/QCoreApplication> | ||||||
|  | #include "codegen/common/logging.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | constexpr int kErrorOutputPathExpected      = 902; | ||||||
|  | constexpr int kErrorInputPathExpected       = 903; | ||||||
|  | constexpr int kErrorSingleInputPathExpected = 904; | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | using common::logError; | ||||||
|  | 
 | ||||||
|  | Options parseOptions() { | ||||||
|  | 	Options result; | ||||||
|  | 	auto args(QCoreApplication::instance()->arguments()); | ||||||
|  | 	for (int i = 1, count = args.size(); i < count; ++i) { // skip first
 | ||||||
|  | 		const auto &arg(args.at(i)); | ||||||
|  | 
 | ||||||
|  | 		// Output path
 | ||||||
|  | 		if (arg == "-o") { | ||||||
|  | 			if (++i == count) { | ||||||
|  | 				logError(kErrorOutputPathExpected, "Command Line") << "output path expected after -o"; | ||||||
|  | 				return Options(); | ||||||
|  | 			} else { | ||||||
|  | 				result.outputPath = args.at(i); | ||||||
|  | 			} | ||||||
|  | 		} else if (arg.startsWith("-o")) { | ||||||
|  | 			result.outputPath = arg.mid(2); | ||||||
|  | 
 | ||||||
|  | 		// Input path
 | ||||||
|  | 		} else { | ||||||
|  | 			if (result.inputPath.isEmpty()) { | ||||||
|  | 				result.inputPath = arg; | ||||||
|  | 			} else { | ||||||
|  | 				logError(kErrorSingleInputPathExpected, "Command Line") << "only one input path expected"; | ||||||
|  | 				return Options(); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if (result.inputPath.isEmpty()) { | ||||||
|  | 		logError(kErrorInputPathExpected, "Command Line") << "input path expected"; | ||||||
|  | 		return Options(); | ||||||
|  | 	} | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,38 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include <QtCore/QStringList> | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | 
 | ||||||
|  | struct Options { | ||||||
|  | 	QString outputPath = "."; | ||||||
|  | 	QString inputPath; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Parsing failed if inputPath is empty in the result.
 | ||||||
|  | Options parseOptions(); | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,66 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/numbers/parsed_file.h" | ||||||
|  | 
 | ||||||
|  | #include <iostream> | ||||||
|  | #include <QtCore/QMap> | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QRegularExpression> | ||||||
|  | #include "codegen/common/basic_tokenized_file.h" | ||||||
|  | #include "codegen/common/logging.h" | ||||||
|  | 
 | ||||||
|  | using BasicToken = codegen::common::BasicTokenizedFile::Token; | ||||||
|  | using BasicType = BasicToken::Type; | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | ParsedFile::ParsedFile(const Options &options) | ||||||
|  | : file_(options.inputPath) | ||||||
|  | , options_(options) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool ParsedFile::read() { | ||||||
|  | 	if (!file_.read()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	auto filepath = QFileInfo(options_.inputPath).absoluteFilePath(); | ||||||
|  | 	do { | ||||||
|  | 		if (auto startToken = file_.getToken(BasicType::Name)) { | ||||||
|  | 		} | ||||||
|  | 		if (file_.atEnd()) { | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 		logErrorUnexpectedToken() << "numbers rule"; | ||||||
|  | 	} while (!failed()); | ||||||
|  | 
 | ||||||
|  | 	if (failed()) { | ||||||
|  | 		result_.data.clear(); | ||||||
|  | 	} | ||||||
|  | 	return !failed(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,74 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <memory> | ||||||
|  | #include <string> | ||||||
|  | #include "codegen/common/basic_tokenized_file.h" | ||||||
|  | #include "codegen/numbers/options.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | 
 | ||||||
|  | struct Rule { | ||||||
|  | }; | ||||||
|  | struct Rules { | ||||||
|  | 	QVector<Rule> data; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Parses an input file to the internal struct.
 | ||||||
|  | class ParsedFile { | ||||||
|  | public: | ||||||
|  | 	explicit ParsedFile(const Options &options); | ||||||
|  | 	ParsedFile(const ParsedFile &other) = delete; | ||||||
|  | 	ParsedFile &operator=(const ParsedFile &other) = delete; | ||||||
|  | 
 | ||||||
|  | 	bool read(); | ||||||
|  | 
 | ||||||
|  | 	Rules getResult() { | ||||||
|  | 		return result_; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 
 | ||||||
|  | 	bool failed() const { | ||||||
|  | 		return failed_ || file_.failed(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Log error to std::cerr with 'code' at the current position in file.
 | ||||||
|  | 	common::LogStream logError(int code) { | ||||||
|  | 		failed_ = true; | ||||||
|  | 		return file_.logError(code); | ||||||
|  | 	} | ||||||
|  | 	common::LogStream logErrorUnexpectedToken() { | ||||||
|  | 		failed_ = true; | ||||||
|  | 		return file_.logErrorUnexpectedToken(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	common::BasicTokenizedFile file_; | ||||||
|  | 	Options options_; | ||||||
|  | 	bool failed_ = false; | ||||||
|  | 	Rules result_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,86 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/numbers/processor.h" | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QFileInfo> | ||||||
|  | #include "codegen/common/cpp_file.h" | ||||||
|  | #include "codegen/numbers/parsed_file.h" | ||||||
|  | #include "codegen/numbers/generator.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | constexpr int kErrorCantWritePath = 851; | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | Processor::Processor(const Options &options) | ||||||
|  | : parser_(std::make_unique<ParsedFile>(options)) | ||||||
|  | , options_(options) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int Processor::launch() { | ||||||
|  | 	if (!parser_->read()) { | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	auto result = parser_->getResult(); | ||||||
|  | 	if (!write(result)) { | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Processor::write(const Rules &rules) const { | ||||||
|  | 	QDir dir(options_.outputPath); | ||||||
|  | 	if (!dir.mkpath(".")) { | ||||||
|  | 		common::logError(kErrorCantWritePath, "Command Line") << "can not open path for writing: " << dir.absolutePath().toStdString(); | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	QFileInfo srcFile(options_.inputPath); | ||||||
|  | 	QString dstFilePath = dir.absolutePath() + "/numbers"; | ||||||
|  | 
 | ||||||
|  | 	common::ProjectInfo project = { | ||||||
|  | 		"codegen_style", | ||||||
|  | 		srcFile.fileName(), | ||||||
|  | 		"stdafx.h", | ||||||
|  | 		false, // forceReGenerate
 | ||||||
|  | 	}; | ||||||
|  | 
 | ||||||
|  | 	Generator generator(rules, dstFilePath, project); | ||||||
|  | 	if (!generator.writeHeader()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (!generator.writeSource()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Processor::~Processor() = default; | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,53 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <memory> | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include "codegen/numbers/options.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace numbers { | ||||||
|  | class ParsedFile; | ||||||
|  | struct Rules; | ||||||
|  | 
 | ||||||
|  | // Walks through a file, parses it and generates number formatter.
 | ||||||
|  | class Processor { | ||||||
|  | public: | ||||||
|  | 	explicit Processor(const Options &options); | ||||||
|  | 	Processor(const Processor &other) = delete; | ||||||
|  | 	Processor &operator=(const Processor &other) = delete; | ||||||
|  | 
 | ||||||
|  | 	// Returns 0 on success.
 | ||||||
|  | 	int launch(); | ||||||
|  | 
 | ||||||
|  | 	~Processor(); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 	bool write(const Rules &rules) const; | ||||||
|  | 
 | ||||||
|  | 	std::unique_ptr<ParsedFile> parser_; | ||||||
|  | 	const Options &options_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace numbers
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -20,45 +20,533 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| #include "codegen/style/generator.h" | #include "codegen/style/generator.h" | ||||||
| 
 | 
 | ||||||
| #include <QtGui/QImage> | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QSet> | ||||||
|  | #include <functional> | ||||||
| #include "codegen/style/parsed_file.h" | #include "codegen/style/parsed_file.h" | ||||||
| 
 | 
 | ||||||
|  | using Module = codegen::style::structure::Module; | ||||||
|  | using Struct = codegen::style::structure::Struct; | ||||||
|  | using Variable = codegen::style::structure::Variable; | ||||||
|  | using Tag = codegen::style::structure::TypeTag; | ||||||
|  | 
 | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
| namespace { | namespace { | ||||||
| 
 | 
 | ||||||
| } // namespace
 | char hexChar(uchar ch) { | ||||||
| 
 | 	if (ch < 10) { | ||||||
| Generator::Generator(const Options &options) | 		return '0' + ch; | ||||||
| : parser_(std::make_unique<ParsedFile>(options)) | 	} else if (ch < 16) { | ||||||
| , options_(options) { | 		return 'a' + (ch - 10); | ||||||
| 
 | 	} | ||||||
|  | 	return '0'; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int Generator::process() { | char hexSecondChar(char ch) { | ||||||
| 	if (!parser_->read()) { | 	return hexChar((*reinterpret_cast<uchar*>(&ch)) << 4); | ||||||
| 		return -1; | } | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	const auto &result = parser_->data(); | char hexFirstChar(char ch) { | ||||||
| 	if (!write(result)) { | 	return hexChar((*reinterpret_cast<uchar*>(&ch)) & 0x0F); | ||||||
| 		return -1; | } | ||||||
| 	} | 
 | ||||||
| 	if (options_.rebuildDependencies) { | QString stringToEncodedString(const std::string &str) { | ||||||
| 		for (auto included : result.includes) { | 	QString result; | ||||||
| 			if (!write(included)) { | 	result.reserve(str.size() * 4); | ||||||
| 				return -1; | 	for (auto ch : str) { | ||||||
| 			} | 		if (ch == '\n') { | ||||||
|  | 			result.append("\\n"); | ||||||
|  | 		} else if (ch == '\t') { | ||||||
|  | 			result.append("\\t"); | ||||||
|  | 		} else if (ch == '"' || ch == '\\') { | ||||||
|  | 			result.append('\\').append(ch); | ||||||
|  | 		} else if (ch < 32 || ch > 127) { | ||||||
|  | 			result.append("\\x").append(hexFirstChar(ch)).append(hexSecondChar(ch)); | ||||||
|  | 		} else { | ||||||
|  | 			result.append(ch); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return 0; | 	return '"' + result + '"'; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Generator::write(const structure::Module &) const { | QString pxValueName(int value) { | ||||||
|  | 	QString result = "px"; | ||||||
|  | 	if (value < 0) { | ||||||
|  | 		value = -value; | ||||||
|  | 		result += 'm'; | ||||||
|  | 	} | ||||||
|  | 	return result + QString::number(value); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | Generator::Generator(const structure::Module &module, const QString &destBasePath, const common::ProjectInfo &project) | ||||||
|  | : module_(module) | ||||||
|  | , basePath_(destBasePath) | ||||||
|  | , baseName_(QFileInfo(basePath_).baseName()) | ||||||
|  | , project_(project) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeHeader() { | ||||||
|  | 	header_ = std::make_unique<common::CppFile>(basePath_ + ".h", project_); | ||||||
|  | 
 | ||||||
|  | 	header_->include("ui/style_core.h").newline(); | ||||||
|  | 
 | ||||||
|  | 	if (!writeHeaderStyleNamespace()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (!writeRefsDeclarations()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return header_->finalize(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeSource() { | ||||||
|  | 	source_ = std::make_unique<common::CppFile>(basePath_ + ".cpp", project_); | ||||||
|  | 
 | ||||||
|  | 	writeIncludesInSource(); | ||||||
|  | 
 | ||||||
|  | 	if (module_.hasVariables()) { | ||||||
|  | 		source_->pushNamespace().newline(); | ||||||
|  | 		source_->stream() << "\
 | ||||||
|  | bool inited = false;\n\ | ||||||
|  | \n\ | ||||||
|  | class Module_" << baseName_ << " : public style::internal::ModuleBase {\n\ | ||||||
|  | public:\n\ | ||||||
|  | 	Module_" << baseName_ << "() { style::internal::registerModule(this); }\n\ | ||||||
|  | 	~Module_" << baseName_ << "() { style::internal::unregisterModule(this); }\n\ | ||||||
|  | \n\ | ||||||
|  | 	void start() override {\n\ | ||||||
|  | 		style::internal::init_" << baseName_ << "();\n\ | ||||||
|  | 	}\n\ | ||||||
|  | 	void stop() override {\n\ | ||||||
|  | 	}\n\ | ||||||
|  | };\n\ | ||||||
|  | Module_" << baseName_ << " registrator;\n"; | ||||||
|  | 		if (!writeVariableDefinitions()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		source_->newline().popNamespace(); | ||||||
|  | 
 | ||||||
|  | 		source_->newline().pushNamespace("st"); | ||||||
|  | 		if (!writeRefsDefinition()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		source_->popNamespace().newline(); | ||||||
|  | 		source_->newline().pushNamespace("style").pushNamespace("internal").newline(); | ||||||
|  | 		if (!writeVariableInit()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return source_->finalize(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Empty result means an error.
 | ||||||
|  | QString Generator::typeToString(structure::Type type) const { | ||||||
|  | 	switch (type.tag) { | ||||||
|  | 	case Tag::Invalid: return QString(); | ||||||
|  | 	case Tag::Int: return "int"; | ||||||
|  | 	case Tag::Double: return "double"; | ||||||
|  | 	case Tag::Pixels: return "int"; | ||||||
|  | 	case Tag::String: return "QString"; | ||||||
|  | 	case Tag::Color: return "style::color"; | ||||||
|  | 	case Tag::Point: return "style::point"; | ||||||
|  | 	case Tag::Sprite: return "style::sprite"; | ||||||
|  | 	case Tag::Size: return "style::size"; | ||||||
|  | 	case Tag::Transition: return "style::transition"; | ||||||
|  | 	case Tag::Cursor: return "style::cursor"; | ||||||
|  | 	case Tag::Align: return "style::align"; | ||||||
|  | 	case Tag::Margins: return "style::margins"; | ||||||
|  | 	case Tag::Font: return "style::font"; | ||||||
|  | 	case Tag::Struct: return "style::" + type.name.back(); | ||||||
|  | 	} | ||||||
|  | 	return QString(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Empty result means an error.
 | ||||||
|  | QString Generator::typeToDefaultValue(structure::Type type) const { | ||||||
|  | 	switch (type.tag) { | ||||||
|  | 	case Tag::Invalid: return QString(); | ||||||
|  | 	case Tag::Int: return "0"; | ||||||
|  | 	case Tag::Double: return "0."; | ||||||
|  | 	case Tag::Pixels: return "0"; | ||||||
|  | 	case Tag::String: return "QString()"; | ||||||
|  | 	case Tag::Color: return "{ Qt::Uninitialized }"; | ||||||
|  | 	case Tag::Point: return "{ 0, 0 }"; | ||||||
|  | 	case Tag::Sprite: return "{ 0, 0, 0, 0 }"; | ||||||
|  | 	case Tag::Size: return "{ 0, 0 }"; | ||||||
|  | 	case Tag::Transition: return "anim::linear"; | ||||||
|  | 	case Tag::Cursor: return "style::cur_default"; | ||||||
|  | 	case Tag::Align: return "style::al_topleft"; | ||||||
|  | 	case Tag::Margins: return "{ 0, 0, 0, 0 }"; | ||||||
|  | 	case Tag::Font: return "{ Qt::Uninitialized }"; | ||||||
|  | 	case Tag::Struct: { | ||||||
|  | 		if (auto realType = module_.findStruct(type.name)) { | ||||||
|  | 			QStringList fields; | ||||||
|  | 			for (auto field : realType->fields) { | ||||||
|  | 				fields.push_back(typeToDefaultValue(field.type)); | ||||||
|  | 			} | ||||||
|  | 			return "{ " + fields.join(", ") + " }"; | ||||||
|  | 		} | ||||||
|  | 		return QString(); | ||||||
|  | 	} break; | ||||||
|  | 	} | ||||||
|  | 	return QString(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Empty result means an error.
 | ||||||
|  | QString Generator::valueAssignmentCode(structure::Value value) const { | ||||||
|  | 	auto copy = value.copyOf(); | ||||||
|  | 	if (!copy.isEmpty()) { | ||||||
|  | 		return "st::" + copy.back(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	switch (value.type().tag) { | ||||||
|  | 	case Tag::Invalid: return QString(); | ||||||
|  | 	case Tag::Int: return QString("%1").arg(value.Int()); | ||||||
|  | 	case Tag::Double: return QString("%1").arg(value.Double()); | ||||||
|  | 	case Tag::Pixels: return pxValueName(value.Int()); | ||||||
|  | 	case Tag::String: return QString("qsl(%1)").arg(stringToEncodedString(value.String())); | ||||||
|  | 	case Tag::Color: return QString("{ %1, %2, %3, %4 }").arg(value.Color().red).arg(value.Color().green).arg(value.Color().blue).arg(value.Color().alpha); | ||||||
|  | 	case Tag::Point: { | ||||||
|  | 		auto v(value.Point()); | ||||||
|  | 		return QString("{ %1, %2 }").arg(pxValueName(v.x)).arg(pxValueName(v.y)); | ||||||
|  | 	} | ||||||
|  | 	case Tag::Sprite: { | ||||||
|  | 		auto v(value.Sprite()); | ||||||
|  | 		return QString("{ %1, %2, %3, %4 }").arg(pxValueName(v.left)).arg(pxValueName(v.top)).arg(pxValueName(v.width)).arg(pxValueName(v.height)); | ||||||
|  | 	} | ||||||
|  | 	case Tag::Size: { | ||||||
|  | 		auto v(value.Size()); | ||||||
|  | 		return QString("{ %1, %2 }").arg(pxValueName(v.width)).arg(pxValueName(v.height)); | ||||||
|  | 	} | ||||||
|  | 	case Tag::Transition: return QString("anim::%1").arg(value.String().c_str()); | ||||||
|  | 	case Tag::Cursor: return QString("style::cur_%1").arg(value.String().c_str()); | ||||||
|  | 	case Tag::Align: return QString("style::al_%1").arg(value.String().c_str()); | ||||||
|  | 	case Tag::Margins: { | ||||||
|  | 		auto v(value.Margins()); | ||||||
|  | 		return QString("{ %1, %2, %3, %4 }").arg(pxValueName(v.left)).arg(pxValueName(v.top)).arg(pxValueName(v.right)).arg(pxValueName(v.bottom)); | ||||||
|  | 	} | ||||||
|  | 	case Tag::Font: { | ||||||
|  | 		auto v(value.Font()); | ||||||
|  | 		QString family = "0"; | ||||||
|  | 		if (!v.family.empty()) { | ||||||
|  | 			auto familyIndex = fontFamilyValues_.value(v.family, -1); | ||||||
|  | 			if (familyIndex < 0) { | ||||||
|  | 				return QString(); | ||||||
|  | 			} else { | ||||||
|  | 				family = QString("font%1index").arg(familyIndex); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return QString("{ %1, %2, %3 }").arg(pxValueName(v.size)).arg(v.flags).arg(family); | ||||||
|  | 	} | ||||||
|  | 	case Tag::Struct: { | ||||||
|  | 		if (!value.Fields()) return QString(); | ||||||
|  | 
 | ||||||
|  | 		QStringList fields; | ||||||
|  | 		for (auto field : *value.Fields()) { | ||||||
|  | 			fields.push_back(valueAssignmentCode(field.variable.value)); | ||||||
|  | 		} | ||||||
|  | 		return "{ " + fields.join(", ") + " }"; | ||||||
|  | 	} break; | ||||||
|  | 	} | ||||||
|  | 	return QString(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeHeaderStyleNamespace() { | ||||||
|  | 	if (!module_.hasStructs() && !module_.hasVariables()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 	header_->pushNamespace("style"); | ||||||
|  | 
 | ||||||
|  | 	if (module_.hasVariables()) { | ||||||
|  | 		header_->pushNamespace("internal").newline(); | ||||||
|  | 		header_->stream() << "void init_" << baseName_ << "();\n\n"; | ||||||
|  | 		header_->popNamespace(); | ||||||
|  | 	} | ||||||
|  | 	if (module_.hasStructs()) { | ||||||
|  | 		header_->newline(); | ||||||
|  | 		if (!writeStructsDefinitions()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	header_->popNamespace().newline(); | ||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Generator::~Generator() = default; | bool Generator::writeStructsDefinitions() { | ||||||
|  | 	if (!module_.hasStructs()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	bool result = module_.enumStructs([this](const Struct &value) -> bool { | ||||||
|  | 		header_->stream() << "struct " << value.name.back() << " {\n"; | ||||||
|  | 		for (const auto &field : value.fields) { | ||||||
|  | 			auto type = typeToString(field.type); | ||||||
|  | 			if (type.isEmpty()) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 			header_->stream() << "\t" << type << " " << field.name.back() << ";\n"; | ||||||
|  | 		} | ||||||
|  | 		header_->stream() << "};\n\n"; | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeRefsDeclarations() { | ||||||
|  | 	if (!module_.hasVariables()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	header_->pushNamespace("st"); | ||||||
|  | 
 | ||||||
|  | 	bool result = module_.enumVariables([this](const Variable &value) -> bool { | ||||||
|  | 		auto name = value.name.back(); | ||||||
|  | 		auto type = typeToString(value.value.type()); | ||||||
|  | 		if (type.isEmpty()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		header_->stream() << "extern const " << type << " &" << name << ";\n"; | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 
 | ||||||
|  | 	header_->popNamespace(); | ||||||
|  | 
 | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeIncludesInSource() { | ||||||
|  | 	if (!module_.hasIncludes()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	bool result = module_.enumIncludes([this](const Module &module) -> bool { | ||||||
|  | 		source_->include("style_" + QFileInfo(module.filepath()).baseName() + ".h"); | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 	source_->newline(); | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeVariableDefinitions() { | ||||||
|  | 	if (!module_.hasVariables()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	source_->newline(); | ||||||
|  | 	bool result = module_.enumVariables([this](const Variable &variable) -> bool { | ||||||
|  | 		auto name = variable.name.back(); | ||||||
|  | 		auto type = typeToString(variable.value.type()); | ||||||
|  | 		if (type.isEmpty()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		source_->stream() << type << " _" << name << " = " << typeToDefaultValue(variable.value.type()) << ";\n"; | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeRefsDefinition() { | ||||||
|  | 	if (!module_.hasVariables()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	source_->newline(); | ||||||
|  | 	bool result = module_.enumVariables([this](const Variable &variable) -> bool { | ||||||
|  | 		auto name = variable.name.back(); | ||||||
|  | 		auto type = typeToString(variable.value.type()); | ||||||
|  | 		if (type.isEmpty()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		source_->stream() << "const " << type << " &" << name << "(_" << name << ");\n"; | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeVariableInit() { | ||||||
|  | 	if (!module_.hasVariables()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!collectUniqueValues()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	bool hasUniqueValues = (!pxValues_.isEmpty() || !fontFamilyValues_.isEmpty()); | ||||||
|  | 	if (hasUniqueValues) { | ||||||
|  | 		source_->pushNamespace(); | ||||||
|  | 		if (!writePxValues()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		if (!writeFontFamilyValues()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		source_->popNamespace().newline(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	source_->stream() << "\
 | ||||||
|  | void init_" << baseName_ << "() {\n\ | ||||||
|  | 	if (inited) return;\n\ | ||||||
|  | 	inited = true;\n\n"; | ||||||
|  | 
 | ||||||
|  | 	if (module_.hasIncludes()) { | ||||||
|  | 		bool writtenAtLeastOne = false; | ||||||
|  | 		bool result = module_.enumIncludes([this,&writtenAtLeastOne](const Module &module) -> bool { | ||||||
|  | 			if (module.hasVariables()) { | ||||||
|  | 				source_->stream() << "\tinit_style_" + QFileInfo(module.filepath()).baseName() + "();\n"; | ||||||
|  | 				writtenAtLeastOne = true; | ||||||
|  | 			} | ||||||
|  | 			return true; | ||||||
|  | 		}); | ||||||
|  | 		if (!result) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		if (writtenAtLeastOne) { | ||||||
|  | 			source_->newline(); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (hasUniqueValues) { | ||||||
|  | 		if (!pxValues_.isEmpty()) { | ||||||
|  | 			source_->stream() << "\tinitPxValues();\n"; | ||||||
|  | 		} | ||||||
|  | 		if (!fontFamilyValues_.isEmpty()) { | ||||||
|  | 			source_->stream() << "\tinitFontFamilyValues();\n"; | ||||||
|  | 		} | ||||||
|  | 		source_->newline(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	bool result = module_.enumVariables([this](const Variable &variable) -> bool { | ||||||
|  | 		auto name = variable.name.back(); | ||||||
|  | 		auto value = valueAssignmentCode(variable.value); | ||||||
|  | 		if (value.isEmpty()) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		source_->stream() << "\t_" << name << " = " << value << ";\n"; | ||||||
|  | 		return true; | ||||||
|  | 	}); | ||||||
|  | 	source_->stream() << "\
 | ||||||
|  | }\n\n"; | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writePxValues() { | ||||||
|  | 	if (pxValues_.isEmpty()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for (auto i = pxValues_.cbegin(), e = pxValues_.cend(); i != e; ++i) { | ||||||
|  | 		source_->stream() << "int " << pxValueName(i.key()) << " = " << i.key() << ";\n"; | ||||||
|  | 	} | ||||||
|  | 	source_->stream() << "\
 | ||||||
|  | void initPxValues() {\n\ | ||||||
|  | 	if (cRetina()) return;\n\ | ||||||
|  | \n\ | ||||||
|  | 	switch (cScale()) {\n"; | ||||||
|  | 	for (int i = 1, scalesCount = scales.size(); i < scalesCount; ++i) { | ||||||
|  | 		source_->stream() << "\tcase " << scaleNames.at(i) << ":\n"; | ||||||
|  | 		for (auto it = pxValues_.cbegin(), e = pxValues_.cend(); it != e; ++it) { | ||||||
|  | 			auto value = it.key(); | ||||||
|  | 			int adjusted = structure::data::pxAdjust(value, scales.at(i)); | ||||||
|  | 			if (adjusted != value) { | ||||||
|  | 				source_->stream() << "\t\t" << pxValueName(value) << " = " << adjusted << ";\n"; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		source_->stream() << "\tbreak;\n"; | ||||||
|  | 	} | ||||||
|  | 	source_->stream() << "\
 | ||||||
|  | 	}\n\ | ||||||
|  | }\n\n"; | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::writeFontFamilyValues() { | ||||||
|  | 	if (fontFamilyValues_.isEmpty()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for (auto i = fontFamilyValues_.cbegin(), e = fontFamilyValues_.cend(); i != e; ++i) { | ||||||
|  | 		source_->stream() << "int font" << i.value() << "index;\n"; | ||||||
|  | 	} | ||||||
|  | 	source_->stream() << "void initFontFamilyValues() {\n"; | ||||||
|  | 	for (auto i = fontFamilyValues_.cbegin(), e = fontFamilyValues_.cend(); i != e; ++i) { | ||||||
|  | 		auto family = stringToEncodedString(i.key()); | ||||||
|  | 		source_->stream() << "\tfont" << i.value() << "index = style::internal::registerFontFamily(" << family << ");\n"; | ||||||
|  | 	} | ||||||
|  | 	source_->stream() << "}\n\n"; | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Generator::collectUniqueValues() { | ||||||
|  | 	int fontFamilyIndex = 0; | ||||||
|  | 	std::function<bool(const Variable&)> collector = [this, &collector, &fontFamilyIndex](const Variable &variable) { | ||||||
|  | 		auto value = variable.value; | ||||||
|  | 		switch (value.type().tag) { | ||||||
|  | 		case Tag::Invalid: | ||||||
|  | 		case Tag::Int: | ||||||
|  | 		case Tag::Double: | ||||||
|  | 		case Tag::String: | ||||||
|  | 		case Tag::Color: | ||||||
|  | 		case Tag::Transition: | ||||||
|  | 		case Tag::Cursor: | ||||||
|  | 		case Tag::Align: break; | ||||||
|  | 		case Tag::Pixels: pxValues_.insert(value.Int(), true); break; | ||||||
|  | 		case Tag::Point: { | ||||||
|  | 			auto v(value.Point()); | ||||||
|  | 			pxValues_.insert(v.x, true); | ||||||
|  | 			pxValues_.insert(v.y, true); | ||||||
|  | 		} break; | ||||||
|  | 		case Tag::Sprite: { | ||||||
|  | 			auto v(value.Sprite()); | ||||||
|  | 			pxValues_.insert(v.left, true); | ||||||
|  | 			pxValues_.insert(v.top, true); | ||||||
|  | 			pxValues_.insert(v.width, true); | ||||||
|  | 			pxValues_.insert(v.height, true); | ||||||
|  | 		} break; | ||||||
|  | 		case Tag::Size: { | ||||||
|  | 			auto v(value.Size()); | ||||||
|  | 			pxValues_.insert(v.width, true); | ||||||
|  | 			pxValues_.insert(v.height, true); | ||||||
|  | 		} break; | ||||||
|  | 		case Tag::Margins: { | ||||||
|  | 			auto v(value.Margins()); | ||||||
|  | 			pxValues_.insert(v.left, true); | ||||||
|  | 			pxValues_.insert(v.top, true); | ||||||
|  | 			pxValues_.insert(v.right, true); | ||||||
|  | 			pxValues_.insert(v.bottom, true); | ||||||
|  | 		} break; | ||||||
|  | 		case Tag::Font: { | ||||||
|  | 			auto v(value.Font()); | ||||||
|  | 			pxValues_.insert(v.size, true); | ||||||
|  | 			if (!v.family.empty() && !fontFamilyValues_.contains(v.family)) { | ||||||
|  | 				fontFamilyValues_.insert(v.family, ++fontFamilyIndex); | ||||||
|  | 			} | ||||||
|  | 		} break; | ||||||
|  | 		case Tag::Struct: { | ||||||
|  | 			auto fields = variable.value.Fields(); | ||||||
|  | 			if (!fields) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			for (auto field : *fields) { | ||||||
|  | 				if (!collector(field.variable)) { | ||||||
|  | 					return false; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} break; | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	}; | ||||||
|  | 	return module_.enumVariables(collector); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| } // namespace style
 | } // namespace style
 | ||||||
| } // namespace codegen
 | } // namespace codegen
 | ||||||
|  |  | ||||||
|  | @ -22,36 +22,53 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| 
 | 
 | ||||||
| #include <memory> | #include <memory> | ||||||
| #include <QtCore/QString> | #include <QtCore/QString> | ||||||
| #include "codegen/style/options.h" | #include <QtCore/QSet> | ||||||
|  | #include "codegen/common/cpp_file.h" | ||||||
|  | #include "codegen/style/structure_types.h" | ||||||
| 
 | 
 | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
| namespace structure { | namespace structure { | ||||||
| struct Module; | class Module; | ||||||
| } // namespace structure
 | } // namespace structure
 | ||||||
| class ParsedFile; |  | ||||||
| 
 | 
 | ||||||
| // Walks through a file, parses it and parses dependency files if necessary.
 |  | ||||||
| class Generator { | class Generator { | ||||||
| public: | public: | ||||||
| 	Generator(const Options &options); | 	Generator(const structure::Module &module, const QString &destBasePath, const common::ProjectInfo &project); | ||||||
| 	Generator(const Generator &other) = delete; | 	Generator(const Generator &other) = delete; | ||||||
| 	Generator &operator=(const Generator &other) = delete; | 	Generator &operator=(const Generator &other) = delete; | ||||||
| 
 | 
 | ||||||
| 	// Returns 0 on success.
 | 	bool writeHeader(); | ||||||
| 	int process(); | 	bool writeSource(); | ||||||
| 
 |  | ||||||
| 	~Generator(); |  | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	bool write(const structure::Module &module) const; | 	QString typeToString(structure::Type type) const; | ||||||
|  | 	QString typeToDefaultValue(structure::Type type) const; | ||||||
|  | 	QString valueAssignmentCode(structure::Value value) const; | ||||||
| 
 | 
 | ||||||
| 	std::unique_ptr<ParsedFile> parser_; | 	bool writeHeaderStyleNamespace(); | ||||||
| 	const Options &options_; | 	bool writeStructsDefinitions(); | ||||||
|  | 	bool writeRefsDeclarations(); | ||||||
| 
 | 
 | ||||||
| 	// List of files we need to generate with other instance of Generator.
 | 	bool writeIncludesInSource(); | ||||||
| 	// It is not empty only if rebuild_ flag is true.
 | 	bool writeVariableDefinitions(); | ||||||
| 	QStringList dependenciesToGenerate_; | 	bool writeRefsDefinition(); | ||||||
|  | 	bool writeVariableInit(); | ||||||
|  | 	bool writePxValues(); | ||||||
|  | 	bool writeFontFamilyValues(); | ||||||
|  | 
 | ||||||
|  | 	bool collectUniqueValues(); | ||||||
|  | 
 | ||||||
|  | 	const structure::Module &module_; | ||||||
|  | 	QString basePath_, baseName_; | ||||||
|  | 	const common::ProjectInfo &project_; | ||||||
|  | 	std::unique_ptr<common::CppFile> source_, header_; | ||||||
|  | 
 | ||||||
|  | 	QMap<int, bool> pxValues_; | ||||||
|  | 	QMap<std::string, int> fontFamilyValues_; | ||||||
|  | 
 | ||||||
|  | 	std::vector<int> scales = { 4, 5, 6, 8 }; // scale / 4 gives our 1.00, 1.25, 1.50, 2.00
 | ||||||
|  | 	std::vector<const char *>scaleNames = { "dbisOne", "dbisOneAndQuarter", "dbisOneAndHalf", "dbisTwo" }; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -19,21 +19,18 @@ Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE | ||||||
| Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
| */ | */ | ||||||
| #include <QtCore/QCoreApplication> | #include <QtCore/QCoreApplication> | ||||||
| #include <QtCore/QTimer> |  | ||||||
| 
 | 
 | ||||||
| #include "codegen/style/generator.h" |  | ||||||
| #include "codegen/style/options.h" | #include "codegen/style/options.h" | ||||||
| 
 | #include "codegen/style/processor.h" | ||||||
| using namespace codegen::style; |  | ||||||
| 
 | 
 | ||||||
| int main(int argc, char *argv[]) { | int main(int argc, char *argv[]) { | ||||||
| 	QCoreApplication app(argc, argv); | 	QCoreApplication app(argc, argv); | ||||||
| 
 | 
 | ||||||
| 	Options options = parseOptions(); | 	auto options = codegen::style::parseOptions(); | ||||||
| 	if (options.inputPath.isEmpty()) { | 	if (options.inputPath.isEmpty()) { | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	Generator generator(options); | 	codegen::style::Processor processor(options); | ||||||
| 	return generator.process(); | 	return processor.launch(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,101 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/style/module.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace style { | ||||||
|  | namespace structure { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | QString fullNameKey(const FullName &name) { | ||||||
|  | 	return name.join('.'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | Module::Module(const QString &fullpath) : fullpath_(fullpath) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void Module::addIncluded(std::unique_ptr<Module> &&value) { | ||||||
|  | 	included_.push_back(std::move(value)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Module::addStruct(const Struct &value) { | ||||||
|  | 	if (findStruct(value.name)) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	structsByName_.insert(fullNameKey(value.name), structs_.size()); | ||||||
|  | 	structs_.push_back(value); | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const Struct *Module::findStruct(const FullName &name) const { | ||||||
|  | 	if (auto result = findStructInModule(name, *this)) { | ||||||
|  | 		return result; | ||||||
|  | 	} | ||||||
|  | 	for (const auto &module : included_) { | ||||||
|  | 		if (auto result = findStructInModule(name, *module)) { | ||||||
|  | 			return result; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nullptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Module::addVariable(const Variable &value) { | ||||||
|  | 	if (findVariable(value.name)) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	variablesByName_.insert(fullNameKey(value.name), variables_.size()); | ||||||
|  | 	variables_.push_back(value); | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const Variable *Module::findVariable(const FullName &name) const { | ||||||
|  | 	if (auto result = findVariableInModule(name, *this)) { | ||||||
|  | 		return result; | ||||||
|  | 	} | ||||||
|  | 	for (const auto &module : included_) { | ||||||
|  | 		if (auto result = findVariableInModule(name, *module)) { | ||||||
|  | 			return result; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nullptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const Struct *Module::findStructInModule(const FullName &name, const Module &module) const { | ||||||
|  | 	auto index = module.structsByName_.value(fullNameKey(name), -1); | ||||||
|  | 	if (index < 0) { | ||||||
|  | 		return nullptr; | ||||||
|  | 	} | ||||||
|  | 	return &module.structs_.at(index); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const Variable *Module::findVariableInModule(const FullName &name, const Module &module) const { | ||||||
|  | 	auto index = module.variablesByName_.value(fullNameKey(name), -1); | ||||||
|  | 	if (index < 0) { | ||||||
|  | 		return nullptr; | ||||||
|  | 	} | ||||||
|  | 	return &module.variables_.at(index); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace structure
 | ||||||
|  | } // namespace style
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,112 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include <QtCore/QList> | ||||||
|  | #include <QtCore/QMap> | ||||||
|  | #include <vector> | ||||||
|  | #include "codegen/style/structure_types.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace style { | ||||||
|  | namespace structure { | ||||||
|  | 
 | ||||||
|  | class Module { | ||||||
|  | public: | ||||||
|  | 
 | ||||||
|  | 	explicit Module(const QString &fullpath); | ||||||
|  | 
 | ||||||
|  | 	QString filepath() const { | ||||||
|  | 		return fullpath_; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	void addIncluded(std::unique_ptr<Module> &&value); | ||||||
|  | 
 | ||||||
|  | 	bool hasIncludes() const { | ||||||
|  | 		return !included_.empty(); | ||||||
|  | 	} | ||||||
|  | 	template <typename F> | ||||||
|  | 	bool enumIncludes(F functor) const { | ||||||
|  | 		for (const auto &module : included_) { | ||||||
|  | 			if (!functor(*module)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Returns false if there is a struct with such name already.
 | ||||||
|  | 	bool addStruct(const Struct &value); | ||||||
|  | 	// Returns nullptr if there is no such struct in result_ or any of included modules.
 | ||||||
|  | 	const Struct *findStruct(const FullName &name) const; | ||||||
|  | 	bool hasStructs() const { | ||||||
|  | 		return !structs_.isEmpty(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	template <typename F> | ||||||
|  | 	bool enumStructs(F functor) const { | ||||||
|  | 		for (const auto &value : structs_) { | ||||||
|  | 			if (!functor(value)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Returns false if there is a variable with such name already.
 | ||||||
|  | 	bool addVariable(const Variable &value); | ||||||
|  | 	// Returns nullptr if there is no such variable in result_ or any of included modules.
 | ||||||
|  | 	const Variable *findVariable(const FullName &name) const; | ||||||
|  | 	bool hasVariables() const { | ||||||
|  | 		return !variables_.isEmpty(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	template <typename F> | ||||||
|  | 	bool enumVariables(F functor) const { | ||||||
|  | 		for (const auto &value : variables_) { | ||||||
|  | 			if (!functor(value)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	explicit operator bool() const { | ||||||
|  | 		return !fullpath_.isEmpty(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 	QString fullpath_; | ||||||
|  | 	std::vector<std::unique_ptr<Module>> included_; | ||||||
|  | 	QList<Struct> structs_; | ||||||
|  | 	QList<Variable> variables_; | ||||||
|  | 	QMap<QString, int> structsByName_; | ||||||
|  | 	QMap<QString, int> variablesByName_; | ||||||
|  | 
 | ||||||
|  | 	const Struct *findStructInModule(const FullName &name, const Module &module) const; | ||||||
|  | 	const Variable *findVariableInModule(const FullName &name, const Module &module) const; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace structure
 | ||||||
|  | } // namespace style
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -32,6 +32,9 @@ using BasicType = BasicToken::Type; | ||||||
| 
 | 
 | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
|  | 
 | ||||||
|  | using structure::logFullName; | ||||||
|  | 
 | ||||||
| namespace { | namespace { | ||||||
| 
 | 
 | ||||||
| constexpr int kErrorInIncluded         = 801; | constexpr int kErrorInIncluded         = 801; | ||||||
|  | @ -65,7 +68,7 @@ bool isValidColor(const QString &str) { | ||||||
| 
 | 
 | ||||||
| uchar readHexUchar(QChar ch) { | uchar readHexUchar(QChar ch) { | ||||||
| 	auto code = ch.unicode(); | 	auto code = ch.unicode(); | ||||||
| 	return (code >= '0' && code <= '9') ? ((code - '0') & 0xFF) : ((code - 'a') & 0xFF); | 	return (code >= '0' && code <= '9') ? ((code - '0') & 0xFF) : ((code + 10 - 'a') & 0xFF); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| uchar readHexUchar(QChar char1, QChar char2) { | uchar readHexUchar(QChar char1, QChar char2) { | ||||||
|  | @ -95,10 +98,6 @@ structure::data::color convertIntColor(int r, int g, int b, int a) { | ||||||
| 	return { uchar(r & 0xFF), uchar(g & 0xFF), uchar(b & 0xFF), uchar(a & 0xFF) }; | 	return { uchar(r & 0xFF), uchar(g & 0xFF), uchar(b & 0xFF), uchar(a & 0xFF) }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::string logFullName(const structure::FullName &name) { |  | ||||||
| 	return name.join('.').toStdString(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::string logType(const structure::Type &type) { | std::string logType(const structure::Type &type) { | ||||||
| 	if (type.tag == structure::TypeTag::Struct) { | 	if (type.tag == structure::TypeTag::Struct) { | ||||||
| 		return "struct " + logFullName(type.name); | 		return "struct " + logFullName(type.name); | ||||||
|  | @ -121,10 +120,6 @@ std::string logType(const structure::Type &type) { | ||||||
| 	return builtInTypes->value(type.tag, "invalid"); | 	return builtInTypes->value(type.tag, "invalid"); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| QString fullNameKey(const structure::FullName &name) { |  | ||||||
| 	return name.join('.'); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool validateAnsiString(const QString &value) { | bool validateAnsiString(const QString &value) { | ||||||
| 	for (auto ch : value) { | 	for (auto ch : value) { | ||||||
| 		if (ch.unicode() > 127) { | 		if (ch.unicode() > 127) { | ||||||
|  | @ -158,67 +153,61 @@ bool ParsedFile::read() { | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	bool noErrors = false; | 	auto filepath = QFileInfo(options_.inputPath).absoluteFilePath(); | ||||||
|  | 	module_ = std::make_unique<structure::Module>(filepath); | ||||||
| 	do { | 	do { | ||||||
| 		if (auto startToken = file_.getToken(BasicType::Name)) { | 		if (auto startToken = file_.getToken(BasicType::Name)) { | ||||||
| 			if (tokenValue(startToken) == "using") { | 			if (tokenValue(startToken) == "using") { | ||||||
| 				if (auto includedResult = readIncluded()) { | 				if (auto includedResult = readIncluded()) { | ||||||
| 					result_.includes.push_back(includedResult); | 					module_->addIncluded(std::move(includedResult)); | ||||||
| 					continue; | 					continue; | ||||||
| 				} | 				} | ||||||
| 			} else if (auto braceOpen = file_.getToken(BasicType::LeftBrace)) { | 			} else if (auto braceOpen = file_.getToken(BasicType::LeftBrace)) { | ||||||
| 				if (auto structResult = readStruct(tokenValue(startToken))) { | 				if (auto structResult = readStruct(tokenValue(startToken))) { | ||||||
| 					if (findStruct(structResult.name)) { | 					if (module_->addStruct(structResult)) { | ||||||
| 						logError(kErrorAlreadyDefined) << "struct '" << logFullName(structResult.name) << "' already defined"; | 						continue; | ||||||
| 						break; |  | ||||||
| 					} | 					} | ||||||
| 					result_.structsByName.insert(fullNameKey(structResult.name), result_.structs.size()); | 					logError(kErrorAlreadyDefined) << "struct '" << logFullName(structResult.name) << "' already defined"; | ||||||
| 					result_.structs.push_back(structResult); | 					break; | ||||||
| 					continue; |  | ||||||
| 				} | 				} | ||||||
| 			} else if (auto colonToken = file_.getToken(BasicType::Colon)) { | 			} else if (auto colonToken = file_.getToken(BasicType::Colon)) { | ||||||
| 				if (auto variableResult = readVariable(tokenValue(startToken))) { | 				if (auto variableResult = readVariable(tokenValue(startToken))) { | ||||||
| 					if (findVariable(variableResult.name)) { | 					if (module_->addVariable(variableResult)) { | ||||||
| 						logError(kErrorAlreadyDefined) << "variable '" << logFullName(variableResult.name) << "' already defined"; | 						continue; | ||||||
| 						break; |  | ||||||
| 					} | 					} | ||||||
| 					result_.variablesByName.insert(fullNameKey(variableResult.name), result_.variables.size()); | 					logError(kErrorAlreadyDefined) << "variable '" << logFullName(variableResult.name) << "' already defined"; | ||||||
| 					result_.variables.push_back(variableResult); | 					break; | ||||||
| 					continue; |  | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		if (!file_.atEnd()) { | 		if (file_.atEnd()) { | ||||||
| 			logErrorUnexpectedToken() << "using keyword, or struct definition, or variable definition"; |  | ||||||
| 		} else { |  | ||||||
| 			noErrors = !failed(); |  | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
|  | 		logErrorUnexpectedToken() << "using keyword, or struct definition, or variable definition"; | ||||||
| 	} while (!failed()); | 	} while (!failed()); | ||||||
| 
 | 
 | ||||||
| 	if (noErrors) { | 	if (failed()) { | ||||||
| 		result_.fullpath = QFileInfo(options_.inputPath).absoluteFilePath(); | 		module_ = nullptr; | ||||||
| 	} | 	} | ||||||
| 	return noErrors; | 	return !failed(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| common::LogStream ParsedFile::logErrorTypeMismatch() { | common::LogStream ParsedFile::logErrorTypeMismatch() { | ||||||
| 	return logError(kErrorTypeMismatch) << "type mismatch: "; | 	return logError(kErrorTypeMismatch) << "type mismatch: "; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| structure::Module ParsedFile::readIncluded() { | ParsedFile::ModulePtr ParsedFile::readIncluded() { | ||||||
| 	structure::Module result; |  | ||||||
| 	if (auto usingFile = assertNextToken(BasicType::String)) { | 	if (auto usingFile = assertNextToken(BasicType::String)) { | ||||||
| 		if (assertNextToken(BasicType::Semicolon)) { | 		if (assertNextToken(BasicType::Semicolon)) { | ||||||
| 			ParsedFile included(includedOptions(tokenValue(usingFile))); | 			ParsedFile included(includedOptions(tokenValue(usingFile))); | ||||||
| 			if (included.read()) { | 			if (included.read()) { | ||||||
| 				result = included.data(); | 				return included.getResult(); | ||||||
| 			} else { | 			} else { | ||||||
| 				logError(kErrorInIncluded) << "error while parsing '" << tokenValue(usingFile).toStdString() << "'"; | 				logError(kErrorInIncluded) << "error while parsing '" << tokenValue(usingFile).toStdString() << "'"; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return result; | 	return nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| structure::Struct ParsedFile::readStruct(const QString &name) { | structure::Struct ParsedFile::readStruct(const QString &name) { | ||||||
|  | @ -268,7 +257,7 @@ structure::Type ParsedFile::readType() { | ||||||
| 			result = builtInType; | 			result = builtInType; | ||||||
| 		} else { | 		} else { | ||||||
| 			auto fullName = composeFullName(name); | 			auto fullName = composeFullName(name); | ||||||
| 			if (findStruct(fullName)) { | 			if (module_->findStruct(fullName)) { | ||||||
| 				result.tag = structure::TypeTag::Struct; | 				result.tag = structure::TypeTag::Struct; | ||||||
| 				result.name = fullName; | 				result.name = fullName; | ||||||
| 			} else { | 			} else { | ||||||
|  | @ -331,11 +320,17 @@ structure::Value ParsedFile::readStructValue() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| structure::Value ParsedFile::defaultConstructedStruct(const structure::FullName &structName) { | structure::Value ParsedFile::defaultConstructedStruct(const structure::FullName &structName) { | ||||||
| 	if (auto pattern = findStruct(structName)) { | 	if (auto pattern = module_->findStruct(structName)) { | ||||||
| 		QList<structure::Variable> fields; | 		QList<structure::data::field> fields; | ||||||
| 		fields.reserve(pattern->fields.size()); | 		fields.reserve(pattern->fields.size()); | ||||||
| 		for (const auto &fieldType : pattern->fields) { | 		for (const auto &fieldType : pattern->fields) { | ||||||
| 			fields.push_back({ fieldType.name, { fieldType.type, Qt::Uninitialized } }); | 			fields.push_back({ | ||||||
|  | 				{ // variable
 | ||||||
|  | 					fieldType.name, | ||||||
|  | 					{ fieldType.type, Qt::Uninitialized }, // value
 | ||||||
|  | 				}, | ||||||
|  | 				structure::data::field::Status::Uninitialized, // status
 | ||||||
|  | 			}); | ||||||
| 		} | 		} | ||||||
| 		return { structName, fields }; | 		return { structName, fields }; | ||||||
| 	} | 	} | ||||||
|  | @ -343,14 +338,14 @@ structure::Value ParsedFile::defaultConstructedStruct(const structure::FullName | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ParsedFile::applyStructParent(structure::Value &result, const structure::FullName &parentName) { | void ParsedFile::applyStructParent(structure::Value &result, const structure::FullName &parentName) { | ||||||
| 	if (auto parent = findVariable(parentName)) { | 	if (auto parent = module_->findVariable(parentName)) { | ||||||
| 		if (parent->value.type() != result.type()) { | 		if (parent->value.type() != result.type()) { | ||||||
| 			logErrorTypeMismatch() << "parent '" << logFullName(parentName) << "' has type '" << logType(parent->value.type()) << "' while child value has type " << logType(result.type()); | 			logErrorTypeMismatch() << "parent '" << logFullName(parentName) << "' has type '" << logType(parent->value.type()) << "' while child value has type " << logType(result.type()); | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		const auto *srcFields(parent->value.Complex()); | 		const auto *srcFields(parent->value.Fields()); | ||||||
| 		auto *dstFields(result.Complex()); | 		auto *dstFields(result.Fields()); | ||||||
| 		if (!srcFields || !dstFields) { | 		if (!srcFields || !dstFields) { | ||||||
| 			logAssert(false) << "struct data check failed"; | 			logAssert(false) << "struct data check failed"; | ||||||
| 			return; | 			return; | ||||||
|  | @ -358,10 +353,17 @@ void ParsedFile::applyStructParent(structure::Value &result, const structure::Fu | ||||||
| 
 | 
 | ||||||
| 		logAssert(srcFields->size() == dstFields->size()) << "struct size check failed"; | 		logAssert(srcFields->size() == dstFields->size()) << "struct size check failed"; | ||||||
| 		for (int i = 0, s = srcFields->size(); i != s; ++i) { | 		for (int i = 0, s = srcFields->size(); i != s; ++i) { | ||||||
| 			const auto &srcValue(srcFields->at(i).value); | 			const auto &srcField(srcFields->at(i)); | ||||||
| 			auto &dstValue((*dstFields)[i].value); | 			auto &dstField((*dstFields)[i]); | ||||||
| 			logAssert(srcValue.type() == dstValue.type()) << "struct field type check failed"; | 			using Status = structure::data::field::Status; | ||||||
| 			dstValue = srcValue; | 			if (srcField.status == Status::Explicit || | ||||||
|  | 				dstField.status == Status::Uninitialized) { | ||||||
|  | 				const auto &srcValue(srcField.variable.value); | ||||||
|  | 				auto &dstValue(dstField.variable.value); | ||||||
|  | 				logAssert(srcValue.type() == dstValue.type()) << "struct field type check failed"; | ||||||
|  | 				dstValue = srcValue; | ||||||
|  | 				dstField.status = Status::Implicit; | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		logError(kErrorIdentifierNotFound) << "parent '" << logFullName(parentName) << "' not found"; | 		logError(kErrorIdentifierNotFound) << "parent '" << logFullName(parentName) << "' not found"; | ||||||
|  | @ -388,18 +390,19 @@ bool ParsedFile::readStructValueInner(structure::Value &result) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool ParsedFile::assignStructField(structure::Value &result, const structure::Variable &field) { | bool ParsedFile::assignStructField(structure::Value &result, const structure::Variable &field) { | ||||||
| 	auto *fields = result.Complex(); | 	auto *fields = result.Fields(); | ||||||
| 	if (!fields) { | 	if (!fields) { | ||||||
| 		logAssert(false) << "struct data check failed"; | 		logAssert(false) << "struct data check failed"; | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
| 	for (auto &already : *fields) { | 	for (auto &already : *fields) { | ||||||
| 		if (already.name == field.name) { | 		if (already.variable.name == field.name) { | ||||||
| 			if (already.value.type() == field.value.type()) { | 			if (already.variable.value.type() == field.value.type()) { | ||||||
| 				already.value = field.value; | 				already.variable.value = field.value; | ||||||
|  | 				already.status = structure::data::field::Status::Explicit; | ||||||
| 				return true; | 				return true; | ||||||
| 			} else { | 			} else { | ||||||
| 				logErrorTypeMismatch() << "field '" << logFullName(already.name) << "' has type '" << logType(already.value.type()) << "' while value has type '" << logType(field.value.type()) << "'"; | 				logErrorTypeMismatch() << "field '" << logFullName(already.variable.name) << "' has type '" << logType(already.variable.value.type()) << "' while value has type '" << logType(field.value.type()) << "'"; | ||||||
| 				return false; | 				return false; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | @ -429,7 +432,7 @@ structure::Value ParsedFile::readPositiveValue() { | ||||||
| 	if (numericToken.type == BasicType::Int) { | 	if (numericToken.type == BasicType::Int) { | ||||||
| 		return { structure::TypeTag::Int, tokenValue(numericToken).toInt() }; | 		return { structure::TypeTag::Int, tokenValue(numericToken).toInt() }; | ||||||
| 	} else if (numericToken.type == BasicType::Double) { | 	} else if (numericToken.type == BasicType::Double) { | ||||||
| 		return { tokenValue(numericToken).toDouble() }; | 		return { structure::TypeTag::Double, tokenValue(numericToken).toDouble() }; | ||||||
| 	} else if (numericToken.type == BasicType::Name) { | 	} else if (numericToken.type == BasicType::Name) { | ||||||
| 		auto value = tokenValue(numericToken); | 		auto value = tokenValue(numericToken); | ||||||
| 		auto match = QRegularExpression("^\\d+px$").match(value); | 		auto match = QRegularExpression("^\\d+px$").match(value); | ||||||
|  | @ -717,7 +720,7 @@ structure::Value ParsedFile::readFontValue() { | ||||||
| structure::Value ParsedFile::readCopyValue() { | structure::Value ParsedFile::readCopyValue() { | ||||||
| 	if (auto copyName = file_.getToken(BasicType::Name)) { | 	if (auto copyName = file_.getToken(BasicType::Name)) { | ||||||
| 		structure::FullName name = { tokenValue(copyName) }; | 		structure::FullName name = { tokenValue(copyName) }; | ||||||
| 		if (auto variable = findVariable(name)) { | 		if (auto variable = module_->findVariable(name)) { | ||||||
| 			return variable->value.makeCopy(variable->name); | 			return variable->value.makeCopy(variable->name); | ||||||
| 		} | 		} | ||||||
| 		logError(kErrorIdentifierNotFound) << "identifier '" << logFullName(name) << "' not found"; | 		logError(kErrorIdentifierNotFound) << "identifier '" << logFullName(name) << "' not found"; | ||||||
|  | @ -725,48 +728,6 @@ structure::Value ParsedFile::readCopyValue() { | ||||||
| 	return {}; | 	return {}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Returns nullptr if there is no such struct in result_ or any of included modules.
 |  | ||||||
| const structure::Struct *ParsedFile::findStruct(const structure::FullName &name) { |  | ||||||
| 	if (auto result = findStructInModule(name, result_)) { |  | ||||||
| 		return result; |  | ||||||
| 	} |  | ||||||
| 	for (const auto &included : result_.includes) { |  | ||||||
| 		if (auto result = findStructInModule(name, included)) { |  | ||||||
| 			return result; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nullptr; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const structure::Struct *ParsedFile::findStructInModule(const structure::FullName &name, const structure::Module &module) { |  | ||||||
| 	auto index = module.structsByName.value(fullNameKey(name), -1); |  | ||||||
| 	if (index < 0) { |  | ||||||
| 		return nullptr; |  | ||||||
| 	} |  | ||||||
| 	return &module.structs.at(index); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Returns nullptr if there is no such variable in result_ or any of included modules.
 |  | ||||||
| const structure::Variable *ParsedFile::findVariable(const structure::FullName &name) { |  | ||||||
| 	if (auto result = findVariableInModule(name, result_)) { |  | ||||||
| 		return result; |  | ||||||
| 	} |  | ||||||
| 	for (const auto &included : result_.includes) { |  | ||||||
| 		if (auto result = findVariableInModule(name, included)) { |  | ||||||
| 			return result; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nullptr; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const structure::Variable *ParsedFile::findVariableInModule(const structure::FullName &name, const structure::Module &module) { |  | ||||||
| 	auto index = module.variablesByName.value(fullNameKey(name), -1); |  | ||||||
| 	if (index < 0) { |  | ||||||
| 		return nullptr; |  | ||||||
| 	} |  | ||||||
| 	return &module.variables.at(index); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| BasicToken ParsedFile::assertNextToken(BasicToken::Type type) { | BasicToken ParsedFile::assertNextToken(BasicToken::Type type) { | ||||||
| 	auto result = file_.getToken(type); | 	auto result = file_.getToken(type); | ||||||
| 	if (!result) { | 	if (!result) { | ||||||
|  |  | ||||||
|  | @ -24,7 +24,7 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| #include <string> | #include <string> | ||||||
| #include "codegen/common/basic_tokenized_file.h" | #include "codegen/common/basic_tokenized_file.h" | ||||||
| #include "codegen/style/options.h" | #include "codegen/style/options.h" | ||||||
| #include "codegen/style/structure.h" | #include "codegen/style/module.h" | ||||||
| 
 | 
 | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
|  | @ -32,14 +32,15 @@ namespace style { | ||||||
| // Parses an input file to the internal struct.
 | // Parses an input file to the internal struct.
 | ||||||
| class ParsedFile { | class ParsedFile { | ||||||
| public: | public: | ||||||
| 	ParsedFile(const Options &options); | 	explicit ParsedFile(const Options &options); | ||||||
| 	ParsedFile(const ParsedFile &other) = delete; | 	ParsedFile(const ParsedFile &other) = delete; | ||||||
| 	ParsedFile &operator=(const ParsedFile &other) = delete; | 	ParsedFile &operator=(const ParsedFile &other) = delete; | ||||||
| 
 | 
 | ||||||
| 	bool read(); | 	bool read(); | ||||||
| 
 | 
 | ||||||
| 	const structure::Module &data() const { | 	using ModulePtr = std::unique_ptr<structure::Module>; | ||||||
| 		return result_; | 	ModulePtr getResult() { | ||||||
|  | 		return std::move(module_); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | @ -66,7 +67,7 @@ private: | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Helper methods for context-dependent reading.
 | 	// Helper methods for context-dependent reading.
 | ||||||
| 	structure::Module readIncluded(); | 	ModulePtr readIncluded(); | ||||||
| 	structure::Struct readStruct(const QString &name); | 	structure::Struct readStruct(const QString &name); | ||||||
| 	structure::Variable readVariable(const QString &name); | 	structure::Variable readVariable(const QString &name); | ||||||
| 
 | 
 | ||||||
|  | @ -96,14 +97,6 @@ private: | ||||||
| 	structure::Value readFontValue(); | 	structure::Value readFontValue(); | ||||||
| 	structure::Value readCopyValue(); | 	structure::Value readCopyValue(); | ||||||
| 
 | 
 | ||||||
| 	// Returns nullptr if there is no such struct in result_ or any of included modules.
 |  | ||||||
| 	const structure::Struct *findStruct(const structure::FullName &name); |  | ||||||
| 	const structure::Struct *findStructInModule(const structure::FullName &name, const structure::Module &module); |  | ||||||
| 
 |  | ||||||
| 	// Returns nullptr if there is no such variable in result_ or any of included modules.
 |  | ||||||
| 	const structure::Variable *findVariable(const structure::FullName &name); |  | ||||||
| 	const structure::Variable *findVariableInModule(const structure::FullName &name, const structure::Module &module); |  | ||||||
| 
 |  | ||||||
| 	// Read next token and fire unexpected token error if it is not of "type".
 | 	// Read next token and fire unexpected token error if it is not of "type".
 | ||||||
| 	using BasicToken = common::BasicTokenizedFile::Token; | 	using BasicToken = common::BasicTokenizedFile::Token; | ||||||
| 	BasicToken assertNextToken(BasicToken::Type type); | 	BasicToken assertNextToken(BasicToken::Type type); | ||||||
|  | @ -117,7 +110,7 @@ private: | ||||||
| 	common::BasicTokenizedFile file_; | 	common::BasicTokenizedFile file_; | ||||||
| 	Options options_; | 	Options options_; | ||||||
| 	bool failed_ = false; | 	bool failed_ = false; | ||||||
| 	structure::Module result_; | 	ModulePtr module_; | ||||||
| 
 | 
 | ||||||
| 	QMap<std::string, structure::Type> typeNames_ = { | 	QMap<std::string, structure::Type> typeNames_ = { | ||||||
| 		{ "int"       , { structure::TypeTag::Int } }, | 		{ "int"       , { structure::TypeTag::Int } }, | ||||||
|  |  | ||||||
|  | @ -0,0 +1,103 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/style/processor.h" | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QFileInfo> | ||||||
|  | #include "codegen/common/cpp_file.h" | ||||||
|  | #include "codegen/style/parsed_file.h" | ||||||
|  | #include "codegen/style/generator.h" | ||||||
|  | #include "codegen/style/sprite_generator.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace style { | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | constexpr int kErrorCantWritePath = 851; | ||||||
|  | 
 | ||||||
|  | QString destFileBaseName(const structure::Module &module) { | ||||||
|  | 	return "style_" + QFileInfo(module.filepath()).baseName(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | Processor::Processor(const Options &options) | ||||||
|  | : parser_(std::make_unique<ParsedFile>(options)) | ||||||
|  | , options_(options) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int Processor::launch() { | ||||||
|  | 	if (!parser_->read()) { | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	auto module = parser_->getResult(); | ||||||
|  | 	if (options_.rebuildDependencies) { | ||||||
|  | 		bool result = module->enumIncludes([this](const structure::Module &included) -> bool { | ||||||
|  | 			return write(included); | ||||||
|  | 		}); | ||||||
|  | 		if (!result) { | ||||||
|  | 			return -1; | ||||||
|  | 		} | ||||||
|  | 	} else if (!write(*module)) { | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool Processor::write(const structure::Module &module) const { | ||||||
|  | 	QDir dir(options_.outputPath); | ||||||
|  | 	if (!dir.mkpath(".")) { | ||||||
|  | 		common::logError(kErrorCantWritePath, "Command Line") << "can not open path for writing: " << dir.absolutePath().toStdString(); | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	QFileInfo srcFile(module.filepath()); | ||||||
|  | 	QString dstFilePath = dir.absolutePath() + '/' + destFileBaseName(module); | ||||||
|  | 
 | ||||||
|  | 	common::ProjectInfo project = { | ||||||
|  | 		"codegen_style", | ||||||
|  | 		srcFile.fileName(), | ||||||
|  | 		"stdafx.h", | ||||||
|  | 		!options_.rebuildDependencies, // forceReGenerate
 | ||||||
|  | 	}; | ||||||
|  | 
 | ||||||
|  | 	SpriteGenerator spriteGenerator(module); | ||||||
|  | 	if (!spriteGenerator.writeSprites()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	Generator generator(module, dstFilePath, project); | ||||||
|  | 	if (!generator.writeHeader()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (!generator.writeSource()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Processor::~Processor() = default; | ||||||
|  | 
 | ||||||
|  | } // namespace style
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,60 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <memory> | ||||||
|  | #include <QtCore/QString> | ||||||
|  | #include "codegen/style/options.h" | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace style { | ||||||
|  | namespace structure { | ||||||
|  | class Module; | ||||||
|  | } // namespace structure
 | ||||||
|  | class ParsedFile; | ||||||
|  | 
 | ||||||
|  | // Walks through a file, parses it and parses dependency files if necessary.
 | ||||||
|  | // Uses Generator class to produce the final output.
 | ||||||
|  | class Processor { | ||||||
|  | public: | ||||||
|  | 	explicit Processor(const Options &options); | ||||||
|  | 	Processor(const Processor &other) = delete; | ||||||
|  | 	Processor &operator=(const Processor &other) = delete; | ||||||
|  | 
 | ||||||
|  | 	// Returns 0 on success.
 | ||||||
|  | 	int launch(); | ||||||
|  | 
 | ||||||
|  | 	~Processor(); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 	bool write(const structure::Module &module) const; | ||||||
|  | 
 | ||||||
|  | 	std::unique_ptr<ParsedFile> parser_; | ||||||
|  | 	const Options &options_; | ||||||
|  | 
 | ||||||
|  | 	// List of files we need to generate with other instance of Generator.
 | ||||||
|  | 	// It is not empty only if rebuild_ flag is true.
 | ||||||
|  | 	QStringList dependenciesToGenerate_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace style
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -0,0 +1,180 @@ | ||||||
|  | /*
 | ||||||
|  | This file is part of Telegram Desktop, | ||||||
|  | the official desktop version of Telegram messaging app, see https://telegram.org
 | ||||||
|  | 
 | ||||||
|  | Telegram Desktop is free software: you can redistribute it and/or modify | ||||||
|  | it under the terms of the GNU General Public License as published by | ||||||
|  | the Free Software Foundation, either version 3 of the License, or | ||||||
|  | (at your option) any later version. | ||||||
|  | 
 | ||||||
|  | It is distributed in the hope that it will be useful, | ||||||
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | GNU General Public License for more details. | ||||||
|  | 
 | ||||||
|  | In addition, as a special exception, the copyright holders give permission | ||||||
|  | to link the code of portions of this program with the OpenSSL library. | ||||||
|  | 
 | ||||||
|  | Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | ||||||
|  | Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | ||||||
|  | */ | ||||||
|  | #include "codegen/style/sprite_generator.h" | ||||||
|  | 
 | ||||||
|  | #include <QtCore/QFileInfo> | ||||||
|  | #include <QtCore/QDir> | ||||||
|  | #include <QtCore/QBuffer> | ||||||
|  | #include <QtGui/QPainter> | ||||||
|  | #include <QtGui/QColor> | ||||||
|  | #include <functional> | ||||||
|  | #include "codegen/style/parsed_file.h" | ||||||
|  | 
 | ||||||
|  | using Module = codegen::style::structure::Module; | ||||||
|  | using Struct = codegen::style::structure::Struct; | ||||||
|  | using Variable = codegen::style::structure::Variable; | ||||||
|  | using Tag = codegen::style::structure::TypeTag; | ||||||
|  | 
 | ||||||
|  | namespace codegen { | ||||||
|  | namespace style { | ||||||
|  | 
 | ||||||
|  | using structure::logFullName; | ||||||
|  | 
 | ||||||
|  | namespace { | ||||||
|  | 
 | ||||||
|  | constexpr int kErrorFileNotFound      = 881; | ||||||
|  | constexpr int kErrorSpritesIntersect  = 882; | ||||||
|  | constexpr int kErrorCouldNotGenerate  = 886; | ||||||
|  | constexpr int kErrorCouldNotSerialize = 886; | ||||||
|  | constexpr int kErrorCouldNotOpen      = 886; | ||||||
|  | constexpr int kErrorCouldNotWrite     = 886; | ||||||
|  | 
 | ||||||
|  | } // namespace
 | ||||||
|  | 
 | ||||||
|  | SpriteGenerator::SpriteGenerator(const structure::Module &module) | ||||||
|  | : module_(module) | ||||||
|  | , basePath_(QFileInfo(module.filepath()).dir().absolutePath()) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool SpriteGenerator::writeSprites() { | ||||||
|  | 	if (!collectSprites()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (sprites_.isEmpty()) { | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	sprite2x_ = QImage(basePath_ + "/art/sprite_200x.png"); | ||||||
|  | 	if (sprite2x_.isNull()) { | ||||||
|  | 		common::logError(kErrorFileNotFound, "/art/sprite_200x.png") << "sprite file was not found"; | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	std::vector<int> sizes = { 5, 6 }; | ||||||
|  | 	std::vector<const char *> postfixes = { "125", "150" }; | ||||||
|  | 	for (int i = 0, l = sizes.size(); i < l; ++i) { | ||||||
|  | 		auto sprite = generateSprite(sizes[i]); | ||||||
|  | 		QString filepath = basePath_ + "/art/sprite_" + postfixes[i] + "x.png"; | ||||||
|  | 		if (sprite.isNull()) { | ||||||
|  | 			common::logError(kErrorCouldNotGenerate, filepath) << "could not generate sprite file"; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		QByteArray spriteData; | ||||||
|  | 		{ | ||||||
|  | 			QBuffer spriteBuffer(&spriteData); | ||||||
|  | 			if (!sprite.save(&spriteBuffer, "PNG")) { | ||||||
|  | 				common::logError(kErrorCouldNotSerialize, filepath) << "could not serialize sprite file"; | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		QFile file(filepath); | ||||||
|  | 		if (file.open(QIODevice::ReadOnly)) { | ||||||
|  | 			if (file.readAll() == spriteData) { | ||||||
|  | 				continue; | ||||||
|  | 			} | ||||||
|  | 			file.close(); | ||||||
|  | 		} | ||||||
|  | 		if (!file.open(QIODevice::WriteOnly)) { | ||||||
|  | 			common::logError(kErrorCouldNotOpen, filepath) << "could not open sprite file for write"; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		if (file.write(spriteData) != spriteData.size()) { | ||||||
|  | 			common::logError(kErrorCouldNotWrite, filepath) << "could not write sprite file"; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Touch resource file.
 | ||||||
|  | 		filepath = basePath_ + "/telegram.qrc"; | ||||||
|  | 		QFile qrc(filepath); | ||||||
|  | 		if (qrc.open(QIODevice::ReadOnly)) { | ||||||
|  | 			auto qrcContent = qrc.readAll(); | ||||||
|  | 			qrc.close(); | ||||||
|  | 			if (!qrc.open(QIODevice::WriteOnly)) { | ||||||
|  | 				common::logError(kErrorCouldNotOpen, filepath) << "could not open .qrc file for write"; | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 			if (qrc.write(qrcContent) != qrcContent.size()) { | ||||||
|  | 				common::logError(kErrorCouldNotWrite, filepath) << "could not write .qrc file"; | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool SpriteGenerator::collectSprites() { | ||||||
|  | 	std::function<bool(const Variable&)> collector = [this, &collector](const Variable &variable) { | ||||||
|  | 		auto value = variable.value; | ||||||
|  | 		if (value.type().tag == Tag::Sprite) { | ||||||
|  | 			auto v(value.Sprite()); | ||||||
|  | 			if (!v.width || !v.height) return true; | ||||||
|  | 
 | ||||||
|  | 			QRect vRect(v.left, v.top, v.width, v.height); | ||||||
|  | 			bool found = false; | ||||||
|  | 			for (auto var : sprites_) { | ||||||
|  | 				auto sprite = var.value.Sprite(); | ||||||
|  | 				QRect spriteRect(sprite.left, sprite.top, sprite.width, sprite.height); | ||||||
|  | 				if (spriteRect == vRect) { | ||||||
|  | 					found = true; | ||||||
|  | 				} else if (spriteRect.intersects(vRect)) { | ||||||
|  | 					common::logError(kErrorSpritesIntersect, module_.filepath()) << "sprite '" << logFullName(variable.name) << "' intersects with '" << logFullName(var.name) << "'"; | ||||||
|  | 					return false; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			if (!found) { | ||||||
|  | 				sprites_.push_back(variable); | ||||||
|  | 			} | ||||||
|  | 		} else if (value.type().tag == Tag::Struct) { | ||||||
|  | 			auto fields = variable.value.Fields(); | ||||||
|  | 			if (!fields) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			for (auto field : *fields) { | ||||||
|  | 				if (!collector(field.variable)) { | ||||||
|  | 					return false; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	}; | ||||||
|  | 	return module_.enumVariables(collector); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | QImage SpriteGenerator::generateSprite(int scale) { | ||||||
|  | 	auto convert = [scale](int value) -> int { return structure::data::pxAdjust(value, scale); }; | ||||||
|  | 	QImage result(convert(sprite2x_.width() / 2), convert(sprite2x_.height() / 2), sprite2x_.format()); | ||||||
|  | 	{ | ||||||
|  | 		QPainter p(&result); | ||||||
|  | 		p.setCompositionMode(QPainter::CompositionMode_Source); | ||||||
|  | 		p.fillRect(0, 0, result.width(), result.height(), QColor(0, 0, 0, 0)); | ||||||
|  | 		for (auto variable : sprites_) { | ||||||
|  | 			auto sprite = variable.value.Sprite(); | ||||||
|  | 			auto copy = sprite2x_.copy(sprite.left * 2, sprite.top * 2, sprite.width * 2, sprite.height * 2); | ||||||
|  | 			copy = copy.scaled(convert(sprite.width), convert(sprite.height), Qt::IgnoreAspectRatio, Qt::SmoothTransformation); | ||||||
|  | 			p.drawImage(convert(sprite.left), convert(sprite.top), copy); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace style
 | ||||||
|  | } // namespace codegen
 | ||||||
|  | @ -20,55 +20,37 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
|  | #include <memory> | ||||||
| #include <QtCore/QString> | #include <QtCore/QString> | ||||||
| #include <QtCore/QList> | #include <QtCore/QSet> | ||||||
| #include <QtCore/QMap> | #include <QtGui/QImage> | ||||||
| #include "codegen/style/structure_types.h" | #include "codegen/style/structure_types.h" | ||||||
| 
 | 
 | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
| namespace structure { | namespace structure { | ||||||
| 
 | class Module; | ||||||
| struct Variable { |  | ||||||
| 	FullName name; |  | ||||||
| 	Value value; |  | ||||||
| 
 |  | ||||||
| 	explicit operator bool() const { |  | ||||||
| 		return !name.isEmpty(); |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct StructField { |  | ||||||
| 	FullName name; |  | ||||||
| 	Type type; |  | ||||||
| 
 |  | ||||||
| 	explicit operator bool() const { |  | ||||||
| 		return !name.isEmpty(); |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct Struct { |  | ||||||
| 	FullName name; |  | ||||||
| 	QList<StructField> fields; |  | ||||||
| 
 |  | ||||||
| 	explicit operator bool() const { |  | ||||||
| 		return !name.isEmpty(); |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct Module { |  | ||||||
| 	QString fullpath; |  | ||||||
| 	QList<Module> includes; |  | ||||||
| 	QList<Struct> structs; |  | ||||||
| 	QList<Variable> variables; |  | ||||||
| 	QMap<QString, int> structsByName; |  | ||||||
| 	QMap<QString, int> variablesByName; |  | ||||||
| 
 |  | ||||||
| 	explicit operator bool() const { |  | ||||||
| 		return !fullpath.isEmpty(); |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| } // namespace structure
 | } // namespace structure
 | ||||||
|  | 
 | ||||||
|  | class SpriteGenerator { | ||||||
|  | public: | ||||||
|  | 	SpriteGenerator(const structure::Module &module); | ||||||
|  | 	SpriteGenerator(const SpriteGenerator &other) = delete; | ||||||
|  | 	SpriteGenerator &operator=(const SpriteGenerator &other) = delete; | ||||||
|  | 
 | ||||||
|  | 	bool writeSprites(); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  | 
 | ||||||
|  | 	bool collectSprites(); | ||||||
|  | 	QImage generateSprite(int scale); // scale = 5 for 125% and 6 for 150%.
 | ||||||
|  | 
 | ||||||
|  | 	const structure::Module &module_; | ||||||
|  | 	QString basePath_; | ||||||
|  | 	QImage sprite2x_; | ||||||
|  | 	QList<structure::Variable> sprites_; | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| } // namespace style
 | } // namespace style
 | ||||||
| } // namespace codegen
 | } // namespace codegen
 | ||||||
|  | @ -20,8 +20,6 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| #include "codegen/style/structure_types.h" | #include "codegen/style/structure_types.h" | ||||||
| 
 | 
 | ||||||
| #include "codegen/style/structure.h" |  | ||||||
| 
 |  | ||||||
| namespace codegen { | namespace codegen { | ||||||
| namespace style { | namespace style { | ||||||
| namespace structure { | namespace structure { | ||||||
|  | @ -117,15 +115,15 @@ struct Value::DataTypes { | ||||||
| 		data::font value_; | 		data::font value_; | ||||||
| 
 | 
 | ||||||
| 	}; | 	}; | ||||||
| 	class TComplex : public DataBase { | 	class TFields : public DataBase { | ||||||
| 	public: | 	public: | ||||||
| 		TComplex(data::complex value) : value_(value) { | 		TFields(data::fields value) : value_(value) { | ||||||
| 		} | 		} | ||||||
| 		const data::complex *Complex() const override { return &value_; } | 		const data::fields *Fields() const override { return &value_; } | ||||||
| 		data::complex *Complex() override { return &value_; } | 		data::fields *Fields() override { return &value_; } | ||||||
| 
 | 
 | ||||||
| 	private: | 	private: | ||||||
| 		data::complex value_; | 		data::fields value_; | ||||||
| 
 | 
 | ||||||
| 	}; | 	}; | ||||||
| }; | }; | ||||||
|  | @ -133,9 +131,6 @@ struct Value::DataTypes { | ||||||
| Value::Value() : Value(TypeTag::Invalid, std::make_shared<DataBase>()) { | Value::Value() : Value(TypeTag::Invalid, std::make_shared<DataBase>()) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Value::Value(double value) : Value(TypeTag::Double, std::make_shared<DataTypes::TDouble>(value)) { |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| Value::Value(data::point value) : Value(TypeTag::Point, std::make_shared<DataTypes::TPoint>(value)) { | Value::Value(data::point value) : Value(TypeTag::Point, std::make_shared<DataTypes::TPoint>(value)) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -154,9 +149,16 @@ Value::Value(data::margins value) : Value(TypeTag::Margins, std::make_shared<Dat | ||||||
| Value::Value(data::font value) : Value(TypeTag::Font, std::make_shared<DataTypes::TFont>(value)) { | Value::Value(data::font value) : Value(TypeTag::Font, std::make_shared<DataTypes::TFont>(value)) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Value::Value(const FullName &type, data::complex value) | Value::Value(const FullName &type, data::fields value) | ||||||
| : type_ { TypeTag::Struct, type } | : type_ { TypeTag::Struct, type } | ||||||
| , data_(std::make_shared<DataTypes::TComplex>(value)) { | , data_(std::make_shared<DataTypes::TFields>(value)) { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Value::Value(TypeTag type, double value) : Value(type, std::make_shared<DataTypes::TDouble>(value)) { | ||||||
|  | 	if (type_.tag != TypeTag::Double) { | ||||||
|  | 		type_.tag = TypeTag::Invalid; | ||||||
|  | 		data_ = std::make_shared<DataBase>(); | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Value::Value(TypeTag type, int value) : Value(type, std::make_shared<DataTypes::TInt>(value)) { | Value::Value(TypeTag type, int value) : Value(type, std::make_shared<DataTypes::TInt>(value)) { | ||||||
|  | @ -189,10 +191,10 @@ Value::Value(Type type, Qt::Initialization) : type_(type) { | ||||||
| 	case TypeTag::Size: data_ = std::make_shared<DataTypes::TSize>(data::size { 0, 0 }); break; | 	case TypeTag::Size: data_ = std::make_shared<DataTypes::TSize>(data::size { 0, 0 }); break; | ||||||
| 	case TypeTag::Transition: data_ = std::make_shared<DataTypes::TString>("linear"); break; | 	case TypeTag::Transition: data_ = std::make_shared<DataTypes::TString>("linear"); break; | ||||||
| 	case TypeTag::Cursor: data_ = std::make_shared<DataTypes::TString>("default"); break; | 	case TypeTag::Cursor: data_ = std::make_shared<DataTypes::TString>("default"); break; | ||||||
| 	case TypeTag::Align: data_ = std::make_shared<DataTypes::TString>("left"); break; | 	case TypeTag::Align: data_ = std::make_shared<DataTypes::TString>("topleft"); break; | ||||||
| 	case TypeTag::Margins: data_ = std::make_shared<DataTypes::TMargins>(data::margins { 0, 0, 0, 0 }); break; | 	case TypeTag::Margins: data_ = std::make_shared<DataTypes::TMargins>(data::margins { 0, 0, 0, 0 }); break; | ||||||
| 	case TypeTag::Font: data_ = std::make_shared<DataTypes::TFont>(data::font { "", 13, 0 }); break; | 	case TypeTag::Font: data_ = std::make_shared<DataTypes::TFont>(data::font { "", 13, 0 }); break; | ||||||
| 	case TypeTag::Struct: data_ = std::make_shared<DataTypes::TComplex>(data::complex {}); break; | 	case TypeTag::Struct: data_ = std::make_shared<DataTypes::TFields>(data::fields {}); break; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -30,6 +30,9 @@ namespace structure { | ||||||
| 
 | 
 | ||||||
| // List of names, like overview.document.bg
 | // List of names, like overview.document.bg
 | ||||||
| using FullName = QStringList; | using FullName = QStringList; | ||||||
|  | inline std::string logFullName(const FullName &name) { | ||||||
|  | 	return name.join('.').toStdString(); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| struct Variable; | struct Variable; | ||||||
| 
 | 
 | ||||||
|  | @ -68,6 +71,10 @@ inline bool operator!=(const Type &a, const Type &b) { | ||||||
| 
 | 
 | ||||||
| namespace data { | namespace data { | ||||||
| 
 | 
 | ||||||
|  | inline int pxAdjust(int value, int scale) { | ||||||
|  | 	return qRound((value * scale / 4.) + (value > 0 ? -0.01 : 0.01)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| struct point { | struct point { | ||||||
| 	int x, y; | 	int x, y; | ||||||
| }; | }; | ||||||
|  | @ -85,29 +92,32 @@ struct margins { | ||||||
| }; | }; | ||||||
| struct font { | struct font { | ||||||
| 	enum Flag { | 	enum Flag { | ||||||
| 		Bold = 0x01, | 		Bold      = 0x01, | ||||||
| 		Italic = 0x02, | 		Italic    = 0x02, | ||||||
| 		Underline = 0x04, | 		Underline = 0x04, | ||||||
| 	}; | 	}; | ||||||
| 	std::string family; | 	std::string family; | ||||||
| 	int size; | 	int size; | ||||||
| 	int flags; | 	int flags; | ||||||
| }; | }; | ||||||
| using complex = QList<Variable>; | struct field; // defined after Variable is defined
 | ||||||
|  | using fields = QList<field>; | ||||||
| 
 | 
 | ||||||
| } // namespace data
 | } // namespace data
 | ||||||
| 
 | 
 | ||||||
| class Value { | class Value { | ||||||
| public: | public: | ||||||
| 	Value(); | 	Value(); | ||||||
| 	Value(double value); |  | ||||||
| 	Value(data::point value); | 	Value(data::point value); | ||||||
| 	Value(data::sprite value); | 	Value(data::sprite value); | ||||||
| 	Value(data::size value); | 	Value(data::size value); | ||||||
| 	Value(data::color value); | 	Value(data::color value); | ||||||
| 	Value(data::margins value); | 	Value(data::margins value); | ||||||
| 	Value(data::font value); | 	Value(data::font value); | ||||||
| 	Value(const FullName &type, data::complex value); | 	Value(const FullName &type, data::fields value); | ||||||
|  | 
 | ||||||
|  | 	// Can be only double.
 | ||||||
|  | 	Value(TypeTag type, double value); | ||||||
| 
 | 
 | ||||||
| 	// Can be int / pixels.
 | 	// Can be int / pixels.
 | ||||||
| 	Value(TypeTag type, int value); | 	Value(TypeTag type, int value); | ||||||
|  | @ -128,8 +138,8 @@ public: | ||||||
| 	data::color Color() const { return data_->Color(); }; | 	data::color Color() const { return data_->Color(); }; | ||||||
| 	data::margins Margins() const { return data_->Margins(); }; | 	data::margins Margins() const { return data_->Margins(); }; | ||||||
| 	data::font Font() const { return data_->Font(); }; | 	data::font Font() const { return data_->Font(); }; | ||||||
| 	const data::complex *Complex() const { return data_->Complex(); }; | 	const data::fields *Fields() const { return data_->Fields(); }; | ||||||
| 	data::complex *Complex() { return data_->Complex(); }; | 	data::fields *Fields() { return data_->Fields(); }; | ||||||
| 
 | 
 | ||||||
| 	explicit operator bool() const { | 	explicit operator bool() const { | ||||||
| 		return type_.tag != TypeTag::Invalid; | 		return type_.tag != TypeTag::Invalid; | ||||||
|  | @ -141,6 +151,10 @@ public: | ||||||
| 		return result; | 		return result; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	const FullName ©Of() const { | ||||||
|  | 		return copyOf_; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
| 	class DataBase { | 	class DataBase { | ||||||
| 	public: | 	public: | ||||||
|  | @ -153,8 +167,8 @@ private: | ||||||
| 		virtual data::color Color() const { return {}; }; | 		virtual data::color Color() const { return {}; }; | ||||||
| 		virtual data::margins Margins() const { return {}; }; | 		virtual data::margins Margins() const { return {}; }; | ||||||
| 		virtual data::font Font() const { return {}; }; | 		virtual data::font Font() const { return {}; }; | ||||||
| 		virtual const data::complex *Complex() const { return nullptr; }; | 		virtual const data::fields *Fields() const { return nullptr; }; | ||||||
| 		virtual data::complex *Complex() { return nullptr; }; | 		virtual data::fields *Fields() { return nullptr; }; | ||||||
| 		virtual ~DataBase() { | 		virtual ~DataBase() { | ||||||
| 		} | 		} | ||||||
| 	}; | 	}; | ||||||
|  | @ -169,6 +183,45 @@ private: | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | struct Variable { | ||||||
|  | 	FullName name; | ||||||
|  | 	Value value; | ||||||
|  | 
 | ||||||
|  | 	explicit operator bool() const { | ||||||
|  | 		return !name.isEmpty(); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | namespace data { | ||||||
|  | struct field { | ||||||
|  | 	enum class Status { | ||||||
|  | 		Uninitialized, | ||||||
|  | 		Implicit, | ||||||
|  | 		Explicit | ||||||
|  | 	}; | ||||||
|  | 	Variable variable; | ||||||
|  | 	Status status; | ||||||
|  | }; | ||||||
|  | } // namespace data
 | ||||||
|  | 
 | ||||||
|  | struct StructField { | ||||||
|  | 	FullName name; | ||||||
|  | 	Type type; | ||||||
|  | 
 | ||||||
|  | 	explicit operator bool() const { | ||||||
|  | 		return !name.isEmpty(); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct Struct { | ||||||
|  | 	FullName name; | ||||||
|  | 	QList<StructField> fields; | ||||||
|  | 
 | ||||||
|  | 	explicit operator bool() const { | ||||||
|  | 		return !name.isEmpty(); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| } // namespace structure
 | } // namespace structure
 | ||||||
| } // namespace style
 | } // namespace style
 | ||||||
| } // namespace codegen
 | } // namespace codegen
 | ||||||
|  |  | ||||||
|  | @ -813,8 +813,7 @@ inline QSharedPointer<T> MakeShared(Args&&... args) { | ||||||
| template <typename T> | template <typename T> | ||||||
| class NeverFreedPointer { | class NeverFreedPointer { | ||||||
| public: | public: | ||||||
| 	explicit NeverFreedPointer() { | 	NeverFreedPointer() = default; | ||||||
| 	} |  | ||||||
| 	NeverFreedPointer(const NeverFreedPointer<T> &other) = delete; | 	NeverFreedPointer(const NeverFreedPointer<T> &other) = delete; | ||||||
| 	NeverFreedPointer &operator=(const NeverFreedPointer<T> &other) = delete; | 	NeverFreedPointer &operator=(const NeverFreedPointer<T> &other) = delete; | ||||||
| 
 | 
 | ||||||
|  | @ -861,7 +860,7 @@ public: | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	T *_p = nullptr; | 	T *_p; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2876,7 +2876,7 @@ void EmojiPan::paintEvent(QPaintEvent *e) { | ||||||
| 					} | 					} | ||||||
| 
 | 
 | ||||||
| 					if (rtl()) selx = width() - selx - st::rbEmoji.width; | 					if (rtl()) selx = width() - selx - st::rbEmoji.width; | ||||||
| 					p.setOpacity(skip ? qMax(1., selx / (skip * st::rbEmoji.width)) : 1.); | 					p.setOpacity(skip ? qMax(1., selx / float64(skip * st::rbEmoji.width)) : 1.); | ||||||
| 					p.fillRect(selx, _iconsTop + st::rbEmoji.height - st::stickerIconPadding, st::rbEmoji.width, st::stickerIconSel, st::stickerIconSelColor); | 					p.fillRect(selx, _iconsTop + st::rbEmoji.height - st::stickerIconPadding, st::rbEmoji.width, st::stickerIconSel, st::stickerIconSelColor); | ||||||
| 
 | 
 | ||||||
| 					float64 o_left = snap(float64(_iconsX.current()) / st::stickerIconLeft.pxWidth(), 0., 1.); | 					float64 o_left = snap(float64(_iconsX.current()) / st::stickerIconLeft.pxWidth(), 0., 1.); | ||||||
|  |  | ||||||
|  | @ -375,7 +375,7 @@ void BoxButton::resizeToText() { | ||||||
| 		resize(_textWidth - _st.width, _st.height); | 		resize(_textWidth - _st.width, _st.height); | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_st.width < _textWidth + (_st.height - _st.font->height)) { | 		if (_st.width < _textWidth + (_st.height - _st.font->height)) { | ||||||
| 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.height - _st.font->height), 1.)); | 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.height - _st.font->height), 1)); | ||||||
| 			_textWidth = _st.font->width(_text); | 			_textWidth = _st.font->width(_text); | ||||||
| 		} | 		} | ||||||
| 		resize(_st.width, _st.height); | 		resize(_st.width, _st.height); | ||||||
|  |  | ||||||
|  | @ -250,7 +250,7 @@ Checkbox::Checkbox(QWidget *parent, const QString &text, bool checked, const sty | ||||||
| 		resize(_textWidth - _st.width, _st.height); | 		resize(_textWidth - _st.width, _st.height); | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_st.width < _st.textPosition.x() + _textWidth + (_st.textPosition.x() - _st.diameter)) { | 		if (_st.width < _st.textPosition.x() + _textWidth + (_st.textPosition.x() - _st.diameter)) { | ||||||
| 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.textPosition.x() + (_st.textPosition.x() - _st.diameter)), 1.)); | 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.textPosition.x() + (_st.textPosition.x() - _st.diameter)), 1)); | ||||||
| 			_textWidth = _st.font->width(_text); | 			_textWidth = _st.font->width(_text); | ||||||
| 		} | 		} | ||||||
| 		resize(_st.width, _st.height); | 		resize(_st.width, _st.height); | ||||||
|  | @ -392,7 +392,7 @@ Radiobutton::Radiobutton(QWidget *parent, const QString &group, int32 value, con | ||||||
| 		resize(_textWidth - _st.width, _st.height); | 		resize(_textWidth - _st.width, _st.height); | ||||||
| 	} else { | 	} else { | ||||||
| 		if (_st.width < _st.textPosition.x() + _textWidth + (_st.textPosition.x() - _st.diameter)) { | 		if (_st.width < _st.textPosition.x() + _textWidth + (_st.textPosition.x() - _st.diameter)) { | ||||||
| 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.textPosition.x() + (_st.textPosition.x() - _st.diameter)), 1.)); | 			_text = _st.font->elided(_fullText, qMax(_st.width - (_st.textPosition.x() + (_st.textPosition.x() - _st.diameter)), 1)); | ||||||
| 			_textWidth = _st.font->width(_text); | 			_textWidth = _st.font->width(_text); | ||||||
| 		} | 		} | ||||||
| 		resize(_st.width, _st.height); | 		resize(_st.width, _st.height); | ||||||
|  |  | ||||||
|  | @ -21,5 +21,5 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include "ui/style_core.h" | #include "ui/style_core.h" | ||||||
| #include "GeneratedFiles/style_classes.h" | #include "GeneratedFiles/styles/style_basic_types.h" | ||||||
| #include "GeneratedFiles/style_auto.h" | #include "GeneratedFiles/styles/style_basic.h" | ||||||
|  |  | ||||||
|  | @ -20,13 +20,63 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org | ||||||
| */ | */ | ||||||
| #include "stdafx.h" | #include "stdafx.h" | ||||||
| 
 | 
 | ||||||
|  | namespace style { | ||||||
| namespace { | namespace { | ||||||
| 	typedef QMap<QString, uint32> FontFamilyMap; | using ModulesList = QList<internal::ModuleBase*>; | ||||||
| 	FontFamilyMap _fontFamilyMap; | NeverFreedPointer<ModulesList> styleModules; | ||||||
|  | 
 | ||||||
|  | typedef QMap<QString, int> FontFamilyMap; | ||||||
|  | FontFamilyMap fontFamilyMap; | ||||||
|  | 
 | ||||||
|  | typedef QVector<QString> FontFamilies; | ||||||
|  | FontFamilies _fontFamilies; | ||||||
|  | 
 | ||||||
|  | typedef QMap<uint32, FontData*> FontDatas; | ||||||
|  | FontDatas fontsMap; | ||||||
|  | 
 | ||||||
|  | typedef QMap<uint32, ColorData*> ColorDatas; | ||||||
|  | ColorDatas colorsMap; | ||||||
|  | 
 | ||||||
|  | int spriteWidthValue = 0; | ||||||
|  | QPixmap *spriteData = nullptr; | ||||||
|  | 
 | ||||||
|  | inline uint32 fontKey(int size, uint32 flags, int family) { | ||||||
|  | 	return (((uint32(family) << 10) | uint32(size)) << 3) | flags; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| namespace style { | } // namespace
 | ||||||
| 	FontData::FontData(uint32 size, uint32 flags, uint32 family, Font *other) : f(_fontFamilies[family]), m(f), _size(size), _flags(flags), _family(family) { | 
 | ||||||
|  | namespace internal { | ||||||
|  | 
 | ||||||
|  | void registerModule(ModuleBase *module) { | ||||||
|  | 	styleModules.makeIfNull(); | ||||||
|  | 	styleModules->push_back(module); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void unregisterModule(ModuleBase *module) { | ||||||
|  | 	styleModules->removeOne(module); | ||||||
|  | 	if (styleModules->isEmpty()) { | ||||||
|  | 		styleModules.clear(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int registerFontFamily(const QString &family) { | ||||||
|  | 	auto result = fontFamilyMap.value(family, -1); | ||||||
|  | 	if (result < 0) { | ||||||
|  | 		result = _fontFamilies.size(); | ||||||
|  | 		fontFamilyMap.insert(family, result); | ||||||
|  | 		_fontFamilies.push_back(family); | ||||||
|  | 	} | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int spriteWidth() { | ||||||
|  | 	return spriteWidthValue; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace internal
 | ||||||
|  | 
 | ||||||
|  | 	FontData::FontData(int size, uint32 flags, int family, Font *other) : f(_fontFamilies[family]), m(f), _size(size), _flags(flags), _family(family) { | ||||||
| 		if (other) { | 		if (other) { | ||||||
| 			memcpy(modified, other, sizeof(modified)); | 			memcpy(modified, other, sizeof(modified)); | ||||||
| 		} else { | 		} else { | ||||||
|  | @ -60,7 +110,7 @@ namespace style { | ||||||
| 		return otherFlagsFont(FontUnderline, set); | 		return otherFlagsFont(FontUnderline, set); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	uint32 FontData::size() const { | 	int FontData::size() const { | ||||||
| 		return _size; | 		return _size; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -68,7 +118,7 @@ namespace style { | ||||||
| 		return _flags; | 		return _flags; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	uint32 FontData::family() const { | 	int FontData::family() const { | ||||||
| 		return _family; | 		return _family; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -80,34 +130,34 @@ namespace style { | ||||||
| 		return modified[newFlags]; | 		return modified[newFlags]; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	Font::Font(uint32 size, uint32 flags, const QString &family) { | 	Font::Font(int size, uint32 flags, const QString &family) { | ||||||
| 		if (_fontFamilyMap.isEmpty()) { | 		if (fontFamilyMap.isEmpty()) { | ||||||
| 			for (uint32 i = 0, s = style::_fontFamilies.size(); i != s; ++i) { | 			for (uint32 i = 0, s = style::_fontFamilies.size(); i != s; ++i) { | ||||||
| 				_fontFamilyMap.insert(style::_fontFamilies.at(i), i); | 				fontFamilyMap.insert(style::_fontFamilies.at(i), i); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		FontFamilyMap::const_iterator i = _fontFamilyMap.constFind(family); | 		auto i = fontFamilyMap.constFind(family); | ||||||
| 		if (i == _fontFamilyMap.cend()) { | 		if (i == fontFamilyMap.cend()) { | ||||||
| 			style::_fontFamilies.push_back(family); | 			style::_fontFamilies.push_back(family); | ||||||
| 			i = _fontFamilyMap.insert(family, style::_fontFamilies.size() - 1); | 			i = fontFamilyMap.insert(family, style::_fontFamilies.size() - 1); | ||||||
| 		} | 		} | ||||||
| 		init(size, flags, i.value(), 0); | 		init(size, flags, i.value(), 0); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	Font::Font(uint32 size, uint32 flags, uint32 family) { | 	Font::Font(int size, uint32 flags, int family) { | ||||||
| 		init(size, flags, family, 0); | 		init(size, flags, family, 0); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	Font::Font(uint32 size, uint32 flags, uint32 family, Font *modified) { | 	Font::Font(int size, uint32 flags, int family, Font *modified) { | ||||||
| 		init(size, flags, family, modified); | 		init(size, flags, family, modified); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	void Font::init(uint32 size, uint32 flags, uint32 family, Font *modified) { | 	void Font::init(int size, uint32 flags, int family, Font *modified) { | ||||||
| 		uint32 key = _fontKey(size, flags, family); | 		uint32 key = fontKey(size, flags, family); | ||||||
| 		FontDatas::const_iterator i = _fontsMap.constFind(key); | 		auto i = fontsMap.constFind(key); | ||||||
| 		if (i == _fontsMap.cend()) { | 		if (i == fontsMap.cend()) { | ||||||
| 			i = _fontsMap.insert(key, new FontData(size, flags, family, modified)); | 			i = fontsMap.insert(key, new FontData(size, flags, family, modified)); | ||||||
| 		} | 		} | ||||||
| 		ptr = i.value(); | 		ptr = i.value(); | ||||||
| 	} | 	} | ||||||
|  | @ -141,9 +191,9 @@ namespace style { | ||||||
| 	} | 	} | ||||||
| 	void Color::init(uchar r, uchar g, uchar b, uchar a) { | 	void Color::init(uchar r, uchar g, uchar b, uchar a) { | ||||||
| 		uint32 key = colorKey(r, g, b, a); | 		uint32 key = colorKey(r, g, b, a); | ||||||
| 		ColorDatas::const_iterator i = _colorsMap.constFind(key); | 		auto i = colorsMap.constFind(key); | ||||||
| 		if (i == _colorsMap.cend()) { | 		if (i == colorsMap.cend()) { | ||||||
| 			i = _colorsMap.insert(key, new ColorData(r, g, b, a)); | 			i = colorsMap.insert(key, new ColorData(r, g, b, a)); | ||||||
| 		} | 		} | ||||||
| 		ptr = i.value(); | 		ptr = i.value(); | ||||||
| 	} | 	} | ||||||
|  | @ -163,16 +213,57 @@ namespace style { | ||||||
| 		b = QBrush(color); | 		b = QBrush(color); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	void stopManager() { | 	void startManager() { | ||||||
| 		for (FontDatas::const_iterator i = _fontsMap.cbegin(), e = _fontsMap.cend(); i != e; ++i) { | 		if (cRetina()) { | ||||||
| 			delete i.value(); | 			cSetRealScale(dbisOne); | ||||||
| 		} | 		} | ||||||
| 		_fontsMap.clear(); |  | ||||||
| 
 | 
 | ||||||
| 		for (ColorDatas::const_iterator i = _colorsMap.cbegin(), e = _colorsMap.cend(); i != e; ++i) { | 		internal::registerFontFamily(qsl("Open Sans")); | ||||||
| 			delete i.value(); | 		QString spriteFilePostfix; | ||||||
|  | 		if (cRetina() || cScale() == dbisTwo) { | ||||||
|  | 			spriteFilePostfix = qsl("_200x"); | ||||||
|  | 		} else if (cScale() == dbisOneAndQuarter) { | ||||||
|  | 			spriteFilePostfix = qsl("_125x"); | ||||||
|  | 		} else if (cScale() == dbisOneAndHalf) { | ||||||
|  | 			spriteFilePostfix = qsl("_150x"); | ||||||
| 		} | 		} | ||||||
| 		_colorsMap.clear(); | 		QString spriteFile = qsl(":/gui/art/sprite") + spriteFilePostfix + qsl(".png"); | ||||||
|  | 		if (rtl()) { | ||||||
|  | 			spriteData = new QPixmap(QPixmap::fromImage(QImage(spriteFile).mirrored(true, false))); | ||||||
|  | 		} else { | ||||||
|  | 			spriteData = new QPixmap(spriteFile); | ||||||
|  | 		} | ||||||
|  | 		if (cRetina()) spriteData->setDevicePixelRatio(cRetinaFactor()); | ||||||
|  | 		spriteWidthValue = spriteData->width(); | ||||||
|  | 
 | ||||||
|  | 		if (styleModules) { | ||||||
|  | 			for_const (auto module, *styleModules) { | ||||||
|  | 				module->start(); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		_fontFamilies.push_back(qsl("Open Sans")); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	void stopManager() { | ||||||
|  | 		if (styleModules) { | ||||||
|  | 			for_const (auto module, *styleModules) { | ||||||
|  | 				module->stop(); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for (auto fontData : fontsMap) { | ||||||
|  | 			delete fontData; | ||||||
|  | 		} | ||||||
|  | 		fontsMap.clear(); | ||||||
|  | 
 | ||||||
|  | 		for (auto colorData : colorsMap) { | ||||||
|  | 			delete colorData; | ||||||
|  | 		} | ||||||
|  | 		colorsMap.clear(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	const QPixmap &spritePixmap() { | ||||||
|  | 		return *spriteData; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -46,14 +46,31 @@ inline QRect centerrect(const QRect &inRect, const QRect &rect) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| namespace style { | namespace style { | ||||||
|  | namespace internal { | ||||||
|  | 
 | ||||||
|  | // Objects of derived classes are created in global scope.
 | ||||||
|  | // They call [un]registerModule() in [de|con]structor.
 | ||||||
|  | class ModuleBase { | ||||||
|  | public: | ||||||
|  | 	virtual void start() = 0; | ||||||
|  | 	virtual void stop() = 0; | ||||||
|  | }; | ||||||
|  | void registerModule(ModuleBase *module); | ||||||
|  | void unregisterModule(ModuleBase *module); | ||||||
|  | 
 | ||||||
|  | int registerFontFamily(const QString &family); | ||||||
|  | 
 | ||||||
|  | int spriteWidth(); | ||||||
|  | 
 | ||||||
|  | } // namespace internal
 | ||||||
| 
 | 
 | ||||||
| 	class FontData; | 	class FontData; | ||||||
| 	class Font { | 	class Font { | ||||||
| 	public: | 	public: | ||||||
| 		Font(Qt::Initialization = Qt::Uninitialized) : ptr(0) { | 		Font(Qt::Initialization = Qt::Uninitialized) : ptr(0) { | ||||||
| 		} | 		} | ||||||
| 		Font(uint32 size, uint32 flags, const QString &family); | 		Font(int size, uint32 flags, const QString &family); | ||||||
| 		Font(uint32 size, uint32 flags = 0, uint32 family = 0); | 		Font(int size, uint32 flags = 0, int family = 0); | ||||||
| 
 | 
 | ||||||
| 		Font &operator=(const Font &other) { | 		Font &operator=(const Font &other) { | ||||||
| 			ptr = other.ptr; | 			ptr = other.ptr; | ||||||
|  | @ -76,36 +93,24 @@ namespace style { | ||||||
| 	private: | 	private: | ||||||
| 		FontData *ptr; | 		FontData *ptr; | ||||||
| 
 | 
 | ||||||
| 		void init(uint32 size, uint32 flags, uint32 family, Font *modified); | 		void init(int size, uint32 flags, int family, Font *modified); | ||||||
| 		friend void startManager(); | 		friend void startManager(); | ||||||
| 
 | 
 | ||||||
| 		Font(FontData *p) : ptr(p) { | 		Font(FontData *p) : ptr(p) { | ||||||
| 		} | 		} | ||||||
| 		Font(uint32 size, uint32 flags, uint32 family, Font *modified); | 		Font(int size, uint32 flags, int family, Font *modified); | ||||||
| 		friend class FontData; | 		friend class FontData; | ||||||
| 
 | 
 | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	enum FontFlagBits { |  | ||||||
| 		FontBoldBit, |  | ||||||
| 		FontItalicBit, |  | ||||||
| 		FontUnderlineBit, |  | ||||||
| 
 |  | ||||||
| 		FontFlagsBits |  | ||||||
| 	}; |  | ||||||
| 
 |  | ||||||
| 	enum FontFlags { | 	enum FontFlags { | ||||||
| 		FontBold = (1 << FontBoldBit), | 		FontBold           = 0x01, | ||||||
| 		FontItalic = (1 << FontItalicBit), | 		FontItalic         = 0x02, | ||||||
| 		FontUnderline = (1 << FontUnderlineBit), | 		FontUnderline      = 0x04, | ||||||
| 
 | 
 | ||||||
| 		FontDifferentFlags = (1 << FontFlagsBits) | 		FontDifferentFlags = 0x08, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	inline uint32 _fontKey(uint32 size, uint32 flags, uint32 family) { |  | ||||||
| 		return (((family << 10) | size) << FontFlagsBits) | flags; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	class FontData { | 	class FontData { | ||||||
| 	public: | 	public: | ||||||
| 
 | 
 | ||||||
|  | @ -126,9 +131,9 @@ namespace style { | ||||||
| 		Font italic(bool set = true) const; | 		Font italic(bool set = true) const; | ||||||
| 		Font underline(bool set = true) const; | 		Font underline(bool set = true) const; | ||||||
| 
 | 
 | ||||||
| 		uint32 size() const; | 		int size() const; | ||||||
| 		uint32 flags() const; | 		uint32 flags() const; | ||||||
| 		uint32 family() const; | 		int family() const; | ||||||
| 
 | 
 | ||||||
| 		QFont f; | 		QFont f; | ||||||
| 		QFontMetrics m; | 		QFontMetrics m; | ||||||
|  | @ -138,10 +143,12 @@ namespace style { | ||||||
| 		mutable Font modified[FontDifferentFlags]; | 		mutable Font modified[FontDifferentFlags]; | ||||||
| 
 | 
 | ||||||
| 		Font otherFlagsFont(uint32 flag, bool set) const; | 		Font otherFlagsFont(uint32 flag, bool set) const; | ||||||
| 		FontData(uint32 size, uint32 flags, uint32 family, Font *other); | 		FontData(int size, uint32 flags, int family, Font *other); | ||||||
| 
 | 
 | ||||||
| 		friend class Font; | 		friend class Font; | ||||||
| 		uint32 _size, _flags, _family; | 		int _size; | ||||||
|  | 		uint32 _flags; | ||||||
|  | 		int _family; | ||||||
| 
 | 
 | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | @ -159,7 +166,7 @@ namespace style { | ||||||
| 	class ColorData; | 	class ColorData; | ||||||
| 	class Color { | 	class Color { | ||||||
| 	public: | 	public: | ||||||
| 		Color(Qt::Initialization = Qt::Uninitialized) : ptr(0), owner(false) { | 		Color(Qt::Initialization = Qt::Uninitialized) { | ||||||
| 		} | 		} | ||||||
| 		Color(const Color &c); | 		Color(const Color &c); | ||||||
| 		Color(const QColor &c); | 		Color(const QColor &c); | ||||||
|  | @ -185,8 +192,8 @@ namespace style { | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 	private: | 	private: | ||||||
| 		ColorData *ptr; | 		ColorData *ptr = nullptr; | ||||||
| 		bool owner; | 		bool owner = false; | ||||||
| 
 | 
 | ||||||
| 		void init(uchar r, uchar g, uchar b, uchar a); | 		void init(uchar r, uchar g, uchar b, uchar a); | ||||||
| 
 | 
 | ||||||
|  | @ -233,17 +240,6 @@ namespace style { | ||||||
| 		return ptr->p; | 		return ptr->p; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	typedef QVector<QString> FontFamilies; |  | ||||||
| 	extern FontFamilies _fontFamilies; |  | ||||||
| 
 |  | ||||||
| 	typedef QMap<uint32, FontData*> FontDatas; |  | ||||||
| 	extern FontDatas _fontsMap; |  | ||||||
| 
 |  | ||||||
| 	typedef QMap<uint32, ColorData*> ColorDatas; |  | ||||||
| 	extern ColorDatas _colorsMap; |  | ||||||
| 
 |  | ||||||
| 	extern int _spriteWidth; |  | ||||||
| 
 |  | ||||||
| 	typedef float64 number; | 	typedef float64 number; | ||||||
| 	typedef QString string; | 	typedef QString string; | ||||||
| 	typedef QRect rect; | 	typedef QRect rect; | ||||||
|  | @ -252,7 +248,7 @@ namespace style { | ||||||
|     public: |     public: | ||||||
|         sprite() { |         sprite() { | ||||||
|         } |         } | ||||||
| 		sprite(int left, int top, int width, int height) : rect(rtl() ? (_spriteWidth - left - width) : left, top, width, height) { | 		sprite(int left, int top, int width, int height) : rect(rtl() ? (internal::spriteWidth() - left - width) : left, top, width, height) { | ||||||
|         } |         } | ||||||
|         inline int pxWidth() const { |         inline int pxWidth() const { | ||||||
|             return rect::width() / cIntRetinaFactor(); |             return rect::width() / cIntRetinaFactor(); | ||||||
|  | @ -314,8 +310,9 @@ namespace style { | ||||||
| 
 | 
 | ||||||
| 	void startManager(); | 	void startManager(); | ||||||
| 	void stopManager(); | 	void stopManager(); | ||||||
|  | 	const QPixmap &spritePixmap(); | ||||||
| 
 | 
 | ||||||
| }; | } // namespace style
 | ||||||
| 
 | 
 | ||||||
| inline QRect centersprite(const QRect &inRect, const style::sprite &sprite) { | inline QRect centersprite(const QRect &inRect, const style::sprite &sprite) { | ||||||
| 	return centerrect(inRect, QRect(QPoint(0, 0), sprite.pxSize())); | 	return centerrect(inRect, QRect(QPoint(0, 0), sprite.pxSize())); | ||||||
|  |  | ||||||
|  | @ -1051,7 +1051,13 @@ | ||||||
|       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> | ||||||
|       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> | ||||||
|     </ClCompile> |     </ClCompile> | ||||||
|     <ClCompile Include="GeneratedFiles\style_auto.cpp" /> |     <ClCompile Include="GeneratedFiles\styles\style_basic.cpp" /> | ||||||
|  |     <ClCompile Include="GeneratedFiles\styles\style_basic_types.cpp" /> | ||||||
|  |     <ClCompile Include="GeneratedFiles\style_auto.cpp"> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild> | ||||||
|  |     </ClCompile> | ||||||
|     <ClCompile Include="SourceFiles\apiwrap.cpp" /> |     <ClCompile Include="SourceFiles\apiwrap.cpp" /> | ||||||
|     <ClCompile Include="SourceFiles\app.cpp" /> |     <ClCompile Include="SourceFiles\app.cpp" /> | ||||||
|     <ClCompile Include="SourceFiles\application.cpp" /> |     <ClCompile Include="SourceFiles\application.cpp" /> | ||||||
|  | @ -1245,6 +1251,8 @@ | ||||||
|       <Outputs Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">.\GeneratedFiles\$(ConfigurationName)\moc_%(Filename).cpp</Outputs> |       <Outputs Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">.\GeneratedFiles\$(ConfigurationName)\moc_%(Filename).cpp</Outputs> | ||||||
|       <Command Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">"$(QTDIR)\bin\moc.exe"  "%(FullPath)" -o ".\GeneratedFiles\$(ConfigurationName)\moc_%(Filename).cpp" "-fstdafx.h" "-f../../SourceFiles/core/basic_types.h"  -DAL_LIBTYPE_STATIC -DUNICODE -DWIN32 -DWIN64 -DHAVE_STDINT_H -DZLIB_WINAPI -DQT_NO_DEBUG -DNDEBUG -D_SCL_SECURE_NO_WARNINGS "-I.\..\..\Libraries\lzma\C" "-I.\..\..\Libraries\libexif-0.6.20" "-I.\..\..\Libraries\zlib-1.2.8" "-I.\..\..\Libraries\openssl\Release\include" "-I.\..\..\Libraries\ffmpeg" "-I.\..\..\Libraries\openal-soft\include" "-I.\SourceFiles" "-I.\GeneratedFiles" "-I.\..\..\Libraries\breakpad\src" "-I.\ThirdParty\minizip" "-I." "-I$(QTDIR)\include" "-I.\GeneratedFiles\$(ConfigurationName)\." "-I.\..\..\Libraries\QtStatic\qtbase\include\QtCore\5.5.1\QtCore" "-I.\..\..\Libraries\QtStatic\qtbase\include\QtGui\5.5.1\QtGui"</Command> |       <Command Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">"$(QTDIR)\bin\moc.exe"  "%(FullPath)" -o ".\GeneratedFiles\$(ConfigurationName)\moc_%(Filename).cpp" "-fstdafx.h" "-f../../SourceFiles/core/basic_types.h"  -DAL_LIBTYPE_STATIC -DUNICODE -DWIN32 -DWIN64 -DHAVE_STDINT_H -DZLIB_WINAPI -DQT_NO_DEBUG -DNDEBUG -D_SCL_SECURE_NO_WARNINGS "-I.\..\..\Libraries\lzma\C" "-I.\..\..\Libraries\libexif-0.6.20" "-I.\..\..\Libraries\zlib-1.2.8" "-I.\..\..\Libraries\openssl\Release\include" "-I.\..\..\Libraries\ffmpeg" "-I.\..\..\Libraries\openal-soft\include" "-I.\SourceFiles" "-I.\GeneratedFiles" "-I.\..\..\Libraries\breakpad\src" "-I.\ThirdParty\minizip" "-I." "-I$(QTDIR)\include" "-I.\GeneratedFiles\$(ConfigurationName)\." "-I.\..\..\Libraries\QtStatic\qtbase\include\QtCore\5.5.1\QtCore" "-I.\..\..\Libraries\QtStatic\qtbase\include\QtGui\5.5.1\QtGui"</Command> | ||||||
|     </CustomBuild> |     </CustomBuild> | ||||||
|  |     <ClInclude Include="GeneratedFiles\styles\style_basic.h" /> | ||||||
|  |     <ClInclude Include="GeneratedFiles\styles\style_basic_types.h" /> | ||||||
|     <ClInclude Include="SourceFiles\core\click_handler.h" /> |     <ClInclude Include="SourceFiles\core\click_handler.h" /> | ||||||
|     <ClInclude Include="SourceFiles\core\click_handler_types.h" /> |     <ClInclude Include="SourceFiles\core\click_handler_types.h" /> | ||||||
|     <ClInclude Include="SourceFiles\dialogs\dialogs_common.h" /> |     <ClInclude Include="SourceFiles\dialogs\dialogs_common.h" /> | ||||||
|  | @ -1608,8 +1616,16 @@ | ||||||
|       <AdditionalInputs Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">$(QTDIR)\bin\moc.exe;%(FullPath)</AdditionalInputs> |       <AdditionalInputs Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">$(QTDIR)\bin\moc.exe;%(FullPath)</AdditionalInputs> | ||||||
|     </CustomBuild> |     </CustomBuild> | ||||||
|     <ClInclude Include="GeneratedFiles\lang_auto.h" /> |     <ClInclude Include="GeneratedFiles\lang_auto.h" /> | ||||||
|     <ClInclude Include="GeneratedFiles\style_auto.h" /> |     <ClInclude Include="GeneratedFiles\style_auto.h"> | ||||||
|     <ClInclude Include="GeneratedFiles\style_classes.h" /> |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="GeneratedFiles\style_classes.h"> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild> | ||||||
|  |     </ClInclude> | ||||||
|     <ClInclude Include="resource.h" /> |     <ClInclude Include="resource.h" /> | ||||||
|     <CustomBuild Include="SourceFiles\apiwrap.h"> |     <CustomBuild Include="SourceFiles\apiwrap.h"> | ||||||
|       <AdditionalInputs Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">$(QTDIR)\bin\moc.exe;%(FullPath)</AdditionalInputs> |       <AdditionalInputs Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">$(QTDIR)\bin\moc.exe;%(FullPath)</AdditionalInputs> | ||||||
|  | @ -2379,7 +2395,13 @@ | ||||||
|     <None Include="Version" /> |     <None Include="Version" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <CodegenStyleItem Include="Resources\all_files.style" /> |     <CodegenStyleItem Include="Resources\all_files.style"> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Deploy|Win32'">true</ExcludedFromBuild> | ||||||
|  |       <ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild> | ||||||
|  |     </CodegenStyleItem> | ||||||
|  |     <CodegenStyleItem Include="Resources\basic.style" /> | ||||||
|  |     <CodegenStyleItem Include="Resources\basic_types.style" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" /> |   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" /> | ||||||
|   <ImportGroup Label="ExtensionTargets"> |   <ImportGroup Label="ExtensionTargets"> | ||||||
|  |  | ||||||
|  | @ -82,6 +82,9 @@ | ||||||
|       <Extensions>cpp;moc</Extensions> |       <Extensions>cpp;moc</Extensions> | ||||||
|       <SourceControlFiles>False</SourceControlFiles> |       <SourceControlFiles>False</SourceControlFiles> | ||||||
|     </Filter> |     </Filter> | ||||||
|  |     <Filter Include="GeneratedFiles\styles"> | ||||||
|  |       <UniqueIdentifier>{3397dfda-79f1-4a4c-940b-be26b001baf5}</UniqueIdentifier> | ||||||
|  |     </Filter> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ClCompile Include="SourceFiles\main.cpp"> |     <ClCompile Include="SourceFiles\main.cpp"> | ||||||
|  | @ -1077,6 +1080,12 @@ | ||||||
|     <ClCompile Include="SourceFiles\ui\text\text_block.cpp"> |     <ClCompile Include="SourceFiles\ui\text\text_block.cpp"> | ||||||
|       <Filter>SourceFiles\ui\text</Filter> |       <Filter>SourceFiles\ui\text</Filter> | ||||||
|     </ClCompile> |     </ClCompile> | ||||||
|  |     <ClCompile Include="GeneratedFiles\styles\style_basic.cpp"> | ||||||
|  |       <Filter>GeneratedFiles\styles</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="GeneratedFiles\styles\style_basic_types.cpp"> | ||||||
|  |       <Filter>GeneratedFiles\styles</Filter> | ||||||
|  |     </ClCompile> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ClInclude Include="SourceFiles\stdafx.h"> |     <ClInclude Include="SourceFiles\stdafx.h"> | ||||||
|  | @ -1235,6 +1244,12 @@ | ||||||
|     <ClInclude Include="SourceFiles\ui\text\text_block.h"> |     <ClInclude Include="SourceFiles\ui\text\text_block.h"> | ||||||
|       <Filter>SourceFiles\ui\text</Filter> |       <Filter>SourceFiles\ui\text</Filter> | ||||||
|     </ClInclude> |     </ClInclude> | ||||||
|  |     <ClInclude Include="GeneratedFiles\styles\style_basic.h"> | ||||||
|  |       <Filter>GeneratedFiles\styles</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="GeneratedFiles\styles\style_basic_types.h"> | ||||||
|  |       <Filter>GeneratedFiles\styles</Filter> | ||||||
|  |     </ClInclude> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <CustomBuild Include="SourceFiles\application.h"> |     <CustomBuild Include="SourceFiles\application.h"> | ||||||
|  | @ -1539,5 +1554,11 @@ | ||||||
|     <CodegenStyleItem Include="Resources\all_files.style"> |     <CodegenStyleItem Include="Resources\all_files.style"> | ||||||
|       <Filter>Resources</Filter> |       <Filter>Resources</Filter> | ||||||
|     </CodegenStyleItem> |     </CodegenStyleItem> | ||||||
|  |     <CodegenStyleItem Include="Resources\basic.style"> | ||||||
|  |       <Filter>Resources</Filter> | ||||||
|  |     </CodegenStyleItem> | ||||||
|  |     <CodegenStyleItem Include="Resources\basic_types.style"> | ||||||
|  |       <Filter>Resources</Filter> | ||||||
|  |     </CodegenStyleItem> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
| </Project> | </Project> | ||||||
|  | @ -0,0 +1,120 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> | ||||||
|  |   <ItemGroup Label="ProjectConfigurations"> | ||||||
|  |     <ProjectConfiguration Include="Debug|Win32"> | ||||||
|  |       <Configuration>Debug</Configuration> | ||||||
|  |       <Platform>Win32</Platform> | ||||||
|  |     </ProjectConfiguration> | ||||||
|  |     <ProjectConfiguration Include="Release|Win32"> | ||||||
|  |       <Configuration>Release</Configuration> | ||||||
|  |       <Platform>Win32</Platform> | ||||||
|  |     </ProjectConfiguration> | ||||||
|  |   </ItemGroup> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\cpp_file.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\generator.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\main.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\options.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\parsed_file.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\processor.cpp" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file_reader.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\const_utf8_string.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\cpp_file.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\logging.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\generator.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\options.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\parsed_file.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\processor.h" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |   <PropertyGroup Label="Globals"> | ||||||
|  |     <ProjectGuid>{7C25BFBD-7930-4DE2-AF33-27CE1CC521E6}</ProjectGuid> | ||||||
|  |     <Keyword>Qt4VSv1.0</Keyword> | ||||||
|  |     <WindowsTargetPlatformVersion>8.1</WindowsTargetPlatformVersion> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" /> | ||||||
|  |   <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration"> | ||||||
|  |     <ConfigurationType>Application</ConfigurationType> | ||||||
|  |     <PlatformToolset>v140</PlatformToolset> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration"> | ||||||
|  |     <ConfigurationType>Application</ConfigurationType> | ||||||
|  |     <PlatformToolset>v140</PlatformToolset> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" /> | ||||||
|  |   <ImportGroup Label="ExtensionSettings"> | ||||||
|  |   </ImportGroup> | ||||||
|  |   <ImportGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="PropertySheets"> | ||||||
|  |     <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" /> | ||||||
|  |   </ImportGroup> | ||||||
|  |   <ImportGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="PropertySheets"> | ||||||
|  |     <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" /> | ||||||
|  |   </ImportGroup> | ||||||
|  |   <PropertyGroup Label="UserMacros" /> | ||||||
|  |   <PropertyGroup> | ||||||
|  |     <_ProjectFileVersion>14.0.24730.2</_ProjectFileVersion> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'"> | ||||||
|  |     <OutDir>$(SolutionDir)$(Platform)\codegen\$(Configuration)\</OutDir> | ||||||
|  |     <IntDir>$(SolutionDir)$(Platform)\obj\$(ProjectName)\$(Configuration)\</IntDir> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'"> | ||||||
|  |     <OutDir>$(SolutionDir)$(Platform)\codegen\$(Configuration)\</OutDir> | ||||||
|  |     <IntDir>$(SolutionDir)$(Platform)\obj\$(ProjectName)\$(Configuration)\</IntDir> | ||||||
|  |   </PropertyGroup> | ||||||
|  |   <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'"> | ||||||
|  |     <ClCompile> | ||||||
|  |       <PreprocessorDefinitions>UNICODE;WIN32;QT_CORE_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions> | ||||||
|  |       <Optimization>Disabled</Optimization> | ||||||
|  |       <DebugInformationFormat>ProgramDatabase</DebugInformationFormat> | ||||||
|  |       <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary> | ||||||
|  |       <AdditionalIncludeDirectories>.;$(QTDIR)\include;.\GeneratedFiles\$(ConfigurationName);.\..\..\..\SourceFiles;$(QTDIR)\include\QtCore;.\..\%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> | ||||||
|  |       <TreatWChar_tAsBuiltInType>true</TreatWChar_tAsBuiltInType> | ||||||
|  |       <WarningLevel>Level4</WarningLevel> | ||||||
|  |       <TreatWarningAsError>true</TreatWarningAsError> | ||||||
|  |       <AdditionalOptions>/w44062 /wd4127 %(AdditionalOptions)</AdditionalOptions> | ||||||
|  |     </ClCompile> | ||||||
|  |     <Link> | ||||||
|  |       <SubSystem>Console</SubSystem> | ||||||
|  |       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> | ||||||
|  |       <AdditionalLibraryDirectories>$(QTDIR)\lib;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> | ||||||
|  |       <GenerateDebugInformation>true</GenerateDebugInformation> | ||||||
|  |       <AdditionalDependencies>ws2_32.lib;qtmaind.lib;qtharfbuzzngd.lib;qtpcred.lib;qtfreetyped.lib;Qt5Cored.lib;%(AdditionalDependencies)</AdditionalDependencies> | ||||||
|  |     </Link> | ||||||
|  |   </ItemDefinitionGroup> | ||||||
|  |   <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'"> | ||||||
|  |     <ClCompile> | ||||||
|  |       <PreprocessorDefinitions>UNICODE;WIN32;QT_NO_DEBUG;NDEBUG;QT_CORE_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions> | ||||||
|  |       <DebugInformationFormat /> | ||||||
|  |       <RuntimeLibrary>MultiThreaded</RuntimeLibrary> | ||||||
|  |       <AdditionalIncludeDirectories>.;$(QTDIR)\include;.\GeneratedFiles\$(ConfigurationName);.\..\..\..\SourceFiles;$(QTDIR)\include\QtCore;.\..\%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> | ||||||
|  |       <TreatWChar_tAsBuiltInType>true</TreatWChar_tAsBuiltInType> | ||||||
|  |       <WarningLevel>Level4</WarningLevel> | ||||||
|  |       <TreatWarningAsError>true</TreatWarningAsError> | ||||||
|  |       <AdditionalOptions>/w44062 /wd4127 %(AdditionalOptions)</AdditionalOptions> | ||||||
|  |     </ClCompile> | ||||||
|  |     <Link> | ||||||
|  |       <SubSystem>Console</SubSystem> | ||||||
|  |       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> | ||||||
|  |       <AdditionalLibraryDirectories>$(QTDIR)\lib;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> | ||||||
|  |       <GenerateDebugInformation>false</GenerateDebugInformation> | ||||||
|  |       <AdditionalDependencies>ws2_32.lib;qtmain.lib;qtharfbuzzng.lib;qtpcre.lib;qtfreetype.lib;Qt5Core.lib;%(AdditionalDependencies)</AdditionalDependencies> | ||||||
|  |     </Link> | ||||||
|  |   </ItemDefinitionGroup> | ||||||
|  |   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" /> | ||||||
|  |   <ImportGroup Label="ExtensionTargets"> | ||||||
|  |   </ImportGroup> | ||||||
|  |   <ProjectExtensions> | ||||||
|  |     <VisualStudio> | ||||||
|  |       <UserProperties MocDir=".\GeneratedFiles\$(ConfigurationName)" UicDir=".\GeneratedFiles" RccDir=".\GeneratedFiles" lupdateOptions="" lupdateOnBuild="0" lreleaseOptions="" Qt5Version_x0020_Win32="QtStatic" MocOptions="" /> | ||||||
|  |     </VisualStudio> | ||||||
|  |   </ProjectExtensions> | ||||||
|  | </Project> | ||||||
|  | @ -0,0 +1,82 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <Filter Include="src"> | ||||||
|  |       <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier> | ||||||
|  |       <Extensions>cpp;cxx;c;def</Extensions> | ||||||
|  |     </Filter> | ||||||
|  |     <Filter Include="src\common"> | ||||||
|  |       <UniqueIdentifier>{77524fac-2d55-437d-810f-c271326c7f95}</UniqueIdentifier> | ||||||
|  |     </Filter> | ||||||
|  |     <Filter Include="src\numbers"> | ||||||
|  |       <UniqueIdentifier>{48ba9931-338f-4476-8d83-600d0ac4440e}</UniqueIdentifier> | ||||||
|  |     </Filter> | ||||||
|  |   </ItemGroup> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\cpp_file.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\main.cpp"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\options.cpp"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\processor.cpp"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\parsed_file.cpp"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\numbers\generator.cpp"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |   </ItemGroup> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file_reader.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\const_utf8_string.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\cpp_file.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\logging.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\options.h"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\processor.h"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\parsed_file.h"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\numbers\generator.h"> | ||||||
|  |       <Filter>src\numbers</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |   </ItemGroup> | ||||||
|  | </Project> | ||||||
|  | @ -15,6 +15,6 @@ | ||||||
|     Inputs="%(CodegenStyleItem.Identity)" |     Inputs="%(CodegenStyleItem.Identity)" | ||||||
|     Outputs=".\GeneratedFiles\styles\style_%(CodegenStyleItem.Filename).h;.\GeneratedFiles\styles\style_%(CodegenStyleItem.Filename).cpp"> |     Outputs=".\GeneratedFiles\styles\style_%(CodegenStyleItem.Filename).h;.\GeneratedFiles\styles\style_%(CodegenStyleItem.Filename).cpp"> | ||||||
|   <Message Text="Generating style: %(CodegenStyleItem.Identity)" /> |   <Message Text="Generating style: %(CodegenStyleItem.Identity)" /> | ||||||
|   <Exec Command="$(SolutionDir)$(Platform)\codegen\$(Configuration)\codegen_style.exe %(CodegenStyleItem.Identity)"/> |   <Exec Command="$(SolutionDir)$(Platform)\codegen\$(Configuration)\codegen_style.exe "-I.\SourceFiles" "-o.\GeneratedFiles\styles" %(CodegenStyleItem.Identity)"/> | ||||||
|  </Target> |  </Target> | ||||||
| </Project> | </Project> | ||||||
|  |  | ||||||
|  | @ -14,11 +14,15 @@ | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\cpp_file.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\generator.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\generator.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\processor.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\module.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\options.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\options.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\parsed_file.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\parsed_file.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\main.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\main.cpp" /> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\sprite_generator.cpp" /> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\structure_types.cpp" /> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\structure_types.cpp" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|  | @ -27,11 +31,14 @@ | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\basic_tokenized_file.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\clean_file.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\common\const_utf8_string.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\const_utf8_string.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\cpp_file.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\common\logging.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\logging.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\generator.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\generator.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\processor.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\options.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\options.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\module.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\parsed_file.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\parsed_file.h" /> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\sprite_generator.h" /> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure_types.h" /> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure_types.h" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <PropertyGroup Label="Globals"> |   <PropertyGroup Label="Globals"> | ||||||
|  | @ -86,7 +93,7 @@ | ||||||
|       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> |       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> | ||||||
|       <AdditionalLibraryDirectories>$(QTDIR)\lib;$(QTDIR)\plugins;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> |       <AdditionalLibraryDirectories>$(QTDIR)\lib;$(QTDIR)\plugins;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> | ||||||
|       <GenerateDebugInformation>true</GenerateDebugInformation> |       <GenerateDebugInformation>true</GenerateDebugInformation> | ||||||
|       <AdditionalDependencies>ws2_32.lib;qtmaind.lib;qtharfbuzzngd.lib;qtpcred.lib;qtfreetyped.lib;imageformats\qwebpd.lib;Qt5Cored.lib;%(AdditionalDependencies)</AdditionalDependencies> |       <AdditionalDependencies>ws2_32.lib;qtmaind.lib;qtharfbuzzngd.lib;qtpcred.lib;qtfreetyped.lib;imageformats\qwebpd.lib;Qt5Cored.lib;Qt5Guid.lib;%(AdditionalDependencies)</AdditionalDependencies> | ||||||
|     </Link> |     </Link> | ||||||
|   </ItemDefinitionGroup> |   </ItemDefinitionGroup> | ||||||
|   <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'"> |   <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'"> | ||||||
|  | @ -94,7 +101,7 @@ | ||||||
|       <PreprocessorDefinitions>UNICODE;WIN32;QT_NO_DEBUG;NDEBUG;QT_CORE_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions> |       <PreprocessorDefinitions>UNICODE;WIN32;QT_NO_DEBUG;NDEBUG;QT_CORE_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions> | ||||||
|       <DebugInformationFormat /> |       <DebugInformationFormat /> | ||||||
|       <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary> |       <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary> | ||||||
|       <AdditionalIncludeDirectories>.;$(QTDIR)\include;.\GeneratedFiles\$(ConfigurationName);$(QTDIR)\include\QtCore;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> |       <AdditionalIncludeDirectories>.;$(QTDIR)\include;.\GeneratedFiles\$(ConfigurationName);.\..\..\..\SourceFiles;$(QTDIR)\include\QtCore;.\..\%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> | ||||||
|       <TreatWChar_tAsBuiltInType>true</TreatWChar_tAsBuiltInType> |       <TreatWChar_tAsBuiltInType>true</TreatWChar_tAsBuiltInType> | ||||||
|       <WarningLevel>Level4</WarningLevel> |       <WarningLevel>Level4</WarningLevel> | ||||||
|       <TreatWarningAsError>true</TreatWarningAsError> |       <TreatWarningAsError>true</TreatWarningAsError> | ||||||
|  | @ -105,7 +112,7 @@ | ||||||
|       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> |       <OutputFile>$(OutDir)\$(ProjectName).exe</OutputFile> | ||||||
|       <AdditionalLibraryDirectories>$(QTDIR)\lib;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> |       <AdditionalLibraryDirectories>$(QTDIR)\lib;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories> | ||||||
|       <GenerateDebugInformation>false</GenerateDebugInformation> |       <GenerateDebugInformation>false</GenerateDebugInformation> | ||||||
|       <AdditionalDependencies>qtmain.lib;Qt5Core.lib;%(AdditionalDependencies)</AdditionalDependencies> |       <AdditionalDependencies>ws2_32.lib;qtmain.lib;qtharfbuzzng.lib;qtpcre.lib;qtfreetype.lib;imageformats\qwebp.lib;Qt5Core.lib;Qt5Gui.lib;%(AdditionalDependencies)</AdditionalDependencies> | ||||||
|     </Link> |     </Link> | ||||||
|   </ItemDefinitionGroup> |   </ItemDefinitionGroup> | ||||||
|   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" /> |   <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" /> | ||||||
|  |  | ||||||
|  | @ -22,9 +22,6 @@ | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp"> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\logging.cpp"> | ||||||
|       <Filter>src\common</Filter> |       <Filter>src\common</Filter> | ||||||
|     </ClCompile> |     </ClCompile> | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\generator.cpp"> |  | ||||||
|       <Filter>src\style</Filter> |  | ||||||
|     </ClCompile> |  | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp"> |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\clean_file.cpp"> | ||||||
|       <Filter>src\common</Filter> |       <Filter>src\common</Filter> | ||||||
|     </ClCompile> |     </ClCompile> | ||||||
|  | @ -40,11 +37,23 @@ | ||||||
|     <ClCompile Include="..\..\..\SourceFiles\codegen\style\structure_types.cpp"> |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\structure_types.cpp"> | ||||||
|       <Filter>src\style</Filter> |       <Filter>src\style</Filter> | ||||||
|     </ClCompile> |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\module.cpp"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\common\cpp_file.cpp"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\processor.cpp"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\generator.cpp"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClCompile> | ||||||
|  |     <ClCompile Include="..\..\..\SourceFiles\codegen\style\sprite_generator.cpp"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClCompile> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\generator.h"> |  | ||||||
|       <Filter>src\style</Filter> |  | ||||||
|     </ClInclude> |  | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.h"> |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\checked_utf8_string.h"> | ||||||
|       <Filter>src\common</Filter> |       <Filter>src\common</Filter> | ||||||
|     </ClInclude> |     </ClInclude> | ||||||
|  | @ -66,14 +75,26 @@ | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\options.h"> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\options.h"> | ||||||
|       <Filter>src\style</Filter> |       <Filter>src\style</Filter> | ||||||
|     </ClInclude> |     </ClInclude> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure.h"> |  | ||||||
|       <Filter>src\style</Filter> |  | ||||||
|     </ClInclude> |  | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\parsed_file.h"> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\parsed_file.h"> | ||||||
|       <Filter>src\style</Filter> |       <Filter>src\style</Filter> | ||||||
|     </ClInclude> |     </ClInclude> | ||||||
|     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure_types.h"> |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\structure_types.h"> | ||||||
|       <Filter>src\style</Filter> |       <Filter>src\style</Filter> | ||||||
|     </ClInclude> |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\module.h"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\common\cpp_file.h"> | ||||||
|  |       <Filter>src\common</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\processor.h"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\generator.h"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClInclude> | ||||||
|  |     <ClInclude Include="..\..\..\SourceFiles\codegen\style\sprite_generator.h"> | ||||||
|  |       <Filter>src\style</Filter> | ||||||
|  |     </ClInclude> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
| </Project> | </Project> | ||||||
		Loading…
	
		Reference in New Issue