545 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			545 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "window/section_widget.h"
 | |
| 
 | |
| #include "mainwidget.h"
 | |
| #include "ui/ui_utility.h"
 | |
| #include "ui/chat/chat_theme.h"
 | |
| #include "ui/painter.h"
 | |
| #include "boxes/premium_preview_box.h"
 | |
| #include "data/data_peer.h"
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_document.h"
 | |
| #include "data/data_document_media.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_cloud_themes.h"
 | |
| #include "data/data_message_reactions.h"
 | |
| #include "data/data_peer_values.h"
 | |
| #include "history/history.h"
 | |
| #include "history/history_item.h"
 | |
| #include "settings/settings_premium.h"
 | |
| #include "main/main_session.h"
 | |
| #include "window/section_memento.h"
 | |
| #include "window/window_slide_animation.h"
 | |
| #include "window/window_session_controller.h"
 | |
| #include "window/themes/window_theme.h"
 | |
| 
 | |
| #include <rpl/range.h>
 | |
| 
 | |
| namespace Window {
 | |
| namespace {
 | |
| 
 | |
| [[nodiscard]] rpl::producer<QString> PeerThemeEmojiValue(
 | |
| 		not_null<PeerData*> peer) {
 | |
| 	return peer->session().changes().peerFlagsValue(
 | |
| 		peer,
 | |
| 		Data::PeerUpdate::Flag::ChatThemeEmoji
 | |
| 	) | rpl::map([=] {
 | |
| 		return peer->themeEmoji();
 | |
| 	});
 | |
| }
 | |
| 
 | |
| struct ResolvedPaper {
 | |
| 	Data::WallPaper paper;
 | |
| 	std::shared_ptr<Data::DocumentMedia> media;
 | |
| };
 | |
| 
 | |
| [[nodiscard]] rpl::producer<const Data::WallPaper*> PeerWallPaperMapped(
 | |
| 		not_null<PeerData*> peer) {
 | |
| 	return peer->session().changes().peerFlagsValue(
 | |
| 		peer,
 | |
| 		Data::PeerUpdate::Flag::ChatWallPaper
 | |
| 	) | rpl::map([=]() -> rpl::producer<const Data::WallPaper*> {
 | |
| 		return WallPaperResolved(&peer->owner(), peer->wallPaper());
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| [[nodiscard]] rpl::producer<std::optional<ResolvedPaper>> PeerWallPaperValue(
 | |
| 		not_null<PeerData*> peer) {
 | |
| 	return PeerWallPaperMapped(
 | |
| 		peer
 | |
| 	) | rpl::map([=](const Data::WallPaper *paper)
 | |
| 	-> rpl::producer<std::optional<ResolvedPaper>> {
 | |
| 		const auto single = [](std::optional<ResolvedPaper> value) {
 | |
| 			return rpl::single(std::move(value));
 | |
| 		};
 | |
| 		if (!paper) {
 | |
| 			return single({});
 | |
| 		}
 | |
| 		const auto document = paper->document();
 | |
| 		auto value = ResolvedPaper{
 | |
| 			*paper,
 | |
| 			document ? document->createMediaView() : nullptr,
 | |
| 		};
 | |
| 		if (!value.media || value.media->loaded(true)) {
 | |
| 			return single(std::move(value));
 | |
| 		}
 | |
| 		paper->loadDocument();
 | |
| 		return single(
 | |
| 			value
 | |
| 		) | rpl::then(document->session().downloaderTaskFinished(
 | |
| 		) | rpl::filter([=] {
 | |
| 			return value.media->loaded(true);
 | |
| 		}) | rpl::take(1) | rpl::map_to(
 | |
| 			std::optional<ResolvedPaper>(value)
 | |
| 		));
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| [[nodiscard]] auto MaybeChatThemeDataValueFromPeer(
 | |
| 	not_null<PeerData*> peer)
 | |
| -> rpl::producer<std::optional<Data::CloudTheme>> {
 | |
| 	return PeerThemeEmojiValue(
 | |
| 		peer
 | |
| 	) | rpl::map([=](const QString &emoji)
 | |
| 	-> rpl::producer<std::optional<Data::CloudTheme>> {
 | |
| 		return peer->owner().cloudThemes().themeForEmojiValue(emoji);
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| [[nodiscard]] rpl::producer<> DebouncedPaletteValue() {
 | |
| 	return [=](auto consumer) {
 | |
| 		auto lifetime = rpl::lifetime();
 | |
| 
 | |
| 		struct State {
 | |
| 			base::has_weak_ptr guard;
 | |
| 			bool scheduled = false;
 | |
| 		};
 | |
| 		const auto state = lifetime.make_state<State>();
 | |
| 
 | |
| 		consumer.put_next_copy(rpl::empty);
 | |
| 		style::PaletteChanged(
 | |
| 		) | rpl::start_with_next([=] {
 | |
| 			if (state->scheduled) {
 | |
| 				return;
 | |
| 			}
 | |
| 			state->scheduled = true;
 | |
| 			Ui::PostponeCall(&state->guard, [=] {
 | |
| 				state->scheduled = false;
 | |
| 				consumer.put_next_copy(rpl::empty);
 | |
| 			});
 | |
| 		}, lifetime);
 | |
| 
 | |
| 		return lifetime;
 | |
| 	};
 | |
| }
 | |
| 
 | |
| struct ResolvedTheme {
 | |
| 	std::optional<Data::CloudTheme> theme;
 | |
| 	std::optional<ResolvedPaper> paper;
 | |
| 	bool dark = false;
 | |
| };
 | |
| 
 | |
| [[nodiscard]] auto MaybeCloudThemeValueFromPeer(
 | |
| 	not_null<PeerData*> peer)
 | |
| -> rpl::producer<ResolvedTheme> {
 | |
| 	return rpl::combine(
 | |
| 		MaybeChatThemeDataValueFromPeer(peer),
 | |
| 		PeerWallPaperValue(peer),
 | |
| 		Theme::IsThemeDarkValue() | rpl::distinct_until_changed()
 | |
| 	) | rpl::map([](
 | |
| 			std::optional<Data::CloudTheme> theme,
 | |
| 			std::optional<ResolvedPaper> paper,
 | |
| 			bool night) -> rpl::producer<ResolvedTheme> {
 | |
| 		if (theme || !paper) {
 | |
| 			return rpl::single<ResolvedTheme>({
 | |
| 				std::move(theme),
 | |
| 				std::move(paper),
 | |
| 				night,
 | |
| 			});
 | |
| 		}
 | |
| 		return DebouncedPaletteValue(
 | |
| 		) | rpl::map([=] {
 | |
| 			return ResolvedTheme{
 | |
| 				.paper = paper,
 | |
| 				.dark = night,
 | |
| 			};
 | |
| 		});
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| rpl::producer<const Data::WallPaper*> WallPaperResolved(
 | |
| 		not_null<Data::Session*> owner,
 | |
| 		const Data::WallPaper *paper) {
 | |
| 	const auto id = paper ? paper->emojiId() : QString();
 | |
| 	if (id.isEmpty()) {
 | |
| 		return rpl::single(paper);
 | |
| 	}
 | |
| 	const auto themes = &owner->cloudThemes();
 | |
| 	auto fromThemes = [=](bool force)
 | |
| 	-> rpl::producer<const Data::WallPaper*> {
 | |
| 		if (themes->chatThemes().empty() && !force) {
 | |
| 			return nullptr;
 | |
| 		}
 | |
| 		return Window::Theme::IsNightModeValue(
 | |
| 		) | rpl::map([=](bool dark) -> const Data::WallPaper* {
 | |
| 			const auto &list = themes->chatThemes();
 | |
| 			const auto i = ranges::find(
 | |
| 				list,
 | |
| 				id,
 | |
| 				&Data::CloudTheme::emoticon);
 | |
| 			if (i != end(list)) {
 | |
| 				using Type = Data::CloudThemeType;
 | |
| 				const auto type = dark ? Type::Dark : Type::Light;
 | |
| 				const auto j = i->settings.find(type);
 | |
| 				if (j != end(i->settings) && j->second.paper) {
 | |
| 					return &*j->second.paper;
 | |
| 				}
 | |
| 			}
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	};
 | |
| 	if (auto result = fromThemes(false)) {
 | |
| 		return result;
 | |
| 	}
 | |
| 	themes->refreshChatThemes();
 | |
| 	return themes->chatThemesUpdated(
 | |
| 	) | rpl::take(1) | rpl::map([=] {
 | |
| 		return fromThemes(true);
 | |
| 	}) | rpl::flatten_latest();
 | |
| }
 | |
| 
 | |
| AbstractSectionWidget::AbstractSectionWidget(
 | |
| 	QWidget *parent,
 | |
| 	not_null<SessionController*> controller,
 | |
| 	rpl::producer<PeerData*> peerForBackground)
 | |
| : RpWidget(parent)
 | |
| , _controller(controller) {
 | |
| 	std::move(
 | |
| 		peerForBackground
 | |
| 	) | rpl::map([=](PeerData *peer) -> rpl::producer<> {
 | |
| 		if (!peer) {
 | |
| 			return rpl::single(rpl::empty) | rpl::then(
 | |
| 				controller->defaultChatTheme()->repaintBackgroundRequests()
 | |
| 			);
 | |
| 		}
 | |
| 		return ChatThemeValueFromPeer(
 | |
| 			controller,
 | |
| 			peer
 | |
| 		) | rpl::map([](const std::shared_ptr<Ui::ChatTheme> &theme) {
 | |
| 			return rpl::single(rpl::empty) | rpl::then(
 | |
| 				theme->repaintBackgroundRequests()
 | |
| 			);
 | |
| 		}) | rpl::flatten_latest();
 | |
| 	}) | rpl::flatten_latest() | rpl::start_with_next([=] {
 | |
| 		update();
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| Main::Session &AbstractSectionWidget::session() const {
 | |
| 	return _controller->session();
 | |
| }
 | |
| 
 | |
| SectionWidget::SectionWidget(
 | |
| 	QWidget *parent,
 | |
| 	not_null<Window::SessionController*> controller,
 | |
| 	rpl::producer<PeerData*> peerForBackground)
 | |
| : AbstractSectionWidget(parent, controller, std::move(peerForBackground)) {
 | |
| }
 | |
| 
 | |
| SectionWidget::SectionWidget(
 | |
| 	QWidget *parent,
 | |
| 	not_null<Window::SessionController*> controller,
 | |
| 	not_null<PeerData*> peerForBackground)
 | |
| : AbstractSectionWidget(
 | |
| 	parent,
 | |
| 	controller,
 | |
| 	rpl::single(peerForBackground.get())) {
 | |
| }
 | |
| 
 | |
| void SectionWidget::setGeometryWithTopMoved(
 | |
| 		const QRect &newGeometry,
 | |
| 		int topDelta) {
 | |
| 	_topDelta = topDelta;
 | |
| 	bool willBeResized = (size() != newGeometry.size());
 | |
| 	if (geometry() != newGeometry) {
 | |
| 		auto weak = Ui::MakeWeak(this);
 | |
| 		setGeometry(newGeometry);
 | |
| 		if (!weak) {
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!willBeResized) {
 | |
| 		resizeEvent(nullptr);
 | |
| 	}
 | |
| 	_topDelta = 0;
 | |
| }
 | |
| 
 | |
| void SectionWidget::showAnimated(
 | |
| 		SlideDirection direction,
 | |
| 		const SectionSlideParams ¶ms) {
 | |
| 	if (_showAnimation) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	showChildren();
 | |
| 	auto myContentCache = grabForShowAnimation(params);
 | |
| 	hideChildren();
 | |
| 	showAnimatedHook(params);
 | |
| 
 | |
| 	_showAnimation = std::make_unique<SlideAnimation>();
 | |
| 	_showAnimation->setDirection(direction);
 | |
| 	_showAnimation->setRepaintCallback([this] { update(); });
 | |
| 	_showAnimation->setFinishedCallback([this] { showFinished(); });
 | |
| 	_showAnimation->setPixmaps(
 | |
| 		params.oldContentCache,
 | |
| 		myContentCache);
 | |
| 	_showAnimation->setTopBarShadow(params.withTopBarShadow);
 | |
| 	_showAnimation->setWithFade(params.withFade);
 | |
| 	_showAnimation->setTopSkip(params.topSkip);
 | |
| 	_showAnimation->setTopBarMask(params.topMask);
 | |
| 	_showAnimation->start();
 | |
| 
 | |
| 	show();
 | |
| }
 | |
| 
 | |
| std::shared_ptr<SectionMemento> SectionWidget::createMemento() {
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| void SectionWidget::showFast() {
 | |
| 	show();
 | |
| 	showFinished();
 | |
| }
 | |
| 
 | |
| QPixmap SectionWidget::grabForShowAnimation(
 | |
| 		const SectionSlideParams ¶ms) {
 | |
| 	return Ui::GrabWidget(this);
 | |
| }
 | |
| 
 | |
| void SectionWidget::PaintBackground(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<Ui::ChatTheme*> theme,
 | |
| 		not_null<QWidget*> widget,
 | |
| 		QRect clip) {
 | |
| 	PaintBackground(
 | |
| 		theme,
 | |
| 		widget,
 | |
| 		controller->content()->height(),
 | |
| 		controller->content()->backgroundFromY(),
 | |
| 		clip);
 | |
| }
 | |
| 
 | |
| void SectionWidget::PaintBackground(
 | |
| 		not_null<Ui::ChatTheme*> theme,
 | |
| 		not_null<QWidget*> widget,
 | |
| 		int fillHeight,
 | |
| 		int fromy,
 | |
| 		QRect clip) {
 | |
| 	auto p = QPainter(widget);
 | |
| 	if (fromy) {
 | |
| 		p.translate(0, fromy);
 | |
| 		clip = clip.translated(0, -fromy);
 | |
| 	}
 | |
| 	PaintBackground(p, theme, QSize(widget->width(), fillHeight), clip);
 | |
| }
 | |
| 
 | |
| void SectionWidget::PaintBackground(
 | |
| 		QPainter &p,
 | |
| 		not_null<Ui::ChatTheme*> theme,
 | |
| 		QSize fill,
 | |
| 		QRect clip) {
 | |
| 	const auto &background = theme->background();
 | |
| 	if (background.colorForFill) {
 | |
| 		p.fillRect(clip, *background.colorForFill);
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto &gradient = background.gradientForFill;
 | |
| 	auto state = theme->backgroundState(fill);
 | |
| 	const auto paintCache = [&](const Ui::CachedBackground &cache) {
 | |
| 		const auto to = QRect(
 | |
| 			QPoint(cache.x, cache.y),
 | |
| 			cache.pixmap.size() / cIntRetinaFactor());
 | |
| 		if (cache.waitingForNegativePattern) {
 | |
| 			// While we wait for pattern being loaded we paint just gradient.
 | |
| 			// But in case of negative patter opacity we just fill-black.
 | |
| 			p.fillRect(to, Qt::black);
 | |
| 		} else if (cache.area == fill) {
 | |
| 			p.drawPixmap(to, cache.pixmap);
 | |
| 		} else {
 | |
| 			const auto sx = fill.width() / float64(cache.area.width());
 | |
| 			const auto sy = fill.height() / float64(cache.area.height());
 | |
| 			const auto round = [](float64 value) -> int {
 | |
| 				return (value >= 0.)
 | |
| 					? int(std::ceil(value))
 | |
| 					: int(std::floor(value));
 | |
| 			};
 | |
| 			const auto sto = QPoint(round(to.x() * sx), round(to.y() * sy));
 | |
| 			p.drawPixmap(
 | |
| 				sto.x(),
 | |
| 				sto.y(),
 | |
| 				round((to.x() + to.width()) * sx) - sto.x(),
 | |
| 				round((to.y() + to.height()) * sy) - sto.y(),
 | |
| 				cache.pixmap);
 | |
| 		}
 | |
| 	};
 | |
| 	const auto hasNow = !state.now.pixmap.isNull();
 | |
| 	const auto goodNow = hasNow && (state.now.area == fill);
 | |
| 	const auto useCache = goodNow || !gradient.isNull();
 | |
| 	if (useCache) {
 | |
| 		const auto fade = (state.shown < 1. && !gradient.isNull());
 | |
| 		if (fade) {
 | |
| 			paintCache(state.was);
 | |
| 			p.setOpacity(state.shown);
 | |
| 		}
 | |
| 		paintCache(state.now);
 | |
| 		if (fade) {
 | |
| 			p.setOpacity(1.);
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto &prepared = background.prepared;
 | |
| 	if (prepared.isNull()) {
 | |
| 		return;
 | |
| 	} else if (background.isPattern) {
 | |
| 		const auto w = prepared.width() * fill.height() / prepared.height();
 | |
| 		const auto cx = qCeil(fill.width() / float64(w));
 | |
| 		const auto cols = (cx / 2) * 2 + 1;
 | |
| 		const auto xshift = (fill.width() - w * cols) / 2;
 | |
| 		for (auto i = 0; i != cols; ++i) {
 | |
| 			p.drawImage(
 | |
| 				QRect(xshift + i * w, 0, w, fill.height()),
 | |
| 				prepared,
 | |
| 				QRect(QPoint(), prepared.size()));
 | |
| 		}
 | |
| 	} else if (background.tile) {
 | |
| 		const auto &tiled = background.preparedForTiled;
 | |
| 		const auto left = clip.left();
 | |
| 		const auto top = clip.top();
 | |
| 		const auto right = clip.left() + clip.width();
 | |
| 		const auto bottom = clip.top() + clip.height();
 | |
| 		const auto w = tiled.width() / cRetinaFactor();
 | |
| 		const auto h = tiled.height() / cRetinaFactor();
 | |
| 		const auto sx = qFloor(left / w);
 | |
| 		const auto sy = qFloor(top / h);
 | |
| 		const auto cx = qCeil(right / w);
 | |
| 		const auto cy = qCeil(bottom / h);
 | |
| 		for (auto i = sx; i < cx; ++i) {
 | |
| 			for (auto j = sy; j < cy; ++j) {
 | |
| 				p.drawImage(QPointF(i * w, j * h), tiled);
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		const auto hq = PainterHighQualityEnabler(p);
 | |
| 		const auto rects = Ui::ComputeChatBackgroundRects(
 | |
| 			fill,
 | |
| 			prepared.size());
 | |
| 		p.drawImage(rects.to, prepared, rects.from);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void SectionWidget::paintEvent(QPaintEvent *e) {
 | |
| 	if (_showAnimation) {
 | |
| 		auto p = QPainter(this);
 | |
| 		_showAnimation->paintContents(p);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool SectionWidget::animatingShow() const {
 | |
| 	return (_showAnimation != nullptr);
 | |
| }
 | |
| 
 | |
| void SectionWidget::showFinished() {
 | |
| 	_showAnimation.reset();
 | |
| 	if (isHidden()) return;
 | |
| 
 | |
| 	showChildren();
 | |
| 	showFinishedHook();
 | |
| 
 | |
| 	setInnerFocus();
 | |
| }
 | |
| 
 | |
| rpl::producer<int> SectionWidget::desiredHeight() const {
 | |
| 	return rpl::single(height());
 | |
| }
 | |
| 
 | |
| SectionWidget::~SectionWidget() = default;
 | |
| 
 | |
| auto ChatThemeValueFromPeer(
 | |
| 	not_null<SessionController*> controller,
 | |
| 	not_null<PeerData*> peer)
 | |
| -> rpl::producer<std::shared_ptr<Ui::ChatTheme>> {
 | |
| 	auto cloud = MaybeCloudThemeValueFromPeer(
 | |
| 		peer
 | |
| 	) | rpl::map([=](ResolvedTheme resolved)
 | |
| 	-> rpl::producer<std::shared_ptr<Ui::ChatTheme>> {
 | |
| 		if (!resolved.theme && !resolved.paper) {
 | |
| 			return rpl::single(controller->defaultChatTheme());
 | |
| 		}
 | |
| 		const auto theme = resolved.theme.value_or(Data::CloudTheme());
 | |
| 		const auto paper = resolved.paper
 | |
| 			? resolved.paper->paper
 | |
| 			: Data::WallPaper(0);
 | |
| 		const auto type = resolved.dark
 | |
| 			? Data::CloudThemeType::Dark
 | |
| 			: Data::CloudThemeType::Light;
 | |
| 		if (paper.document()
 | |
| 			&& resolved.paper->media
 | |
| 			&& !resolved.paper->media->loaded()
 | |
| 			&& !controller->chatThemeAlreadyCached(theme, paper, type)) {
 | |
| 			return rpl::single(controller->defaultChatTheme());
 | |
| 		}
 | |
| 		return controller->cachedChatThemeValue(theme, paper, type);
 | |
| 	}) | rpl::flatten_latest(
 | |
| 	) | rpl::distinct_until_changed();
 | |
| 
 | |
| 	return rpl::combine(
 | |
| 		std::move(cloud),
 | |
| 		controller->peerThemeOverrideValue()
 | |
| 	) | rpl::map([=](
 | |
| 			std::shared_ptr<Ui::ChatTheme> &&cloud,
 | |
| 			PeerThemeOverride &&overriden) {
 | |
| 		return (overriden.peer == peer.get()
 | |
| 			&& Ui::Emoji::Find(peer->themeEmoji()) != overriden.emoji)
 | |
| 			? std::move(overriden.theme)
 | |
| 			: std::move(cloud);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| bool ShowSendPremiumError(
 | |
| 		not_null<SessionController*> controller,
 | |
| 		not_null<DocumentData*> document) {
 | |
| 	return ShowSendPremiumError(controller->uiShow(), document);
 | |
| }
 | |
| 
 | |
| bool ShowSendPremiumError(
 | |
| 		std::shared_ptr<ChatHelpers::Show> show,
 | |
| 		not_null<DocumentData*> document) {
 | |
| 	if (!document->isPremiumSticker()
 | |
| 		|| document->session().premium()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	ShowStickerPreviewBox(std::move(show), document);
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool ShowReactPremiumError(
 | |
| 		not_null<SessionController*> controller,
 | |
| 		not_null<HistoryItem*> item,
 | |
| 		const Data::ReactionId &id) {
 | |
| 	if (item->reactionsAreTags()) {
 | |
| 		if (controller->session().premium()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		ShowPremiumPreviewBox(controller, PremiumFeature::TagsForMessages);
 | |
| 		return true;
 | |
| 	} else if (controller->session().premium()
 | |
| 		|| ranges::contains(item->chosenReactions(), id)
 | |
| 		|| item->history()->peer->isBroadcast()) {
 | |
| 		return false;
 | |
| 	} else if (!id.custom()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	ShowPremiumPreviewBox(controller, PremiumFeature::InfiniteReactions);
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| } // namespace Window
 | 
