934 lines
		
	
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			934 lines
		
	
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of Desktop App Toolkit,
 | |
| // a set of libraries for developing nice desktop applications.
 | |
| //
 | |
| // For license and copyright information please follow this link:
 | |
| // https://github.com/desktop-app/legal/blob/master/LEGAL
 | |
| //
 | |
| #include "ui/widgets/checkbox.h"
 | |
| 
 | |
| #include "ui/effects/ripple_animation.h"
 | |
| #include "ui/basic_click_handlers.h"
 | |
| #include "ui/ui_utility.h"
 | |
| #include "ui/painter.h"
 | |
| 
 | |
| #include <QtGui/QtEvents>
 | |
| #include <QtCore/QtMath>
 | |
| 
 | |
| namespace Ui {
 | |
| namespace {
 | |
| 
 | |
| TextParseOptions _checkboxOptions = {
 | |
| 	TextParseMultiline, // flags
 | |
| 	0, // maxw
 | |
| 	0, // maxh
 | |
| 	Qt::LayoutDirectionAuto, // dir
 | |
| };
 | |
| 
 | |
| TextParseOptions _checkboxRichOptions = {
 | |
| 	TextParseMultiline, // flags
 | |
| 	0, // maxw
 | |
| 	0, // maxh
 | |
| 	Qt::LayoutDirectionAuto, // dir
 | |
| };
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| AbstractCheckView::AbstractCheckView(int duration, bool checked, Fn<void()> updateCallback)
 | |
| : _duration(duration)
 | |
| , _checked(checked)
 | |
| , _updateCallback(std::move(updateCallback)) {
 | |
| }
 | |
| 
 | |
| void AbstractCheckView::setChecked(bool checked, anim::type animated) {
 | |
| 	const auto changed = (_checked != checked);
 | |
| 	_checked = checked;
 | |
| 	if (animated == anim::type::instant) {
 | |
| 		finishAnimating();
 | |
| 		if (_updateCallback) {
 | |
| 			_updateCallback();
 | |
| 		}
 | |
| 	} else if (changed) {
 | |
| 		_toggleAnimation.start(
 | |
| 			[=] { if (_updateCallback) _updateCallback(); },
 | |
| 			_checked ? 0. : 1.,
 | |
| 			_checked ? 1. : 0.,
 | |
| 			_duration);
 | |
| 	}
 | |
| 	checkedChangedHook(animated);
 | |
| 	if (changed) {
 | |
| 		_checks.fire_copy(_checked);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void AbstractCheckView::setUpdateCallback(Fn<void()> updateCallback) {
 | |
| 	_updateCallback = std::move(updateCallback);
 | |
| }
 | |
| 
 | |
| void AbstractCheckView::update() {
 | |
| 	if (_updateCallback) {
 | |
| 		_updateCallback();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void AbstractCheckView::finishAnimating() {
 | |
| 	_toggleAnimation.stop();
 | |
| }
 | |
| 
 | |
| float64 AbstractCheckView::currentAnimationValue() {
 | |
| 	return _toggleAnimation.value(_checked ? 1. : 0.);
 | |
| }
 | |
| 
 | |
| bool AbstractCheckView::animating() const {
 | |
| 	return _toggleAnimation.animating();
 | |
| }
 | |
| 
 | |
| ToggleView::ToggleView(
 | |
| 	const style::Toggle &st,
 | |
| 	bool checked,
 | |
| 	Fn<void()> updateCallback)
 | |
| : AbstractCheckView(st.duration, checked, std::move(updateCallback))
 | |
| , _st(&st) {
 | |
| }
 | |
| 
 | |
| QSize ToggleView::getSize() const {
 | |
| 	return QSize(2 * _st->border + _st->diameter + _st->width, 2 * _st->border + _st->diameter);
 | |
| }
 | |
| 
 | |
| void ToggleView::setStyle(const style::Toggle &st) {
 | |
| 	_st = &st;
 | |
| }
 | |
| 
 | |
| void ToggleView::paint(QPainter &p, int left, int top, int outerWidth) {
 | |
| 	left += _st->border;
 | |
| 	top += _st->border;
 | |
| 
 | |
| 	PainterHighQualityEnabler hq(p);
 | |
| 	auto toggled = currentAnimationValue();
 | |
| 	auto fullWidth = _st->diameter + _st->width;
 | |
| 	auto innerDiameter = _st->diameter - 2 * _st->shift;
 | |
| 	auto innerRadius = float64(innerDiameter) / 2.;
 | |
| 	auto toggleLeft = left + anim::interpolate(0, fullWidth - _st->diameter, toggled);
 | |
| 	auto bgRect = style::rtlrect(left + _st->shift, top + _st->shift, fullWidth - 2 * _st->shift, innerDiameter, outerWidth);
 | |
| 	auto fgRect = style::rtlrect(toggleLeft, top, _st->diameter, _st->diameter, outerWidth);
 | |
| 	auto fgBrush = anim::brush(_st->untoggledFg, _st->toggledFg, toggled);
 | |
| 
 | |
| 	p.setPen(Qt::NoPen);
 | |
| 	p.setBrush(fgBrush);
 | |
| 	p.drawRoundedRect(bgRect, innerRadius, innerRadius);
 | |
| 
 | |
| 	auto pen = anim::pen(_st->untoggledFg, _st->toggledFg, toggled);
 | |
| 	pen.setWidth(_st->border);
 | |
| 	p.setPen(pen);
 | |
| 	p.setBrush(anim::brush(_st->untoggledBg, _st->toggledBg, toggled));
 | |
| 	p.drawEllipse(fgRect);
 | |
| 
 | |
| 	if (_locked || _st->xsize > 0) {
 | |
| 		p.setPen(Qt::NoPen);
 | |
| 		p.setBrush(fgBrush);
 | |
| 		if (_locked) {
 | |
| 			const auto color = anim::color(_st->untoggledFg, _st->toggledFg, toggled);
 | |
| 			_st->lockIcon.paint(p, toggleLeft, top, outerWidth, color);
 | |
| 		} else {
 | |
| 			paintXV(p, toggleLeft, top, outerWidth, toggled, fgBrush);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ToggleView::paintXV(QPainter &p, int left, int top, int outerWidth, float64 toggled, const QBrush &brush) {
 | |
| 	Expects(_st->vsize > 0);
 | |
| 	Expects(_st->stroke > 0);
 | |
| 
 | |
| 	const auto stroke = (0. + _st->stroke) / M_SQRT2;
 | |
| 	if (toggled < 1) {
 | |
| 		// Just X or X->V.
 | |
| 		const auto xSize = 0. + _st->xsize;
 | |
| 		const auto xLeft = left + (_st->diameter - xSize) / 2.;
 | |
| 		const auto xTop = top + (_st->diameter - xSize) / 2.;
 | |
| 		QPointF pathX[] = {
 | |
| 			{ xLeft, xTop + stroke },
 | |
| 			{ xLeft + stroke, xTop },
 | |
| 			{ xLeft + (xSize / 2.), xTop + (xSize / 2.) - stroke },
 | |
| 			{ xLeft + xSize - stroke, xTop },
 | |
| 			{ xLeft + xSize, xTop + stroke },
 | |
| 			{ xLeft + (xSize / 2.) + stroke, xTop + (xSize / 2.) },
 | |
| 			{ xLeft + xSize, xTop + xSize - stroke },
 | |
| 			{ xLeft + xSize - stroke, xTop + xSize },
 | |
| 			{ xLeft + (xSize / 2.), xTop + (xSize / 2.) + stroke },
 | |
| 			{ xLeft + stroke, xTop + xSize },
 | |
| 			{ xLeft, xTop + xSize - stroke },
 | |
| 			{ xLeft + (xSize / 2.) - stroke, xTop + (xSize / 2.) },
 | |
| 		};
 | |
| 		for (auto &point : pathX) {
 | |
| 			point = style::rtlpoint(point, outerWidth);
 | |
| 		}
 | |
| 		if (toggled > 0) {
 | |
| 			// X->V.
 | |
| 			const auto vSize = 0. + _st->vsize;
 | |
| 			const auto fSize = (xSize + vSize - 2. * stroke);
 | |
| 			const auto vLeft = left + (_st->diameter - fSize) / 2.;
 | |
| 			const auto vTop = 0. + xTop + _st->vshift;
 | |
| 			QPointF pathV[] = {
 | |
| 				{ vLeft, vTop + xSize - vSize + stroke },
 | |
| 				{ vLeft + stroke, vTop + xSize - vSize },
 | |
| 				{ vLeft + vSize - stroke, vTop + xSize - 2 * stroke },
 | |
| 				{ vLeft + fSize - stroke, vTop },
 | |
| 				{ vLeft + fSize, vTop + stroke },
 | |
| 				{ vLeft + vSize, vTop + xSize - stroke },
 | |
| 				{ vLeft + vSize, vTop + xSize - stroke },
 | |
| 				{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 				{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 				{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 				{ vLeft + vSize - 2 * stroke, vTop + xSize - stroke },
 | |
| 				{ vLeft + vSize - 2 * stroke, vTop + xSize - stroke },
 | |
| 			};
 | |
| 			for (auto &point : pathV) {
 | |
| 				point = style::rtlpoint(point, outerWidth);
 | |
| 			}
 | |
| 			p.fillPath(anim::interpolate(pathX, pathV, toggled), brush);
 | |
| 		} else {
 | |
| 			// Just X.
 | |
| 			p.fillPath(anim::path(pathX), brush);
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Just V.
 | |
| 		const auto xSize = 0. + _st->xsize;
 | |
| 		const auto xTop = top + (_st->diameter - xSize) / 2.;
 | |
| 		const auto vSize = 0. + _st->vsize;
 | |
| 		const auto fSize = (xSize + vSize - 2. * stroke);
 | |
| 		const auto vLeft = left + (_st->diameter - (_st->xsize + _st->vsize - 2. * stroke)) / 2.;
 | |
| 		const auto vTop = 0. + xTop + _st->vshift;
 | |
| 		QPointF pathV[] = {
 | |
| 			{ vLeft, vTop + xSize - vSize + stroke },
 | |
| 			{ vLeft + stroke, vTop + xSize - vSize },
 | |
| 			{ vLeft + vSize - stroke, vTop + xSize - 2 * stroke },
 | |
| 			{ vLeft + fSize - stroke, vTop },
 | |
| 			{ vLeft + fSize, vTop + stroke },
 | |
| 			{ vLeft + vSize, vTop + xSize - stroke },
 | |
| 			{ vLeft + vSize, vTop + xSize - stroke },
 | |
| 			{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 			{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 			{ vLeft + vSize - stroke, vTop + xSize },
 | |
| 			{ vLeft + vSize - 2 * stroke, vTop + xSize - stroke },
 | |
| 			{ vLeft + vSize - 2 * stroke, vTop + xSize - stroke },
 | |
| 		};
 | |
| 
 | |
| 		p.fillPath(anim::path(pathV), brush);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QSize ToggleView::rippleSize() const {
 | |
| 	return getSize() + 2 * QSize(_st->rippleAreaPadding, _st->rippleAreaPadding);
 | |
| }
 | |
| 
 | |
| QImage ToggleView::prepareRippleMask() const {
 | |
| 	auto size = rippleSize();
 | |
| 	return RippleAnimation::RoundRectMask(size, size.height() / 2);
 | |
| }
 | |
| 
 | |
| bool ToggleView::checkRippleStartPosition(QPoint position) const {
 | |
| 	return QRect(QPoint(0, 0), rippleSize()).contains(position);
 | |
| }
 | |
| 
 | |
| void ToggleView::setLocked(bool locked) {
 | |
| 	if (_locked != locked) {
 | |
| 		_locked = locked;
 | |
| 		update();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| CheckView::CheckView(const style::Check &st, bool checked, Fn<void()> updateCallback) : AbstractCheckView(st.duration, checked, std::move(updateCallback))
 | |
| , _st(&st) {
 | |
| }
 | |
| 
 | |
| QSize CheckView::getSize() const {
 | |
| 	return QSize(_st->diameter, _st->diameter);
 | |
| }
 | |
| 
 | |
| void CheckView::setStyle(const style::Check &st) {
 | |
| 	_st = &st;
 | |
| }
 | |
| 
 | |
| void CheckView::paint(QPainter &p, int left, int top, int outerWidth) {
 | |
| 	auto toggled = currentAnimationValue();
 | |
| 	auto pen = _untoggledOverride
 | |
| 		? anim::pen(*_untoggledOverride, _st->toggledFg, toggled)
 | |
| 		: anim::pen(_st->untoggledFg, _st->toggledFg, toggled);
 | |
| 	pen.setWidth(_st->thickness);
 | |
| 	p.setPen(pen);
 | |
| 	p.setBrush(anim::brush(
 | |
| 		_st->bg,
 | |
| 		(_untoggledOverride
 | |
| 			? anim::color(*_untoggledOverride, _st->toggledFg, toggled)
 | |
| 			: anim::color(_st->untoggledFg, _st->toggledFg, toggled)),
 | |
| 		toggled));
 | |
| 
 | |
| 	{
 | |
| 		PainterHighQualityEnabler hq(p);
 | |
| 		p.drawRoundedRect(style::rtlrect(QRectF(left, top, _st->diameter, _st->diameter).marginsRemoved(QMarginsF(_st->thickness / 2., _st->thickness / 2., _st->thickness / 2., _st->thickness / 2.)), outerWidth), st::roundRadiusSmall - (_st->thickness / 2.), st::roundRadiusSmall - (_st->thickness / 2.));
 | |
| 	}
 | |
| 
 | |
| 	if (toggled > 0) {
 | |
| 		_st->icon.paint(p, QPoint(left, top), outerWidth);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QSize CheckView::rippleSize() const {
 | |
| 	return getSize() + 2 * QSize(_st->rippleAreaPadding, _st->rippleAreaPadding);
 | |
| }
 | |
| 
 | |
| QImage CheckView::prepareRippleMask() const {
 | |
| 	return RippleAnimation::EllipseMask(rippleSize());
 | |
| }
 | |
| 
 | |
| bool CheckView::checkRippleStartPosition(QPoint position) const {
 | |
| 	return QRect(QPoint(0, 0), rippleSize()).contains(position);
 | |
| }
 | |
| 
 | |
| void CheckView::setUntoggledOverride(
 | |
| 		std::optional<QColor> untoggledOverride) {
 | |
| 	_untoggledOverride = untoggledOverride;
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| RadioView::RadioView(
 | |
| 	const style::Radio &st,
 | |
| 	bool checked,
 | |
| 	Fn<void()> updateCallback)
 | |
| : AbstractCheckView(st.duration, checked, std::move(updateCallback))
 | |
| , _st(&st) {
 | |
| }
 | |
| 
 | |
| QSize RadioView::getSize() const {
 | |
| 	return QSize(_st->diameter, _st->diameter);
 | |
| }
 | |
| 
 | |
| void RadioView::setStyle(const style::Radio &st) {
 | |
| 	_st = &st;
 | |
| }
 | |
| 
 | |
| void RadioView::paint(QPainter &p, int left, int top, int outerWidth) {
 | |
| 	PainterHighQualityEnabler hq(p);
 | |
| 
 | |
| 	auto toggled = currentAnimationValue();
 | |
| 	auto pen = _toggledOverride
 | |
| 		? (_untoggledOverride
 | |
| 			? anim::pen(*_untoggledOverride, *_toggledOverride, toggled)
 | |
| 			: anim::pen(_st->untoggledFg, *_toggledOverride, toggled))
 | |
| 		: (_untoggledOverride
 | |
| 			? anim::pen(*_untoggledOverride, _st->toggledFg, toggled)
 | |
| 			: anim::pen(_st->untoggledFg, _st->toggledFg, toggled));
 | |
| 	pen.setWidth(_st->thickness);
 | |
| 	p.setPen(pen);
 | |
| 	p.setBrush(_st->bg);
 | |
| 	//int32 skip = qCeil(_st->thickness / 2.);
 | |
| 	//p.drawEllipse(_checkRect.marginsRemoved(QMargins(skip, skip, skip, skip)));
 | |
| 	const auto skip = (_st->outerSkip / 10.) + (_st->thickness / 2);
 | |
| 	p.drawEllipse(style::rtlrect(QRectF(left, top, _st->diameter, _st->diameter).marginsRemoved(QMarginsF(skip, skip, skip, skip)), outerWidth));
 | |
| 
 | |
| 	if (toggled > 0) {
 | |
| 		p.setPen(Qt::NoPen);
 | |
| 		p.setBrush(_toggledOverride
 | |
| 			? (_untoggledOverride
 | |
| 				? anim::brush(*_untoggledOverride, *_toggledOverride, toggled)
 | |
| 				: anim::brush(_st->untoggledFg, *_toggledOverride, toggled))
 | |
| 			: (_untoggledOverride
 | |
| 				? anim::brush(*_untoggledOverride, _st->toggledFg, toggled)
 | |
| 				: anim::brush(_st->untoggledFg, _st->toggledFg, toggled)));
 | |
| 
 | |
| 		const auto skip0 = _st->diameter / 2., skip1 = _st->skip / 10., checkSkip = skip0 * (1. - toggled) + skip1 * toggled;
 | |
| 		p.drawEllipse(style::rtlrect(QRectF(left, top, _st->diameter, _st->diameter).marginsRemoved(QMarginsF(checkSkip, checkSkip, checkSkip, checkSkip)), outerWidth));
 | |
| 		//int32 fskip = qFloor(checkSkip), cskip = qCeil(checkSkip);
 | |
| 		//if (2 * fskip < _checkRect.width()) {
 | |
| 		//	if (fskip != cskip) {
 | |
| 		//		p.setOpacity(float64(cskip) - checkSkip);
 | |
| 		//		p.drawEllipse(_checkRect.marginsRemoved(QMargins(fskip, fskip, fskip, fskip)));
 | |
| 		//		p.setOpacity(1.);
 | |
| 		//	}
 | |
| 		//	if (2 * cskip < _checkRect.width()) {
 | |
| 		//		p.drawEllipse(_checkRect.marginsRemoved(QMargins(cskip, cskip, cskip, cskip)));
 | |
| 		//	}
 | |
| 		//}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QSize RadioView::rippleSize() const {
 | |
| 	return getSize() + 2 * QSize(_st->rippleAreaPadding, _st->rippleAreaPadding);
 | |
| }
 | |
| 
 | |
| QImage RadioView::prepareRippleMask() const {
 | |
| 	return RippleAnimation::EllipseMask(rippleSize());
 | |
| }
 | |
| 
 | |
| bool RadioView::checkRippleStartPosition(QPoint position) const {
 | |
| 	return QRect(QPoint(0, 0), rippleSize()).contains(position);
 | |
| }
 | |
| 
 | |
| void RadioView::setToggledOverride(std::optional<QColor> toggledOverride) {
 | |
| 	_toggledOverride = toggledOverride;
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void RadioView::setUntoggledOverride(
 | |
| 		std::optional<QColor> untoggledOverride) {
 | |
| 	_untoggledOverride = untoggledOverride;
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	const QString &text,
 | |
| 	bool checked,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Check &checkSt)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	rpl::single(text) | rpl::map(TextWithEntities::Simple),
 | |
| 	st,
 | |
| 	std::make_unique<CheckView>(
 | |
| 		checkSt,
 | |
| 		checked)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	const TextWithEntities &text,
 | |
| 	bool checked,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Check &checkSt)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	rpl::single(text),
 | |
| 	st,
 | |
| 	std::make_unique<CheckView>(
 | |
| 		checkSt,
 | |
| 		checked)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	const QString &text,
 | |
| 	bool checked,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Toggle &toggleSt)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	rpl::single(text) | rpl::map(TextWithEntities::Simple),
 | |
| 	st,
 | |
| 	std::make_unique<ToggleView>(
 | |
| 		toggleSt,
 | |
| 		checked)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	rpl::producer<QString> &&text,
 | |
| 	bool checked,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Check &checkSt)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	std::move(text) | rpl::map(TextWithEntities::Simple),
 | |
| 	st,
 | |
| 	std::make_unique<CheckView>(
 | |
| 		checkSt,
 | |
| 		checked)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	rpl::producer<QString> &&text,
 | |
| 	bool checked,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Toggle &toggleSt)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	std::move(text) | rpl::map(TextWithEntities::Simple),
 | |
| 	st,
 | |
| 	std::make_unique<ToggleView>(
 | |
| 		toggleSt,
 | |
| 		checked)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	const QString &text,
 | |
| 	const style::Checkbox &st,
 | |
| 	std::unique_ptr<AbstractCheckView> check)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	rpl::single(text) | rpl::map(TextWithEntities::Simple),
 | |
| 	st,
 | |
| 	std::move(check)) {
 | |
| }
 | |
| 
 | |
| Checkbox::Checkbox(
 | |
| 	QWidget *parent,
 | |
| 	rpl::producer<TextWithEntities> &&text,
 | |
| 	const style::Checkbox &st,
 | |
| 	std::unique_ptr<AbstractCheckView> check)
 | |
| : RippleButton(parent, st.ripple)
 | |
| , _st(st)
 | |
| , _check(std::move(check))
 | |
| , _text(
 | |
| 		_st.style,
 | |
| 		QString(),
 | |
| 		_checkboxOptions,
 | |
| 		countTextMinWidth()) {
 | |
| 	_check->setUpdateCallback([=] { update(); });
 | |
| 	resizeToText();
 | |
| 	setCursor(style::cur_pointer);
 | |
| 	std::move(
 | |
| 		text
 | |
| 	) | rpl::start_with_next([=](TextWithEntities &&value) {
 | |
| 		if (value.entities.empty()) {
 | |
| 			setText(base::take(value.text));
 | |
| 		} else {
 | |
| 			_text.setMarkedText(
 | |
| 				_st.style,
 | |
| 				std::move(value),
 | |
| 				_checkboxRichOptions);
 | |
| 			resizeToText();
 | |
| 			if (_text.hasLinks()) {
 | |
| 				setMouseTracking(true);
 | |
| 			}
 | |
| 			update();
 | |
| 		}
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| int Checkbox::countTextMinWidth() const {
 | |
| 	const auto leftSkip = _st.checkPosition.x()
 | |
| 		+ checkRect().width()
 | |
| 		+ _st.textPosition.x();
 | |
| 	return (_st.width > 0)
 | |
| 		? std::max(_st.width - leftSkip, 1)
 | |
| 		: QFIXED_MAX;
 | |
| }
 | |
| 
 | |
| QRect Checkbox::checkRect() const {
 | |
| 	auto size = _check->getSize();
 | |
| 	return QRect({
 | |
| 		(_checkAlignment & Qt::AlignHCenter)
 | |
| 			? (width() - size.width()) / 2
 | |
| 			: (_checkAlignment & Qt::AlignRight)
 | |
| 				? (width() - _st.checkPosition.x() - size.width())
 | |
| 				: _st.checkPosition.x(),
 | |
| 		(_checkAlignment & Qt::AlignVCenter)
 | |
| 			? (height() - size.height()) / 2
 | |
| 			: (_checkAlignment & Qt::AlignBottom)
 | |
| 				? (height() - _st.checkPosition.y() - size.height())
 | |
| 				: _st.checkPosition.y()
 | |
| 	}, size);
 | |
| }
 | |
| 
 | |
| void Checkbox::setText(const QString &text, bool rich) {
 | |
| 	_text.setText(_st.style, text, rich ? _checkboxRichOptions : _checkboxOptions);
 | |
| 	resizeToText();
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void Checkbox::setCheckAlignment(style::align alignment) {
 | |
| 	if (_checkAlignment != alignment) {
 | |
| 		_checkAlignment = alignment;
 | |
| 		resizeToText();
 | |
| 		update();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::setAllowTextLines(int lines) {
 | |
| 	_allowTextLines = lines;
 | |
| 	resizeToText();
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void Checkbox::setTextBreakEverywhere(bool allow) {
 | |
| 	_textBreakEverywhere = allow;
 | |
| }
 | |
| 
 | |
| void Checkbox::setLink(uint16 lnkIndex, const ClickHandlerPtr &lnk) {
 | |
| 	_text.setLink(lnkIndex, lnk);
 | |
| }
 | |
| 
 | |
| void Checkbox::setLinksTrusted() {
 | |
| 	static const auto TrustedLinksFilter = [](
 | |
| 			const ClickHandlerPtr &link,
 | |
| 			Qt::MouseButton button) {
 | |
| 		if (const auto url = dynamic_cast<UrlClickHandler*>(link.get())) {
 | |
| 			url->UrlClickHandler::onClick({ button });
 | |
| 			return false;
 | |
| 		}
 | |
| 		return true;
 | |
| 	};
 | |
| 	setClickHandlerFilter(TrustedLinksFilter);
 | |
| }
 | |
| 
 | |
| void Checkbox::setClickHandlerFilter(ClickHandlerFilter &&filter) {
 | |
| 	_clickHandlerFilter = std::move(filter);
 | |
| }
 | |
| 
 | |
| bool Checkbox::checked() const {
 | |
| 	return _check->checked();
 | |
| }
 | |
| 
 | |
| rpl::producer<bool> Checkbox::checkedChanges() const {
 | |
| 	return _checkedChanges.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<bool> Checkbox::checkedValue() const {
 | |
| 	return _checkedChanges.events_starting_with(checked());
 | |
| }
 | |
| 
 | |
| void Checkbox::resizeToText() {
 | |
| 	if (_st.width <= 0) {
 | |
| 		resizeToWidth(_text.maxWidth() - _st.width);
 | |
| 	} else {
 | |
| 		resizeToWidth(_st.width);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::setChecked(bool checked, NotifyAboutChange notify) {
 | |
| 	if (_check->checked() != checked) {
 | |
| 		_check->setChecked(checked, anim::type::normal);
 | |
| 		if (notify == NotifyAboutChange::Notify) {
 | |
| 			_checkedChanges.fire_copy(checked);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::finishAnimating() {
 | |
| 	_check->finishAnimating();
 | |
| }
 | |
| 
 | |
| int Checkbox::naturalWidth() const {
 | |
| 	if (_st.width > 0) {
 | |
| 		return _st.width;
 | |
| 	}
 | |
| 	auto result = _st.checkPosition.x() + _check->getSize().width();
 | |
| 	if (!_text.isEmpty()) {
 | |
| 		result += _st.textPosition.x() + _text.maxWidth();
 | |
| 	}
 | |
| 	return result - _st.width;
 | |
| }
 | |
| 
 | |
| void Checkbox::paintEvent(QPaintEvent *e) {
 | |
| 	Painter p(this);
 | |
| 
 | |
| 	auto check = checkRect();
 | |
| 	auto active = _check->currentAnimationValue();
 | |
| 	if (isDisabled()) {
 | |
| 		p.setOpacity(_st.disabledOpacity);
 | |
| 	} else {
 | |
| 		auto color = anim::color(_st.rippleBg, _st.rippleBgActive, active);
 | |
| 		paintRipple(p, check.topLeft() + _st.rippleAreaPosition, &color);
 | |
| 	}
 | |
| 
 | |
| 	auto realCheckRect = myrtlrect(check);
 | |
| 	if (realCheckRect.intersects(e->rect())) {
 | |
| 		if (isDisabled()) {
 | |
| 			p.drawPixmapLeft(check.left(), check.top(), width(), _checkCache);
 | |
| 		} else {
 | |
| 			_check->paint(p, check.left(), check.top(), width());
 | |
| 		}
 | |
| 	}
 | |
| 	if (realCheckRect.contains(e->rect()) || _text.isEmpty()) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	const auto alignLeft = (_checkAlignment & Qt::AlignLeft);
 | |
| 	const auto alignRight = (_checkAlignment & Qt::AlignRight);
 | |
| 	const auto centered = ((_checkAlignment & Qt::AlignHCenter) != 0);
 | |
| 	const auto textSkip = _st.checkPosition.x()
 | |
| 		+ check.width()
 | |
| 		+ _st.textPosition.x();
 | |
| 	const auto availableTextWidth = centered
 | |
| 		? std::max(width() - _st.margin.left() - _st.margin.right(), 1)
 | |
| 		: std::max(width() - textSkip, 1);
 | |
| 	const auto textTop = _st.margin.top() + _st.textPosition.y();
 | |
| 
 | |
| 	p.setPen(anim::pen(_st.textFg, _st.textFgActive, active));
 | |
| 	if (alignLeft) {
 | |
| 		if (!_allowTextLines) {
 | |
| 			_text.drawLeft(
 | |
| 				p,
 | |
| 				textSkip,
 | |
| 				textTop,
 | |
| 				availableTextWidth,
 | |
| 				width());
 | |
| 		} else {
 | |
| 			_text.drawLeftElided(
 | |
| 				p,
 | |
| 				textSkip,
 | |
| 				textTop,
 | |
| 				availableTextWidth,
 | |
| 				width(),
 | |
| 				_allowTextLines,
 | |
| 				style::al_left,
 | |
| 				0,
 | |
| 				-1,
 | |
| 				0,
 | |
| 				_textBreakEverywhere);
 | |
| 		}
 | |
| 	} else if (alignRight) {
 | |
| 		if (!_allowTextLines) {
 | |
| 			_text.drawRight(
 | |
| 				p,
 | |
| 				textSkip,
 | |
| 				textTop,
 | |
| 				availableTextWidth,
 | |
| 				width());
 | |
| 		} else {
 | |
| 			_text.drawRightElided(
 | |
| 				p,
 | |
| 				textSkip,
 | |
| 				textTop,
 | |
| 				availableTextWidth,
 | |
| 				width(),
 | |
| 				_allowTextLines,
 | |
| 				style::al_left,
 | |
| 				0,
 | |
| 				-1,
 | |
| 				0,
 | |
| 				_textBreakEverywhere);
 | |
| 		}
 | |
| 	} else if (!_allowTextLines
 | |
| 		|| (_text.countHeight(availableTextWidth)
 | |
| 			< (_allowTextLines + 1) * _st.style.font->height)) {
 | |
| 		_text.drawLeft(
 | |
| 			p,
 | |
| 			_st.margin.left(),
 | |
| 			textTop,
 | |
| 			width() - _st.margin.left() - _st.margin.right(),
 | |
| 			width(),
 | |
| 			style::al_top);
 | |
| 	} else {
 | |
| 		_text.drawLeftElided(
 | |
| 			p,
 | |
| 			_st.margin.left(),
 | |
| 			textTop,
 | |
| 			width() - _st.margin.left() - _st.margin.right(),
 | |
| 			width(),
 | |
| 			_allowTextLines,
 | |
| 			style::al_top,
 | |
| 			0,
 | |
| 			-1,
 | |
| 			0,
 | |
| 			_textBreakEverywhere);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::mousePressEvent(QMouseEvent *e) {
 | |
| 	RippleButton::mousePressEvent(e);
 | |
| 	ClickHandler::pressed();
 | |
| }
 | |
| 
 | |
| void Checkbox::mouseMoveEvent(QMouseEvent *e) {
 | |
| 	RippleButton::mouseMoveEvent(e);
 | |
| 	const auto state = getTextState(e->pos());
 | |
| 	if (state.link != ClickHandler::getActive()) {
 | |
| 		ClickHandler::setActive(state.link, this);
 | |
| 		update();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::mouseReleaseEvent(QMouseEvent *e) {
 | |
| 	const auto weak = Ui::MakeWeak(this);
 | |
| 	if (auto activated = _activatingHandler = ClickHandler::unpressed()) {
 | |
| 		// _clickHandlerFilter may delete `this`. In that case we don't want
 | |
| 		// to try to show a context menu or smth like that.
 | |
| 		const auto button = e->button();
 | |
| 		crl::on_main(this, [=] {
 | |
| 			const auto guard = window();
 | |
| 			if (!_clickHandlerFilter
 | |
| 				|| _clickHandlerFilter(activated, button)) {
 | |
| 				ActivateClickHandler(guard, activated, button);
 | |
| 			}
 | |
| 		});
 | |
| 	}
 | |
| 	RippleButton::mouseReleaseEvent(e);
 | |
| 	if (weak) {
 | |
| 		_activatingHandler = nullptr;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::leaveEventHook(QEvent *e) {
 | |
| 	RippleButton::leaveEventHook(e);
 | |
| 	ClickHandler::clearActive(this);
 | |
| }
 | |
| 
 | |
| Text::StateResult Checkbox::getTextState(const QPoint &m) const {
 | |
| 	if (!(_checkAlignment & Qt::AlignLeft)) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	const auto check = checkRect();
 | |
| 	const auto textSkip = _st.checkPosition.x()
 | |
| 		+ check.width()
 | |
| 		+ _st.textPosition.x();
 | |
| 	const auto availableTextWidth = std::max(width() - textSkip, 1);
 | |
| 	const auto textTop = _st.margin.top() + _st.textPosition.y();
 | |
| 	return !_allowTextLines
 | |
| 		? _text.getStateElided(
 | |
| 			m - QPoint(textSkip, textTop),
 | |
| 			availableTextWidth,
 | |
| 			{})
 | |
| 		: _text.getStateElidedLeft(
 | |
| 			m - QPoint(textSkip, textTop),
 | |
| 			availableTextWidth,
 | |
| 			width(),
 | |
| 			{});
 | |
| }
 | |
| 
 | |
| QPixmap Checkbox::grabCheckCache() const {
 | |
| 	auto checkSize = _check->getSize();
 | |
| 	auto image = QImage(
 | |
| 		checkSize * style::DevicePixelRatio(),
 | |
| 		QImage::Format_ARGB32_Premultiplied);
 | |
| 	image.fill(Qt::transparent);
 | |
| 	image.setDevicePixelRatio(style::DevicePixelRatio());
 | |
| 	{
 | |
| 		Painter p(&image);
 | |
| 		_check->paint(p, 0, 0, checkSize.width());
 | |
| 	}
 | |
| 	return PixmapFromImage(std::move(image));
 | |
| }
 | |
| 
 | |
| void Checkbox::onStateChanged(State was, StateChangeSource source) {
 | |
| 	RippleButton::onStateChanged(was, source);
 | |
| 
 | |
| 	if (isDisabled() && !(was & StateFlag::Disabled)) {
 | |
| 		setCursor(style::cur_default);
 | |
| 		finishAnimating();
 | |
| 		_checkCache = grabCheckCache();
 | |
| 	} else if (!isDisabled() && (was & StateFlag::Disabled)) {
 | |
| 		setCursor(style::cur_pointer);
 | |
| 		_checkCache = QPixmap();
 | |
| 	}
 | |
| 
 | |
| 	auto now = state();
 | |
| 	if (!isDisabled() && (was & StateFlag::Over) && (now & StateFlag::Over)) {
 | |
| 		if ((was & StateFlag::Down) && !(now & StateFlag::Down)) {
 | |
| 			handlePress();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Checkbox::handlePress() {
 | |
| 	if (!_activatingHandler) {
 | |
| 		setChecked(!checked());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int Checkbox::resizeGetHeight(int newWidth) {
 | |
| 	const auto result = _check->getSize().height();
 | |
| 	const auto centered = ((_checkAlignment & Qt::AlignHCenter) != 0);
 | |
| 	if (!centered && _allowTextLines == 1) {
 | |
| 		return result;
 | |
| 	}
 | |
| 	const auto textSkip = _st.checkPosition.x()
 | |
| 		+ checkRect().width()
 | |
| 		+ _st.textPosition.x();
 | |
| 	const auto fullWidth = _st.margin.left() + newWidth + _st.margin.right();
 | |
| 	const auto availableTextWidth = centered
 | |
| 		? std::max(newWidth, 1)
 | |
| 		: std::max(fullWidth - textSkip, 1);
 | |
| 	const auto textHeight = _text.countHeight(availableTextWidth);
 | |
| 	const auto textBottom = _st.textPosition.y()
 | |
| 		+ (_allowTextLines
 | |
| 			? std::min(textHeight, _allowTextLines * _st.style.font->height)
 | |
| 			: textHeight);
 | |
| 	return std::max(result, textBottom);
 | |
| }
 | |
| 
 | |
| QImage Checkbox::prepareRippleMask() const {
 | |
| 	return _check->prepareRippleMask();
 | |
| }
 | |
| 
 | |
| QPoint Checkbox::prepareRippleStartPosition() const {
 | |
| 	if (isDisabled()) {
 | |
| 		return DisabledRippleStartPosition();
 | |
| 	}
 | |
| 	auto position = myrtlpoint(mapFromGlobal(QCursor::pos()))
 | |
| 		- checkRect().topLeft()
 | |
| 		- _st.rippleAreaPosition;
 | |
| 	return _check->checkRippleStartPosition(position)
 | |
| 		? position
 | |
| 		: DisabledRippleStartPosition();
 | |
| }
 | |
| 
 | |
| void RadiobuttonGroup::setValue(int value) {
 | |
| 	if (_hasValue && _value == value) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_hasValue = true;
 | |
| 	_value = value;
 | |
| 	for (const auto button : _buttons) {
 | |
| 		button->handleNewGroupValue(_value);
 | |
| 	}
 | |
| 	if (const auto callback = _changedCallback) {
 | |
| 		callback(_value);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| Radiobutton::Radiobutton(
 | |
| 	QWidget *parent,
 | |
| 	const std::shared_ptr<RadiobuttonGroup> &group,
 | |
| 	int value,
 | |
| 	const QString &text,
 | |
| 	const style::Checkbox &st,
 | |
| 	const style::Radio &radioSt)
 | |
| : Radiobutton(
 | |
| 	parent,
 | |
| 	group,
 | |
| 	value,
 | |
| 	text,
 | |
| 	st,
 | |
| 	std::make_unique<RadioView>(
 | |
| 		radioSt,
 | |
| 		(group->hasValue() && group->value() == value))) {
 | |
| }
 | |
| 
 | |
| Radiobutton::Radiobutton(
 | |
| 	QWidget *parent,
 | |
| 	const std::shared_ptr<RadiobuttonGroup> &group,
 | |
| 	int value,
 | |
| 	const QString &text,
 | |
| 	const style::Checkbox &st,
 | |
| 	std::unique_ptr<AbstractCheckView> check)
 | |
| : Checkbox(
 | |
| 	parent,
 | |
| 	text,
 | |
| 	st,
 | |
| 	std::move(check))
 | |
| , _group(group)
 | |
| , _value(value) {
 | |
| 	using namespace rpl::mappers;
 | |
| 
 | |
| 	checkbox()->setChecked(group->hasValue() && group->value() == value);
 | |
| 	_group->registerButton(this);
 | |
| 	checkbox()->checkedChanges(
 | |
| 	) | rpl::filter(
 | |
| 		_1
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		_group->setValue(_value);
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void Radiobutton::handleNewGroupValue(int value) {
 | |
| 	auto checked = (value == _value);
 | |
| 	if (checkbox()->checked() != checked) {
 | |
| 		checkbox()->setChecked(
 | |
| 			checked,
 | |
| 			Ui::Checkbox::NotifyAboutChange::DontNotify);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Radiobutton::handlePress() {
 | |
| 	if (!checkbox()->checked()) {
 | |
| 		checkbox()->setChecked(true);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| Radiobutton::~Radiobutton() {
 | |
| 	_group->unregisterButton(this);
 | |
| }
 | |
| 
 | |
| } // namespace Ui
 | 
