/* 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 "data/data_document_resolver.h" #include "facades.h" #include "base/platform/base_platform_info.h" #include "ui/boxes/confirm_box.h" #include "core/application.h" #include "core/core_settings.h" #include "core/mime_type.h" #include "data/data_document.h" #include "data/data_document_media.h" #include "data/data_file_click_handler.h" #include "data/data_file_origin.h" #include "data/data_session.h" #include "history/view/media/history_view_gif.h" #include "history/history.h" #include "history/history_item.h" #include "media/player/media_player_instance.h" #include "platform/platform_file_utilities.h" #include "ui/chat/chat_theme.h" #include "ui/text/text_utilities.h" #include "window/window_session_controller.h" #include #include #include namespace Data { namespace { void LaunchWithWarning( // not_null controller, const QString &name, HistoryItem *item) { const auto isExecutable = Data::IsExecutableName(name); const auto isIpReveal = Data::IsIpRevealingName(name); auto &app = Core::App(); const auto warn = [&] { if (item && item->history()->peer->isVerified()) { return false; } return (isExecutable && app.settings().exeLaunchWarning()) || (isIpReveal && app.settings().ipRevealWarning()); }(); const auto extension = '.' + Data::FileExtension(name); if (Platform::IsWindows() && extension == u"."_q) { // If you launch a file without extension, like "test", in case // there is an executable file with the same name in this folder, // like "test.bat", the executable file will be launched. // // Now we always force an Open With dialog box for such files. crl::on_main([=] { Platform::File::UnsafeShowOpenWith(name); }); return; } else if (!warn) { File::Launch(name); return; } const auto callback = [=, &app](bool checked) { if (checked) { if (isExecutable) { app.settings().setExeLaunchWarning(false); } else if (isIpReveal) { app.settings().setIpRevealWarning(false); } app.saveSettingsDelayed(); } File::Launch(name); }; auto text = isExecutable ? tr::lng_launch_exe_warning( lt_extension, rpl::single(Ui::Text::Bold(extension)), Ui::Text::WithEntities) : tr::lng_launch_svg_warning(Ui::Text::WithEntities); Ui::show(Box( std::move(text), tr::lng_launch_exe_dont_ask(tr::now), (isExecutable ? tr::lng_launch_exe_sure : tr::lng_continue)(), callback)); } } // namespace QString FileExtension(const QString &filepath) { const auto reversed = ranges::views::reverse(filepath); const auto last = ranges::find_first_of(reversed, ".\\/"); if (last == reversed.end() || *last != '.') { return QString(); } return QString(last.base(), last - reversed.begin()); } #if 0 bool IsValidMediaFile(const QString &filepath) { static const auto kExtensions = [] { const auto list = qsl("\ 16svx 2sf 3g2 3gp 8svx aac aaf aif aifc aiff amr amv ape asf ast au aup \ avchd avi brstm bwf cam cdda cust dat divx drc dsh dsf dts dtshd dtsma \ dvr-ms dwd evo f4a f4b f4p f4v fla flac flr flv gif gifv gsf gsm gym iff \ ifo it jam la ly m1v m2p m2ts m2v m4a m4p m4v mcf mid mk3d mka mks mkv mng \ mov mp1 mp2 mp3 mp4 minipsf mod mpc mpe mpeg mpg mpv mscz mt2 mus mxf mxl \ niff nsf nsv off ofr ofs ogg ogv opus ots pac ps psf psf2 psflib ptb qsf \ qt ra raw rka rm rmj rmvb roq s3m shn sib sid smi smp sol spc spx ssf svi \ swa swf tak ts tta txm usf vgm vob voc vox vqf wav webm wma wmv wrap wtv \ wv xm xml ym yuv").split(' '); return base::flat_set(list.begin(), list.end()); }(); return ranges::binary_search( kExtensions, FileExtension(filepath).toLower()); } #endif bool IsExecutableName(const QString &filepath) { static const auto kExtensions = [] { const auto joined = #ifdef Q_OS_MAC qsl("\ applescript action app bin command csh osx workflow terminal url caction \ mpkg pkg scpt scptd xhtm webarchive"); #elif defined Q_OS_UNIX // Q_OS_MAC qsl("bin csh deb desktop ksh out pet pkg pup rpm run sh shar \ slp zsh"); #else // Q_OS_MAC || Q_OS_UNIX qsl("\ ad ade adp app application appref-ms asp asx bas bat bin cab cdxml cer cfg \ chi chm cmd cnt com cpl crt csh der diagcab dll drv eml exe fon fxp gadget \ grp hlp hpj hta htt inf ini ins inx isp isu its jar jnlp job js jse key ksh \ lnk local lua mad maf mag mam manifest maq mar mas mat mau mav maw mcf mda \ mdb mde mdt mdw mdz mht mhtml mjs mmc mof msc msg msh msh1 msh2 msh1xml \ msh2xml mshxml msi msp mst ops osd paf pcd phar php php3 php4 php5 php7 phps \ php-s pht phtml pif pl plg pm pod prf prg ps1 ps2 ps1xml ps2xml psc1 psc2 \ psd1 psm1 pssc pst py py3 pyc pyd pyi pyo pyw pywz pyz rb reg rgs scf scr \ sct search-ms settingcontent-ms sh shb shs slk sys t tmp u3p url vb vbe vbp \ vbs vbscript vdx vsmacros vsd vsdm vsdx vss vssm vssx vst vstm vstx vsw vsx \ vtx website ws wsc wsf wsh xbap xll xnk xs"); #endif // !Q_OS_MAC && !Q_OS_UNIX const auto list = joined.split(' '); return base::flat_set(list.begin(), list.end()); }(); return ranges::binary_search( kExtensions, FileExtension(filepath).toLower()); } bool IsIpRevealingName(const QString &filepath) { static const auto kExtensions = [] { const auto joined = u"htm html svg"_q; const auto list = joined.split(' '); return base::flat_set(list.begin(), list.end()); }(); static const auto kMimeTypes = [] { const auto joined = u"text/html image/svg+xml"_q; const auto list = joined.split(' '); return base::flat_set(list.begin(), list.end()); }(); return ranges::binary_search( kExtensions, FileExtension(filepath).toLower() ) || ranges::binary_search( kMimeTypes, QMimeDatabase().mimeTypeForFile(QFileInfo(filepath)).name() ); } base::binary_guard ReadBackgroundImageAsync( not_null media, FnMut postprocess, FnMut done) { auto result = base::binary_guard(); const auto gzipSvg = media->owner()->isPatternWallPaperSVG(); crl::async([ gzipSvg, bytes = media->bytes(), path = media->owner()->filepath(), postprocess = std::move(postprocess), guard = result.make_guard(), callback = std::move(done) ]() mutable { auto image = Ui::ReadBackgroundImage(path, bytes, gzipSvg); if (postprocess) { image = postprocess(std::move(image)); } crl::on_main(std::move(guard), [ image = std::move(image), callback = std::move(callback) ]() mutable { callback(std::move(image)); }); }); return result; } void ResolveDocument( Window::SessionController *controller, not_null document, HistoryItem *item) { if (document->isNull()) { return; } const auto msgId = item ? item->fullId() : FullMsgId(); const auto showDocument = [&] { if (cUseExternalVideoPlayer() && document->isVideoFile() && !document->filepath().isEmpty()) { File::Launch(document->location(false).fname); } else if (controller) { controller->openDocument(document, msgId, true); } }; const auto media = document->createMediaView(); const auto openImageInApp = [&] { if (document->size >= Images::kReadBytesLimit) { return false; } const auto &location = document->location(true); if (!location.isEmpty() && location.accessEnable()) { const auto guard = gsl::finally([&] { location.accessDisable(); }); const auto path = location.name(); if (Core::MimeTypeForFile(QFileInfo(path)).name().startsWith("image/") && QImageReader(path).canRead()) { showDocument(); return true; } } else if (document->mimeString().startsWith("image/") && !media->bytes().isEmpty()) { auto bytes = media->bytes(); auto buffer = QBuffer(&bytes); if (QImageReader(&buffer).canRead()) { showDocument(); return true; } } return false; }; const auto &location = document->location(true); if (document->isTheme() && media->loaded(true)) { showDocument(); location.accessDisable(); } else if (media->canBePlayed(item)) { if (document->isAudioFile() || document->isVoiceMessage() || document->isVideoMessage()) { ::Media::Player::instance()->playPause({ document, msgId }); } else if (item && document->isAnimation() && HistoryView::Gif::CanPlayInline(document)) { document->owner().requestAnimationPlayInline(item); } else { showDocument(); } } else { document->saveFromDataSilent(); if (!openImageInApp()) { if (!document->filepath(true).isEmpty()) { LaunchWithWarning(location.name(), item); } else if (document->status == FileReady || document->status == FileDownloadFailed) { DocumentSaveClickHandler::Save( item ? item->fullId() : Data::FileOrigin(), document); } } } } } // namespace Data