/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtCore module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QCORE_MAC_P_H #define QCORE_MAC_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists for the convenience // of other Qt classes. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include "private/qglobal_p.h" #ifndef __IMAGECAPTURE__ # define __IMAGECAPTURE__ #endif // -------------------------------------------------------------------------- #if !defined(QT_BOOTSTRAPPED) && (QT_MACOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_12) || !defined(Q_OS_MACOS)) #define QT_USE_APPLE_ACTIVITIES #if defined(OS_ACTIVITY_OBJECT_API) #error The file has already been included #endif // We runtime-check all use of the activity APIs, so we can safely build // with them included, even if the deployment target is macOS 10.11 #if QT_MACOS_DEPLOYMENT_TARGET_BELOW(__MAC_10_12) #undef __MAC_OS_X_VERSION_MIN_REQUIRED #define __MAC_OS_X_VERSION_MIN_REQUIRED __MAC_10_12 #define DID_OVERRIDE_DEPLOYMENT_TARGET #endif #include #if !OS_ACTIVITY_OBJECT_API #error "Expected activity API to be available" #endif #if defined(DID_OVERRIDE_DEPLOYMENT_TARGET) #undef __MAC_OS_X_VERSION_MIN_REQUIRED #define __MAC_OS_X_VERSION_MIN_REQUIRED __MAC_10_11 #undef DID_OVERRIDE_DEPLOYMENT_TARGET #endif #endif // -------------------------------------------------------------------------- #if defined(QT_BOOTSTRAPPED) #include #else #include #endif #ifdef __OBJC__ #include #endif #include "qstring.h" #include "qscopedpointer.h" #if defined( __OBJC__) && defined(QT_NAMESPACE) #define QT_NAMESPACE_ALIAS_OBJC_CLASS(__KLASS__) @compatibility_alias __KLASS__ QT_MANGLE_NAMESPACE(__KLASS__) #else #define QT_NAMESPACE_ALIAS_OBJC_CLASS(__KLASS__) #endif QT_BEGIN_NAMESPACE template class QAppleRefCounted { public: QAppleRefCounted(const T &t = T()) : value(t) {} QAppleRefCounted(QAppleRefCounted &&other) : value(other.value) { other.value = T(); } QAppleRefCounted(const QAppleRefCounted &other) : value(other.value) { if (value) RetainFunction(value); } ~QAppleRefCounted() { if (value) ReleaseFunction(value); } operator T() { return value; } void swap(QAppleRefCounted &other) Q_DECL_NOEXCEPT_EXPR(noexcept(qSwap(value, other.value))) { qSwap(value, other.value); } QAppleRefCounted &operator=(const QAppleRefCounted &other) { QAppleRefCounted copy(other); swap(copy); return *this; } QAppleRefCounted &operator=(QAppleRefCounted &&other) { QAppleRefCounted moved(std::move(other)); swap(moved); return *this; } T *operator&() { return &value; } protected: T value; }; #ifdef Q_OS_MACOS class QMacRootLevelAutoReleasePool { public: QMacRootLevelAutoReleasePool(); ~QMacRootLevelAutoReleasePool(); private: QScopedPointer pool; }; #endif /* Helper class that automates refernce counting for CFtypes. After constructing the QCFType object, it can be copied like a value-based type. Note that you must own the object you are wrapping. This is typically the case if you get the object from a Core Foundation function with the word "Create" or "Copy" in it. If you got the object from a "Get" function, either retain it or use constructFromGet(). One exception to this rule is the HIThemeGet*Shape functions, which in reality are "Copy" functions. */ template class QCFType : public QAppleRefCounted { public: using QAppleRefCounted::QAppleRefCounted; template X as() const { return reinterpret_cast(this->value); } static QCFType constructFromGet(const T &t) { if (t) CFRetain(t); return QCFType(t); } }; class Q_CORE_EXPORT QCFString : public QCFType { public: inline QCFString(const QString &str) : QCFType(0), string(str) {} inline QCFString(const CFStringRef cfstr = 0) : QCFType(cfstr) {} inline QCFString(const QCFType &other) : QCFType(other) {} operator QString() const; operator CFStringRef() const; private: QString string; }; #ifdef Q_OS_OSX Q_CORE_EXPORT QChar qt_mac_qtKey2CocoaKey(Qt::Key key); Q_CORE_EXPORT Qt::Key qt_mac_cocoaKey2QtKey(QChar keyCode); #endif #ifndef QT_NO_DEBUG_STREAM QDebug operator<<(QDebug debug, const QMacAutoReleasePool *pool); #endif Q_CORE_EXPORT void qt_apple_check_os_version(); Q_CORE_EXPORT bool qt_apple_isApplicationExtension(); #if !defined(QT_BOOTSTRAPPED) && !defined(Q_OS_WATCHOS) QT_END_NAMESPACE # if defined(Q_OS_MACOS) Q_FORWARD_DECLARE_OBJC_CLASS(NSApplication); using AppleApplication = NSApplication; # else Q_FORWARD_DECLARE_OBJC_CLASS(UIApplication); using AppleApplication = UIApplication; # endif QT_BEGIN_NAMESPACE Q_CORE_EXPORT AppleApplication *qt_apple_sharedApplication(); #endif // -------------------------------------------------------------------------- #if !defined(QT_BOOTSTRAPPED) && (QT_MACOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_12) || !defined(Q_OS_MACOS)) #define QT_USE_APPLE_UNIFIED_LOGGING QT_END_NAMESPACE #include // The compiler isn't smart enough to realize that we're calling these functions // guarded by __builtin_available, so we need to also tag each function with the // runtime requirements. #include #define OS_LOG_AVAILABILITY API_AVAILABLE(macos(10.12), ios(10.0), tvos(10.0), watchos(3.0)) QT_BEGIN_NAMESPACE class Q_CORE_EXPORT AppleUnifiedLogger { public: static bool messageHandler(QtMsgType msgType, const QMessageLogContext &context, const QString &message, const QString &subsystem = QString()) OS_LOG_AVAILABILITY; private: static os_log_type_t logTypeForMessageType(QtMsgType msgType) OS_LOG_AVAILABILITY; static os_log_t cachedLog(const QString &subsystem, const QString &category) OS_LOG_AVAILABILITY; }; #undef OS_LOG_AVAILABILITY #endif // -------------------------------------------------------------------------- #if defined(QT_USE_APPLE_ACTIVITIES) QT_END_NAMESPACE #include #define OS_ACTIVITY_AVAILABILITY API_AVAILABLE(macos(10.12), ios(10.0), tvos(10.0), watchos(3.0)) #define OS_ACTIVITY_AVAILABILITY_CHECK __builtin_available(macOS 10.12, iOS 10, tvOS 10, watchOS 3, *) QT_BEGIN_NAMESPACE template using QAppleOsType = QAppleRefCounted; class Q_CORE_EXPORT QAppleLogActivity { public: QAppleLogActivity() : activity(nullptr) {} QAppleLogActivity(os_activity_t activity) OS_ACTIVITY_AVAILABILITY : activity(activity) {} ~QAppleLogActivity() { if (activity) leave(); } QAppleLogActivity(const QAppleLogActivity &) = delete; QAppleLogActivity& operator=(const QAppleLogActivity &) = delete; QAppleLogActivity(QAppleLogActivity&& other) : activity(other.activity), state(other.state) { other.activity = nullptr; } QAppleLogActivity& operator=(QAppleLogActivity &&other) { if (this != &other) { activity = other.activity; state = other.state; other.activity = nullptr; } return *this; } QAppleLogActivity&& enter() { if (activity) { if (OS_ACTIVITY_AVAILABILITY_CHECK) os_activity_scope_enter(static_cast(*this), &state); } return std::move(*this); } void leave() { if (activity) { if (OS_ACTIVITY_AVAILABILITY_CHECK) os_activity_scope_leave(&state); } } operator os_activity_t() OS_ACTIVITY_AVAILABILITY { return reinterpret_cast(static_cast(activity)); } private: // Work around API_AVAILABLE not working for templates by using void* QAppleOsType activity; os_activity_scope_state_s state; }; #define QT_APPLE_LOG_ACTIVITY_CREATE(condition, description, parent) []() { \ if (!(condition)) \ return QAppleLogActivity(); \ if (OS_ACTIVITY_AVAILABILITY_CHECK) \ return QAppleLogActivity(os_activity_create(description, parent, OS_ACTIVITY_FLAG_DEFAULT)); \ return QAppleLogActivity(); \ }() #define QT_VA_ARGS_CHOOSE(_1, _2, _3, _4, _5, _6, _7, _8, _9, N, ...) N #define QT_VA_ARGS_COUNT(...) QT_VA_ARGS_CHOOSE(__VA_ARGS__, 9, 8, 7, 6, 5, 4, 3, 2, 1) #define QT_OVERLOADED_MACRO(MACRO, ...) _QT_OVERLOADED_MACRO(MACRO, QT_VA_ARGS_COUNT(__VA_ARGS__))(__VA_ARGS__) #define _QT_OVERLOADED_MACRO(MACRO, ARGC) _QT_OVERLOADED_MACRO_EXPAND(MACRO, ARGC) #define _QT_OVERLOADED_MACRO_EXPAND(MACRO, ARGC) MACRO##ARGC #define QT_APPLE_LOG_ACTIVITY_WITH_PARENT3(condition, description, parent) QT_APPLE_LOG_ACTIVITY_CREATE(condition, description, parent) #define QT_APPLE_LOG_ACTIVITY_WITH_PARENT2(description, parent) QT_APPLE_LOG_ACTIVITY_WITH_PARENT3(true, description, parent) #define QT_APPLE_LOG_ACTIVITY_WITH_PARENT(...) QT_OVERLOADED_MACRO(QT_APPLE_LOG_ACTIVITY_WITH_PARENT, __VA_ARGS__) #define QT_APPLE_LOG_ACTIVITY2(condition, description) QT_APPLE_LOG_ACTIVITY_CREATE(condition, description, OS_ACTIVITY_CURRENT) #define QT_APPLE_LOG_ACTIVITY1(description) QT_APPLE_LOG_ACTIVITY2(true, description) #define QT_APPLE_LOG_ACTIVITY(...) QT_OVERLOADED_MACRO(QT_APPLE_LOG_ACTIVITY, __VA_ARGS__) #define QT_APPLE_SCOPED_LOG_ACTIVITY(...) QAppleLogActivity scopedLogActivity = QT_APPLE_LOG_ACTIVITY(__VA_ARGS__).enter(); #else // No-ops for macOS 10.11. We don't need to provide QT_APPLE_SCOPED_LOG_ACTIVITY, // as all the call sites for that are in code that's only built on 10.12 and above. #define QT_APPLE_LOG_ACTIVITY_WITH_PARENT(...) #define QT_APPLE_LOG_ACTIVITY(...) #endif // QT_DARWIN_PLATFORM_SDK_EQUAL_OR_ABOVE // ------------------------------------------------------------------------- QT_END_NAMESPACE #endif // QCORE_MAC_P_H