mirror of https://github.com/procxx/kepka.git
				
				
				
			Improve new settings design.
This commit is contained in:
		
							parent
							
								
									ffc4cd3415
								
							
						
					
					
						commit
						c2039da600
					
				
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 408 B | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 1.2 KiB | 
|  | @ -267,7 +267,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| "lng_bio_placeholder" = "Bio"; | "lng_bio_placeholder" = "Bio"; | ||||||
| "lng_bio_about" = "You can add a few lines about yourself. Anyone who opens your profile will see this text."; | "lng_bio_about" = "You can add a few lines about yourself. Anyone who opens your profile will see this text."; | ||||||
| 
 | 
 | ||||||
| "lng_settings_section_info" = "Info"; | "lng_settings_section_info" = "My info"; | ||||||
| "lng_settings_username" = "Username:"; | "lng_settings_username" = "Username:"; | ||||||
| "lng_settings_choose_username" = "Choose username"; | "lng_settings_choose_username" = "Choose username"; | ||||||
| "lng_settings_empty_bio" = "None"; | "lng_settings_empty_bio" = "None"; | ||||||
|  | @ -353,6 +353,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| "lng_settings_calls_title" = "Calls"; | "lng_settings_calls_title" = "Calls"; | ||||||
| "lng_settings_peer_to_peer" = "Peer-to-Peer"; | "lng_settings_peer_to_peer" = "Peer-to-Peer"; | ||||||
| "lng_settings_peer_to_peer_about" = "Disabling peer-to-peer will relay all calls through Telegram servers to avoid revealing your IP address, but may slightly decrease audio quality."; | "lng_settings_peer_to_peer_about" = "Disabling peer-to-peer will relay all calls through Telegram servers to avoid revealing your IP address, but may slightly decrease audio quality."; | ||||||
|  | "lng_settings_advanced" = "Advanced"; | ||||||
|  | "lng_settings_stickers_emoji" = "Stickers and emoji"; | ||||||
|  | "lng_settings_themes" = "Themes"; | ||||||
|  | "lng_settings_chat_other" = "Other"; | ||||||
| 
 | 
 | ||||||
| "lng_backgrounds_header" = "Choose your new chat background"; | "lng_backgrounds_header" = "Choose your new chat background"; | ||||||
| "lng_theme_sure_keep" = "Keep this theme?"; | "lng_theme_sure_keep" = "Keep this theme?"; | ||||||
|  |  | ||||||
|  | @ -192,7 +192,7 @@ void LocalStorageBox::Row::step_radial(TimeMs ms, bool timer) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| rpl::producer<> LocalStorageBox::Row::clearRequests() const { | rpl::producer<> LocalStorageBox::Row::clearRequests() const { | ||||||
| 	return _clear->clicks(); | 	return _clear->clicks() | rpl::map([] { return rpl::empty_value(); }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int LocalStorageBox::Row::resizeGetHeight(int newWidth) { | int LocalStorageBox::Row::resizeGetHeight(int newWidth) { | ||||||
|  |  | ||||||
|  | @ -266,7 +266,7 @@ ProgressWidget::ProgressWidget( | ||||||
| 
 | 
 | ||||||
| rpl::producer<> ProgressWidget::cancelClicks() const { | rpl::producer<> ProgressWidget::cancelClicks() const { | ||||||
| 	return _cancel | 	return _cancel | ||||||
| 		? _cancel->clicks() | 		? (_cancel->clicks() | rpl::map([] { return rpl::empty_value(); })) | ||||||
| 		: (rpl::never<>() | rpl::type_erased()); | 		: (rpl::never<>() | rpl::type_erased()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -322,7 +322,10 @@ void ProgressWidget::showDone() { | ||||||
| 	if (_done->width() < desired) { | 	if (_done->width() < desired) { | ||||||
| 		_done->setFullWidth(desired); | 		_done->setFullWidth(desired); | ||||||
| 	} | 	} | ||||||
| 	_done->clicks() | rpl::start_to_stream(_doneClicks, _done->lifetime()); | 	_done->clicks( | ||||||
|  | 	) | rpl::map([] { | ||||||
|  | 		return rpl::empty_value(); | ||||||
|  | 	}) | rpl::start_to_stream(_doneClicks, _done->lifetime()); | ||||||
| 	setupBottomButton(_done.get()); | 	setupBottomButton(_done.get()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -524,7 +524,10 @@ void SettingsWidget::refreshButtons( | ||||||
| 		: nullptr; | 		: nullptr; | ||||||
| 	if (start) { | 	if (start) { | ||||||
| 		start->show(); | 		start->show(); | ||||||
| 		_startClicks = start->clicks(); | 		_startClicks = start->clicks( | ||||||
|  | 		) | rpl::map([] { | ||||||
|  | 			return rpl::empty_value(); | ||||||
|  | 		}); | ||||||
| 
 | 
 | ||||||
| 		container->sizeValue( | 		container->sizeValue( | ||||||
| 		) | rpl::start_with_next([=](QSize size) { | 		) | rpl::start_with_next([=](QSize size) { | ||||||
|  | @ -539,7 +542,10 @@ void SettingsWidget::refreshButtons( | ||||||
| 		langFactory(lng_cancel), | 		langFactory(lng_cancel), | ||||||
| 		st::defaultBoxButton); | 		st::defaultBoxButton); | ||||||
| 	cancel->show(); | 	cancel->show(); | ||||||
| 	_cancelClicks = cancel->clicks(); | 	_cancelClicks = cancel->clicks( | ||||||
|  | 	) | rpl::map([] { | ||||||
|  | 		return rpl::empty_value(); | ||||||
|  | 	}); | ||||||
| 
 | 
 | ||||||
| 	rpl::combine( | 	rpl::combine( | ||||||
| 		container->sizeValue(), | 		container->sizeValue(), | ||||||
|  |  | ||||||
|  | @ -30,7 +30,7 @@ TopBar::TopBar(QWidget *parent, Content &&content) | ||||||
| 	updateData(std::move(content)); | 	updateData(std::move(content)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| rpl::producer<> TopBar::clicks() const { | rpl::producer<Qt::MouseButton> TopBar::clicks() const { | ||||||
| 	return _button->clicks(); | 	return _button->clicks(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ class TopBar : public Ui::RpWidget { | ||||||
| public: | public: | ||||||
| 	TopBar(QWidget *parent, Content &&content); | 	TopBar(QWidget *parent, Content &&content); | ||||||
| 
 | 
 | ||||||
| 	rpl::producer<> clicks() const; | 	rpl::producer<Qt::MouseButton> clicks() const; | ||||||
| 
 | 
 | ||||||
| 	void updateData(Content &&content); | 	void updateData(Content &&content); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -50,7 +50,7 @@ public: | ||||||
| 
 | 
 | ||||||
| 	rpl::producer<int> scrollHeightValue() const; | 	rpl::producer<int> scrollHeightValue() const; | ||||||
| 	rpl::producer<int> desiredHeightValue() const override; | 	rpl::producer<int> desiredHeightValue() const override; | ||||||
| 	rpl::producer<bool> desiredShadowVisibility() const; | 	virtual rpl::producer<bool> desiredShadowVisibility() const; | ||||||
| 	bool hasTopBarShadow() const; | 	bool hasTopBarShadow() const; | ||||||
| 
 | 
 | ||||||
| 	virtual void setInnerFocus(); | 	virtual void setInnerFocus(); | ||||||
|  |  | ||||||
|  | @ -99,8 +99,10 @@ void TopBar::enableBackButton() { | ||||||
| 		st::infoTopBarScale); | 		st::infoTopBarScale); | ||||||
| 	_back->setDuration(st::infoTopBarDuration); | 	_back->setDuration(st::infoTopBarDuration); | ||||||
| 	_back->toggle(!selectionMode(), anim::type::instant); | 	_back->toggle(!selectionMode(), anim::type::instant); | ||||||
| 	_back->entity()->clicks() | 	_back->entity()->clicks( | ||||||
| 		| rpl::start_to_stream(_backClicks, _back->lifetime()); | 	) | rpl::map([] { | ||||||
|  | 		return rpl::empty_value(); | ||||||
|  | 	}) | rpl::start_to_stream(_backClicks, _back->lifetime()); | ||||||
| 	registerToggleControlCallback(_back.data(), [=] { | 	registerToggleControlCallback(_back.data(), [=] { | ||||||
| 		return !selectionMode(); | 		return !selectionMode(); | ||||||
| 	}); | 	}); | ||||||
|  | @ -434,7 +436,9 @@ void TopBar::createSelectionControls() { | ||||||
| 		st::infoTopBarScale)); | 		st::infoTopBarScale)); | ||||||
| 	_cancelSelection->setDuration(st::infoTopBarDuration); | 	_cancelSelection->setDuration(st::infoTopBarDuration); | ||||||
| 	_cancelSelection->entity()->clicks( | 	_cancelSelection->entity()->clicks( | ||||||
| 	) | rpl::start_to_stream( | 	) | rpl::map([] { | ||||||
|  | 		return rpl::empty_value(); | ||||||
|  | 	}) | rpl::start_to_stream( | ||||||
| 		_cancelSelectionClicks, | 		_cancelSelectionClicks, | ||||||
| 		_cancelSelection->lifetime()); | 		_cancelSelection->lifetime()); | ||||||
| 	_selectionText = wrap(Ui::CreateChild<Ui::FadeWrap<Ui::LabelWithNumbers>>( | 	_selectionText = wrap(Ui::CreateChild<Ui::FadeWrap<Ui::LabelWithNumbers>>( | ||||||
|  | @ -605,7 +609,7 @@ rpl::producer<QString> TitleValue( | ||||||
| 			case Section::SettingsType::PrivacySecurity: | 			case Section::SettingsType::PrivacySecurity: | ||||||
| 				return lng_settings_section_privacy; | 				return lng_settings_section_privacy; | ||||||
| 			case Section::SettingsType::General: | 			case Section::SettingsType::General: | ||||||
| 				return lng_settings_section_general; | 				return lng_settings_advanced; | ||||||
| 			case Section::SettingsType::Chat: | 			case Section::SettingsType::Chat: | ||||||
| 				return lng_settings_section_chat_settings; | 				return lng_settings_section_chat_settings; | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | @ -71,7 +71,7 @@ void Button::paintEvent(QPaintEvent *e) { | ||||||
| 	Painter p(this); | 	Painter p(this); | ||||||
| 
 | 
 | ||||||
| 	auto ms = getms(); | 	auto ms = getms(); | ||||||
| 	auto paintOver = (isOver() || isDown()); | 	auto paintOver = (isOver() || isDown()) && !isDisabled(); | ||||||
| 	p.fillRect(e->rect(), paintOver ? _st.textBgOver : _st.textBg); | 	p.fillRect(e->rect(), paintOver ? _st.textBgOver : _st.textBg); | ||||||
| 
 | 
 | ||||||
| 	paintRipple(p, 0, 0, ms); | 	paintRipple(p, 0, 0, ms); | ||||||
|  | @ -113,10 +113,13 @@ int Button::resizeGetHeight(int newWidth) { | ||||||
| void Button::onStateChanged( | void Button::onStateChanged( | ||||||
| 		State was, | 		State was, | ||||||
| 		StateChangeSource source) { | 		StateChangeSource source) { | ||||||
| 	RippleButton::onStateChanged(was, source); | 	if (!isDisabled() || !isDown()) { | ||||||
|  | 		RippleButton::onStateChanged(was, source); | ||||||
|  | 	} | ||||||
| 	if (_toggle) { | 	if (_toggle) { | ||||||
| 		_toggle->setStyle(isOver() ? _st.toggleOver : _st.toggle); | 		_toggle->setStyle(isOver() ? _st.toggleOver : _st.toggle); | ||||||
| 	} | 	} | ||||||
|  | 	setPointerCursor(!isDisabled()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Button::setText(QString &&text) { | void Button::setText(QString &&text) { | ||||||
|  |  | ||||||
|  | @ -294,6 +294,8 @@ void Cover::initViewers() { | ||||||
| 		) | rpl::start_with_next( | 		) | rpl::start_with_next( | ||||||
| 			[this] { refreshUploadPhotoOverlay(); }, | 			[this] { refreshUploadPhotoOverlay(); }, | ||||||
| 			lifetime()); | 			lifetime()); | ||||||
|  | 	} else if (_peer->isSelf()) { | ||||||
|  | 		refreshUploadPhotoOverlay(); | ||||||
| 	} | 	} | ||||||
| 	VerifiedValue( | 	VerifiedValue( | ||||||
| 		_peer | 		_peer | ||||||
|  | @ -304,12 +306,12 @@ void Cover::initViewers() { | ||||||
| 
 | 
 | ||||||
| void Cover::refreshUploadPhotoOverlay() { | void Cover::refreshUploadPhotoOverlay() { | ||||||
| 	_userpic->switchChangePhotoOverlay([&] { | 	_userpic->switchChangePhotoOverlay([&] { | ||||||
| 		if (auto chat = _peer->asChat()) { | 		if (const auto chat = _peer->asChat()) { | ||||||
| 			return chat->canEdit(); | 			return chat->canEdit(); | ||||||
| 		} else if (auto channel = _peer->asChannel()) { | 		} else if (const auto channel = _peer->asChannel()) { | ||||||
| 			return channel->canEditInformation(); | 			return channel->canEditInformation(); | ||||||
| 		} | 		} | ||||||
| 		return false; | 		return _peer->isSelf(); | ||||||
| 	}()); | 	}()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -86,6 +86,12 @@ void Widget::saveChanges(FnMut<void()> done) { | ||||||
| 	_inner->sectionSaveChanges(std::move(done)); | 	_inner->sectionSaveChanges(std::move(done)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | rpl::producer<bool> Widget::desiredShadowVisibility() const { | ||||||
|  | 	return (_type == Type::Main || _type == Type::Information) | ||||||
|  | 		? ContentWidget::desiredShadowVisibility() | ||||||
|  | 		: rpl::single(true); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| std::unique_ptr<ContentMemento> Widget::doCreateMemento() { | std::unique_ptr<ContentMemento> Widget::doCreateMemento() { | ||||||
| 	auto result = std::make_unique<Memento>(self(), _type); | 	auto result = std::make_unique<Memento>(self(), _type); | ||||||
| 	saveState(result.get()); | 	saveState(result.get()); | ||||||
|  |  | ||||||
|  | @ -66,6 +66,8 @@ public: | ||||||
| 	rpl::producer<bool> canSaveChanges() const override; | 	rpl::producer<bool> canSaveChanges() const override; | ||||||
| 	void saveChanges(FnMut<void()> done) override; | 	void saveChanges(FnMut<void()> done) override; | ||||||
| 
 | 
 | ||||||
|  | 	rpl::producer<bool> desiredShadowVisibility() const override; | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
| 	void saveState(not_null<Memento*> memento); | 	void saveState(not_null<Memento*> memento); | ||||||
| 	void restoreState(not_null<Memento*> memento); | 	void restoreState(not_null<Memento*> memento); | ||||||
|  |  | ||||||
|  | @ -81,10 +81,12 @@ public: | ||||||
| 	void setError(bool error); | 	void setError(bool error); | ||||||
| 
 | 
 | ||||||
| 	rpl::producer<> deleteClicks() const { | 	rpl::producer<> deleteClicks() const { | ||||||
| 		return _delete->entity()->clicks(); | 		return _delete->entity()->clicks( | ||||||
|  | 		) | rpl::map([] { return rpl::empty_value(); }); | ||||||
| 	} | 	} | ||||||
| 	rpl::producer<> restoreClicks() const { | 	rpl::producer<> restoreClicks() const { | ||||||
| 		return _restore->entity()->clicks(); | 		return _restore->entity()->clicks( | ||||||
|  | 		) | rpl::map([] { return rpl::empty_value(); }); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
|  |  | ||||||
|  | @ -10,7 +10,6 @@ using "ui/widgets/widgets.style"; | ||||||
| using "info/info.style"; | using "info/info.style"; | ||||||
| using "boxes/boxes.style"; | using "boxes/boxes.style"; | ||||||
| 
 | 
 | ||||||
| settingsFirstDividerSkip: 3px; |  | ||||||
| settingsSectionButton: InfoProfileButton(infoProfileButton) { | settingsSectionButton: InfoProfileButton(infoProfileButton) { | ||||||
| 	font: boxTextFont; | 	font: boxTextFont; | ||||||
| 	padding: margins(79px, 13px, 22px, 11px); | 	padding: margins(79px, 13px, 22px, 11px); | ||||||
|  | @ -22,7 +21,7 @@ settingsButton: InfoProfileButton(settingsSectionButton) { | ||||||
| 	padding: margins(22px, 13px, 22px, 11px); | 	padding: margins(22px, 13px, 22px, 11px); | ||||||
| } | } | ||||||
| settingsSectionSkip: 9px; | settingsSectionSkip: 9px; | ||||||
| settingsSectionIconPosition: point(22px, 9px); | settingsSectionIconLeft: 22px; | ||||||
| settingsSeparatorPadding: margins(22px, infoProfileSkip, 0px, infoProfileSkip); | settingsSeparatorPadding: margins(22px, infoProfileSkip, 0px, infoProfileSkip); | ||||||
| settingsButtonRightPosition: point(28px, 13px); | settingsButtonRightPosition: point(28px, 13px); | ||||||
| settingsButtonRight: FlatLabel(defaultFlatLabel) { | settingsButtonRight: FlatLabel(defaultFlatLabel) { | ||||||
|  | @ -48,7 +47,6 @@ settingsUpdate: InfoProfileButton(infoMainButton, settingsButton) { | ||||||
| settingsUpdateStatePosition: point(24px, 29px); | settingsUpdateStatePosition: point(24px, 29px); | ||||||
| settingsDividerLabelPadding: margins(22px, 10px, 22px, 19px); | settingsDividerLabelPadding: margins(22px, 10px, 22px, 19px); | ||||||
| 
 | 
 | ||||||
| settingsIconSetPhoto: icon {{ "settings_set_photo", menuIconFg }}; |  | ||||||
| settingsIconInformation: icon {{ "settings_information", menuIconFg }}; | settingsIconInformation: icon {{ "settings_information", menuIconFg }}; | ||||||
| settingsIconNotifications: icon {{ "settings_notifications", menuIconFg }}; | settingsIconNotifications: icon {{ "settings_notifications", menuIconFg }}; | ||||||
| settingsIconChat: icon {{ "settings_chat", menuIconFg }}; | settingsIconChat: icon {{ "settings_chat", menuIconFg }}; | ||||||
|  | @ -68,6 +66,7 @@ settingsLink: boxLinkButton; | ||||||
| settingsAdvancedNotificationsPadding: margins(22px, 20px, 10px, 10px); | settingsAdvancedNotificationsPadding: margins(22px, 20px, 10px, 10px); | ||||||
| settingsLinkLabel: defaultFlatLabel; | settingsLinkLabel: defaultFlatLabel; | ||||||
| settingsCheckboxesSkip: 12px; | settingsCheckboxesSkip: 12px; | ||||||
|  | settingsStickersEmojiPadding: 17px; | ||||||
| 
 | 
 | ||||||
| settingsSendType: settingsCheckbox; | settingsSendType: settingsCheckbox; | ||||||
| settingsSendTypePadding: margins(22px, 5px, 10px, 5px); | settingsSendTypePadding: margins(22px, 5px, 10px, 5px); | ||||||
|  | @ -100,7 +99,20 @@ settingsCloudPasswordLabel: FlatLabel(defaultFlatLabel) { | ||||||
| } | } | ||||||
| settingsCloudPasswordLabelPadding: margins(22px, 8px, 10px, 8px); | settingsCloudPasswordLabelPadding: margins(22px, 8px, 10px, 8px); | ||||||
| 
 | 
 | ||||||
| settingsInfoRowHeight: 62px; | settingsInfoPhotoHeight: 175px; | ||||||
|  | settingsInfoPhotoSize: 84px; | ||||||
|  | settingsInfoPhoto: UserpicButton(defaultUserpicButton) { | ||||||
|  | 	size: size(settingsInfoPhotoSize, settingsInfoPhotoSize); | ||||||
|  | 	photoSize: settingsInfoPhotoSize; | ||||||
|  | } | ||||||
|  | settingsInfoPhotoTop: 17px; | ||||||
|  | settingsInfoPhotoSkip: 16px; | ||||||
|  | settingsInfoPhotoSet: defaultActiveButton; | ||||||
|  | 
 | ||||||
|  | settingsInfoRow: InfoProfileButton(settingsButton) { | ||||||
|  | 	height: 62px; | ||||||
|  | 	padding: margins(0px, 0px, 0px, 0px); | ||||||
|  | } | ||||||
| settingsInfoIconPosition: point(22px, 18px); | settingsInfoIconPosition: point(22px, 18px); | ||||||
| settingsInfoValue: FlatLabel(defaultFlatLabel) { | settingsInfoValue: FlatLabel(defaultFlatLabel) { | ||||||
| 	textFg: windowFg; | 	textFg: windowFg; | ||||||
|  | @ -114,24 +126,11 @@ settingsInfoAbout: FlatLabel(settingsInfoValue) { | ||||||
| } | } | ||||||
| settingsInfoAboutPosition: point(78px, 34px); | settingsInfoAboutPosition: point(78px, 34px); | ||||||
| settingsInfoRightSkip: 60px; | settingsInfoRightSkip: 60px; | ||||||
| settingsInfoEditPosition: point(0px, 6px); | settingsInfoEditRight: 14px; | ||||||
| settingsInfoEdit: IconButton(defaultIconButton) { | settingsInfoEditIconOver: icon {{ "settings_edit", menuIconFgOver }}; | ||||||
| 	width: 56px; |  | ||||||
| 	height: 56px; |  | ||||||
| 
 |  | ||||||
| 	icon: icon {{ "settings_edit", menuIconFg }}; |  | ||||||
| 	iconOver: icon {{ "settings_edit", menuIconFgOver }}; |  | ||||||
| 	iconPosition: point(14px, 14px); |  | ||||||
| 
 |  | ||||||
| 	rippleAreaPosition: point(8px, 8px); |  | ||||||
| 	rippleAreaSize: 40px; |  | ||||||
| 	ripple: RippleAnimation(defaultRippleAnimation) { |  | ||||||
| 		color: windowBgOver; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| settingsBio: InputField(defaultInputField) { | settingsBio: InputField(defaultInputField) { | ||||||
| 	textBg: transparent; | 	textBg: transparent; | ||||||
| 	textMargins: margins(2px, 7px, 2px, 0px); | 	textMargins: margins(0px, 7px, 0px, 13px); | ||||||
| 
 | 
 | ||||||
| 	placeholderFg: placeholderFg; | 	placeholderFg: placeholderFg; | ||||||
| 	placeholderFgActive: placeholderFgActive; | 	placeholderFgActive: placeholderFgActive; | ||||||
|  | @ -140,9 +139,6 @@ settingsBio: InputField(defaultInputField) { | ||||||
| 	placeholderScale: 0.; | 	placeholderScale: 0.; | ||||||
| 	placeholderFont: normalFont; | 	placeholderFont: normalFont; | ||||||
| 
 | 
 | ||||||
| 	border: 0px; |  | ||||||
| 	borderActive: 0px; |  | ||||||
| 
 |  | ||||||
| 	heightMin: 32px; | 	heightMin: 32px; | ||||||
| 
 | 
 | ||||||
| 	font: boxTextFont; | 	font: boxTextFont; | ||||||
|  | @ -157,3 +153,4 @@ settingsBioCountdown: FlatLabel(defaultFlatLabel) { | ||||||
| 	style: boxTextStyle; | 	style: boxTextStyle; | ||||||
| 	textFg: windowSubTextFg; | 	textFg: windowSubTextFg; | ||||||
| } | } | ||||||
|  | settingsBioLabelPadding: margins(22px, 11px, 22px, 0px); | ||||||
|  |  | ||||||
|  | @ -369,9 +369,10 @@ void DownloadPathRow::setupControls() { | ||||||
| 
 | 
 | ||||||
| #endif // OS_WIN_STORE
 | #endif // OS_WIN_STORE
 | ||||||
| 
 | 
 | ||||||
| void SetupChatOptions(not_null<Ui::VerticalLayout*> container) { | void SetupStickersEmoji(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); | 	AddSkip(container, st::settingsStickersEmojiPadding); | ||||||
| 	AddSkip(container, st::settingsCheckboxesSkip); | 
 | ||||||
|  | 	AddSubsectionTitle(container, lng_settings_stickers_emoji); | ||||||
| 
 | 
 | ||||||
| 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | ||||||
| 	const auto inner = wrap.data(); | 	const auto inner = wrap.data(); | ||||||
|  | @ -422,65 +423,31 @@ void SetupChatOptions(not_null<Ui::VerticalLayout*> container) { | ||||||
| 			Local::writeUserSettings(); | 			Local::writeUserSettings(); | ||||||
| 		}); | 		}); | ||||||
| 
 | 
 | ||||||
| 	const auto dontask = inner->add( | 	AddButton( | ||||||
| 		checkbox( | 		container, | ||||||
| 			lng_download_path_dont_ask, | 		lng_stickers_you_have, | ||||||
| 			!Global::AskDownloadPath()), | 		st::settingsButton | ||||||
| #ifndef OS_WIN_STORE | 	)->addClickHandler([] { | ||||||
| 		st::settingsAskPathPadding); | 		Ui::show(Box<StickersBox>(StickersBox::Section::Installed)); | ||||||
| #else // OS_WIN_STORE
 | 	}); | ||||||
| 		st::settingsCheckboxPadding); |  | ||||||
| #endif // OS_WIN_STORE
 |  | ||||||
| 
 |  | ||||||
| #ifndef OS_WIN_STORE |  | ||||||
| 	const auto showpath = Ui::AttachAsChild( |  | ||||||
| 		dontask, |  | ||||||
| 		rpl::event_stream<bool>()); |  | ||||||
| 	const auto padding = st::settingsDownloadPathPadding; |  | ||||||
| 	const auto path = container->add( |  | ||||||
| 		object_ptr<Ui::SlideWrap<DownloadPathRow>>( |  | ||||||
| 			container, |  | ||||||
| 			object_ptr<DownloadPathRow>(container), |  | ||||||
| 			QMargins( |  | ||||||
| 				(padding.left() |  | ||||||
| 					+ st::settingsCheckbox.checkPosition.x() |  | ||||||
| 					+ st::defaultCheck.diameter |  | ||||||
| 					+ st::settingsCheckbox.textPosition.x() |  | ||||||
| 					- st::settingsCheckbox.margin.left()), |  | ||||||
| 				padding.top(), |  | ||||||
| 				padding.right(), |  | ||||||
| 				padding.bottom()))); |  | ||||||
| 	AddSkip(container, st::settingsCheckboxPadding.bottom()); |  | ||||||
| 	path->toggleOn( |  | ||||||
| 		showpath->events_starting_with(!Global::AskDownloadPath())); |  | ||||||
| #endif // OS_WIN_STORE
 |  | ||||||
| 
 |  | ||||||
| 	base::ObservableViewer( |  | ||||||
| 		dontask->checkedChanged |  | ||||||
| 	) | rpl::start_with_next([=](bool checked) { |  | ||||||
| 		Global::SetAskDownloadPath(!checked); |  | ||||||
| 		Local::writeUserSettings(); |  | ||||||
| 
 |  | ||||||
| #ifndef OS_WIN_STORE |  | ||||||
| 		showpath->fire_copy(checked); |  | ||||||
| #endif // OS_WIN_STORE
 |  | ||||||
| 
 |  | ||||||
| 	}, inner->lifetime()); |  | ||||||
| 
 | 
 | ||||||
| 	AddSkip(container, st::settingsCheckboxesSkip); | 	AddSkip(container, st::settingsCheckboxesSkip); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupSendKey(not_null<Ui::VerticalLayout*> container) { | void SetupChatOther(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); | 	AddDivider(container); | ||||||
| 	const auto skip = st::settingsSendTypeSkip; | 	AddSkip(container, st::settingsSectionSkip); | ||||||
| 	const auto full = st::settingsCheckboxesSkip + skip; | 
 | ||||||
| 	AddSkip(container, full); | 	AddSubsectionTitle(container, lng_settings_chat_other); | ||||||
|  | 
 | ||||||
|  | 	AddSkip(container, st::settingsSendTypeSkip); | ||||||
| 
 | 
 | ||||||
| 	enum class SendByType { | 	enum class SendByType { | ||||||
| 		Enter, | 		Enter, | ||||||
| 		CtrlEnter, | 		CtrlEnter, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | 	const auto skip = st::settingsSendTypeSkip; | ||||||
| 	const auto group = std::make_shared<Ui::RadioenumGroup<SendByType>>( | 	const auto group = std::make_shared<Ui::RadioenumGroup<SendByType>>( | ||||||
| 		cCtrlEnter() ? SendByType::CtrlEnter : SendByType::Enter); | 		cCtrlEnter() ? SendByType::CtrlEnter : SendByType::Enter); | ||||||
| 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | ||||||
|  | @ -491,8 +458,6 @@ void SetupSendKey(not_null<Ui::VerticalLayout*> container) { | ||||||
| 			std::move(wrap), | 			std::move(wrap), | ||||||
| 			QMargins(0, skip, 0, skip))); | 			QMargins(0, skip, 0, skip))); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(container, full); |  | ||||||
| 
 |  | ||||||
| 	const auto add = [&]( | 	const auto add = [&]( | ||||||
| 			SendByType value, | 			SendByType value, | ||||||
| 			LangKey key, | 			LangKey key, | ||||||
|  | @ -525,11 +490,55 @@ void SetupSendKey(not_null<Ui::VerticalLayout*> container) { | ||||||
| 		} | 		} | ||||||
| 		Local::writeUserSettings(); | 		Local::writeUserSettings(); | ||||||
| 	}); | 	}); | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| void SetupMediaOptions(not_null<Ui::VerticalLayout*> container) { | 	AddSkip(inner, st::settingsCheckboxesSkip); | ||||||
| 	AddDivider(container); | 
 | ||||||
| 	AddSkip(container); | 	const auto dontask = inner->add( | ||||||
|  | 		object_ptr<Ui::Checkbox>( | ||||||
|  | 			inner, | ||||||
|  | 			lang(lng_download_path_dont_ask), | ||||||
|  | 			!Global::AskDownloadPath(), | ||||||
|  | 			st::settingsCheckbox), | ||||||
|  | #ifndef OS_WIN_STORE | ||||||
|  | 		st::settingsAskPathPadding); | ||||||
|  | #else // OS_WIN_STORE
 | ||||||
|  | 		st::settingsCheckboxPadding); | ||||||
|  | #endif // OS_WIN_STORE
 | ||||||
|  | 
 | ||||||
|  | #ifndef OS_WIN_STORE | ||||||
|  | 	const auto showpath = Ui::AttachAsChild( | ||||||
|  | 		dontask, | ||||||
|  | 		rpl::event_stream<bool>()); | ||||||
|  | 	const auto padding = st::settingsDownloadPathPadding; | ||||||
|  | 	const auto path = container->add( | ||||||
|  | 		object_ptr<Ui::SlideWrap<DownloadPathRow>>( | ||||||
|  | 			container, | ||||||
|  | 			object_ptr<DownloadPathRow>(container), | ||||||
|  | 			QMargins( | ||||||
|  | 			(padding.left() | ||||||
|  | 				+ st::settingsCheckbox.checkPosition.x() | ||||||
|  | 				+ st::defaultCheck.diameter | ||||||
|  | 				+ st::settingsCheckbox.textPosition.x() | ||||||
|  | 				- st::settingsCheckbox.margin.left()), | ||||||
|  | 				padding.top(), | ||||||
|  | 				padding.right(), | ||||||
|  | 				padding.bottom()))); | ||||||
|  | 	AddSkip(container, st::settingsCheckboxPadding.bottom()); | ||||||
|  | 	path->toggleOn( | ||||||
|  | 		showpath->events_starting_with(!Global::AskDownloadPath())); | ||||||
|  | #endif // OS_WIN_STORE
 | ||||||
|  | 
 | ||||||
|  | 	base::ObservableViewer( | ||||||
|  | 		dontask->checkedChanged | ||||||
|  | 	) | rpl::start_with_next([=](bool checked) { | ||||||
|  | 		Global::SetAskDownloadPath(!checked); | ||||||
|  | 		Local::writeUserSettings(); | ||||||
|  | 
 | ||||||
|  | #ifndef OS_WIN_STORE | ||||||
|  | 		showpath->fire_copy(checked); | ||||||
|  | #endif // OS_WIN_STORE
 | ||||||
|  | 
 | ||||||
|  | 	}, inner->lifetime()); | ||||||
| 
 | 
 | ||||||
| 	AddButton( | 	AddButton( | ||||||
| 		container, | 		container, | ||||||
|  | @ -539,20 +548,12 @@ void SetupMediaOptions(not_null<Ui::VerticalLayout*> container) { | ||||||
| 		Ui::show(Box<AutoDownloadBox>()); | 		Ui::show(Box<AutoDownloadBox>()); | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| 	AddButton( | 	AddSkip(container, st::settingsCheckboxesSkip); | ||||||
| 		container, |  | ||||||
| 		lng_stickers_you_have, |  | ||||||
| 		st::settingsButton |  | ||||||
| 	)->addClickHandler([] { |  | ||||||
| 		Ui::show(Box<StickersBox>(StickersBox::Section::Installed)); |  | ||||||
| 	}); |  | ||||||
| 
 |  | ||||||
| 	AddSkip(container); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupChatBackground(not_null<Ui::VerticalLayout*> container) { | void SetupChatBackground(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); | 	AddDivider(container); | ||||||
| 	AddSkip(container, st::settingsCheckboxesSkip); | 	AddSkip(container, st::settingsSectionSkip); | ||||||
| 
 | 
 | ||||||
| 	AddSubsectionTitle(container, lng_settings_section_background); | 	AddSubsectionTitle(container, lng_settings_section_background); | ||||||
| 
 | 
 | ||||||
|  | @ -673,6 +674,8 @@ void SetupThemeOptions(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); | 	AddDivider(container); | ||||||
| 	AddSkip(container); | 	AddSkip(container); | ||||||
| 
 | 
 | ||||||
|  | 	AddSubsectionTitle(container, lng_settings_themes); | ||||||
|  | 
 | ||||||
| 	SetupNightMode(container); | 	SetupNightMode(container); | ||||||
| 
 | 
 | ||||||
| 	AddButton( | 	AddButton( | ||||||
|  | @ -698,12 +701,10 @@ Chat::Chat(QWidget *parent, not_null<UserData*> self) | ||||||
| void Chat::setupContent() { | void Chat::setupContent() { | ||||||
| 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(content, st::settingsFirstDividerSkip); | 	SetupStickersEmoji(content); | ||||||
| 	SetupChatOptions(content); |  | ||||||
| 	SetupSendKey(content); |  | ||||||
| 	SetupMediaOptions(content); |  | ||||||
| 	SetupChatBackground(content); | 	SetupChatBackground(content); | ||||||
| 	SetupThemeOptions(content); | 	SetupThemeOptions(content); | ||||||
|  | 	SetupChatOther(content); | ||||||
| 
 | 
 | ||||||
| 	Ui::ResizeFitChild(this, content); | 	Ui::ResizeFitChild(this, content); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -32,17 +32,17 @@ object_ptr<Section> CreateSection( | ||||||
| 		UserData *self) { | 		UserData *self) { | ||||||
| 	switch (type) { | 	switch (type) { | ||||||
| 	case Type::Main: | 	case Type::Main: | ||||||
| 		return object_ptr<::Settings::Main>(parent, controller, self); | 		return object_ptr<Main>(parent, controller, self); | ||||||
| 	case Type::Information: | 	case Type::Information: | ||||||
| 		return object_ptr<::Settings::Information>(parent, self); | 		return object_ptr<Information>(parent, controller, self); | ||||||
| 	case Type::Notifications: | 	case Type::Notifications: | ||||||
| 		return object_ptr<::Settings::Notifications>(parent, self); | 		return object_ptr<Notifications>(parent, self); | ||||||
| 	case Type::PrivacySecurity: | 	case Type::PrivacySecurity: | ||||||
| 		return object_ptr<::Settings::PrivacySecurity>(parent, self); | 		return object_ptr<PrivacySecurity>(parent, self); | ||||||
| 	case Type::General: | 	case Type::General: | ||||||
| 		return object_ptr<::Settings::General>(parent, self); | 		return object_ptr<General>(parent, self); | ||||||
| 	case Type::Chat: | 	case Type::Chat: | ||||||
| 		return object_ptr<::Settings::Chat>(parent, self); | 		return object_ptr<Chat>(parent, self); | ||||||
| 	} | 	} | ||||||
| 	Unexpected("Settings section type in Widget::createInnerWidget."); | 	Unexpected("Settings section type in Widget::createInnerWidget."); | ||||||
| } | } | ||||||
|  | @ -78,26 +78,36 @@ not_null<Button*> AddButton( | ||||||
| 		LangKey text, | 		LangKey text, | ||||||
| 		const style::InfoProfileButton &st, | 		const style::InfoProfileButton &st, | ||||||
| 		const style::icon *leftIcon) { | 		const style::icon *leftIcon) { | ||||||
|  | 	return AddButton(container, Lang::Viewer(text), st, leftIcon); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | not_null<Button*> AddButton( | ||||||
|  | 		not_null<Ui::VerticalLayout*> container, | ||||||
|  | 		rpl::producer<QString> text, | ||||||
|  | 		const style::InfoProfileButton &st, | ||||||
|  | 		const style::icon *leftIcon) { | ||||||
| 	const auto result = container->add(object_ptr<Button>( | 	const auto result = container->add(object_ptr<Button>( | ||||||
| 		container, | 		container, | ||||||
| 		Lang::Viewer(text), | 		std::move(text), | ||||||
| 		st)); | 		st)); | ||||||
| 	if (leftIcon) { | 	if (leftIcon) { | ||||||
| 		const auto icon = Ui::CreateChild<Ui::RpWidget>(result); | 		const auto icon = Ui::CreateChild<Ui::RpWidget>(result); | ||||||
| 		icon->setAttribute(Qt::WA_TransparentForMouseEvents); | 		icon->setAttribute(Qt::WA_TransparentForMouseEvents); | ||||||
| 		icon->resize(leftIcon->size()); | 		icon->resize(leftIcon->size()); | ||||||
| 		result->widthValue( | 		result->sizeValue( | ||||||
| 		) | rpl::start_with_next([=](int width) { | 		) | rpl::start_with_next([=](QSize size) { | ||||||
| 			icon->moveToLeft( | 			icon->moveToLeft( | ||||||
| 				st::settingsSectionIconPosition.x(), | 				st::settingsSectionIconLeft, | ||||||
| 				st::settingsSectionIconPosition.y(), | 				(size.height() - icon->height()) / 2, | ||||||
| 				width); | 				size.width()); | ||||||
| 		}, icon->lifetime()); | 		}, icon->lifetime()); | ||||||
| 		icon->paintRequest( | 		icon->paintRequest( | ||||||
| 		) | rpl::start_with_next([=] { | 		) | rpl::start_with_next([=] { | ||||||
| 			Painter p(icon); | 			Painter p(icon); | ||||||
| 			const auto width = icon->width(); | 			const auto width = icon->width(); | ||||||
| 			if (result->isOver()) { | 			const auto paintOver = (result->isOver() || result->isDown()) | ||||||
|  | 				&& !result->isDisabled(); | ||||||
|  | 			if (paintOver) { | ||||||
| 				leftIcon->paint(p, QPoint(), width, st::menuIconFgOver->c); | 				leftIcon->paint(p, QPoint(), width, st::menuIconFgOver->c); | ||||||
| 			} else { | 			} else { | ||||||
| 				leftIcon->paint(p, QPoint(), width); | 				leftIcon->paint(p, QPoint(), width); | ||||||
|  |  | ||||||
|  | @ -75,6 +75,11 @@ not_null<Button*> AddButton( | ||||||
| 	LangKey text, | 	LangKey text, | ||||||
| 	const style::InfoProfileButton &st, | 	const style::InfoProfileButton &st, | ||||||
| 	const style::icon *leftIcon = nullptr); | 	const style::icon *leftIcon = nullptr); | ||||||
|  | not_null<Button*> AddButton( | ||||||
|  | 	not_null<Ui::VerticalLayout*> container, | ||||||
|  | 	rpl::producer<QString> text, | ||||||
|  | 	const style::InfoProfileButton &st, | ||||||
|  | 	const style::icon *leftIcon = nullptr); | ||||||
| not_null<Button*> AddButtonWithLabel( | not_null<Button*> AddButtonWithLabel( | ||||||
| 	not_null<Ui::VerticalLayout*> container, | 	not_null<Ui::VerticalLayout*> container, | ||||||
| 	LangKey text, | 	LangKey text, | ||||||
|  |  | ||||||
|  | @ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "ui/wrap/vertical_layout.h" | #include "ui/wrap/vertical_layout.h" | ||||||
| #include "ui/wrap/slide_wrap.h" | #include "ui/wrap/slide_wrap.h" | ||||||
| #include "ui/widgets/labels.h" | #include "ui/widgets/labels.h" | ||||||
|  | #include "ui/widgets/checkbox.h" | ||||||
| #include "boxes/local_storage_box.h" | #include "boxes/local_storage_box.h" | ||||||
| #include "boxes/connection_box.h" | #include "boxes/connection_box.h" | ||||||
| #include "boxes/about_box.h" | #include "boxes/about_box.h" | ||||||
|  | @ -68,7 +69,6 @@ void SetupConnectionType(not_null<Ui::VerticalLayout*> container) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupStorageAndConnection(not_null<Ui::VerticalLayout*> container) { | void SetupStorageAndConnection(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); |  | ||||||
| 	AddSkip(container); | 	AddSkip(container); | ||||||
| 
 | 
 | ||||||
| 	AddButton( | 	AddButton( | ||||||
|  | @ -230,44 +230,50 @@ bool HasTray() { | ||||||
| 	return cSupportTray() || (cPlatform() == dbipWindows); | 	return cSupportTray() || (cPlatform() == dbipWindows); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupTray(not_null<Ui::VerticalLayout*> container) { | void SetupTrayContent(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	if (!HasTray()) { | 	const auto checkbox = [&](LangKey label, bool checked) { | ||||||
| 		return; | 		return object_ptr<Ui::Checkbox>( | ||||||
| 	} | 			container, | ||||||
|  | 			lang(label), | ||||||
|  | 			checked, | ||||||
|  | 			st::settingsCheckbox); | ||||||
|  | 	}; | ||||||
|  | 	const auto addCheckbox = [&](LangKey label, bool checked) { | ||||||
|  | 		return container->add( | ||||||
|  | 			checkbox(label, checked), | ||||||
|  | 			st::settingsCheckboxPadding); | ||||||
|  | 	}; | ||||||
|  | 	const auto addSlidingCheckbox = [&](LangKey label, bool checked) { | ||||||
|  | 		return container->add( | ||||||
|  | 			object_ptr<Ui::SlideWrap<Ui::Checkbox>>( | ||||||
|  | 				container, | ||||||
|  | 				checkbox(label, checked), | ||||||
|  | 				st::settingsCheckboxPadding)); | ||||||
|  | 	}; | ||||||
| 
 | 
 | ||||||
| 	const auto trayEnabler = Ui::AttachAsChild( |  | ||||||
| 		container, |  | ||||||
| 		rpl::event_stream<bool>()); |  | ||||||
| 	const auto trayEnabled = [] { | 	const auto trayEnabled = [] { | ||||||
| 		const auto workMode = Global::WorkMode().value(); | 		const auto workMode = Global::WorkMode().value(); | ||||||
| 		return (workMode == dbiwmTrayOnly) | 		return (workMode == dbiwmTrayOnly) | ||||||
| 			|| (workMode == dbiwmWindowAndTray); | 			|| (workMode == dbiwmWindowAndTray); | ||||||
| 	}; | 	}; | ||||||
| 	const auto tray = AddButton( | 	const auto tray = addCheckbox( | ||||||
| 		container, |  | ||||||
| 		lng_settings_workmode_tray, | 		lng_settings_workmode_tray, | ||||||
| 		st::settingsGeneralButton | 		trayEnabled()); | ||||||
| 	)->toggleOn(trayEnabler->events_starting_with(trayEnabled())); |  | ||||||
| 
 | 
 | ||||||
| 	const auto taskbarEnabled = [] { | 	const auto taskbarEnabled = [] { | ||||||
| 		const auto workMode = Global::WorkMode().value(); | 		const auto workMode = Global::WorkMode().value(); | ||||||
| 		return (workMode == dbiwmWindowOnly) | 		return (workMode == dbiwmWindowOnly) | ||||||
| 			|| (workMode == dbiwmWindowAndTray); | 			|| (workMode == dbiwmWindowAndTray); | ||||||
| 	}; | 	}; | ||||||
| 	const auto taskbarEnabler = Ui::AttachAsChild( |  | ||||||
| 		container, |  | ||||||
| 		rpl::event_stream<bool>()); |  | ||||||
| 	const auto taskbar = (cPlatform() == dbipWindows) | 	const auto taskbar = (cPlatform() == dbipWindows) | ||||||
| 		? AddButton( | 		? addCheckbox( | ||||||
| 			container, |  | ||||||
| 			lng_settings_workmode_window, | 			lng_settings_workmode_window, | ||||||
| 			st::settingsGeneralButton | 			taskbarEnabled()) | ||||||
| 		)->toggleOn(taskbarEnabler->events_starting_with(taskbarEnabled())) |  | ||||||
| 		: nullptr; | 		: nullptr; | ||||||
| 
 | 
 | ||||||
| 	const auto updateWorkmode = [=] { | 	const auto updateWorkmode = [=] { | ||||||
| 		const auto newMode = tray->toggled() | 		const auto newMode = tray->checked() | ||||||
| 			? ((!taskbar || taskbar->toggled()) | 			? ((!taskbar || taskbar->checked()) | ||||||
| 				? dbiwmWindowAndTray | 				? dbiwmWindowAndTray | ||||||
| 				: dbiwmTrayOnly) | 				: dbiwmTrayOnly) | ||||||
| 			: dbiwmWindowOnly; | 			: dbiwmWindowOnly; | ||||||
|  | @ -279,24 +285,26 @@ void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
| 		Local::writeSettings(); | 		Local::writeSettings(); | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	tray->toggledValue( | 	base::ObservableViewer( | ||||||
|  | 		tray->checkedChanged | ||||||
| 	) | rpl::filter([=](bool checked) { | 	) | rpl::filter([=](bool checked) { | ||||||
| 		return (checked != trayEnabled()); | 		return (checked != trayEnabled()); | ||||||
| 	}) | rpl::start_with_next([=](bool checked) { | 	}) | rpl::start_with_next([=](bool checked) { | ||||||
| 		if (!checked && taskbar && !taskbar->toggled()) { | 		if (!checked && taskbar && !taskbar->checked()) { | ||||||
| 			taskbarEnabler->fire(true); | 			taskbar->setChecked(true); | ||||||
| 		} else { | 		} else { | ||||||
| 			updateWorkmode(); | 			updateWorkmode(); | ||||||
| 		} | 		} | ||||||
| 	}, tray->lifetime()); | 	}, tray->lifetime()); | ||||||
| 
 | 
 | ||||||
| 	if (taskbar) { | 	if (taskbar) { | ||||||
| 		taskbar->toggledValue( | 		base::ObservableViewer( | ||||||
|  | 			taskbar->checkedChanged | ||||||
| 		) | rpl::filter([=](bool checked) { | 		) | rpl::filter([=](bool checked) { | ||||||
| 			return (checked != taskbarEnabled()); | 			return (checked != taskbarEnabled()); | ||||||
| 		}) | rpl::start_with_next([=](bool checked) { | 		}) | rpl::start_with_next([=](bool checked) { | ||||||
| 			if (!checked && !tray->toggled()) { | 			if (!checked && !tray->checked()) { | ||||||
| 				trayEnabler->fire(true); | 				tray->setChecked(true); | ||||||
| 			} else { | 			} else { | ||||||
| 				updateWorkmode(); | 				updateWorkmode(); | ||||||
| 			} | 			} | ||||||
|  | @ -305,32 +313,22 @@ void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
| 
 | 
 | ||||||
| #ifndef OS_WIN_STORE | #ifndef OS_WIN_STORE | ||||||
| 	if (cPlatform() == dbipWindows) { | 	if (cPlatform() == dbipWindows) { | ||||||
| 		const auto autostart = AddButton( |  | ||||||
| 			container, |  | ||||||
| 			lng_settings_auto_start, |  | ||||||
| 			st::settingsGeneralButton |  | ||||||
| 		)->toggleOn(rpl::single(cAutoStart())); |  | ||||||
| 		const auto minimized = container->add( |  | ||||||
| 			object_ptr<Ui::SlideWrap<Button>>( |  | ||||||
| 				container, |  | ||||||
| 				object_ptr<Button>( |  | ||||||
| 					container, |  | ||||||
| 					Lang::Viewer(lng_settings_start_min), |  | ||||||
| 					st::settingsGeneralButton))); |  | ||||||
| 		const auto sendto = AddButton( |  | ||||||
| 			container, |  | ||||||
| 			lng_settings_add_sendto, |  | ||||||
| 			st::settingsGeneralButton |  | ||||||
| 		)->toggleOn(rpl::single(cSendToMenu())); |  | ||||||
| 
 |  | ||||||
| 		const auto minimizedToggler = Ui::AttachAsChild( |  | ||||||
| 			minimized, |  | ||||||
| 			rpl::event_stream<bool>()); |  | ||||||
| 		const auto minimizedToggled = [] { | 		const auto minimizedToggled = [] { | ||||||
| 			return cStartMinimized() && !Global::LocalPasscode(); | 			return cStartMinimized() && !Global::LocalPasscode(); | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		autostart->toggledValue( | 		const auto autostart = addCheckbox( | ||||||
|  | 			lng_settings_auto_start, | ||||||
|  | 			cAutoStart()); | ||||||
|  | 		const auto minimized = addSlidingCheckbox( | ||||||
|  | 			lng_settings_start_min, | ||||||
|  | 			minimizedToggled()); | ||||||
|  | 		const auto sendto = addCheckbox( | ||||||
|  | 			lng_settings_add_sendto, | ||||||
|  | 			cSendToMenu()); | ||||||
|  | 
 | ||||||
|  | 		base::ObservableViewer( | ||||||
|  | 			autostart->checkedChanged | ||||||
| 		) | rpl::filter([](bool checked) { | 		) | rpl::filter([](bool checked) { | ||||||
| 			return (checked != cAutoStart()); | 			return (checked != cAutoStart()); | ||||||
| 		}) | rpl::start_with_next([=](bool checked) { | 		}) | rpl::start_with_next([=](bool checked) { | ||||||
|  | @ -338,22 +336,25 @@ void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
| 			psAutoStart(checked); | 			psAutoStart(checked); | ||||||
| 			if (checked) { | 			if (checked) { | ||||||
| 				Local::writeSettings(); | 				Local::writeSettings(); | ||||||
| 			} else if (minimized->entity()->toggled()) { | 			} else if (minimized->entity()->checked()) { | ||||||
| 				minimizedToggler->fire(false); | 				minimized->entity()->setChecked(false); | ||||||
| 			} else { | 			} else { | ||||||
| 				Local::writeSettings(); | 				Local::writeSettings(); | ||||||
| 			} | 			} | ||||||
| 		}, autostart->lifetime()); | 		}, autostart->lifetime()); | ||||||
| 
 | 
 | ||||||
| 		minimized->entity()->toggleOn( | 		minimized->toggleOn(rpl::single( | ||||||
| 			minimizedToggler->events_starting_with(minimizedToggled())); | 			autostart->checked() | ||||||
| 		minimized->toggleOn(autostart->toggledValue()); | 		) | rpl::then(base::ObservableViewer( | ||||||
| 		minimized->entity()->toggledValue( | 			autostart->checkedChanged | ||||||
|  | 		))); | ||||||
|  | 		base::ObservableViewer( | ||||||
|  | 			minimized->entity()->checkedChanged | ||||||
| 		) | rpl::filter([=](bool checked) { | 		) | rpl::filter([=](bool checked) { | ||||||
| 			return (checked != minimizedToggled()); | 			return (checked != minimizedToggled()); | ||||||
| 		}) | rpl::start_with_next([=](bool checked) { | 		}) | rpl::start_with_next([=](bool checked) { | ||||||
| 			if (Global::LocalPasscode()) { | 			if (Global::LocalPasscode()) { | ||||||
| 				minimizedToggler->fire(false); | 				minimized->entity()->setChecked(false); | ||||||
| 				Ui::show(Box<InformBox>( | 				Ui::show(Box<InformBox>( | ||||||
| 					lang(lng_error_start_minimized_passcoded))); | 					lang(lng_error_start_minimized_passcoded))); | ||||||
| 			} else { | 			} else { | ||||||
|  | @ -365,10 +366,11 @@ void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
| 		base::ObservableViewer( | 		base::ObservableViewer( | ||||||
| 			Global::RefLocalPasscodeChanged() | 			Global::RefLocalPasscodeChanged() | ||||||
| 		) | rpl::start_with_next([=] { | 		) | rpl::start_with_next([=] { | ||||||
| 			minimizedToggler->fire(minimizedToggled()); | 			minimized->entity()->setChecked(minimizedToggled()); | ||||||
| 		}, minimized->lifetime()); | 		}, minimized->lifetime()); | ||||||
| 
 | 
 | ||||||
| 		sendto->toggledValue( | 		base::ObservableViewer( | ||||||
|  | 			sendto->checkedChanged | ||||||
| 		) | rpl::filter([](bool checked) { | 		) | rpl::filter([](bool checked) { | ||||||
| 			return (checked != cSendToMenu()); | 			return (checked != cSendToMenu()); | ||||||
| 		}) | rpl::start_with_next([](bool checked) { | 		}) | rpl::start_with_next([](bool checked) { | ||||||
|  | @ -380,6 +382,21 @@ void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
| #endif // OS_WIN_STORE
 | #endif // OS_WIN_STORE
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void SetupTray(not_null<Ui::VerticalLayout*> container) { | ||||||
|  | 	if (!HasTray()) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | ||||||
|  | 	SetupTrayContent(wrap.data()); | ||||||
|  | 
 | ||||||
|  | 	container->add(object_ptr<Ui::OverrideMargins>( | ||||||
|  | 		container, | ||||||
|  | 		std::move(wrap))); | ||||||
|  | 
 | ||||||
|  | 	AddSkip(container, st::settingsCheckboxesSkip); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| General::General(QWidget *parent, UserData *self) | General::General(QWidget *parent, UserData *self) | ||||||
| : Section(parent) { | : Section(parent) { | ||||||
| 	setupContent(); | 	setupContent(); | ||||||
|  | @ -388,21 +405,21 @@ General::General(QWidget *parent, UserData *self) | ||||||
| void General::setupContent() { | void General::setupContent() { | ||||||
| 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(content, st::settingsFirstDividerSkip); |  | ||||||
| 
 |  | ||||||
| 	if (!Core::UpdaterDisabled()) { | 	if (!Core::UpdaterDisabled()) { | ||||||
| 		AddDivider(content); |  | ||||||
| 		AddSkip(content); | 		AddSkip(content); | ||||||
| 		SetupUpdate(content); | 		SetupUpdate(content); | ||||||
| 		AddSkip(content); | 		AddSkip(content); | ||||||
| 	} | 	} | ||||||
|  | 	if (!Core::UpdaterDisabled()) { | ||||||
|  | 		AddDivider(content); | ||||||
|  | 	} | ||||||
|  | 	SetupStorageAndConnection(content); | ||||||
| 	if (HasTray()) { | 	if (HasTray()) { | ||||||
| 		AddDivider(content); | 		AddDivider(content); | ||||||
| 		AddSkip(content); | 		AddSkip(content); | ||||||
| 		SetupTray(content); | 		SetupTray(content); | ||||||
| 		AddSkip(content); | 		AddSkip(content); | ||||||
| 	} | 	} | ||||||
| 	SetupStorageAndConnection(content); |  | ||||||
| 
 | 
 | ||||||
| 	Ui::ResizeFitChild(this, content); | 	Ui::ResizeFitChild(this, content); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -13,34 +13,127 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "ui/widgets/labels.h" | #include "ui/widgets/labels.h" | ||||||
| #include "ui/widgets/buttons.h" | #include "ui/widgets/buttons.h" | ||||||
| #include "ui/widgets/input_fields.h" | #include "ui/widgets/input_fields.h" | ||||||
|  | #include "ui/widgets/popup_menu.h" | ||||||
|  | #include "ui/special_buttons.h" | ||||||
| #include "boxes/add_contact_box.h" | #include "boxes/add_contact_box.h" | ||||||
|  | #include "boxes/confirm_box.h" | ||||||
| #include "boxes/change_phone_box.h" | #include "boxes/change_phone_box.h" | ||||||
|  | #include "boxes/photo_crop_box.h" | ||||||
| #include "boxes/username_box.h" | #include "boxes/username_box.h" | ||||||
| #include "info/profile/info_profile_values.h" | #include "info/profile/info_profile_values.h" | ||||||
|  | #include "info/profile/info_profile_button.h" | ||||||
| #include "lang/lang_keys.h" | #include "lang/lang_keys.h" | ||||||
| #include "auth_session.h" | #include "auth_session.h" | ||||||
| #include "apiwrap.h" | #include "apiwrap.h" | ||||||
|  | #include "core/file_utilities.h" | ||||||
|  | #include "styles/style_boxes.h" | ||||||
| #include "styles/style_settings.h" | #include "styles/style_settings.h" | ||||||
| 
 | 
 | ||||||
| namespace Settings { | namespace Settings { | ||||||
| namespace { | namespace { | ||||||
| 
 | 
 | ||||||
|  | void SetupPhoto( | ||||||
|  | 		not_null<Ui::VerticalLayout*> container, | ||||||
|  | 		not_null<Window::Controller*> controller, | ||||||
|  | 		not_null<UserData*> self) { | ||||||
|  | 	const auto wrap = container->add(object_ptr<BoxContentDivider>( | ||||||
|  | 		container, | ||||||
|  | 		st::settingsInfoPhotoHeight)); | ||||||
|  | 	const auto photo = Ui::CreateChild<Ui::UserpicButton>( | ||||||
|  | 		wrap, | ||||||
|  | 		controller, | ||||||
|  | 		self, | ||||||
|  | 		Ui::UserpicButton::Role::OpenPhoto, | ||||||
|  | 		st::settingsInfoPhoto); | ||||||
|  | 	const auto upload = Ui::CreateChild<Ui::RoundButton>( | ||||||
|  | 		wrap, | ||||||
|  | 		langFactory(lng_settings_upload), | ||||||
|  | 		st::settingsInfoPhotoSet); | ||||||
|  | 	upload->setFullRadius(true); | ||||||
|  | 	upload->addClickHandler([=] { | ||||||
|  | 		const auto imageExtensions = cImgExtensions(); | ||||||
|  | 		const auto filter = qsl("Image files (*") | ||||||
|  | 			+ imageExtensions.join(qsl(" *")) | ||||||
|  | 			+ qsl(");;") | ||||||
|  | 			+ FileDialog::AllFilesFilter(); | ||||||
|  | 		const auto callback = [=](const FileDialog::OpenResult &result) { | ||||||
|  | 			if (result.paths.isEmpty() && result.remoteContent.isEmpty()) { | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			const auto image = result.remoteContent.isEmpty() | ||||||
|  | 				? App::readImage(result.paths.front()) | ||||||
|  | 				: App::readImage(result.remoteContent); | ||||||
|  | 			if (image.isNull() | ||||||
|  | 				|| image.width() > 10 * image.height() | ||||||
|  | 				|| image.height() > 10 * image.width()) { | ||||||
|  | 				Ui::show(Box<InformBox>(lang(lng_bad_photo))); | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			auto box = Ui::show(Box<PhotoCropBox>(image, self)); | ||||||
|  | 			box->ready( | ||||||
|  | 			) | rpl::start_with_next([=](QImage &&image) { | ||||||
|  | 				Auth().api().uploadPeerPhoto(self, std::move(image)); | ||||||
|  | 			}, box->lifetime()); | ||||||
|  | 		}; | ||||||
|  | 		FileDialog::GetOpenPath( | ||||||
|  | 			upload, | ||||||
|  | 			lang(lng_choose_image), | ||||||
|  | 			filter, | ||||||
|  | 			crl::guard(upload, callback)); | ||||||
|  | 	}); | ||||||
|  | 	rpl::combine( | ||||||
|  | 		wrap->widthValue(), | ||||||
|  | 		photo->widthValue(), | ||||||
|  | 		upload->widthValue() | ||||||
|  | 	) | rpl::start_with_next([=](int max, int photoWidth, int uploadWidth) { | ||||||
|  | 		photo->moveToLeft( | ||||||
|  | 			(max - photoWidth) / 2, | ||||||
|  | 			st::settingsInfoPhotoTop); | ||||||
|  | 		upload->moveToLeft( | ||||||
|  | 			(max - uploadWidth) / 2, | ||||||
|  | 			(st::settingsInfoPhotoTop | ||||||
|  | 				+ photo->height() | ||||||
|  | 				+ st::settingsInfoPhotoSkip)); | ||||||
|  | 	}, photo->lifetime()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ShowMenu( | ||||||
|  | 		QWidget *parent, | ||||||
|  | 		const QString ©Button, | ||||||
|  | 		const QString &text) { | ||||||
|  | 	const auto menu = new Ui::PopupMenu(parent); | ||||||
|  | 
 | ||||||
|  | 	menu->addAction(copyButton, [=] { | ||||||
|  | 		QApplication::clipboard()->setText(text); | ||||||
|  | 	}); | ||||||
|  | 	menu->popup(QCursor::pos()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void AddRow( | void AddRow( | ||||||
| 		not_null<Ui::VerticalLayout*> container, | 		not_null<Ui::VerticalLayout*> container, | ||||||
| 		rpl::producer<QString> label, | 		rpl::producer<QString> label, | ||||||
| 		rpl::producer<TextWithEntities> value, | 		rpl::producer<TextWithEntities> value, | ||||||
| 		const QString ©Text, | 		const QString ©Button, | ||||||
| 		const style::IconButton &editSt, |  | ||||||
| 		Fn<void()> edit, | 		Fn<void()> edit, | ||||||
| 		const style::icon &icon) { | 		const style::icon &icon) { | ||||||
| 	const auto wrap = container->add(object_ptr<Ui::FixedHeightWidget>( | 	const auto wrap = AddButton( | ||||||
| 		container, | 		container, | ||||||
| 		st::settingsInfoRowHeight)); | 		rpl::single(QString()), | ||||||
| 
 | 		st::settingsInfoRow, | ||||||
| 	wrap->paintRequest( | 		&icon); | ||||||
| 	) | rpl::start_with_next([=, &icon] { | 	const auto forcopy = Ui::AttachAsChild(wrap, QString()); | ||||||
| 		Painter p(wrap); | 	wrap->setAcceptBoth(); | ||||||
| 		icon.paint(p, st::settingsInfoIconPosition, wrap->width()); | 	wrap->clicks( | ||||||
|  | 	) | rpl::filter([=] { | ||||||
|  | 		return !wrap->isDisabled(); | ||||||
|  | 	}) | rpl::start_with_next([=](Qt::MouseButton button) { | ||||||
|  | 		if (button == Qt::LeftButton) { | ||||||
|  | 			edit(); | ||||||
|  | 		} else if (!forcopy->isEmpty()) { | ||||||
|  | 			ShowMenu(wrap, copyButton, *forcopy); | ||||||
|  | 		} | ||||||
| 	}, wrap->lifetime()); | 	}, wrap->lifetime()); | ||||||
| 
 | 
 | ||||||
| 	auto existing = base::duplicate( | 	auto existing = base::duplicate( | ||||||
|  | @ -48,8 +141,15 @@ void AddRow( | ||||||
| 	) | rpl::map([](const TextWithEntities &text) { | 	) | rpl::map([](const TextWithEntities &text) { | ||||||
| 		return text.entities.isEmpty(); | 		return text.entities.isEmpty(); | ||||||
| 	}); | 	}); | ||||||
|  | 	base::duplicate( | ||||||
|  | 		value | ||||||
|  | 	) | rpl::filter([](const TextWithEntities &text) { | ||||||
|  | 		return text.entities.isEmpty(); | ||||||
|  | 	}) | rpl::start_with_next([=](const TextWithEntities &text) { | ||||||
|  | 		*forcopy = text.text; | ||||||
|  | 	}, wrap->lifetime()); | ||||||
| 	const auto text = Ui::CreateChild<Ui::FlatLabel>( | 	const auto text = Ui::CreateChild<Ui::FlatLabel>( | ||||||
| 		wrap, | 		wrap.get(), | ||||||
| 		std::move(value), | 		std::move(value), | ||||||
| 		st::settingsInfoValue); | 		st::settingsInfoValue); | ||||||
| 	text->setClickHandlerFilter([=](auto&&...) { | 	text->setClickHandlerFilter([=](auto&&...) { | ||||||
|  | @ -59,24 +159,32 @@ void AddRow( | ||||||
| 	base::duplicate( | 	base::duplicate( | ||||||
| 		existing | 		existing | ||||||
| 	) | rpl::start_with_next([=](bool existing) { | 	) | rpl::start_with_next([=](bool existing) { | ||||||
|  | 		wrap->setDisabled(!existing); | ||||||
|  | 		text->setAttribute(Qt::WA_TransparentForMouseEvents, existing); | ||||||
| 		text->setSelectable(existing); | 		text->setSelectable(existing); | ||||||
| 		text->setDoubleClickSelectsParagraph(existing); | 		text->setDoubleClickSelectsParagraph(existing); | ||||||
| 		text->setContextCopyText(existing ? copyText : QString()); |  | ||||||
| 	}, text->lifetime()); | 	}, text->lifetime()); | ||||||
| 
 | 
 | ||||||
| 	const auto about = Ui::CreateChild<Ui::FlatLabel>( | 	const auto about = Ui::CreateChild<Ui::FlatLabel>( | ||||||
| 		wrap, | 		wrap.get(), | ||||||
| 		std::move(label), | 		std::move(label), | ||||||
| 		st::settingsInfoAbout); | 		st::settingsInfoAbout); | ||||||
|  | 	about->setAttribute(Qt::WA_TransparentForMouseEvents); | ||||||
| 
 | 
 | ||||||
| 	const auto button = Ui::CreateChild<Ui::IconButton>( | 	const auto button = Ui::CreateChild<Ui::RpWidget>(wrap.get()); | ||||||
| 		wrap, | 	button->resize(st::settingsInfoEditIconOver.size()); | ||||||
| 		editSt); | 	button->setAttribute(Qt::WA_TransparentForMouseEvents); | ||||||
| 	button->addClickHandler(edit); | 	button->paintRequest( | ||||||
| 	button->showOn(std::move(existing)); | 	) | rpl::filter([=] { | ||||||
|  | 		return (wrap->isOver() || wrap->isDown()) && !wrap->isDisabled(); | ||||||
|  | 	}) | rpl::start_with_next([=](QRect clip) { | ||||||
|  | 		Painter p(button); | ||||||
|  | 		st::settingsInfoEditIconOver.paint(p, QPoint(), button->width()); | ||||||
|  | 	}, button->lifetime()); | ||||||
| 
 | 
 | ||||||
| 	wrap->widthValue( | 	wrap->sizeValue( | ||||||
| 	) | rpl::start_with_next([=](int width) { | 	) | rpl::start_with_next([=](QSize size) { | ||||||
|  | 		const auto width = size.width(); | ||||||
| 		text->resizeToWidth(width | 		text->resizeToWidth(width | ||||||
| 			- st::settingsInfoValuePosition.x() | 			- st::settingsInfoValuePosition.x() | ||||||
| 			- st::settingsInfoRightSkip); | 			- st::settingsInfoRightSkip); | ||||||
|  | @ -92,8 +200,8 @@ void AddRow( | ||||||
| 			st::settingsInfoAboutPosition.y(), | 			st::settingsInfoAboutPosition.y(), | ||||||
| 			width); | 			width); | ||||||
| 		button->moveToRight( | 		button->moveToRight( | ||||||
| 			st::settingsInfoEditPosition.x(), | 			st::settingsInfoEditRight, | ||||||
| 			st::settingsInfoEditPosition.y(), | 			(size.height() - button->height()) / 2, | ||||||
| 			width); | 			width); | ||||||
| 	}, wrap->lifetime()); | 	}, wrap->lifetime()); | ||||||
| } | } | ||||||
|  | @ -101,7 +209,6 @@ void AddRow( | ||||||
| void SetupRows( | void SetupRows( | ||||||
| 		not_null<Ui::VerticalLayout*> container, | 		not_null<Ui::VerticalLayout*> container, | ||||||
| 		not_null<UserData*> self) { | 		not_null<UserData*> self) { | ||||||
| 	AddDivider(container); |  | ||||||
| 	AddSkip(container); | 	AddSkip(container); | ||||||
| 
 | 
 | ||||||
| 	AddRow( | 	AddRow( | ||||||
|  | @ -109,7 +216,6 @@ void SetupRows( | ||||||
| 		Lang::Viewer(lng_settings_name_label), | 		Lang::Viewer(lng_settings_name_label), | ||||||
| 		Info::Profile::NameValue(self), | 		Info::Profile::NameValue(self), | ||||||
| 		lang(lng_profile_copy_fullname), | 		lang(lng_profile_copy_fullname), | ||||||
| 		st::settingsInfoEdit, |  | ||||||
| 		[=] { Ui::show(Box<EditNameBox>(self)); }, | 		[=] { Ui::show(Box<EditNameBox>(self)); }, | ||||||
| 		st::settingsInfoName); | 		st::settingsInfoName); | ||||||
| 
 | 
 | ||||||
|  | @ -118,7 +224,6 @@ void SetupRows( | ||||||
| 		Lang::Viewer(lng_settings_phone_label), | 		Lang::Viewer(lng_settings_phone_label), | ||||||
| 		Info::Profile::PhoneValue(self), | 		Info::Profile::PhoneValue(self), | ||||||
| 		lang(lng_profile_copy_phone), | 		lang(lng_profile_copy_phone), | ||||||
| 		st::settingsInfoEdit, |  | ||||||
| 		[] { Ui::show(Box<ChangePhoneBox>()); }, | 		[] { Ui::show(Box<ChangePhoneBox>()); }, | ||||||
| 		st::settingsInfoPhone); | 		st::settingsInfoPhone); | ||||||
| 
 | 
 | ||||||
|  | @ -154,7 +259,6 @@ void SetupRows( | ||||||
| 		std::move(label), | 		std::move(label), | ||||||
| 		std::move(value), | 		std::move(value), | ||||||
| 		lang(lng_context_copy_mention), | 		lang(lng_context_copy_mention), | ||||||
| 		st::settingsInfoEdit, |  | ||||||
| 		[=] { Ui::show(Box<UsernameBox>()); }, | 		[=] { Ui::show(Box<UsernameBox>()); }, | ||||||
| 		st::settingsInfoUsername); | 		st::settingsInfoUsername); | ||||||
| 
 | 
 | ||||||
|  | @ -257,8 +361,14 @@ BioManager SetupBio( | ||||||
| 	bio->setInstantReplacesEnabled(Global::ReplaceEmojiValue()); | 	bio->setInstantReplacesEnabled(Global::ReplaceEmojiValue()); | ||||||
| 	updated(); | 	updated(); | ||||||
| 
 | 
 | ||||||
|  | 	container->add( | ||||||
|  | 		object_ptr<Ui::FlatLabel>( | ||||||
|  | 			container, | ||||||
|  | 			Lang::Viewer(lng_settings_about_bio), | ||||||
|  | 			st::boxDividerLabel), | ||||||
|  | 		st::settingsBioLabelPadding); | ||||||
|  | 
 | ||||||
| 	AddSkip(container); | 	AddSkip(container); | ||||||
| 	AddDividerText(container, Lang::Viewer(lng_settings_about_bio)); |  | ||||||
| 
 | 
 | ||||||
| 	return BioManager{ | 	return BioManager{ | ||||||
| 		changed->events() | rpl::distinct_until_changed(), | 		changed->events() | rpl::distinct_until_changed(), | ||||||
|  | @ -268,10 +378,13 @@ BioManager SetupBio( | ||||||
| 
 | 
 | ||||||
| } // namespace
 | } // namespace
 | ||||||
| 
 | 
 | ||||||
| Information::Information(QWidget *parent, not_null<UserData*> self) | Information::Information( | ||||||
|  | 	QWidget *parent, | ||||||
|  | 	not_null<Window::Controller*> controller, | ||||||
|  | 	not_null<UserData*> self) | ||||||
| : Section(parent) | : Section(parent) | ||||||
| , _self(self) { | , _self(self) { | ||||||
| 	setupContent(); | 	setupContent(controller); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| rpl::producer<bool> Information::sectionCanSaveChanges() { | rpl::producer<bool> Information::sectionCanSaveChanges() { | ||||||
|  | @ -282,10 +395,10 @@ void Information::sectionSaveChanges(FnMut<void()> done) { | ||||||
| 	_save(std::move(done)); | 	_save(std::move(done)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Information::setupContent() { | void Information::setupContent(not_null<Window::Controller*> controller) { | ||||||
| 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(content, st::settingsFirstDividerSkip); | 	SetupPhoto(content, controller, _self); | ||||||
| 	SetupRows(content, _self); | 	SetupRows(content, _self); | ||||||
| 	auto manager = SetupBio(content, _self); | 	auto manager = SetupBio(content, _self); | ||||||
| 	_canSaveChanges = std::move(manager.canSave); | 	_canSaveChanges = std::move(manager.canSave); | ||||||
|  |  | ||||||
|  | @ -13,13 +13,16 @@ namespace Settings { | ||||||
| 
 | 
 | ||||||
| class Information : public Section { | class Information : public Section { | ||||||
| public: | public: | ||||||
| 	Information(QWidget *parent, not_null<UserData*> self); | 	Information( | ||||||
|  | 		QWidget *parent, | ||||||
|  | 		not_null<Window::Controller*> controller, | ||||||
|  | 		not_null<UserData*> self); | ||||||
| 
 | 
 | ||||||
| 	rpl::producer<bool> sectionCanSaveChanges() override; | 	rpl::producer<bool> sectionCanSaveChanges() override; | ||||||
| 	void sectionSaveChanges(FnMut<void()> done) override; | 	void sectionSaveChanges(FnMut<void()> done) override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| 	void setupContent(); | 	void setupContent(not_null<Window::Controller*> controller); | ||||||
| 
 | 
 | ||||||
| 	not_null<UserData*> _self; | 	not_null<UserData*> _self; | ||||||
| 	rpl::variable<bool> _canSaveChanges; | 	rpl::variable<bool> _canSaveChanges; | ||||||
|  |  | ||||||
|  | @ -12,7 +12,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "boxes/language_box.h" | #include "boxes/language_box.h" | ||||||
| #include "boxes/confirm_box.h" | #include "boxes/confirm_box.h" | ||||||
| #include "boxes/about_box.h" | #include "boxes/about_box.h" | ||||||
| #include "boxes/photo_crop_box.h" |  | ||||||
| #include "ui/wrap/vertical_layout.h" | #include "ui/wrap/vertical_layout.h" | ||||||
| #include "ui/widgets/labels.h" | #include "ui/widgets/labels.h" | ||||||
| #include "ui/widgets/discrete_sliders.h" | #include "ui/widgets/discrete_sliders.h" | ||||||
|  | @ -27,58 +26,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| 
 | 
 | ||||||
| namespace Settings { | namespace Settings { | ||||||
| 
 | 
 | ||||||
| void SetupUploadPhotoButton( |  | ||||||
| 		not_null<Ui::VerticalLayout*> container, |  | ||||||
| 		not_null<UserData*> self) { |  | ||||||
| 	AddDivider(container); |  | ||||||
| 	AddSkip(container, st::settingsSetPhotoSkip); |  | ||||||
| 
 |  | ||||||
| 	const auto upload = [=] { |  | ||||||
| 		const auto imageExtensions = cImgExtensions(); |  | ||||||
| 		const auto filter = qsl("Image files (*") |  | ||||||
| 			+ imageExtensions.join(qsl(" *")) |  | ||||||
| 			+ qsl(");;") |  | ||||||
| 			+ FileDialog::AllFilesFilter(); |  | ||||||
| 		const auto callback = [=](const FileDialog::OpenResult &result) { |  | ||||||
| 			if (result.paths.isEmpty() && result.remoteContent.isEmpty()) { |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			const auto image = result.remoteContent.isEmpty() |  | ||||||
| 				? App::readImage(result.paths.front()) |  | ||||||
| 				: App::readImage(result.remoteContent); |  | ||||||
| 			if (image.isNull() |  | ||||||
| 				|| image.width() > 10 * image.height() |  | ||||||
| 				|| image.height() > 10 * image.width()) { |  | ||||||
| 				Ui::show(Box<InformBox>(lang(lng_bad_photo))); |  | ||||||
| 				return; |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			auto box = Ui::show(Box<PhotoCropBox>(image, self)); |  | ||||||
| 			box->ready( |  | ||||||
| 			) | rpl::start_with_next([=](QImage &&image) { |  | ||||||
| 				Auth().api().uploadPeerPhoto(self, std::move(image)); |  | ||||||
| 			}, box->lifetime()); |  | ||||||
| 		}; |  | ||||||
| 		FileDialog::GetOpenPath( |  | ||||||
| 			container.get(), |  | ||||||
| 			lang(lng_choose_image), |  | ||||||
| 			filter, |  | ||||||
| 			crl::guard(container, callback)); |  | ||||||
| 	}; |  | ||||||
| 	AddButton( |  | ||||||
| 		container, |  | ||||||
| 		lng_settings_upload, |  | ||||||
| 		st::settingsSectionButton, |  | ||||||
| 		&st::settingsIconSetPhoto |  | ||||||
| 	)->addClickHandler(App::LambdaDelayed( |  | ||||||
| 		st::settingsSectionButton.ripple.hideDuration, |  | ||||||
| 		container, |  | ||||||
| 		upload)); |  | ||||||
| 
 |  | ||||||
| 	AddSkip(container, st::settingsSetPhotoSkip); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void SetupLanguageButton(not_null<Ui::VerticalLayout*> container) { | void SetupLanguageButton(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	const auto button = AddButtonWithLabel( | 	const auto button = AddButtonWithLabel( | ||||||
| 		container, | 		container, | ||||||
|  | @ -121,14 +68,14 @@ void SetupSections( | ||||||
| 		lng_settings_section_privacy, | 		lng_settings_section_privacy, | ||||||
| 		Type::PrivacySecurity, | 		Type::PrivacySecurity, | ||||||
| 		&st::settingsIconPrivacySecurity); | 		&st::settingsIconPrivacySecurity); | ||||||
| 	addSection( |  | ||||||
| 		lng_settings_section_general, |  | ||||||
| 		Type::General, |  | ||||||
| 		&st::settingsIconGeneral); |  | ||||||
| 	addSection( | 	addSection( | ||||||
| 		lng_settings_section_chat_settings, | 		lng_settings_section_chat_settings, | ||||||
| 		Type::Chat, | 		Type::Chat, | ||||||
| 		&st::settingsIconChat); | 		&st::settingsIconChat); | ||||||
|  | 	addSection( | ||||||
|  | 		lng_settings_advanced, | ||||||
|  | 		Type::General, | ||||||
|  | 		&st::settingsIconGeneral); | ||||||
| 
 | 
 | ||||||
| 	SetupLanguageButton(container); | 	SetupLanguageButton(container); | ||||||
| 
 | 
 | ||||||
|  | @ -309,7 +256,6 @@ void Main::setupContent(not_null<Window::Controller*> controller) { | ||||||
| 		controller)); | 		controller)); | ||||||
| 	cover->setOnlineCount(rpl::single(0)); | 	cover->setOnlineCount(rpl::single(0)); | ||||||
| 
 | 
 | ||||||
| 	SetupUploadPhotoButton(content, _self); |  | ||||||
| 	SetupSections(content, [=](Type type) { | 	SetupSections(content, [=](Type type) { | ||||||
| 		_showOther.fire_copy(type); | 		_showOther.fire_copy(type); | ||||||
| 	}); | 	}); | ||||||
|  |  | ||||||
|  | @ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL | ||||||
| #include "ui/widgets/checkbox.h" | #include "ui/widgets/checkbox.h" | ||||||
| #include "ui/widgets/buttons.h" | #include "ui/widgets/buttons.h" | ||||||
| #include "lang/lang_keys.h" | #include "lang/lang_keys.h" | ||||||
|  | #include "info/profile/info_profile_button.h" | ||||||
| #include "storage/localstorage.h" | #include "storage/localstorage.h" | ||||||
| #include "window/notifications_manager.h" | #include "window/notifications_manager.h" | ||||||
| #include "boxes/notifications_box.h" | #include "boxes/notifications_box.h" | ||||||
|  | @ -74,15 +75,26 @@ void SetupNotificationsContent(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	const auto native = nativeNotificationsKey | 	const auto native = nativeNotificationsKey | ||||||
| 		? addCheckbox(nativeNotificationsKey, Global::NativeNotifications()) | 		? addCheckbox(nativeNotificationsKey, Global::NativeNotifications()) | ||||||
| 		: nullptr; | 		: nullptr; | ||||||
| 	const auto advanced = (cPlatform() != dbipMac) | 
 | ||||||
|  | 	const auto advancedSlide = (cPlatform() != dbipMac) | ||||||
| 		? container->add( | 		? container->add( | ||||||
| 			object_ptr<Ui::SlideWrap<Ui::LinkButton>>( | 			object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>( | ||||||
| 				container, | 				container, | ||||||
| 				object_ptr<Ui::LinkButton>( | 				object_ptr<Ui::VerticalLayout>(container))) | ||||||
| 					container, | 		: nullptr; | ||||||
| 					lang(lng_settings_advanced_notifications), | 	const auto advancedWrap = advancedSlide | ||||||
| 					st::settingsLink), | 		? advancedSlide->entity() | ||||||
| 				st::settingsAdvancedNotificationsPadding)) | 		: nullptr; | ||||||
|  | 	if (advancedWrap) { | ||||||
|  | 		AddSkip(advancedWrap, st::settingsCheckboxesSkip); | ||||||
|  | 		AddDivider(advancedWrap); | ||||||
|  | 		AddSkip(advancedWrap, st::settingsCheckboxesSkip); | ||||||
|  | 	} | ||||||
|  | 	const auto advanced = advancedWrap | ||||||
|  | 		? AddButton( | ||||||
|  | 			advancedWrap, | ||||||
|  | 			lng_settings_advanced_notifications, | ||||||
|  | 			st::settingsButton).get() | ||||||
| 		: nullptr; | 		: nullptr; | ||||||
| 
 | 
 | ||||||
| 	if (!name->entity()->checked()) { | 	if (!name->entity()->checked()) { | ||||||
|  | @ -92,8 +104,8 @@ void SetupNotificationsContent(not_null<Ui::VerticalLayout*> container) { | ||||||
| 		name->hide(anim::type::instant); | 		name->hide(anim::type::instant); | ||||||
| 		preview->hide(anim::type::instant); | 		preview->hide(anim::type::instant); | ||||||
| 	} | 	} | ||||||
| 	if (native && advanced && Global::NativeNotifications()) { | 	if (native && advancedSlide && Global::NativeNotifications()) { | ||||||
| 		advanced->hide(anim::type::instant); | 		advancedSlide->hide(anim::type::instant); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	using Change = Window::Notifications::ChangeType; | 	using Change = Window::Notifications::ChangeType; | ||||||
|  | @ -187,22 +199,21 @@ void SetupNotificationsContent(not_null<Ui::VerticalLayout*> container) { | ||||||
| 
 | 
 | ||||||
| 			Auth().notifications().createManager(); | 			Auth().notifications().createManager(); | ||||||
| 
 | 
 | ||||||
| 			if (advanced) { | 			if (advancedSlide) { | ||||||
| 				advanced->toggle( | 				advancedSlide->toggle( | ||||||
| 					!Global::NativeNotifications(), | 					!Global::NativeNotifications(), | ||||||
| 					anim::type::normal); | 					anim::type::normal); | ||||||
| 			} | 			} | ||||||
| 		}, native->lifetime()); | 		}, native->lifetime()); | ||||||
| 	} | 	} | ||||||
| 	if (advanced) { | 	if (advanced) { | ||||||
| 		advanced->entity()->addClickHandler([=] { | 		advanced->addClickHandler([=] { | ||||||
| 			Ui::show(Box<NotificationsBox>()); | 			Ui::show(Box<NotificationsBox>()); | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupNotifications(not_null<Ui::VerticalLayout*> container) { | void SetupNotifications(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); |  | ||||||
| 	AddSkip(container, st::settingsCheckboxesSkip); | 	AddSkip(container, st::settingsCheckboxesSkip); | ||||||
| 
 | 
 | ||||||
| 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | 	auto wrap = object_ptr<Ui::VerticalLayout>(container); | ||||||
|  | @ -226,7 +237,6 @@ Notifications::Notifications(QWidget *parent, not_null<UserData*> self) | ||||||
| void Notifications::setupContent() { | void Notifications::setupContent() { | ||||||
| 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(content, st::settingsFirstDividerSkip); |  | ||||||
| 	SetupNotifications(content); | 	SetupNotifications(content); | ||||||
| 
 | 
 | ||||||
| 	Ui::ResizeFitChild(this, content); | 	Ui::ResizeFitChild(this, content); | ||||||
|  |  | ||||||
|  | @ -57,7 +57,6 @@ QString PrivacyBase(ApiWrap::Privacy::Option option) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SetupPrivacy(not_null<Ui::VerticalLayout*> container) { | void SetupPrivacy(not_null<Ui::VerticalLayout*> container) { | ||||||
| 	AddDivider(container); |  | ||||||
| 	AddSkip(container); | 	AddSkip(container); | ||||||
| 
 | 
 | ||||||
| 	AddSubsectionTitle(container, lng_settings_privacy_title); | 	AddSubsectionTitle(container, lng_settings_privacy_title); | ||||||
|  | @ -482,7 +481,6 @@ PrivacySecurity::PrivacySecurity(QWidget *parent, not_null<UserData*> self) | ||||||
| void PrivacySecurity::setupContent() { | void PrivacySecurity::setupContent() { | ||||||
| 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | 	const auto content = Ui::CreateChild<Ui::VerticalLayout>(this); | ||||||
| 
 | 
 | ||||||
| 	AddSkip(content, st::settingsFirstDividerSkip); |  | ||||||
| 	SetupPrivacy(content); | 	SetupPrivacy(content); | ||||||
| 	SetupSecurity(content); | 	SetupSecurity(content); | ||||||
| 	SetupCalls(content); | 	SetupCalls(content); | ||||||
|  |  | ||||||
|  | @ -78,7 +78,7 @@ void AbstractButton::mouseReleaseEvent(QMouseEvent *e) { | ||||||
| 				emit clicked(); | 				emit clicked(); | ||||||
| 			} | 			} | ||||||
| 			if (weak) { | 			if (weak) { | ||||||
| 				_clicks.fire({}); | 				_clicks.fire(e->button()); | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			setOver(false, StateChangeSource::ByHover); | 			setOver(false, StateChangeSource::ByHover); | ||||||
|  |  | ||||||
|  | @ -43,7 +43,7 @@ public: | ||||||
| 		_clickedCallback = std::move(callback); | 		_clickedCallback = std::move(callback); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	rpl::producer<> clicks() const { | 	rpl::producer<Qt::MouseButton> clicks() const { | ||||||
| 		return _clicks.events(); | 		return _clicks.events(); | ||||||
| 	} | 	} | ||||||
| 	template <typename Handler> | 	template <typename Handler> | ||||||
|  | @ -100,7 +100,7 @@ private: | ||||||
| 
 | 
 | ||||||
| 	Fn<void()> _clickedCallback; | 	Fn<void()> _clickedCallback; | ||||||
| 
 | 
 | ||||||
| 	rpl::event_stream<> _clicks; | 	rpl::event_stream<Qt::MouseButton> _clicks; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -266,6 +266,11 @@ void RoundButton::setFullWidth(int newFullWidth) { | ||||||
| 	refreshText(); | 	refreshText(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void RoundButton::setFullRadius(bool enabled) { | ||||||
|  | 	_fullRadius = enabled; | ||||||
|  | 	update(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void RoundButton::refreshText() { | void RoundButton::refreshText() { | ||||||
| 	_text = computeFullText(); | 	_text = computeFullText(); | ||||||
| 	_textWidth = _text.isEmpty() ? 0 : _st.font->width(_text); | 	_textWidth = _text.isEmpty() ? 0 : _st.font->width(_text); | ||||||
|  | @ -319,12 +324,24 @@ void RoundButton::paintEvent(QPaintEvent *e) { | ||||||
| 	if (_fullWidthOverride < 0) { | 	if (_fullWidthOverride < 0) { | ||||||
| 		rounded = QRect(0, rounded.top(), innerWidth - _fullWidthOverride, rounded.height()); | 		rounded = QRect(0, rounded.top(), innerWidth - _fullWidthOverride, rounded.height()); | ||||||
| 	} | 	} | ||||||
| 	App::roundRect(p, myrtlrect(rounded), _st.textBg, ImageRoundRadius::Small); | 	const auto drawRect = [&](const style::color &color) { | ||||||
|  | 		const auto fill = myrtlrect(rounded); | ||||||
|  | 		if (_fullRadius) { | ||||||
|  | 			const auto radius = rounded.height() / 2; | ||||||
|  | 			PainterHighQualityEnabler hq(p); | ||||||
|  | 			p.setPen(Qt::NoPen); | ||||||
|  | 			p.setBrush(color); | ||||||
|  | 			p.drawRoundedRect(fill, radius, radius); | ||||||
|  | 		} else { | ||||||
|  | 			App::roundRect(p, fill, color, ImageRoundRadius::Small); | ||||||
|  | 		} | ||||||
|  | 	}; | ||||||
|  | 	drawRect(_st.textBg); | ||||||
| 
 | 
 | ||||||
| 	auto over = isOver(); | 	auto over = isOver(); | ||||||
| 	auto down = isDown(); | 	auto down = isDown(); | ||||||
| 	if (over || down) { | 	if (over || down) { | ||||||
| 		App::roundRect(p, myrtlrect(rounded), _st.textBgOver, ImageRoundRadius::Small); | 		drawRect(_st.textBgOver); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	auto ms = getms(); | 	auto ms = getms(); | ||||||
|  | @ -369,7 +386,9 @@ QImage RoundButton::prepareRippleMask() const { | ||||||
| 	if (_fullWidthOverride < 0) { | 	if (_fullWidthOverride < 0) { | ||||||
| 		rounded = QRect(0, rounded.top(), innerWidth - _fullWidthOverride, rounded.height()); | 		rounded = QRect(0, rounded.top(), innerWidth - _fullWidthOverride, rounded.height()); | ||||||
| 	} | 	} | ||||||
| 	return RippleAnimation::roundRectMask(rounded.size(), st::buttonRadius); | 	return RippleAnimation::roundRectMask( | ||||||
|  | 		rounded.size(), | ||||||
|  | 		_fullRadius ? (rounded.height() / 2) : st::buttonRadius); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| QPoint RoundButton::prepareRippleStartPosition() const { | QPoint RoundButton::prepareRippleStartPosition() const { | ||||||
|  |  | ||||||
|  | @ -120,6 +120,7 @@ public: | ||||||
| 	int contentWidth() const; | 	int contentWidth() const; | ||||||
| 
 | 
 | ||||||
| 	void setFullWidth(int newFullWidth); | 	void setFullWidth(int newFullWidth); | ||||||
|  | 	void setFullRadius(bool enabled); | ||||||
| 
 | 
 | ||||||
| 	enum class TextTransform { | 	enum class TextTransform { | ||||||
| 		NoTransform, | 		NoTransform, | ||||||
|  | @ -153,6 +154,7 @@ private: | ||||||
| 	const style::RoundButton &_st; | 	const style::RoundButton &_st; | ||||||
| 
 | 
 | ||||||
| 	TextTransform _transform = TextTransform::ToUpper; | 	TextTransform _transform = TextTransform::ToUpper; | ||||||
|  | 	bool _fullRadius = false; | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -87,13 +87,15 @@ void SeparatePanel::updateTitlePosition() { | ||||||
| 
 | 
 | ||||||
| rpl::producer<> SeparatePanel::backRequests() const { | rpl::producer<> SeparatePanel::backRequests() const { | ||||||
| 	return rpl::merge( | 	return rpl::merge( | ||||||
| 		_back->entity()->clicks(), | 		_back->entity()->clicks( | ||||||
|  | 		) | rpl::map([] { return rpl::empty_value(); }), | ||||||
| 		_synteticBackRequests.events()); | 		_synteticBackRequests.events()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| rpl::producer<> SeparatePanel::closeRequests() const { | rpl::producer<> SeparatePanel::closeRequests() const { | ||||||
| 	return rpl::merge( | 	return rpl::merge( | ||||||
| 		_close->clicks(), | 		_close->clicks( | ||||||
|  | 		) | rpl::map([] { return rpl::empty_value(); }), | ||||||
| 		_userCloseRequests.events()); | 		_userCloseRequests.events()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -293,11 +293,15 @@ void TermsBox::prepare() { | ||||||
| 			return false; | 			return false; | ||||||
| 		} | 		} | ||||||
| 		return true; | 		return true; | ||||||
|  | 	}) | rpl::map([] { | ||||||
|  | 		return rpl::empty_value(); | ||||||
| 	}) | rpl::start_to_stream(_agreeClicks, lifetime()); | 	}) | rpl::start_to_stream(_agreeClicks, lifetime()); | ||||||
| 
 | 
 | ||||||
| 	if (_cancel) { | 	if (_cancel) { | ||||||
| 		addButton(_cancel, [=] {})->clicks( | 		addButton(_cancel, [=] {})->clicks( | ||||||
| 		) | rpl::start_to_stream(_cancelClicks, lifetime()); | 		) | rpl::map([] { | ||||||
|  | 			return rpl::empty_value(); | ||||||
|  | 		}) | rpl::start_to_stream(_cancelClicks, lifetime()); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (age) { | 	if (age) { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue