@ -0,0 +1,28 @@ |
||||
# This file configures the analyzer, which statically analyzes Dart code to |
||||
# check for errors, warnings, and lints. |
||||
# |
||||
# The issues identified by the analyzer are surfaced in the UI of Dart-enabled |
||||
# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be |
||||
# invoked from the command line by running `flutter analyze`. |
||||
|
||||
# The following line activates a set of recommended lints for Flutter apps, |
||||
# packages, and plugins designed to encourage good coding practices. |
||||
include: package:flutter_lints/flutter.yaml |
||||
|
||||
linter: |
||||
# The lint rules applied to this project can be customized in the |
||||
# section below to disable rules from the `package:flutter_lints/flutter.yaml` |
||||
# included above or to enable additional rules. A list of all available lints |
||||
# and their documentation is published at https://dart.dev/lints. |
||||
# |
||||
# Instead of disabling a lint rule for the entire project in the |
||||
# section below, it can also be suppressed for a single line of code |
||||
# or a specific dart file by using the `// ignore: name_of_lint` and |
||||
# `// ignore_for_file: name_of_lint` syntax on the line or in the file |
||||
# producing the lint. |
||||
rules: |
||||
# avoid_print: false # Uncomment to disable the `avoid_print` rule |
||||
# prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule |
||||
|
||||
# Additional information about this file can be found at |
||||
# https://dart.dev/guides/language/analysis-options |
@ -1,8 +1,7 @@ |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||
package="com.example.app"> |
||||
<!-- Flutter needs it to communicate with the running application |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"> |
||||
<!-- The INTERNET permission is required for development. Specifically, |
||||
the Flutter tool needs it to communicate with the running application |
||||
to allow setting breakpoints, to provide hot reload, etc. |
||||
--> |
||||
<uses-permission android:name="android.permission.INTERNET"/> |
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> |
||||
</manifest> |
||||
|
@ -1,8 +1,7 @@ |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||
package="com.example.app"> |
||||
<!-- Flutter needs it to communicate with the running application |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"> |
||||
<!-- The INTERNET permission is required for development. Specifically, |
||||
the Flutter tool needs it to communicate with the running application |
||||
to allow setting breakpoints, to provide hot reload, etc. |
||||
--> |
||||
<uses-permission android:name="android.permission.INTERNET"/> |
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> |
||||
</manifest> |
||||
|
@ -1,3 +1,6 @@ |
||||
org.gradle.jvmargs=-Xmx1536M |
||||
android.useAndroidX=true |
||||
android.enableJetifier=true |
||||
android.defaults.buildfeatures.buildconfig=true |
||||
android.nonTransitiveRClass=false |
||||
android.nonFinalResIds=false |
||||
|
@ -1,6 +1,5 @@ |
||||
#Fri Jun 23 08:50:38 CEST 2017 |
||||
distributionBase=GRADLE_USER_HOME |
||||
distributionPath=wrapper/dists |
||||
zipStoreBase=GRADLE_USER_HOME |
||||
zipStorePath=wrapper/dists |
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-all.zip |
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.0-all.zip |
||||
|
@ -1,11 +1,20 @@ |
||||
include ':app' |
||||
pluginManagement { |
||||
def flutterSdkPath = { |
||||
def properties = new Properties() |
||||
file("local.properties").withInputStream { properties.load(it) } |
||||
def flutterSdkPath = properties.getProperty("flutter.sdk") |
||||
assert flutterSdkPath != null, "flutter.sdk not set in local.properties" |
||||
return flutterSdkPath |
||||
} |
||||
settings.ext.flutterSdkPath = flutterSdkPath() |
||||
|
||||
def localPropertiesFile = new File(rootProject.projectDir, "local.properties") |
||||
def properties = new Properties() |
||||
includeBuild("${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle") |
||||
|
||||
assert localPropertiesFile.exists() |
||||
localPropertiesFile.withReader("UTF-8") { reader -> properties.load(reader) } |
||||
plugins { |
||||
id "dev.flutter.flutter-gradle-plugin" version "1.0.0" apply false |
||||
} |
||||
} |
||||
|
||||
def flutterSdkPath = properties.getProperty("flutter.sdk") |
||||
assert flutterSdkPath != null, "flutter.sdk not set in local.properties" |
||||
apply from: "$flutterSdkPath/packages/flutter_tools/gradle/app_plugin_loader.gradle" |
||||
include ":app" |
||||
|
||||
apply from: "${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle/app_plugin_loader.gradle" |
||||
|
@ -1,6 +0,0 @@ |
||||
#import <Flutter/Flutter.h> |
||||
#import <UIKit/UIKit.h> |
||||
|
||||
@interface AppDelegate : FlutterAppDelegate |
||||
|
||||
@end |
@ -1,13 +0,0 @@ |
||||
#import "AppDelegate.h" |
||||
#import "GeneratedPluginRegistrant.h" |
||||
|
||||
@implementation AppDelegate |
||||
|
||||
- (BOOL)application:(UIApplication *)application |
||||
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { |
||||
[GeneratedPluginRegistrant registerWithRegistry:self]; |
||||
// Override point for customization after application launch. |
||||
return [super application:application didFinishLaunchingWithOptions:launchOptions]; |
||||
} |
||||
|
||||
@end |
Before Width: | Height: | Size: 564 B After Width: | Height: | Size: 295 B |
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 406 B |
Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 450 B |
Before Width: | Height: | Size: 1.0 KiB After Width: | Height: | Size: 282 B |
Before Width: | Height: | Size: 1.7 KiB After Width: | Height: | Size: 462 B |
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 704 B |
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 406 B |
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 586 B |
Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 862 B |
Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 862 B |
Before Width: | Height: | Size: 3.7 KiB After Width: | Height: | Size: 1.6 KiB |
Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 762 B |
Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 1.2 KiB |
Before Width: | Height: | Size: 3.5 KiB After Width: | Height: | Size: 1.4 KiB |
@ -1,9 +0,0 @@ |
||||
#import <Flutter/Flutter.h> |
||||
#import <UIKit/UIKit.h> |
||||
#import "AppDelegate.h" |
||||
|
||||
int main(int argc, char* argv[]) { |
||||
@autoreleasepool { |
||||
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class])); |
||||
} |
||||
} |
@ -0,0 +1,12 @@ |
||||
import Flutter |
||||
import UIKit |
||||
import XCTest |
||||
|
||||
class RunnerTests: XCTestCase { |
||||
|
||||
func testExample() { |
||||
// If you add code to the Runner application, consider adding tests here. |
||||
// See https://developer.apple.com/documentation/xctest for more information about using XCTest. |
||||
} |
||||
|
||||
} |
@ -1,3 +1,3 @@ |
||||
class PlatformViewRegistry { |
||||
static void registerViewFactory(String viewId, dynamic cb) {} |
||||
} |
||||
// class PlatformViewRegistry { |
||||
// static void registerViewFactory(String viewId, dynamic cb) {} |
||||
// } |
||||
|
@ -1,7 +1,7 @@ |
||||
import 'dart:ui' if (dart.library.html) 'dart:ui_web' as ui; |
||||
// import 'dart:ui' if (dart.library.html) 'dart:ui_web' as ui; |
||||
|
||||
class PlatformViewRegistry { |
||||
static void registerViewFactory(String viewId, dynamic cb) { |
||||
ui.platformViewRegistry.registerViewFactory(viewId, cb); |
||||
} |
||||
} |
||||
// class PlatformViewRegistry { |
||||
// static void registerViewFactory(String viewId, dynamic cb) { |
||||
// ui.platformViewRegistry.registerViewFactory(viewId, cb); |
||||
// } |
||||
// } |
||||
|
@ -1 +1 @@ |
||||
flutter/ephemeral |
||||
flutter/ephemeral |
||||
|
@ -1,106 +1,139 @@ |
||||
cmake_minimum_required(VERSION 3.10) |
||||
project(runner LANGUAGES CXX) |
||||
|
||||
set(BINARY_NAME "app") |
||||
set(APPLICATION_ID "com.example.app") |
||||
|
||||
cmake_policy(SET CMP0063 NEW) |
||||
|
||||
set(CMAKE_INSTALL_RPATH "$ORIGIN/lib") |
||||
|
||||
# Configure build options. |
||||
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) |
||||
set(CMAKE_BUILD_TYPE "Debug" CACHE |
||||
STRING "Flutter build mode" FORCE) |
||||
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS |
||||
"Debug" "Profile" "Release") |
||||
endif() |
||||
|
||||
# Compilation settings that should be applied to most targets. |
||||
function(APPLY_STANDARD_SETTINGS TARGET) |
||||
target_compile_features(${TARGET} PUBLIC cxx_std_14) |
||||
target_compile_options(${TARGET} PRIVATE -Wall -Werror) |
||||
target_compile_options(${TARGET} PRIVATE "$<$<NOT:$<CONFIG:Debug>>:-O3>") |
||||
target_compile_definitions(${TARGET} PRIVATE "$<$<NOT:$<CONFIG:Debug>>:NDEBUG>") |
||||
endfunction() |
||||
|
||||
set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") |
||||
|
||||
# Flutter library and tool build rules. |
||||
add_subdirectory(${FLUTTER_MANAGED_DIR}) |
||||
|
||||
# System-level dependencies. |
||||
find_package(PkgConfig REQUIRED) |
||||
pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) |
||||
|
||||
add_definitions(-DAPPLICATION_ID="${APPLICATION_ID}") |
||||
|
||||
# Application build |
||||
add_executable(${BINARY_NAME} |
||||
"main.cc" |
||||
"my_application.cc" |
||||
"${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" |
||||
) |
||||
apply_standard_settings(${BINARY_NAME}) |
||||
target_link_libraries(${BINARY_NAME} PRIVATE flutter) |
||||
target_link_libraries(${BINARY_NAME} PRIVATE PkgConfig::GTK) |
||||
add_dependencies(${BINARY_NAME} flutter_assemble) |
||||
# Only the install-generated bundle's copy of the executable will launch |
||||
# correctly, since the resources must in the right relative locations. To avoid |
||||
# people trying to run the unbundled copy, put it in a subdirectory instead of |
||||
# the default top-level location. |
||||
set_target_properties(${BINARY_NAME} |
||||
PROPERTIES |
||||
RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/intermediates_do_not_run" |
||||
) |
||||
|
||||
# Generated plugin build rules, which manage building the plugins and adding |
||||
# them to the application. |
||||
include(flutter/generated_plugins.cmake) |
||||
|
||||
|
||||
# === Installation === |
||||
# By default, "installing" just makes a relocatable bundle in the build |
||||
# directory. |
||||
set(BUILD_BUNDLE_DIR "${PROJECT_BINARY_DIR}/bundle") |
||||
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) |
||||
set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) |
||||
endif() |
||||
|
||||
# Start with a clean build bundle directory every time. |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${BUILD_BUNDLE_DIR}/\") |
||||
" COMPONENT Runtime) |
||||
|
||||
set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") |
||||
set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}/lib") |
||||
|
||||
install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
if(PLUGIN_BUNDLED_LIBRARIES) |
||||
install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" |
||||
DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endif() |
||||
|
||||
# Fully re-copy the assets directory on each build to avoid having stale files |
||||
# from a previous install. |
||||
set(FLUTTER_ASSET_DIR_NAME "flutter_assets") |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") |
||||
" COMPONENT Runtime) |
||||
install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" |
||||
DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) |
||||
|
||||
# Install the AOT library on non-Debug builds only. |
||||
if(NOT CMAKE_BUILD_TYPE MATCHES "Debug") |
||||
install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endif() |
||||
# Project-level configuration. |
||||
cmake_minimum_required(VERSION 3.10) |
||||
project(runner LANGUAGES CXX) |
||||
|
||||
# The name of the executable created for the application. Change this to change |
||||
# the on-disk name of your application. |
||||
set(BINARY_NAME "example") |
||||
# The unique GTK application identifier for this application. See: |
||||
# https://wiki.gnome.org/HowDoI/ChooseApplicationID |
||||
set(APPLICATION_ID "com.example.example") |
||||
|
||||
# Explicitly opt in to modern CMake behaviors to avoid warnings with recent |
||||
# versions of CMake. |
||||
cmake_policy(SET CMP0063 NEW) |
||||
|
||||
# Load bundled libraries from the lib/ directory relative to the binary. |
||||
set(CMAKE_INSTALL_RPATH "$ORIGIN/lib") |
||||
|
||||
# Root filesystem for cross-building. |
||||
if(FLUTTER_TARGET_PLATFORM_SYSROOT) |
||||
set(CMAKE_SYSROOT ${FLUTTER_TARGET_PLATFORM_SYSROOT}) |
||||
set(CMAKE_FIND_ROOT_PATH ${CMAKE_SYSROOT}) |
||||
set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) |
||||
set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY) |
||||
set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY) |
||||
set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY) |
||||
endif() |
||||
|
||||
# Define build configuration options. |
||||
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) |
||||
set(CMAKE_BUILD_TYPE "Debug" CACHE |
||||
STRING "Flutter build mode" FORCE) |
||||
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS |
||||
"Debug" "Profile" "Release") |
||||
endif() |
||||
|
||||
# Compilation settings that should be applied to most targets. |
||||
# |
||||
# Be cautious about adding new options here, as plugins use this function by |
||||
# default. In most cases, you should add new options to specific targets instead |
||||
# of modifying this function. |
||||
function(APPLY_STANDARD_SETTINGS TARGET) |
||||
target_compile_features(${TARGET} PUBLIC cxx_std_14) |
||||
target_compile_options(${TARGET} PRIVATE -Wall -Werror) |
||||
target_compile_options(${TARGET} PRIVATE "$<$<NOT:$<CONFIG:Debug>>:-O3>") |
||||
target_compile_definitions(${TARGET} PRIVATE "$<$<NOT:$<CONFIG:Debug>>:NDEBUG>") |
||||
endfunction() |
||||
|
||||
# Flutter library and tool build rules. |
||||
set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") |
||||
add_subdirectory(${FLUTTER_MANAGED_DIR}) |
||||
|
||||
# System-level dependencies. |
||||
find_package(PkgConfig REQUIRED) |
||||
pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) |
||||
|
||||
add_definitions(-DAPPLICATION_ID="${APPLICATION_ID}") |
||||
|
||||
# Define the application target. To change its name, change BINARY_NAME above, |
||||
# not the value here, or `flutter run` will no longer work. |
||||
# |
||||
# Any new source files that you add to the application should be added here. |
||||
add_executable(${BINARY_NAME} |
||||
"main.cc" |
||||
"my_application.cc" |
||||
"${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" |
||||
) |
||||
|
||||
# Apply the standard set of build settings. This can be removed for applications |
||||
# that need different build settings. |
||||
apply_standard_settings(${BINARY_NAME}) |
||||
|
||||
# Add dependency libraries. Add any application-specific dependencies here. |
||||
target_link_libraries(${BINARY_NAME} PRIVATE flutter) |
||||
target_link_libraries(${BINARY_NAME} PRIVATE PkgConfig::GTK) |
||||
|
||||
# Run the Flutter tool portions of the build. This must not be removed. |
||||
add_dependencies(${BINARY_NAME} flutter_assemble) |
||||
|
||||
# Only the install-generated bundle's copy of the executable will launch |
||||
# correctly, since the resources must in the right relative locations. To avoid |
||||
# people trying to run the unbundled copy, put it in a subdirectory instead of |
||||
# the default top-level location. |
||||
set_target_properties(${BINARY_NAME} |
||||
PROPERTIES |
||||
RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/intermediates_do_not_run" |
||||
) |
||||
|
||||
|
||||
# Generated plugin build rules, which manage building the plugins and adding |
||||
# them to the application. |
||||
include(flutter/generated_plugins.cmake) |
||||
|
||||
|
||||
# === Installation === |
||||
# By default, "installing" just makes a relocatable bundle in the build |
||||
# directory. |
||||
set(BUILD_BUNDLE_DIR "${PROJECT_BINARY_DIR}/bundle") |
||||
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) |
||||
set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) |
||||
endif() |
||||
|
||||
# Start with a clean build bundle directory every time. |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${BUILD_BUNDLE_DIR}/\") |
||||
" COMPONENT Runtime) |
||||
|
||||
set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") |
||||
set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}/lib") |
||||
|
||||
install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
foreach(bundled_library ${PLUGIN_BUNDLED_LIBRARIES}) |
||||
install(FILES "${bundled_library}" |
||||
DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endforeach(bundled_library) |
||||
|
||||
# Fully re-copy the assets directory on each build to avoid having stale files |
||||
# from a previous install. |
||||
set(FLUTTER_ASSET_DIR_NAME "flutter_assets") |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") |
||||
" COMPONENT Runtime) |
||||
install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" |
||||
DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) |
||||
|
||||
# Install the AOT library on non-Debug builds only. |
||||
if(NOT CMAKE_BUILD_TYPE MATCHES "Debug") |
||||
install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endif() |
||||
|
@ -1,91 +1,88 @@ |
||||
cmake_minimum_required(VERSION 3.10) |
||||
|
||||
set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") |
||||
|
||||
# Configuration provided via flutter tool. |
||||
include(${EPHEMERAL_DIR}/generated_config.cmake) |
||||
|
||||
# TODO: Move the rest of this into files in ephemeral. See |
||||
# https://github.com/flutter/flutter/issues/57146. |
||||
|
||||
# Serves the same purpose as list(TRANSFORM ... PREPEND ...), |
||||
# which isn't available in 3.10. |
||||
function(list_prepend LIST_NAME PREFIX) |
||||
set(NEW_LIST "") |
||||
foreach(element ${${LIST_NAME}}) |
||||
list(APPEND NEW_LIST "${PREFIX}${element}") |
||||
endforeach(element) |
||||
set(${LIST_NAME} "${NEW_LIST}" PARENT_SCOPE) |
||||
endfunction() |
||||
|
||||
# === Flutter Library === |
||||
# System-level dependencies. |
||||
find_package(PkgConfig REQUIRED) |
||||
pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) |
||||
pkg_check_modules(GLIB REQUIRED IMPORTED_TARGET glib-2.0) |
||||
pkg_check_modules(GIO REQUIRED IMPORTED_TARGET gio-2.0) |
||||
pkg_check_modules(BLKID REQUIRED IMPORTED_TARGET blkid) |
||||
pkg_check_modules(LZMA REQUIRED IMPORTED_TARGET liblzma) |
||||
|
||||
set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/libflutter_linux_gtk.so") |
||||
|
||||
# Published to parent scope for install step. |
||||
set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) |
||||
set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) |
||||
set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) |
||||
set(AOT_LIBRARY "${PROJECT_DIR}/build/lib/libapp.so" PARENT_SCOPE) |
||||
|
||||
list(APPEND FLUTTER_LIBRARY_HEADERS |
||||
"fl_basic_message_channel.h" |
||||
"fl_binary_codec.h" |
||||
"fl_binary_messenger.h" |
||||
"fl_dart_project.h" |
||||
"fl_engine.h" |
||||
"fl_json_message_codec.h" |
||||
"fl_json_method_codec.h" |
||||
"fl_message_codec.h" |
||||
"fl_method_call.h" |
||||
"fl_method_channel.h" |
||||
"fl_method_codec.h" |
||||
"fl_method_response.h" |
||||
"fl_plugin_registrar.h" |
||||
"fl_plugin_registry.h" |
||||
"fl_standard_message_codec.h" |
||||
"fl_standard_method_codec.h" |
||||
"fl_string_codec.h" |
||||
"fl_value.h" |
||||
"fl_view.h" |
||||
"flutter_linux.h" |
||||
) |
||||
list_prepend(FLUTTER_LIBRARY_HEADERS "${EPHEMERAL_DIR}/flutter_linux/") |
||||
add_library(flutter INTERFACE) |
||||
target_include_directories(flutter INTERFACE |
||||
"${EPHEMERAL_DIR}" |
||||
) |
||||
target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}") |
||||
target_link_libraries(flutter INTERFACE |
||||
PkgConfig::GTK |
||||
PkgConfig::GLIB |
||||
PkgConfig::GIO |
||||
PkgConfig::BLKID |
||||
PkgConfig::LZMA |
||||
) |
||||
add_dependencies(flutter flutter_assemble) |
||||
|
||||
# === Flutter tool backend === |
||||
# _phony_ is a non-existent file to force this command to run every time, |
||||
# since currently there's no way to get a full input/output list from the |
||||
# flutter tool. |
||||
add_custom_command( |
||||
OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} |
||||
${CMAKE_CURRENT_BINARY_DIR}/_phony_ |
||||
COMMAND ${CMAKE_COMMAND} -E env |
||||
${FLUTTER_TOOL_ENVIRONMENT} |
||||
"${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.sh" |
||||
${FLUTTER_TARGET_PLATFORM} ${CMAKE_BUILD_TYPE} |
||||
VERBATIM |
||||
) |
||||
add_custom_target(flutter_assemble DEPENDS |
||||
"${FLUTTER_LIBRARY}" |
||||
${FLUTTER_LIBRARY_HEADERS} |
||||
) |
||||
# This file controls Flutter-level build steps. It should not be edited. |
||||
cmake_minimum_required(VERSION 3.10) |
||||
|
||||
set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") |
||||
|
||||
# Configuration provided via flutter tool. |
||||
include(${EPHEMERAL_DIR}/generated_config.cmake) |
||||
|
||||
# TODO: Move the rest of this into files in ephemeral. See |
||||
# https://github.com/flutter/flutter/issues/57146. |
||||
|
||||
# Serves the same purpose as list(TRANSFORM ... PREPEND ...), |
||||
# which isn't available in 3.10. |
||||
function(list_prepend LIST_NAME PREFIX) |
||||
set(NEW_LIST "") |
||||
foreach(element ${${LIST_NAME}}) |
||||
list(APPEND NEW_LIST "${PREFIX}${element}") |
||||
endforeach(element) |
||||
set(${LIST_NAME} "${NEW_LIST}" PARENT_SCOPE) |
||||
endfunction() |
||||
|
||||
# === Flutter Library === |
||||
# System-level dependencies. |
||||
find_package(PkgConfig REQUIRED) |
||||
pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) |
||||
pkg_check_modules(GLIB REQUIRED IMPORTED_TARGET glib-2.0) |
||||
pkg_check_modules(GIO REQUIRED IMPORTED_TARGET gio-2.0) |
||||
|
||||
set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/libflutter_linux_gtk.so") |
||||
|
||||
# Published to parent scope for install step. |
||||
set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) |
||||
set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) |
||||
set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) |
||||
set(AOT_LIBRARY "${PROJECT_DIR}/build/lib/libapp.so" PARENT_SCOPE) |
||||
|
||||
list(APPEND FLUTTER_LIBRARY_HEADERS |
||||
"fl_basic_message_channel.h" |
||||
"fl_binary_codec.h" |
||||
"fl_binary_messenger.h" |
||||
"fl_dart_project.h" |
||||
"fl_engine.h" |
||||
"fl_json_message_codec.h" |
||||
"fl_json_method_codec.h" |
||||
"fl_message_codec.h" |
||||
"fl_method_call.h" |
||||
"fl_method_channel.h" |
||||
"fl_method_codec.h" |
||||
"fl_method_response.h" |
||||
"fl_plugin_registrar.h" |
||||
"fl_plugin_registry.h" |
||||
"fl_standard_message_codec.h" |
||||
"fl_standard_method_codec.h" |
||||
"fl_string_codec.h" |
||||
"fl_value.h" |
||||
"fl_view.h" |
||||
"flutter_linux.h" |
||||
) |
||||
list_prepend(FLUTTER_LIBRARY_HEADERS "${EPHEMERAL_DIR}/flutter_linux/") |
||||
add_library(flutter INTERFACE) |
||||
target_include_directories(flutter INTERFACE |
||||
"${EPHEMERAL_DIR}" |
||||
) |
||||
target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}") |
||||
target_link_libraries(flutter INTERFACE |
||||
PkgConfig::GTK |
||||
PkgConfig::GLIB |
||||
PkgConfig::GIO |
||||
) |
||||
add_dependencies(flutter flutter_assemble) |
||||
|
||||
# === Flutter tool backend === |
||||
# _phony_ is a non-existent file to force this command to run every time, |
||||
# since currently there's no way to get a full input/output list from the |
||||
# flutter tool. |
||||
add_custom_command( |
||||
OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} |
||||
${CMAKE_CURRENT_BINARY_DIR}/_phony_ |
||||
COMMAND ${CMAKE_COMMAND} -E env |
||||
${FLUTTER_TOOL_ENVIRONMENT} |
||||
"${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.sh" |
||||
${FLUTTER_TARGET_PLATFORM} ${CMAKE_BUILD_TYPE} |
||||
VERBATIM |
||||
) |
||||
add_custom_target(flutter_assemble DEPENDS |
||||
"${FLUTTER_LIBRARY}" |
||||
${FLUTTER_LIBRARY_HEADERS} |
||||
) |
||||
|
@ -1,6 +1,6 @@ |
||||
#include "my_application.h" |
||||
|
||||
int main(int argc, char** argv) { |
||||
g_autoptr(MyApplication) app = my_application_new(); |
||||
return g_application_run(G_APPLICATION(app), argc, argv); |
||||
} |
||||
#include "my_application.h" |
||||
|
||||
int main(int argc, char** argv) { |
||||
g_autoptr(MyApplication) app = my_application_new(); |
||||
return g_application_run(G_APPLICATION(app), argc, argv); |
||||
} |
||||
|
@ -1,104 +1,104 @@ |
||||
#include "my_application.h" |
||||
|
||||
#include <flutter_linux/flutter_linux.h> |
||||
#ifdef GDK_WINDOWING_X11 |
||||
#include <gdk/gdkx.h> |
||||
#endif |
||||
|
||||
#include "flutter/generated_plugin_registrant.h" |
||||
|
||||
struct _MyApplication { |
||||
GtkApplication parent_instance; |
||||
char** dart_entrypoint_arguments; |
||||
}; |
||||
|
||||
G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION) |
||||
|
||||
// Implements GApplication::activate.
|
||||
static void my_application_activate(GApplication* application) { |
||||
MyApplication* self = MY_APPLICATION(application); |
||||
GtkWindow* window = |
||||
GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application))); |
||||
|
||||
// Use a header bar when running in GNOME as this is the common style used
|
||||
// by applications and is the setup most users will be using (e.g. Ubuntu
|
||||
// desktop).
|
||||
// If running on X and not using GNOME then just use a traditional title bar
|
||||
// in case the window manager does more exotic layout, e.g. tiling.
|
||||
// If running on Wayland assume the header bar will work (may need changing
|
||||
// if future cases occur).
|
||||
gboolean use_header_bar = FALSE; |
||||
#ifdef GDK_WINDOWING_X11 |
||||
GdkScreen *screen = gtk_window_get_screen(window); |
||||
if (GDK_IS_X11_SCREEN(screen)) { |
||||
const gchar* wm_name = gdk_x11_screen_get_window_manager_name(screen); |
||||
if (g_strcmp0(wm_name, "GNOME Shell") != 0) { |
||||
use_header_bar = FALSE; |
||||
} |
||||
} |
||||
#endif |
||||
if (use_header_bar) { |
||||
GtkHeaderBar *header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); |
||||
gtk_widget_show(GTK_WIDGET(header_bar)); |
||||
gtk_header_bar_set_title(header_bar, "app"); |
||||
gtk_header_bar_set_show_close_button(header_bar, TRUE); |
||||
gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); |
||||
} |
||||
else { |
||||
gtk_window_set_title(window, "app"); |
||||
} |
||||
|
||||
gtk_window_set_default_size(window, 1280, 720); |
||||
gtk_widget_show(GTK_WIDGET(window)); |
||||
|
||||
g_autoptr(FlDartProject) project = fl_dart_project_new(); |
||||
fl_dart_project_set_dart_entrypoint_arguments(project, self->dart_entrypoint_arguments); |
||||
|
||||
FlView* view = fl_view_new(project); |
||||
gtk_widget_show(GTK_WIDGET(view)); |
||||
gtk_container_add(GTK_CONTAINER(window), GTK_WIDGET(view)); |
||||
|
||||
fl_register_plugins(FL_PLUGIN_REGISTRY(view)); |
||||
|
||||
gtk_widget_grab_focus(GTK_WIDGET(view)); |
||||
} |
||||
|
||||
// Implements GApplication::local_command_line.
|
||||
static gboolean my_application_local_command_line(GApplication* application, gchar ***arguments, int *exit_status) { |
||||
MyApplication* self = MY_APPLICATION(application); |
||||
// Strip out the first argument as it is the binary name.
|
||||
self->dart_entrypoint_arguments = g_strdupv(*arguments + 1); |
||||
|
||||
g_autoptr(GError) error = nullptr; |
||||
if (!g_application_register(application, nullptr, &error)) { |
||||
g_warning("Failed to register: %s", error->message); |
||||
*exit_status = 1; |
||||
return TRUE; |
||||
} |
||||
|
||||
g_application_activate(application); |
||||
*exit_status = 0; |
||||
|
||||
return TRUE; |
||||
} |
||||
|
||||
// Implements GObject::dispose.
|
||||
static void my_application_dispose(GObject *object) { |
||||
MyApplication* self = MY_APPLICATION(object); |
||||
g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev); |
||||
G_OBJECT_CLASS(my_application_parent_class)->dispose(object); |
||||
} |
||||
|
||||
static void my_application_class_init(MyApplicationClass* klass) { |
||||
G_APPLICATION_CLASS(klass)->activate = my_application_activate; |
||||
G_APPLICATION_CLASS(klass)->local_command_line = my_application_local_command_line; |
||||
G_OBJECT_CLASS(klass)->dispose = my_application_dispose; |
||||
} |
||||
|
||||
static void my_application_init(MyApplication* self) {} |
||||
|
||||
MyApplication* my_application_new() { |
||||
return MY_APPLICATION(g_object_new(my_application_get_type(), |
||||
"application-id", APPLICATION_ID, |
||||
nullptr)); |
||||
} |
||||
#include "my_application.h" |
||||
|
||||
#include <flutter_linux/flutter_linux.h> |
||||
#ifdef GDK_WINDOWING_X11 |
||||
#include <gdk/gdkx.h> |
||||
#endif |
||||
|
||||
#include "flutter/generated_plugin_registrant.h" |
||||
|
||||
struct _MyApplication { |
||||
GtkApplication parent_instance; |
||||
char** dart_entrypoint_arguments; |
||||
}; |
||||
|
||||
G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION) |
||||
|
||||
// Implements GApplication::activate.
|
||||
static void my_application_activate(GApplication* application) { |
||||
MyApplication* self = MY_APPLICATION(application); |
||||
GtkWindow* window = |
||||
GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application))); |
||||
|
||||
// Use a header bar when running in GNOME as this is the common style used
|
||||
// by applications and is the setup most users will be using (e.g. Ubuntu
|
||||
// desktop).
|
||||
// If running on X and not using GNOME then just use a traditional title bar
|
||||
// in case the window manager does more exotic layout, e.g. tiling.
|
||||
// If running on Wayland assume the header bar will work (may need changing
|
||||
// if future cases occur).
|
||||
gboolean use_header_bar = TRUE; |
||||
#ifdef GDK_WINDOWING_X11 |
||||
GdkScreen* screen = gtk_window_get_screen(window); |
||||
if (GDK_IS_X11_SCREEN(screen)) { |
||||
const gchar* wm_name = gdk_x11_screen_get_window_manager_name(screen); |
||||
if (g_strcmp0(wm_name, "GNOME Shell") != 0) { |
||||
use_header_bar = FALSE; |
||||
} |
||||
} |
||||
#endif |
||||
if (use_header_bar) { |
||||
GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); |
||||
gtk_widget_show(GTK_WIDGET(header_bar)); |
||||
gtk_header_bar_set_title(header_bar, "example"); |
||||
gtk_header_bar_set_show_close_button(header_bar, TRUE); |
||||
gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); |
||||
} else { |
||||
gtk_window_set_title(window, "example"); |
||||
} |
||||
|
||||
gtk_window_set_default_size(window, 1280, 720); |
||||
gtk_widget_show(GTK_WIDGET(window)); |
||||
|
||||
g_autoptr(FlDartProject) project = fl_dart_project_new(); |
||||
fl_dart_project_set_dart_entrypoint_arguments(project, self->dart_entrypoint_arguments); |
||||
|
||||
FlView* view = fl_view_new(project); |
||||
gtk_widget_show(GTK_WIDGET(view)); |
||||
gtk_container_add(GTK_CONTAINER(window), GTK_WIDGET(view)); |
||||
|
||||
fl_register_plugins(FL_PLUGIN_REGISTRY(view)); |
||||
|
||||
gtk_widget_grab_focus(GTK_WIDGET(view)); |
||||
} |
||||
|
||||
// Implements GApplication::local_command_line.
|
||||
static gboolean my_application_local_command_line(GApplication* application, gchar*** arguments, int* exit_status) { |
||||
MyApplication* self = MY_APPLICATION(application); |
||||
// Strip out the first argument as it is the binary name.
|
||||
self->dart_entrypoint_arguments = g_strdupv(*arguments + 1); |
||||
|
||||
g_autoptr(GError) error = nullptr; |
||||
if (!g_application_register(application, nullptr, &error)) { |
||||
g_warning("Failed to register: %s", error->message); |
||||
*exit_status = 1; |
||||
return TRUE; |
||||
} |
||||
|
||||
g_application_activate(application); |
||||
*exit_status = 0; |
||||
|
||||
return TRUE; |
||||
} |
||||
|
||||
// Implements GObject::dispose.
|
||||
static void my_application_dispose(GObject* object) { |
||||
MyApplication* self = MY_APPLICATION(object); |
||||
g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev); |
||||
G_OBJECT_CLASS(my_application_parent_class)->dispose(object); |
||||
} |
||||
|
||||
static void my_application_class_init(MyApplicationClass* klass) { |
||||
G_APPLICATION_CLASS(klass)->activate = my_application_activate; |
||||
G_APPLICATION_CLASS(klass)->local_command_line = my_application_local_command_line; |
||||
G_OBJECT_CLASS(klass)->dispose = my_application_dispose; |
||||
} |
||||
|
||||
static void my_application_init(MyApplication* self) {} |
||||
|
||||
MyApplication* my_application_new() { |
||||
return MY_APPLICATION(g_object_new(my_application_get_type(), |
||||
"application-id", APPLICATION_ID, |
||||
"flags", G_APPLICATION_NON_UNIQUE, |
||||
nullptr)); |
||||
} |
||||
|
@ -1,18 +1,18 @@ |
||||
#ifndef FLUTTER_MY_APPLICATION_H_ |
||||
#define FLUTTER_MY_APPLICATION_H_ |
||||
|
||||
#include <gtk/gtk.h> |
||||
|
||||
G_DECLARE_FINAL_TYPE(MyApplication, my_application, MY, APPLICATION, |
||||
GtkApplication) |
||||
|
||||
/**
|
||||
* my_application_new: |
||||
* |
||||
* Creates a new Flutter-based application. |
||||
* |
||||
* Returns: a new #MyApplication. |
||||
*/ |
||||
MyApplication* my_application_new(); |
||||
|
||||
#endif // FLUTTER_MY_APPLICATION_H_
|
||||
#ifndef FLUTTER_MY_APPLICATION_H_ |
||||
#define FLUTTER_MY_APPLICATION_H_ |
||||
|
||||
#include <gtk/gtk.h> |
||||
|
||||
G_DECLARE_FINAL_TYPE(MyApplication, my_application, MY, APPLICATION, |
||||
GtkApplication) |
||||
|
||||
/**
|
||||
* my_application_new: |
||||
* |
||||
* Creates a new Flutter-based application. |
||||
* |
||||
* Returns: a new #MyApplication. |
||||
*/ |
||||
MyApplication* my_application_new(); |
||||
|
||||
#endif // FLUTTER_MY_APPLICATION_H_
|
||||
|
@ -0,0 +1,12 @@ |
||||
import FlutterMacOS |
||||
import Cocoa |
||||
import XCTest |
||||
|
||||
class RunnerTests: XCTestCase { |
||||
|
||||
func testExample() { |
||||
// If you add code to the Runner application, consider adding tests here. |
||||
// See https://developer.apple.com/documentation/xctest for more information about using XCTest. |
||||
} |
||||
|
||||
} |
After Width: | Height: | Size: 5.5 KiB |
After Width: | Height: | Size: 20 KiB |
@ -1,17 +1,17 @@ |
||||
flutter/ephemeral/ |
||||
|
||||
# Visual Studio user-specific files. |
||||
*.suo |
||||
*.user |
||||
*.userosscache |
||||
*.sln.docstates |
||||
|
||||
# Visual Studio build-related files. |
||||
x64/ |
||||
x86/ |
||||
|
||||
# Visual Studio cache files |
||||
# files ending in .cache can be ignored |
||||
*.[Cc]ache |
||||
# but keep track of directories ending in .cache |
||||
!*.[Cc]ache/ |
||||
flutter/ephemeral/ |
||||
|
||||
# Visual Studio user-specific files. |
||||
*.suo |
||||
*.user |
||||
*.userosscache |
||||
*.sln.docstates |
||||
|
||||
# Visual Studio build-related files. |
||||
x64/ |
||||
x86/ |
||||
|
||||
# Visual Studio cache files |
||||
# files ending in .cache can be ignored |
||||
*.[Cc]ache |
||||
# but keep track of directories ending in .cache |
||||
!*.[Cc]ache/ |
||||
|
@ -1,95 +1,102 @@ |
||||
cmake_minimum_required(VERSION 3.15) |
||||
project(app LANGUAGES CXX) |
||||
|
||||
set(BINARY_NAME "app") |
||||
|
||||
cmake_policy(SET CMP0063 NEW) |
||||
|
||||
set(CMAKE_INSTALL_RPATH "$ORIGIN/lib") |
||||
|
||||
# Configure build options. |
||||
get_property(IS_MULTICONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG) |
||||
if(IS_MULTICONFIG) |
||||
set(CMAKE_CONFIGURATION_TYPES "Debug;Profile;Release" |
||||
CACHE STRING "" FORCE) |
||||
else() |
||||
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) |
||||
set(CMAKE_BUILD_TYPE "Debug" CACHE |
||||
STRING "Flutter build mode" FORCE) |
||||
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS |
||||
"Debug" "Profile" "Release") |
||||
endif() |
||||
endif() |
||||
|
||||
set(CMAKE_EXE_LINKER_FLAGS_PROFILE "${CMAKE_EXE_LINKER_FLAGS_RELEASE}") |
||||
set(CMAKE_SHARED_LINKER_FLAGS_PROFILE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE}") |
||||
set(CMAKE_C_FLAGS_PROFILE "${CMAKE_C_FLAGS_RELEASE}") |
||||
set(CMAKE_CXX_FLAGS_PROFILE "${CMAKE_CXX_FLAGS_RELEASE}") |
||||
|
||||
# Use Unicode for all projects. |
||||
add_definitions(-DUNICODE -D_UNICODE) |
||||
|
||||
# Compilation settings that should be applied to most targets. |
||||
function(APPLY_STANDARD_SETTINGS TARGET) |
||||
target_compile_features(${TARGET} PUBLIC cxx_std_17) |
||||
target_compile_options(${TARGET} PRIVATE /W4 /WX /wd"4100") |
||||
target_compile_options(${TARGET} PRIVATE /EHsc) |
||||
target_compile_definitions(${TARGET} PRIVATE "_HAS_EXCEPTIONS=0") |
||||
target_compile_definitions(${TARGET} PRIVATE "$<$<CONFIG:Debug>:_DEBUG>") |
||||
endfunction() |
||||
|
||||
set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") |
||||
|
||||
# Flutter library and tool build rules. |
||||
add_subdirectory(${FLUTTER_MANAGED_DIR}) |
||||
|
||||
# Application build |
||||
add_subdirectory("runner") |
||||
|
||||
# Generated plugin build rules, which manage building the plugins and adding |
||||
# them to the application. |
||||
include(flutter/generated_plugins.cmake) |
||||
|
||||
|
||||
# === Installation === |
||||
# Support files are copied into place next to the executable, so that it can |
||||
# run in place. This is done instead of making a separate bundle (as on Linux) |
||||
# so that building and running from within Visual Studio will work. |
||||
set(BUILD_BUNDLE_DIR "$<TARGET_FILE_DIR:${BINARY_NAME}>") |
||||
# Make the "install" step default, as it's required to run. |
||||
set(CMAKE_VS_INCLUDE_INSTALL_TO_DEFAULT_BUILD 1) |
||||
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) |
||||
set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) |
||||
endif() |
||||
|
||||
set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") |
||||
set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}") |
||||
|
||||
install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
if(PLUGIN_BUNDLED_LIBRARIES) |
||||
install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" |
||||
DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endif() |
||||
|
||||
# Fully re-copy the assets directory on each build to avoid having stale files |
||||
# from a previous install. |
||||
set(FLUTTER_ASSET_DIR_NAME "flutter_assets") |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") |
||||
" COMPONENT Runtime) |
||||
install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" |
||||
DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) |
||||
|
||||
# Install the AOT library on non-Debug builds only. |
||||
install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
CONFIGURATIONS Profile;Release |
||||
COMPONENT Runtime) |
||||
# Project-level configuration. |
||||
cmake_minimum_required(VERSION 3.14) |
||||
project(example LANGUAGES CXX) |
||||
|
||||
# The name of the executable created for the application. Change this to change |
||||
# the on-disk name of your application. |
||||
set(BINARY_NAME "example") |
||||
|
||||
# Explicitly opt in to modern CMake behaviors to avoid warnings with recent |
||||
# versions of CMake. |
||||
cmake_policy(VERSION 3.14...3.25) |
||||
|
||||
# Define build configuration option. |
||||
get_property(IS_MULTICONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG) |
||||
if(IS_MULTICONFIG) |
||||
set(CMAKE_CONFIGURATION_TYPES "Debug;Profile;Release" |
||||
CACHE STRING "" FORCE) |
||||
else() |
||||
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) |
||||
set(CMAKE_BUILD_TYPE "Debug" CACHE |
||||
STRING "Flutter build mode" FORCE) |
||||
set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS |
||||
"Debug" "Profile" "Release") |
||||
endif() |
||||
endif() |
||||
# Define settings for the Profile build mode. |
||||
set(CMAKE_EXE_LINKER_FLAGS_PROFILE "${CMAKE_EXE_LINKER_FLAGS_RELEASE}") |
||||
set(CMAKE_SHARED_LINKER_FLAGS_PROFILE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE}") |
||||
set(CMAKE_C_FLAGS_PROFILE "${CMAKE_C_FLAGS_RELEASE}") |
||||
set(CMAKE_CXX_FLAGS_PROFILE "${CMAKE_CXX_FLAGS_RELEASE}") |
||||
|
||||
# Use Unicode for all projects. |
||||
add_definitions(-DUNICODE -D_UNICODE) |
||||
|
||||
# Compilation settings that should be applied to most targets. |
||||
# |
||||
# Be cautious about adding new options here, as plugins use this function by |
||||
# default. In most cases, you should add new options to specific targets instead |
||||
# of modifying this function. |
||||
function(APPLY_STANDARD_SETTINGS TARGET) |
||||
target_compile_features(${TARGET} PUBLIC cxx_std_17) |
||||
target_compile_options(${TARGET} PRIVATE /W4 /WX /wd"4100") |
||||
target_compile_options(${TARGET} PRIVATE /EHsc) |
||||
target_compile_definitions(${TARGET} PRIVATE "_HAS_EXCEPTIONS=0") |
||||
target_compile_definitions(${TARGET} PRIVATE "$<$<CONFIG:Debug>:_DEBUG>") |
||||
endfunction() |
||||
|
||||
# Flutter library and tool build rules. |
||||
set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") |
||||
add_subdirectory(${FLUTTER_MANAGED_DIR}) |
||||
|
||||
# Application build; see runner/CMakeLists.txt. |
||||
add_subdirectory("runner") |
||||
|
||||
|
||||
# Generated plugin build rules, which manage building the plugins and adding |
||||
# them to the application. |
||||
include(flutter/generated_plugins.cmake) |
||||
|
||||
|
||||
# === Installation === |
||||
# Support files are copied into place next to the executable, so that it can |
||||
# run in place. This is done instead of making a separate bundle (as on Linux) |
||||
# so that building and running from within Visual Studio will work. |
||||
set(BUILD_BUNDLE_DIR "$<TARGET_FILE_DIR:${BINARY_NAME}>") |
||||
# Make the "install" step default, as it's required to run. |
||||
set(CMAKE_VS_INCLUDE_INSTALL_TO_DEFAULT_BUILD 1) |
||||
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) |
||||
set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) |
||||
endif() |
||||
|
||||
set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") |
||||
set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}") |
||||
|
||||
install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
|
||||
if(PLUGIN_BUNDLED_LIBRARIES) |
||||
install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" |
||||
DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" |
||||
COMPONENT Runtime) |
||||
endif() |
||||
|
||||
# Fully re-copy the assets directory on each build to avoid having stale files |
||||
# from a previous install. |
||||
set(FLUTTER_ASSET_DIR_NAME "flutter_assets") |
||||
install(CODE " |
||||
file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") |
||||
" COMPONENT Runtime) |
||||
install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" |
||||
DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) |
||||
|
||||
# Install the AOT library on non-Debug builds only. |
||||
install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" |
||||
CONFIGURATIONS Profile;Release |
||||
COMPONENT Runtime) |
||||
|
@ -1,103 +1,104 @@ |
||||
cmake_minimum_required(VERSION 3.15) |
||||
|
||||
set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") |
||||
|
||||
# Configuration provided via flutter tool. |
||||
include(${EPHEMERAL_DIR}/generated_config.cmake) |
||||
|
||||
# TODO: Move the rest of this into files in ephemeral. See |
||||
# https://github.com/flutter/flutter/issues/57146. |
||||
set(WRAPPER_ROOT "${EPHEMERAL_DIR}/cpp_client_wrapper") |
||||
|
||||
# === Flutter Library === |
||||
set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/flutter_windows.dll") |
||||
|
||||
# Published to parent scope for install step. |
||||
set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) |
||||
set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) |
||||
set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) |
||||
set(AOT_LIBRARY "${PROJECT_DIR}/build/windows/app.so" PARENT_SCOPE) |
||||
|
||||
list(APPEND FLUTTER_LIBRARY_HEADERS |
||||
"flutter_export.h" |
||||
"flutter_windows.h" |
||||
"flutter_messenger.h" |
||||
"flutter_plugin_registrar.h" |
||||
"flutter_texture_registrar.h" |
||||
) |
||||
list(TRANSFORM FLUTTER_LIBRARY_HEADERS PREPEND "${EPHEMERAL_DIR}/") |
||||
add_library(flutter INTERFACE) |
||||
target_include_directories(flutter INTERFACE |
||||
"${EPHEMERAL_DIR}" |
||||
) |
||||
target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}.lib") |
||||
add_dependencies(flutter flutter_assemble) |
||||
|
||||
# === Wrapper === |
||||
list(APPEND CPP_WRAPPER_SOURCES_CORE |
||||
"core_implementations.cc" |
||||
"standard_codec.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_CORE PREPEND "${WRAPPER_ROOT}/") |
||||
list(APPEND CPP_WRAPPER_SOURCES_PLUGIN |
||||
"plugin_registrar.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_PLUGIN PREPEND "${WRAPPER_ROOT}/") |
||||
list(APPEND CPP_WRAPPER_SOURCES_APP |
||||
"flutter_engine.cc" |
||||
"flutter_view_controller.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_APP PREPEND "${WRAPPER_ROOT}/") |
||||
|
||||
# Wrapper sources needed for a plugin. |
||||
add_library(flutter_wrapper_plugin STATIC |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
) |
||||
apply_standard_settings(flutter_wrapper_plugin) |
||||
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||
POSITION_INDEPENDENT_CODE ON) |
||||
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||
CXX_VISIBILITY_PRESET hidden) |
||||
target_link_libraries(flutter_wrapper_plugin PUBLIC flutter) |
||||
target_include_directories(flutter_wrapper_plugin PUBLIC |
||||
"${WRAPPER_ROOT}/include" |
||||
) |
||||
add_dependencies(flutter_wrapper_plugin flutter_assemble) |
||||
|
||||
# Wrapper sources needed for the runner. |
||||
add_library(flutter_wrapper_app STATIC |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
) |
||||
apply_standard_settings(flutter_wrapper_app) |
||||
target_link_libraries(flutter_wrapper_app PUBLIC flutter) |
||||
target_include_directories(flutter_wrapper_app PUBLIC |
||||
"${WRAPPER_ROOT}/include" |
||||
) |
||||
add_dependencies(flutter_wrapper_app flutter_assemble) |
||||
|
||||
# === Flutter tool backend === |
||||
# _phony_ is a non-existent file to force this command to run every time, |
||||
# since currently there's no way to get a full input/output list from the |
||||
# flutter tool. |
||||
set(PHONY_OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/_phony_") |
||||
set_source_files_properties("${PHONY_OUTPUT}" PROPERTIES SYMBOLIC TRUE) |
||||
add_custom_command( |
||||
OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} |
||||
${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
${PHONY_OUTPUT} |
||||
COMMAND ${CMAKE_COMMAND} -E env |
||||
${FLUTTER_TOOL_ENVIRONMENT} |
||||
"${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.bat" |
||||
windows-x64 $<CONFIG> |
||||
VERBATIM |
||||
) |
||||
add_custom_target(flutter_assemble DEPENDS |
||||
"${FLUTTER_LIBRARY}" |
||||
${FLUTTER_LIBRARY_HEADERS} |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
) |
||||
# This file controls Flutter-level build steps. It should not be edited. |
||||
cmake_minimum_required(VERSION 3.14) |
||||
|
||||
set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") |
||||
|
||||
# Configuration provided via flutter tool. |
||||
include(${EPHEMERAL_DIR}/generated_config.cmake) |
||||
|
||||
# TODO: Move the rest of this into files in ephemeral. See |
||||
# https://github.com/flutter/flutter/issues/57146. |
||||
set(WRAPPER_ROOT "${EPHEMERAL_DIR}/cpp_client_wrapper") |
||||
|
||||
# === Flutter Library === |
||||
set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/flutter_windows.dll") |
||||
|
||||
# Published to parent scope for install step. |
||||
set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) |
||||
set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) |
||||
set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) |
||||
set(AOT_LIBRARY "${PROJECT_DIR}/build/windows/app.so" PARENT_SCOPE) |
||||
|
||||
list(APPEND FLUTTER_LIBRARY_HEADERS |
||||
"flutter_export.h" |
||||
"flutter_windows.h" |
||||
"flutter_messenger.h" |
||||
"flutter_plugin_registrar.h" |
||||
"flutter_texture_registrar.h" |
||||
) |
||||
list(TRANSFORM FLUTTER_LIBRARY_HEADERS PREPEND "${EPHEMERAL_DIR}/") |
||||
add_library(flutter INTERFACE) |
||||
target_include_directories(flutter INTERFACE |
||||
"${EPHEMERAL_DIR}" |
||||
) |
||||
target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}.lib") |
||||
add_dependencies(flutter flutter_assemble) |
||||
|
||||
# === Wrapper === |
||||
list(APPEND CPP_WRAPPER_SOURCES_CORE |
||||
"core_implementations.cc" |
||||
"standard_codec.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_CORE PREPEND "${WRAPPER_ROOT}/") |
||||
list(APPEND CPP_WRAPPER_SOURCES_PLUGIN |
||||
"plugin_registrar.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_PLUGIN PREPEND "${WRAPPER_ROOT}/") |
||||
list(APPEND CPP_WRAPPER_SOURCES_APP |
||||
"flutter_engine.cc" |
||||
"flutter_view_controller.cc" |
||||
) |
||||
list(TRANSFORM CPP_WRAPPER_SOURCES_APP PREPEND "${WRAPPER_ROOT}/") |
||||
|
||||
# Wrapper sources needed for a plugin. |
||||
add_library(flutter_wrapper_plugin STATIC |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
) |
||||
apply_standard_settings(flutter_wrapper_plugin) |
||||
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||
POSITION_INDEPENDENT_CODE ON) |
||||
set_target_properties(flutter_wrapper_plugin PROPERTIES |
||||
CXX_VISIBILITY_PRESET hidden) |
||||
target_link_libraries(flutter_wrapper_plugin PUBLIC flutter) |
||||
target_include_directories(flutter_wrapper_plugin PUBLIC |
||||
"${WRAPPER_ROOT}/include" |
||||
) |
||||
add_dependencies(flutter_wrapper_plugin flutter_assemble) |
||||
|
||||
# Wrapper sources needed for the runner. |
||||
add_library(flutter_wrapper_app STATIC |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
) |
||||
apply_standard_settings(flutter_wrapper_app) |
||||
target_link_libraries(flutter_wrapper_app PUBLIC flutter) |
||||
target_include_directories(flutter_wrapper_app PUBLIC |
||||
"${WRAPPER_ROOT}/include" |
||||
) |
||||
add_dependencies(flutter_wrapper_app flutter_assemble) |
||||
|
||||
# === Flutter tool backend === |
||||
# _phony_ is a non-existent file to force this command to run every time, |
||||
# since currently there's no way to get a full input/output list from the |
||||
# flutter tool. |
||||
set(PHONY_OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/_phony_") |
||||
set_source_files_properties("${PHONY_OUTPUT}" PROPERTIES SYMBOLIC TRUE) |
||||
add_custom_command( |
||||
OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} |
||||
${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
${PHONY_OUTPUT} |
||||
COMMAND ${CMAKE_COMMAND} -E env |
||||
${FLUTTER_TOOL_ENVIRONMENT} |
||||
"${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.bat" |
||||
windows-x64 $<CONFIG> |
||||
VERBATIM |
||||
) |
||||
add_custom_target(flutter_assemble DEPENDS |
||||
"${FLUTTER_LIBRARY}" |
||||
${FLUTTER_LIBRARY_HEADERS} |
||||
${CPP_WRAPPER_SOURCES_CORE} |
||||
${CPP_WRAPPER_SOURCES_PLUGIN} |
||||
${CPP_WRAPPER_SOURCES_APP} |
||||
) |
||||
|
@ -1,18 +1,40 @@ |
||||
cmake_minimum_required(VERSION 3.15) |
||||
project(runner LANGUAGES CXX) |
||||
|
||||
add_executable(${BINARY_NAME} WIN32 |
||||
"flutter_window.cpp" |
||||
"main.cpp" |
||||
"run_loop.cpp" |
||||
"utils.cpp" |
||||
"win32_window.cpp" |
||||
"${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" |
||||
"Runner.rc" |
||||
"runner.exe.manifest" |
||||
) |
||||
apply_standard_settings(${BINARY_NAME}) |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "NOMINMAX") |
||||
target_link_libraries(${BINARY_NAME} PRIVATE flutter flutter_wrapper_app) |
||||
target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") |
||||
add_dependencies(${BINARY_NAME} flutter_assemble) |
||||
cmake_minimum_required(VERSION 3.14) |
||||
project(runner LANGUAGES CXX) |
||||
|
||||
# Define the application target. To change its name, change BINARY_NAME in the |
||||
# top-level CMakeLists.txt, not the value here, or `flutter run` will no longer |
||||
# work. |
||||
# |
||||
# Any new source files that you add to the application should be added here. |
||||
add_executable(${BINARY_NAME} WIN32 |
||||
"flutter_window.cpp" |
||||
"main.cpp" |
||||
"utils.cpp" |
||||
"win32_window.cpp" |
||||
"${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" |
||||
"Runner.rc" |
||||
"runner.exe.manifest" |
||||
) |
||||
|
||||
# Apply the standard set of build settings. This can be removed for applications |
||||
# that need different build settings. |
||||
apply_standard_settings(${BINARY_NAME}) |
||||
|
||||
# Add preprocessor definitions for the build version. |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION=\"${FLUTTER_VERSION}\"") |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MAJOR=${FLUTTER_VERSION_MAJOR}") |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MINOR=${FLUTTER_VERSION_MINOR}") |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_PATCH=${FLUTTER_VERSION_PATCH}") |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_BUILD=${FLUTTER_VERSION_BUILD}") |
||||
|
||||
# Disable Windows macros that collide with C++ standard library functions. |
||||
target_compile_definitions(${BINARY_NAME} PRIVATE "NOMINMAX") |
||||
|
||||
# Add dependency libraries and include directories. Add any application-specific |
||||
# dependencies here. |
||||
target_link_libraries(${BINARY_NAME} PRIVATE flutter flutter_wrapper_app) |
||||
target_link_libraries(${BINARY_NAME} PRIVATE "dwmapi.lib") |
||||
target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") |
||||
|
||||
# Run the Flutter tool portions of the build. This must not be removed. |
||||
add_dependencies(${BINARY_NAME} flutter_assemble) |
||||
|
@ -1,121 +1,121 @@ |
||||
// Microsoft Visual C++ generated resource script. |
||||
// |
||||
#pragma code_page(65001) |
||||
#include "resource.h" |
||||
|
||||
#define APSTUDIO_READONLY_SYMBOLS |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Generated from the TEXTINCLUDE 2 resource. |
||||
// |
||||
#include "winres.h" |
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
#undef APSTUDIO_READONLY_SYMBOLS |
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// English (United States) resources |
||||
|
||||
#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU) |
||||
LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US |
||||
|
||||
#ifdef APSTUDIO_INVOKED |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// TEXTINCLUDE |
||||
// |
||||
|
||||
1 TEXTINCLUDE |
||||
BEGIN |
||||
"resource.h\0" |
||||
END |
||||
|
||||
2 TEXTINCLUDE |
||||
BEGIN |
||||
"#include ""winres.h""\r\n" |
||||
"\0" |
||||
END |
||||
|
||||
3 TEXTINCLUDE |
||||
BEGIN |
||||
"\r\n" |
||||
"\0" |
||||
END |
||||
|
||||
#endif // APSTUDIO_INVOKED |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Icon |
||||
// |
||||
|
||||
// Icon with lowest ID value placed first to ensure application icon |
||||
// remains consistent on all systems. |
||||
IDI_APP_ICON ICON "resources\\app_icon.ico" |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Version |
||||
// |
||||
|
||||
#if defined(FLUTTER_VERSION_MAJOR) && defined(FLUTTER_VERSION_MINOR) && defined(FLUTTER_VERSION_PATCH) && defined(FLUTTER_VERSION_BUILD) |
||||
#define VERSION_AS_NUMBER FLUTTER_VERSION_MAJOR,FLUTTER_VERSION_MINOR,FLUTTER_VERSION_PATCH,FLUTTER_VERSION_BUILD |
||||
#else |
||||
#define VERSION_AS_NUMBER 1,0,0,0 |
||||
#endif |
||||
|
||||
#if defined(FLUTTER_VERSION) |
||||
#define VERSION_AS_STRING FLUTTER_VERSION |
||||
#else |
||||
#define VERSION_AS_STRING "1.0.0" |
||||
#endif |
||||
|
||||
VS_VERSION_INFO VERSIONINFO |
||||
FILEVERSION VERSION_AS_NUMBER |
||||
PRODUCTVERSION VERSION_AS_NUMBER |
||||
FILEFLAGSMASK VS_FFI_FILEFLAGSMASK |
||||
#ifdef _DEBUG |
||||
FILEFLAGS VS_FF_DEBUG |
||||
#else |
||||
FILEFLAGS 0x0L |
||||
#endif |
||||
FILEOS VOS__WINDOWS32 |
||||
FILETYPE VFT_APP |
||||
FILESUBTYPE 0x0L |
||||
BEGIN |
||||
BLOCK "StringFileInfo" |
||||
BEGIN |
||||
BLOCK "040904e4" |
||||
BEGIN |
||||
VALUE "CompanyName", "com.example" "\0" |
||||
VALUE "FileDescription", "A new Flutter project." "\0" |
||||
VALUE "FileVersion", VERSION_AS_STRING "\0" |
||||
VALUE "InternalName", "app" "\0" |
||||
VALUE "LegalCopyright", "Copyright (C) 2021 com.example. All rights reserved." "\0" |
||||
VALUE "OriginalFilename", "app.exe" "\0" |
||||
VALUE "ProductName", "app" "\0" |
||||
VALUE "ProductVersion", VERSION_AS_STRING "\0" |
||||
END |
||||
END |
||||
BLOCK "VarFileInfo" |
||||
BEGIN |
||||
VALUE "Translation", 0x409, 1252 |
||||
END |
||||
END |
||||
|
||||
#endif // English (United States) resources |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
|
||||
|
||||
|
||||
#ifndef APSTUDIO_INVOKED |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Generated from the TEXTINCLUDE 3 resource. |
||||
// |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
#endif // not APSTUDIO_INVOKED |
||||
// Microsoft Visual C++ generated resource script. |
||||
// |
||||
#pragma code_page(65001) |
||||
#include "resource.h" |
||||
|
||||
#define APSTUDIO_READONLY_SYMBOLS |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Generated from the TEXTINCLUDE 2 resource. |
||||
// |
||||
#include "winres.h" |
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
#undef APSTUDIO_READONLY_SYMBOLS |
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// English (United States) resources |
||||
|
||||
#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU) |
||||
LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US |
||||
|
||||
#ifdef APSTUDIO_INVOKED |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// TEXTINCLUDE |
||||
// |
||||
|
||||
1 TEXTINCLUDE |
||||
BEGIN |
||||
"resource.h\0" |
||||
END |
||||
|
||||
2 TEXTINCLUDE |
||||
BEGIN |
||||
"#include ""winres.h""\r\n" |
||||
"\0" |
||||
END |
||||
|
||||
3 TEXTINCLUDE |
||||
BEGIN |
||||
"\r\n" |
||||
"\0" |
||||
END |
||||
|
||||
#endif // APSTUDIO_INVOKED |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Icon |
||||
// |
||||
|
||||
// Icon with lowest ID value placed first to ensure application icon |
||||
// remains consistent on all systems. |
||||
IDI_APP_ICON ICON "resources\\app_icon.ico" |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Version |
||||
// |
||||
|
||||
#if defined(FLUTTER_VERSION_MAJOR) && defined(FLUTTER_VERSION_MINOR) && defined(FLUTTER_VERSION_PATCH) && defined(FLUTTER_VERSION_BUILD) |
||||
#define VERSION_AS_NUMBER FLUTTER_VERSION_MAJOR,FLUTTER_VERSION_MINOR,FLUTTER_VERSION_PATCH,FLUTTER_VERSION_BUILD |
||||
#else |
||||
#define VERSION_AS_NUMBER 1,0,0,0 |
||||
#endif |
||||
|
||||
#if defined(FLUTTER_VERSION) |
||||
#define VERSION_AS_STRING FLUTTER_VERSION |
||||
#else |
||||
#define VERSION_AS_STRING "1.0.0" |
||||
#endif |
||||
|
||||
VS_VERSION_INFO VERSIONINFO |
||||
FILEVERSION VERSION_AS_NUMBER |
||||
PRODUCTVERSION VERSION_AS_NUMBER |
||||
FILEFLAGSMASK VS_FFI_FILEFLAGSMASK |
||||
#ifdef _DEBUG |
||||
FILEFLAGS VS_FF_DEBUG |
||||
#else |
||||
FILEFLAGS 0x0L |
||||
#endif |
||||
FILEOS VOS__WINDOWS32 |
||||
FILETYPE VFT_APP |
||||
FILESUBTYPE 0x0L |
||||
BEGIN |
||||
BLOCK "StringFileInfo" |
||||
BEGIN |
||||
BLOCK "040904e4" |
||||
BEGIN |
||||
VALUE "CompanyName", "com.example" "\0" |
||||
VALUE "FileDescription", "example" "\0" |
||||
VALUE "FileVersion", VERSION_AS_STRING "\0" |
||||
VALUE "InternalName", "example" "\0" |
||||
VALUE "LegalCopyright", "Copyright (C) 2023 com.example. All rights reserved." "\0" |
||||
VALUE "OriginalFilename", "example.exe" "\0" |
||||
VALUE "ProductName", "example" "\0" |
||||
VALUE "ProductVersion", VERSION_AS_STRING "\0" |
||||
END |
||||
END |
||||
BLOCK "VarFileInfo" |
||||
BEGIN |
||||
VALUE "Translation", 0x409, 1252 |
||||
END |
||||
END |
||||
|
||||
#endif // English (United States) resources |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
|
||||
|
||||
|
||||
#ifndef APSTUDIO_INVOKED |
||||
///////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// Generated from the TEXTINCLUDE 3 resource. |
||||
// |
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////// |
||||
#endif // not APSTUDIO_INVOKED |
||||
|
@ -1,64 +1,71 @@ |
||||
#include "flutter_window.h" |
||||
|
||||
#include <optional> |
||||
|
||||
#include "flutter/generated_plugin_registrant.h" |
||||
|
||||
FlutterWindow::FlutterWindow(RunLoop* run_loop, |
||||
const flutter::DartProject& project) |
||||
: run_loop_(run_loop), project_(project) {} |
||||
|
||||
FlutterWindow::~FlutterWindow() {} |
||||
|
||||
bool FlutterWindow::OnCreate() { |
||||
if (!Win32Window::OnCreate()) { |
||||
return false; |
||||
} |
||||
|
||||
RECT frame = GetClientArea(); |
||||
|
||||
// The size here must match the window dimensions to avoid unnecessary surface
|
||||
// creation / destruction in the startup path.
|
||||
flutter_controller_ = std::make_unique<flutter::FlutterViewController>( |
||||
frame.right - frame.left, frame.bottom - frame.top, project_); |
||||
// Ensure that basic setup of the controller was successful.
|
||||
if (!flutter_controller_->engine() || !flutter_controller_->view()) { |
||||
return false; |
||||
} |
||||
RegisterPlugins(flutter_controller_->engine()); |
||||
run_loop_->RegisterFlutterInstance(flutter_controller_->engine()); |
||||
SetChildContent(flutter_controller_->view()->GetNativeWindow()); |
||||
return true; |
||||
} |
||||
|
||||
void FlutterWindow::OnDestroy() { |
||||
if (flutter_controller_) { |
||||
run_loop_->UnregisterFlutterInstance(flutter_controller_->engine()); |
||||
flutter_controller_ = nullptr; |
||||
} |
||||
|
||||
Win32Window::OnDestroy(); |
||||
} |
||||
|
||||
LRESULT |
||||
FlutterWindow::MessageHandler(HWND hwnd, UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
// Give Flutter, including plugins, an opporutunity to handle window messages.
|
||||
if (flutter_controller_) { |
||||
std::optional<LRESULT> result = |
||||
flutter_controller_->HandleTopLevelWindowProc(hwnd, message, wparam, |
||||
lparam); |
||||
if (result) { |
||||
return *result; |
||||
} |
||||
} |
||||
|
||||
switch (message) { |
||||
case WM_FONTCHANGE: |
||||
flutter_controller_->engine()->ReloadSystemFonts(); |
||||
break; |
||||
} |
||||
|
||||
return Win32Window::MessageHandler(hwnd, message, wparam, lparam); |
||||
} |
||||
#include "flutter_window.h" |
||||
|
||||
#include <optional> |
||||
|
||||
#include "flutter/generated_plugin_registrant.h" |
||||
|
||||
FlutterWindow::FlutterWindow(const flutter::DartProject& project) |
||||
: project_(project) {} |
||||
|
||||
FlutterWindow::~FlutterWindow() {} |
||||
|
||||
bool FlutterWindow::OnCreate() { |
||||
if (!Win32Window::OnCreate()) { |
||||
return false; |
||||
} |
||||
|
||||
RECT frame = GetClientArea(); |
||||
|
||||
// The size here must match the window dimensions to avoid unnecessary surface
|
||||
// creation / destruction in the startup path.
|
||||
flutter_controller_ = std::make_unique<flutter::FlutterViewController>( |
||||
frame.right - frame.left, frame.bottom - frame.top, project_); |
||||
// Ensure that basic setup of the controller was successful.
|
||||
if (!flutter_controller_->engine() || !flutter_controller_->view()) { |
||||
return false; |
||||
} |
||||
RegisterPlugins(flutter_controller_->engine()); |
||||
SetChildContent(flutter_controller_->view()->GetNativeWindow()); |
||||
|
||||
flutter_controller_->engine()->SetNextFrameCallback([&]() { |
||||
this->Show(); |
||||
}); |
||||
|
||||
// Flutter can complete the first frame before the "show window" callback is
|
||||
// registered. The following call ensures a frame is pending to ensure the
|
||||
// window is shown. It is a no-op if the first frame hasn't completed yet.
|
||||
flutter_controller_->ForceRedraw(); |
||||
|
||||
return true; |
||||
} |
||||
|
||||
void FlutterWindow::OnDestroy() { |
||||
if (flutter_controller_) { |
||||
flutter_controller_ = nullptr; |
||||
} |
||||
|
||||
Win32Window::OnDestroy(); |
||||
} |
||||
|
||||
LRESULT |
||||
FlutterWindow::MessageHandler(HWND hwnd, UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
// Give Flutter, including plugins, an opportunity to handle window messages.
|
||||
if (flutter_controller_) { |
||||
std::optional<LRESULT> result = |
||||
flutter_controller_->HandleTopLevelWindowProc(hwnd, message, wparam, |
||||
lparam); |
||||
if (result) { |
||||
return *result; |
||||
} |
||||
} |
||||
|
||||
switch (message) { |
||||
case WM_FONTCHANGE: |
||||
flutter_controller_->engine()->ReloadSystemFonts(); |
||||
break; |
||||
} |
||||
|
||||
return Win32Window::MessageHandler(hwnd, message, wparam, lparam); |
||||
} |
||||
|
@ -1,39 +1,33 @@ |
||||
#ifndef RUNNER_FLUTTER_WINDOW_H_ |
||||
#define RUNNER_FLUTTER_WINDOW_H_ |
||||
|
||||
#include <flutter/dart_project.h> |
||||
#include <flutter/flutter_view_controller.h> |
||||
|
||||
#include <memory> |
||||
|
||||
#include "run_loop.h" |
||||
#include "win32_window.h" |
||||
|
||||
// A window that does nothing but host a Flutter view.
|
||||
class FlutterWindow : public Win32Window { |
||||
public: |
||||
// Creates a new FlutterWindow driven by the |run_loop|, hosting a
|
||||
// Flutter view running |project|.
|
||||
explicit FlutterWindow(RunLoop* run_loop, |
||||
const flutter::DartProject& project); |
||||
virtual ~FlutterWindow(); |
||||
|
||||
protected: |
||||
// Win32Window:
|
||||
bool OnCreate() override; |
||||
void OnDestroy() override; |
||||
LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, |
||||
LPARAM const lparam) noexcept override; |
||||
|
||||
private: |
||||
// The run loop driving events for this window.
|
||||
RunLoop* run_loop_; |
||||
|
||||
// The project to run.
|
||||
flutter::DartProject project_; |
||||
|
||||
// The Flutter instance hosted by this window.
|
||||
std::unique_ptr<flutter::FlutterViewController> flutter_controller_; |
||||
}; |
||||
|
||||
#endif // RUNNER_FLUTTER_WINDOW_H_
|
||||
#ifndef RUNNER_FLUTTER_WINDOW_H_ |
||||
#define RUNNER_FLUTTER_WINDOW_H_ |
||||
|
||||
#include <flutter/dart_project.h> |
||||
#include <flutter/flutter_view_controller.h> |
||||
|
||||
#include <memory> |
||||
|
||||
#include "win32_window.h" |
||||
|
||||
// A window that does nothing but host a Flutter view.
|
||||
class FlutterWindow : public Win32Window { |
||||
public: |
||||
// Creates a new FlutterWindow hosting a Flutter view running |project|.
|
||||
explicit FlutterWindow(const flutter::DartProject& project); |
||||
virtual ~FlutterWindow(); |
||||
|
||||
protected: |
||||
// Win32Window:
|
||||
bool OnCreate() override; |
||||
void OnDestroy() override; |
||||
LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, |
||||
LPARAM const lparam) noexcept override; |
||||
|
||||
private: |
||||
// The project to run.
|
||||
flutter::DartProject project_; |
||||
|
||||
// The Flutter instance hosted by this window.
|
||||
std::unique_ptr<flutter::FlutterViewController> flutter_controller_; |
||||
}; |
||||
|
||||
#endif // RUNNER_FLUTTER_WINDOW_H_
|
||||
|
@ -1,42 +1,43 @@ |
||||
#include <flutter/dart_project.h> |
||||
#include <flutter/flutter_view_controller.h> |
||||
#include <windows.h> |
||||
|
||||
#include "flutter_window.h" |
||||
#include "run_loop.h" |
||||
#include "utils.h" |
||||
|
||||
int APIENTRY wWinMain(_In_ HINSTANCE instance, _In_opt_ HINSTANCE prev, |
||||
_In_ wchar_t *command_line, _In_ int show_command) { |
||||
// Attach to console when present (e.g., 'flutter run') or create a
|
||||
// new console when running with a debugger.
|
||||
if (!::AttachConsole(ATTACH_PARENT_PROCESS) && ::IsDebuggerPresent()) { |
||||
CreateAndAttachConsole(); |
||||
} |
||||
|
||||
// Initialize COM, so that it is available for use in the library and/or
|
||||
// plugins.
|
||||
::CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED); |
||||
|
||||
RunLoop run_loop; |
||||
|
||||
flutter::DartProject project(L"data"); |
||||
|
||||
std::vector<std::string> command_line_arguments = |
||||
GetCommandLineArguments(); |
||||
|
||||
project.set_dart_entrypoint_arguments(std::move(command_line_arguments)); |
||||
|
||||
FlutterWindow window(&run_loop, project); |
||||
Win32Window::Point origin(10, 10); |
||||
Win32Window::Size size(1280, 720); |
||||
if (!window.CreateAndShow(L"app", origin, size)) { |
||||
return EXIT_FAILURE; |
||||
} |
||||
window.SetQuitOnClose(true); |
||||
|
||||
run_loop.Run(); |
||||
|
||||
::CoUninitialize(); |
||||
return EXIT_SUCCESS; |
||||
} |
||||
#include <flutter/dart_project.h> |
||||
#include <flutter/flutter_view_controller.h> |
||||
#include <windows.h> |
||||
|
||||
#include "flutter_window.h" |
||||
#include "utils.h" |
||||
|
||||
int APIENTRY wWinMain(_In_ HINSTANCE instance, _In_opt_ HINSTANCE prev, |
||||
_In_ wchar_t *command_line, _In_ int show_command) { |
||||
// Attach to console when present (e.g., 'flutter run') or create a
|
||||
// new console when running with a debugger.
|
||||
if (!::AttachConsole(ATTACH_PARENT_PROCESS) && ::IsDebuggerPresent()) { |
||||
CreateAndAttachConsole(); |
||||
} |
||||
|
||||
// Initialize COM, so that it is available for use in the library and/or
|
||||
// plugins.
|
||||
::CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED); |
||||
|
||||
flutter::DartProject project(L"data"); |
||||
|
||||
std::vector<std::string> command_line_arguments = |
||||
GetCommandLineArguments(); |
||||
|
||||
project.set_dart_entrypoint_arguments(std::move(command_line_arguments)); |
||||
|
||||
FlutterWindow window(project); |
||||
Win32Window::Point origin(10, 10); |
||||
Win32Window::Size size(1280, 720); |
||||
if (!window.Create(L"example", origin, size)) { |
||||
return EXIT_FAILURE; |
||||
} |
||||
window.SetQuitOnClose(true); |
||||
|
||||
::MSG msg; |
||||
while (::GetMessage(&msg, nullptr, 0, 0)) { |
||||
::TranslateMessage(&msg); |
||||
::DispatchMessage(&msg); |
||||
} |
||||
|
||||
::CoUninitialize(); |
||||
return EXIT_SUCCESS; |
||||
} |
||||
|
@ -1,16 +1,16 @@ |
||||
//{{NO_DEPENDENCIES}}
|
||||
// Microsoft Visual C++ generated include file.
|
||||
// Used by Runner.rc
|
||||
//
|
||||
#define IDI_APP_ICON 101 |
||||
|
||||
// Next default values for new objects
|
||||
//
|
||||
#ifdef APSTUDIO_INVOKED |
||||
#ifndef APSTUDIO_READONLY_SYMBOLS |
||||
#define _APS_NEXT_RESOURCE_VALUE 102 |
||||
#define _APS_NEXT_COMMAND_VALUE 40001 |
||||
#define _APS_NEXT_CONTROL_VALUE 1001 |
||||
#define _APS_NEXT_SYMED_VALUE 101 |
||||
#endif |
||||
#endif |
||||
//{{NO_DEPENDENCIES}}
|
||||
// Microsoft Visual C++ generated include file.
|
||||
// Used by Runner.rc
|
||||
//
|
||||
#define IDI_APP_ICON 101 |
||||
|
||||
// Next default values for new objects
|
||||
//
|
||||
#ifdef APSTUDIO_INVOKED |
||||
#ifndef APSTUDIO_READONLY_SYMBOLS |
||||
#define _APS_NEXT_RESOURCE_VALUE 102 |
||||
#define _APS_NEXT_COMMAND_VALUE 40001 |
||||
#define _APS_NEXT_CONTROL_VALUE 1001 |
||||
#define _APS_NEXT_SYMED_VALUE 101 |
||||
#endif |
||||
#endif |
||||
|
@ -1,66 +0,0 @@ |
||||
#include "run_loop.h" |
||||
|
||||
#include <windows.h> |
||||
|
||||
#include <algorithm> |
||||
|
||||
RunLoop::RunLoop() {} |
||||
|
||||
RunLoop::~RunLoop() {} |
||||
|
||||
void RunLoop::Run() { |
||||
bool keep_running = true; |
||||
TimePoint next_flutter_event_time = TimePoint::clock::now(); |
||||
while (keep_running) { |
||||
std::chrono::nanoseconds wait_duration = |
||||
std::max(std::chrono::nanoseconds(0), |
||||
next_flutter_event_time - TimePoint::clock::now()); |
||||
::MsgWaitForMultipleObjects( |
||||
0, nullptr, FALSE, static_cast<DWORD>(wait_duration.count() / 1000), |
||||
QS_ALLINPUT); |
||||
bool processed_events = false; |
||||
MSG message; |
||||
// All pending Windows messages must be processed; MsgWaitForMultipleObjects
|
||||
// won't return again for items left in the queue after PeekMessage.
|
||||
while (::PeekMessage(&message, nullptr, 0, 0, PM_REMOVE)) { |
||||
processed_events = true; |
||||
if (message.message == WM_QUIT) { |
||||
keep_running = false; |
||||
break; |
||||
} |
||||
::TranslateMessage(&message); |
||||
::DispatchMessage(&message); |
||||
// Allow Flutter to process messages each time a Windows message is
|
||||
// processed, to prevent starvation.
|
||||
next_flutter_event_time = |
||||
std::min(next_flutter_event_time, ProcessFlutterMessages()); |
||||
} |
||||
// If the PeekMessage loop didn't run, process Flutter messages.
|
||||
if (!processed_events) { |
||||
next_flutter_event_time = |
||||
std::min(next_flutter_event_time, ProcessFlutterMessages()); |
||||
} |
||||
} |
||||
} |
||||
|
||||
void RunLoop::RegisterFlutterInstance( |
||||
flutter::FlutterEngine* flutter_instance) { |
||||
flutter_instances_.insert(flutter_instance); |
||||
} |
||||
|
||||
void RunLoop::UnregisterFlutterInstance( |
||||
flutter::FlutterEngine* flutter_instance) { |
||||
flutter_instances_.erase(flutter_instance); |
||||
} |
||||
|
||||
RunLoop::TimePoint RunLoop::ProcessFlutterMessages() { |
||||
TimePoint next_event_time = TimePoint::max(); |
||||
for (auto instance : flutter_instances_) { |
||||
std::chrono::nanoseconds wait_duration = instance->ProcessMessages(); |
||||
if (wait_duration != std::chrono::nanoseconds::max()) { |
||||
next_event_time = |
||||
std::min(next_event_time, TimePoint::clock::now() + wait_duration); |
||||
} |
||||
} |
||||
return next_event_time; |
||||
} |
@ -1,40 +0,0 @@ |
||||
#ifndef RUNNER_RUN_LOOP_H_ |
||||
#define RUNNER_RUN_LOOP_H_ |
||||
|
||||
#include <flutter/flutter_engine.h> |
||||
|
||||
#include <chrono> |
||||
#include <set> |
||||
|
||||
// A runloop that will service events for Flutter instances as well
|
||||
// as native messages.
|
||||
class RunLoop { |
||||
public: |
||||
RunLoop(); |
||||
~RunLoop(); |
||||
|
||||
// Prevent copying
|
||||
RunLoop(RunLoop const&) = delete; |
||||
RunLoop& operator=(RunLoop const&) = delete; |
||||
|
||||
// Runs the run loop until the application quits.
|
||||
void Run(); |
||||
|
||||
// Registers the given Flutter instance for event servicing.
|
||||
void RegisterFlutterInstance( |
||||
flutter::FlutterEngine* flutter_instance); |
||||
|
||||
// Unregisters the given Flutter instance from event servicing.
|
||||
void UnregisterFlutterInstance( |
||||
flutter::FlutterEngine* flutter_instance); |
||||
|
||||
private: |
||||
using TimePoint = std::chrono::steady_clock::time_point; |
||||
|
||||
// Processes all currently pending messages for registered Flutter instances.
|
||||
TimePoint ProcessFlutterMessages(); |
||||
|
||||
std::set<flutter::FlutterEngine*> flutter_instances_; |
||||
}; |
||||
|
||||
#endif // RUNNER_RUN_LOOP_H_
|
@ -1,20 +1,20 @@ |
||||
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> |
||||
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0"> |
||||
<application xmlns="urn:schemas-microsoft-com:asm.v3"> |
||||
<windowsSettings> |
||||
<dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness> |
||||
</windowsSettings> |
||||
</application> |
||||
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> |
||||
<application> |
||||
<!-- Windows 10 --> |
||||
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/> |
||||
<!-- Windows 8.1 --> |
||||
<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/> |
||||
<!-- Windows 8 --> |
||||
<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/> |
||||
<!-- Windows 7 --> |
||||
<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/> |
||||
</application> |
||||
</compatibility> |
||||
</assembly> |
||||
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> |
||||
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0"> |
||||
<application xmlns="urn:schemas-microsoft-com:asm.v3"> |
||||
<windowsSettings> |
||||
<dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness> |
||||
</windowsSettings> |
||||
</application> |
||||
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> |
||||
<application> |
||||
<!-- Windows 10 and Windows 11 --> |
||||
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}"/> |
||||
<!-- Windows 8.1 --> |
||||
<supportedOS Id="{1f676c76-80e1-4239-95bb-83d0f6d0da78}"/> |
||||
<!-- Windows 8 --> |
||||
<supportedOS Id="{4a2f28e3-53b9-4441-ba9c-d69d4a4a6e38}"/> |
||||
<!-- Windows 7 --> |
||||
<supportedOS Id="{35138b9a-5d96-4fbd-8e2d-a2440225f93a}"/> |
||||
</application> |
||||
</compatibility> |
||||
</assembly> |
||||
|
@ -1,64 +1,65 @@ |
||||
#include "utils.h" |
||||
|
||||
#include <flutter_windows.h> |
||||
#include <io.h> |
||||
#include <stdio.h> |
||||
#include <windows.h> |
||||
|
||||
#include <iostream> |
||||
|
||||
void CreateAndAttachConsole() { |
||||
if (::AllocConsole()) { |
||||
FILE *unused; |
||||
if (freopen_s(&unused, "CONOUT$", "w", stdout)) { |
||||
_dup2(_fileno(stdout), 1); |
||||
} |
||||
if (freopen_s(&unused, "CONOUT$", "w", stderr)) { |
||||
_dup2(_fileno(stdout), 2); |
||||
} |
||||
std::ios::sync_with_stdio(); |
||||
FlutterDesktopResyncOutputStreams(); |
||||
} |
||||
} |
||||
|
||||
std::vector<std::string> GetCommandLineArguments() { |
||||
// Convert the UTF-16 command line arguments to UTF-8 for the Engine to use.
|
||||
int argc; |
||||
wchar_t** argv = ::CommandLineToArgvW(::GetCommandLineW(), &argc); |
||||
if (argv == nullptr) { |
||||
return std::vector<std::string>(); |
||||
} |
||||
|
||||
std::vector<std::string> command_line_arguments; |
||||
|
||||
// Skip the first argument as it's the binary name.
|
||||
for (int i = 1; i < argc; i++) { |
||||
command_line_arguments.push_back(Utf8FromUtf16(argv[i])); |
||||
} |
||||
|
||||
::LocalFree(argv); |
||||
|
||||
return command_line_arguments; |
||||
} |
||||
|
||||
std::string Utf8FromUtf16(const wchar_t* utf16_string) { |
||||
if (utf16_string == nullptr) { |
||||
return std::string(); |
||||
} |
||||
int target_length = ::WideCharToMultiByte( |
||||
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||
-1, nullptr, 0, nullptr, nullptr); |
||||
if (target_length == 0) { |
||||
return std::string(); |
||||
} |
||||
std::string utf8_string; |
||||
utf8_string.resize(target_length); |
||||
int converted_length = ::WideCharToMultiByte( |
||||
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||
-1, utf8_string.data(), |
||||
target_length, nullptr, nullptr); |
||||
if (converted_length == 0) { |
||||
return std::string(); |
||||
} |
||||
return utf8_string; |
||||
} |
||||
#include "utils.h" |
||||
|
||||
#include <flutter_windows.h> |
||||
#include <io.h> |
||||
#include <stdio.h> |
||||
#include <windows.h> |
||||
|
||||
#include <iostream> |
||||
|
||||
void CreateAndAttachConsole() { |
||||
if (::AllocConsole()) { |
||||
FILE *unused; |
||||
if (freopen_s(&unused, "CONOUT$", "w", stdout)) { |
||||
_dup2(_fileno(stdout), 1); |
||||
} |
||||
if (freopen_s(&unused, "CONOUT$", "w", stderr)) { |
||||
_dup2(_fileno(stdout), 2); |
||||
} |
||||
std::ios::sync_with_stdio(); |
||||
FlutterDesktopResyncOutputStreams(); |
||||
} |
||||
} |
||||
|
||||
std::vector<std::string> GetCommandLineArguments() { |
||||
// Convert the UTF-16 command line arguments to UTF-8 for the Engine to use.
|
||||
int argc; |
||||
wchar_t** argv = ::CommandLineToArgvW(::GetCommandLineW(), &argc); |
||||
if (argv == nullptr) { |
||||
return std::vector<std::string>(); |
||||
} |
||||
|
||||
std::vector<std::string> command_line_arguments; |
||||
|
||||
// Skip the first argument as it's the binary name.
|
||||
for (int i = 1; i < argc; i++) { |
||||
command_line_arguments.push_back(Utf8FromUtf16(argv[i])); |
||||
} |
||||
|
||||
::LocalFree(argv); |
||||
|
||||
return command_line_arguments; |
||||
} |
||||
|
||||
std::string Utf8FromUtf16(const wchar_t* utf16_string) { |
||||
if (utf16_string == nullptr) { |
||||
return std::string(); |
||||
} |
||||
int target_length = ::WideCharToMultiByte( |
||||
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||
-1, nullptr, 0, nullptr, nullptr) |
||||
-1; // remove the trailing null character
|
||||
int input_length = (int)wcslen(utf16_string); |
||||
std::string utf8_string; |
||||
if (target_length <= 0 || target_length > utf8_string.max_size()) { |
||||
return utf8_string; |
||||
} |
||||
utf8_string.resize(target_length); |
||||
int converted_length = ::WideCharToMultiByte( |
||||
CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, |
||||
input_length, utf8_string.data(), target_length, nullptr, nullptr); |
||||
if (converted_length == 0) { |
||||
return std::string(); |
||||
} |
||||
return utf8_string; |
||||
} |
||||
|
@ -1,19 +1,19 @@ |
||||
#ifndef RUNNER_UTILS_H_ |
||||
#define RUNNER_UTILS_H_ |
||||
|
||||
#include <string> |
||||
#include <vector> |
||||
|
||||
// Creates a console for the process, and redirects stdout and stderr to
|
||||
// it for both the runner and the Flutter library.
|
||||
void CreateAndAttachConsole(); |
||||
|
||||
// Takes a null-terminated wchar_t* encoded in UTF-16 and returns a std::string
|
||||
// encoded in UTF-8. Returns an empty std::string on failure.
|
||||
std::string Utf8FromUtf16(const wchar_t* utf16_string); |
||||
|
||||
// Gets the command line arguments passed in as a std::vector<std::string>,
|
||||
// encoded in UTF-8. Returns an empty std::vector<std::string> on failure.
|
||||
std::vector<std::string> GetCommandLineArguments(); |
||||
|
||||
#endif // RUNNER_UTILS_H_
|
||||
#ifndef RUNNER_UTILS_H_ |
||||
#define RUNNER_UTILS_H_ |
||||
|
||||
#include <string> |
||||
#include <vector> |
||||
|
||||
// Creates a console for the process, and redirects stdout and stderr to
|
||||
// it for both the runner and the Flutter library.
|
||||
void CreateAndAttachConsole(); |
||||
|
||||
// Takes a null-terminated wchar_t* encoded in UTF-16 and returns a std::string
|
||||
// encoded in UTF-8. Returns an empty std::string on failure.
|
||||
std::string Utf8FromUtf16(const wchar_t* utf16_string); |
||||
|
||||
// Gets the command line arguments passed in as a std::vector<std::string>,
|
||||
// encoded in UTF-8. Returns an empty std::vector<std::string> on failure.
|
||||
std::vector<std::string> GetCommandLineArguments(); |
||||
|
||||
#endif // RUNNER_UTILS_H_
|
||||
|
@ -1,245 +1,288 @@ |
||||
#include "win32_window.h" |
||||
|
||||
#include <flutter_windows.h> |
||||
|
||||
#include "resource.h" |
||||
|
||||
namespace { |
||||
|
||||
constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW"; |
||||
|
||||
// The number of Win32Window objects that currently exist.
|
||||
static int g_active_window_count = 0; |
||||
|
||||
using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd); |
||||
|
||||
// Scale helper to convert logical scaler values to physical using passed in
|
||||
// scale factor
|
||||
int Scale(int source, double scale_factor) { |
||||
return static_cast<int>(source * scale_factor); |
||||
} |
||||
|
||||
// Dynamically loads the |EnableNonClientDpiScaling| from the User32 module.
|
||||
// This API is only needed for PerMonitor V1 awareness mode.
|
||||
void EnableFullDpiSupportIfAvailable(HWND hwnd) { |
||||
HMODULE user32_module = LoadLibraryA("User32.dll"); |
||||
if (!user32_module) { |
||||
return; |
||||
} |
||||
auto enable_non_client_dpi_scaling = |
||||
reinterpret_cast<EnableNonClientDpiScaling*>( |
||||
GetProcAddress(user32_module, "EnableNonClientDpiScaling")); |
||||
if (enable_non_client_dpi_scaling != nullptr) { |
||||
enable_non_client_dpi_scaling(hwnd); |
||||
FreeLibrary(user32_module); |
||||
} |
||||
} |
||||
|
||||
} // namespace
|
||||
|
||||
// Manages the Win32Window's window class registration.
|
||||
class WindowClassRegistrar { |
||||
public: |
||||
~WindowClassRegistrar() = default; |
||||
|
||||
// Returns the singleton registar instance.
|
||||
static WindowClassRegistrar* GetInstance() { |
||||
if (!instance_) { |
||||
instance_ = new WindowClassRegistrar(); |
||||
} |
||||
return instance_; |
||||
} |
||||
|
||||
// Returns the name of the window class, registering the class if it hasn't
|
||||
// previously been registered.
|
||||
const wchar_t* GetWindowClass(); |
||||
|
||||
// Unregisters the window class. Should only be called if there are no
|
||||
// instances of the window.
|
||||
void UnregisterWindowClass(); |
||||
|
||||
private: |
||||
WindowClassRegistrar() = default; |
||||
|
||||
static WindowClassRegistrar* instance_; |
||||
|
||||
bool class_registered_ = false; |
||||
}; |
||||
|
||||
WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr; |
||||
|
||||
const wchar_t* WindowClassRegistrar::GetWindowClass() { |
||||
if (!class_registered_) { |
||||
WNDCLASS window_class{}; |
||||
window_class.hCursor = LoadCursor(nullptr, IDC_ARROW); |
||||
window_class.lpszClassName = kWindowClassName; |
||||
window_class.style = CS_HREDRAW | CS_VREDRAW; |
||||
window_class.cbClsExtra = 0; |
||||
window_class.cbWndExtra = 0; |
||||
window_class.hInstance = GetModuleHandle(nullptr); |
||||
window_class.hIcon = |
||||
LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON)); |
||||
window_class.hbrBackground = 0; |
||||
window_class.lpszMenuName = nullptr; |
||||
window_class.lpfnWndProc = Win32Window::WndProc; |
||||
RegisterClass(&window_class); |
||||
class_registered_ = true; |
||||
} |
||||
return kWindowClassName; |
||||
} |
||||
|
||||
void WindowClassRegistrar::UnregisterWindowClass() { |
||||
UnregisterClass(kWindowClassName, nullptr); |
||||
class_registered_ = false; |
||||
} |
||||
|
||||
Win32Window::Win32Window() { |
||||
++g_active_window_count; |
||||
} |
||||
|
||||
Win32Window::~Win32Window() { |
||||
--g_active_window_count; |
||||
Destroy(); |
||||
} |
||||
|
||||
bool Win32Window::CreateAndShow(const std::wstring& title, |
||||
const Point& origin, |
||||
const Size& size) { |
||||
Destroy(); |
||||
|
||||
const wchar_t* window_class = |
||||
WindowClassRegistrar::GetInstance()->GetWindowClass(); |
||||
|
||||
const POINT target_point = {static_cast<LONG>(origin.x), |
||||
static_cast<LONG>(origin.y)}; |
||||
HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST); |
||||
UINT dpi = FlutterDesktopGetDpiForMonitor(monitor); |
||||
double scale_factor = dpi / 96.0; |
||||
|
||||
HWND window = CreateWindow( |
||||
window_class, title.c_str(), WS_OVERLAPPEDWINDOW | WS_VISIBLE, |
||||
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor), |
||||
Scale(size.width, scale_factor), Scale(size.height, scale_factor), |
||||
nullptr, nullptr, GetModuleHandle(nullptr), this); |
||||
|
||||
if (!window) { |
||||
return false; |
||||
} |
||||
|
||||
return OnCreate(); |
||||
} |
||||
|
||||
// static
|
||||
LRESULT CALLBACK Win32Window::WndProc(HWND const window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
if (message == WM_NCCREATE) { |
||||
auto window_struct = reinterpret_cast<CREATESTRUCT*>(lparam); |
||||
SetWindowLongPtr(window, GWLP_USERDATA, |
||||
reinterpret_cast<LONG_PTR>(window_struct->lpCreateParams)); |
||||
|
||||
auto that = static_cast<Win32Window*>(window_struct->lpCreateParams); |
||||
EnableFullDpiSupportIfAvailable(window); |
||||
that->window_handle_ = window; |
||||
} else if (Win32Window* that = GetThisFromHandle(window)) { |
||||
return that->MessageHandler(window, message, wparam, lparam); |
||||
} |
||||
|
||||
return DefWindowProc(window, message, wparam, lparam); |
||||
} |
||||
|
||||
LRESULT |
||||
Win32Window::MessageHandler(HWND hwnd, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
switch (message) { |
||||
case WM_DESTROY: |
||||
window_handle_ = nullptr; |
||||
Destroy(); |
||||
if (quit_on_close_) { |
||||
PostQuitMessage(0); |
||||
} |
||||
return 0; |
||||
|
||||
case WM_DPICHANGED: { |
||||
auto newRectSize = reinterpret_cast<RECT*>(lparam); |
||||
LONG newWidth = newRectSize->right - newRectSize->left; |
||||
LONG newHeight = newRectSize->bottom - newRectSize->top; |
||||
|
||||
SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth, |
||||
newHeight, SWP_NOZORDER | SWP_NOACTIVATE); |
||||
|
||||
return 0; |
||||
} |
||||
case WM_SIZE: { |
||||
RECT rect = GetClientArea(); |
||||
if (child_content_ != nullptr) { |
||||
// Size and position the child window.
|
||||
MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left, |
||||
rect.bottom - rect.top, TRUE); |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
case WM_ACTIVATE: |
||||
if (child_content_ != nullptr) { |
||||
SetFocus(child_content_); |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
return DefWindowProc(window_handle_, message, wparam, lparam); |
||||
} |
||||
|
||||
void Win32Window::Destroy() { |
||||
OnDestroy(); |
||||
|
||||
if (window_handle_) { |
||||
DestroyWindow(window_handle_); |
||||
window_handle_ = nullptr; |
||||
} |
||||
if (g_active_window_count == 0) { |
||||
WindowClassRegistrar::GetInstance()->UnregisterWindowClass(); |
||||
} |
||||
} |
||||
|
||||
Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept { |
||||
return reinterpret_cast<Win32Window*>( |
||||
GetWindowLongPtr(window, GWLP_USERDATA)); |
||||
} |
||||
|
||||
void Win32Window::SetChildContent(HWND content) { |
||||
child_content_ = content; |
||||
SetParent(content, window_handle_); |
||||
RECT frame = GetClientArea(); |
||||
|
||||
MoveWindow(content, frame.left, frame.top, frame.right - frame.left, |
||||
frame.bottom - frame.top, true); |
||||
|
||||
SetFocus(child_content_); |
||||
} |
||||
|
||||
RECT Win32Window::GetClientArea() { |
||||
RECT frame; |
||||
GetClientRect(window_handle_, &frame); |
||||
return frame; |
||||
} |
||||
|
||||
HWND Win32Window::GetHandle() { |
||||
return window_handle_; |
||||
} |
||||
|
||||
void Win32Window::SetQuitOnClose(bool quit_on_close) { |
||||
quit_on_close_ = quit_on_close; |
||||
} |
||||
|
||||
bool Win32Window::OnCreate() { |
||||
// No-op; provided for subclasses.
|
||||
return true; |
||||
} |
||||
|
||||
void Win32Window::OnDestroy() { |
||||
// No-op; provided for subclasses.
|
||||
} |
||||
#include "win32_window.h" |
||||
|
||||
#include <dwmapi.h> |
||||
#include <flutter_windows.h> |
||||
|
||||
#include "resource.h" |
||||
|
||||
namespace { |
||||
|
||||
/// Window attribute that enables dark mode window decorations.
|
||||
///
|
||||
/// Redefined in case the developer's machine has a Windows SDK older than
|
||||
/// version 10.0.22000.0.
|
||||
/// See: https://docs.microsoft.com/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute
|
||||
#ifndef DWMWA_USE_IMMERSIVE_DARK_MODE |
||||
#define DWMWA_USE_IMMERSIVE_DARK_MODE 20 |
||||
#endif |
||||
|
||||
constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW"; |
||||
|
||||
/// Registry key for app theme preference.
|
||||
///
|
||||
/// A value of 0 indicates apps should use dark mode. A non-zero or missing
|
||||
/// value indicates apps should use light mode.
|
||||
constexpr const wchar_t kGetPreferredBrightnessRegKey[] = |
||||
L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize"; |
||||
constexpr const wchar_t kGetPreferredBrightnessRegValue[] = L"AppsUseLightTheme"; |
||||
|
||||
// The number of Win32Window objects that currently exist.
|
||||
static int g_active_window_count = 0; |
||||
|
||||
using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd); |
||||
|
||||
// Scale helper to convert logical scaler values to physical using passed in
|
||||
// scale factor
|
||||
int Scale(int source, double scale_factor) { |
||||
return static_cast<int>(source * scale_factor); |
||||
} |
||||
|
||||
// Dynamically loads the |EnableNonClientDpiScaling| from the User32 module.
|
||||
// This API is only needed for PerMonitor V1 awareness mode.
|
||||
void EnableFullDpiSupportIfAvailable(HWND hwnd) { |
||||
HMODULE user32_module = LoadLibraryA("User32.dll"); |
||||
if (!user32_module) { |
||||
return; |
||||
} |
||||
auto enable_non_client_dpi_scaling = |
||||
reinterpret_cast<EnableNonClientDpiScaling*>( |
||||
GetProcAddress(user32_module, "EnableNonClientDpiScaling")); |
||||
if (enable_non_client_dpi_scaling != nullptr) { |
||||
enable_non_client_dpi_scaling(hwnd); |
||||
} |
||||
FreeLibrary(user32_module); |
||||
} |
||||
|
||||
} // namespace
|
||||
|
||||
// Manages the Win32Window's window class registration.
|
||||
class WindowClassRegistrar { |
||||
public: |
||||
~WindowClassRegistrar() = default; |
||||
|
||||
// Returns the singleton registrar instance.
|
||||
static WindowClassRegistrar* GetInstance() { |
||||
if (!instance_) { |
||||
instance_ = new WindowClassRegistrar(); |
||||
} |
||||
return instance_; |
||||
} |
||||
|
||||
// Returns the name of the window class, registering the class if it hasn't
|
||||
// previously been registered.
|
||||
const wchar_t* GetWindowClass(); |
||||
|
||||
// Unregisters the window class. Should only be called if there are no
|
||||
// instances of the window.
|
||||
void UnregisterWindowClass(); |
||||
|
||||
private: |
||||
WindowClassRegistrar() = default; |
||||
|
||||
static WindowClassRegistrar* instance_; |
||||
|
||||
bool class_registered_ = false; |
||||
}; |
||||
|
||||
WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr; |
||||
|
||||
const wchar_t* WindowClassRegistrar::GetWindowClass() { |
||||
if (!class_registered_) { |
||||
WNDCLASS window_class{}; |
||||
window_class.hCursor = LoadCursor(nullptr, IDC_ARROW); |
||||
window_class.lpszClassName = kWindowClassName; |
||||
window_class.style = CS_HREDRAW | CS_VREDRAW; |
||||
window_class.cbClsExtra = 0; |
||||
window_class.cbWndExtra = 0; |
||||
window_class.hInstance = GetModuleHandle(nullptr); |
||||
window_class.hIcon = |
||||
LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON)); |
||||
window_class.hbrBackground = 0; |
||||
window_class.lpszMenuName = nullptr; |
||||
window_class.lpfnWndProc = Win32Window::WndProc; |
||||
RegisterClass(&window_class); |
||||
class_registered_ = true; |
||||
} |
||||
return kWindowClassName; |
||||
} |
||||
|
||||
void WindowClassRegistrar::UnregisterWindowClass() { |
||||
UnregisterClass(kWindowClassName, nullptr); |
||||
class_registered_ = false; |
||||
} |
||||
|
||||
Win32Window::Win32Window() { |
||||
++g_active_window_count; |
||||
} |
||||
|
||||
Win32Window::~Win32Window() { |
||||
--g_active_window_count; |
||||
Destroy(); |
||||
} |
||||
|
||||
bool Win32Window::Create(const std::wstring& title, |
||||
const Point& origin, |
||||
const Size& size) { |
||||
Destroy(); |
||||
|
||||
const wchar_t* window_class = |
||||
WindowClassRegistrar::GetInstance()->GetWindowClass(); |
||||
|
||||
const POINT target_point = {static_cast<LONG>(origin.x), |
||||
static_cast<LONG>(origin.y)}; |
||||
HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST); |
||||
UINT dpi = FlutterDesktopGetDpiForMonitor(monitor); |
||||
double scale_factor = dpi / 96.0; |
||||
|
||||
HWND window = CreateWindow( |
||||
window_class, title.c_str(), WS_OVERLAPPEDWINDOW, |
||||
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor), |
||||
Scale(size.width, scale_factor), Scale(size.height, scale_factor), |
||||
nullptr, nullptr, GetModuleHandle(nullptr), this); |
||||
|
||||
if (!window) { |
||||
return false; |
||||
} |
||||
|
||||
UpdateTheme(window); |
||||
|
||||
return OnCreate(); |
||||
} |
||||
|
||||
bool Win32Window::Show() { |
||||
return ShowWindow(window_handle_, SW_SHOWNORMAL); |
||||
} |
||||
|
||||
// static
|
||||
LRESULT CALLBACK Win32Window::WndProc(HWND const window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
if (message == WM_NCCREATE) { |
||||
auto window_struct = reinterpret_cast<CREATESTRUCT*>(lparam); |
||||
SetWindowLongPtr(window, GWLP_USERDATA, |
||||
reinterpret_cast<LONG_PTR>(window_struct->lpCreateParams)); |
||||
|
||||
auto that = static_cast<Win32Window*>(window_struct->lpCreateParams); |
||||
EnableFullDpiSupportIfAvailable(window); |
||||
that->window_handle_ = window; |
||||
} else if (Win32Window* that = GetThisFromHandle(window)) { |
||||
return that->MessageHandler(window, message, wparam, lparam); |
||||
} |
||||
|
||||
return DefWindowProc(window, message, wparam, lparam); |
||||
} |
||||
|
||||
LRESULT |
||||
Win32Window::MessageHandler(HWND hwnd, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept { |
||||
switch (message) { |
||||
case WM_DESTROY: |
||||
window_handle_ = nullptr; |
||||
Destroy(); |
||||
if (quit_on_close_) { |
||||
PostQuitMessage(0); |
||||
} |
||||
return 0; |
||||
|
||||
case WM_DPICHANGED: { |
||||
auto newRectSize = reinterpret_cast<RECT*>(lparam); |
||||
LONG newWidth = newRectSize->right - newRectSize->left; |
||||
LONG newHeight = newRectSize->bottom - newRectSize->top; |
||||
|
||||
SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth, |
||||
newHeight, SWP_NOZORDER | SWP_NOACTIVATE); |
||||
|
||||
return 0; |
||||
} |
||||
case WM_SIZE: { |
||||
RECT rect = GetClientArea(); |
||||
if (child_content_ != nullptr) { |
||||
// Size and position the child window.
|
||||
MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left, |
||||
rect.bottom - rect.top, TRUE); |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
case WM_ACTIVATE: |
||||
if (child_content_ != nullptr) { |
||||
SetFocus(child_content_); |
||||
} |
||||
return 0; |
||||
|
||||
case WM_DWMCOLORIZATIONCOLORCHANGED: |
||||
UpdateTheme(hwnd); |
||||
return 0; |
||||
} |
||||
|
||||
return DefWindowProc(window_handle_, message, wparam, lparam); |
||||
} |
||||
|
||||
void Win32Window::Destroy() { |
||||
OnDestroy(); |
||||
|
||||
if (window_handle_) { |
||||
DestroyWindow(window_handle_); |
||||
window_handle_ = nullptr; |
||||
} |
||||
if (g_active_window_count == 0) { |
||||
WindowClassRegistrar::GetInstance()->UnregisterWindowClass(); |
||||
} |
||||
} |
||||
|
||||
Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept { |
||||
return reinterpret_cast<Win32Window*>( |
||||
GetWindowLongPtr(window, GWLP_USERDATA)); |
||||
} |
||||
|
||||
void Win32Window::SetChildContent(HWND content) { |
||||
child_content_ = content; |
||||
SetParent(content, window_handle_); |
||||
RECT frame = GetClientArea(); |
||||
|
||||
MoveWindow(content, frame.left, frame.top, frame.right - frame.left, |
||||
frame.bottom - frame.top, true); |
||||
|
||||
SetFocus(child_content_); |
||||
} |
||||
|
||||
RECT Win32Window::GetClientArea() { |
||||
RECT frame; |
||||
GetClientRect(window_handle_, &frame); |
||||
return frame; |
||||
} |
||||
|
||||
HWND Win32Window::GetHandle() { |
||||
return window_handle_; |
||||
} |
||||
|
||||
void Win32Window::SetQuitOnClose(bool quit_on_close) { |
||||
quit_on_close_ = quit_on_close; |
||||
} |
||||
|
||||
bool Win32Window::OnCreate() { |
||||
// No-op; provided for subclasses.
|
||||
return true; |
||||
} |
||||
|
||||
void Win32Window::OnDestroy() { |
||||
// No-op; provided for subclasses.
|
||||
} |
||||
|
||||
void Win32Window::UpdateTheme(HWND const window) { |
||||
DWORD light_mode; |
||||
DWORD light_mode_size = sizeof(light_mode); |
||||
LSTATUS result = RegGetValue(HKEY_CURRENT_USER, kGetPreferredBrightnessRegKey, |
||||
kGetPreferredBrightnessRegValue, |
||||
RRF_RT_REG_DWORD, nullptr, &light_mode, |
||||
&light_mode_size); |
||||
|
||||
if (result == ERROR_SUCCESS) { |
||||
BOOL enable_dark_mode = light_mode == 0; |
||||
DwmSetWindowAttribute(window, DWMWA_USE_IMMERSIVE_DARK_MODE, |
||||
&enable_dark_mode, sizeof(enable_dark_mode)); |
||||
} |
||||
} |
||||
|
@ -1,98 +1,102 @@ |
||||
#ifndef RUNNER_WIN32_WINDOW_H_ |
||||
#define RUNNER_WIN32_WINDOW_H_ |
||||
|
||||
#include <windows.h> |
||||
|
||||
#include <functional> |
||||
#include <memory> |
||||
#include <string> |
||||
|
||||
// A class abstraction for a high DPI-aware Win32 Window. Intended to be
|
||||
// inherited from by classes that wish to specialize with custom
|
||||
// rendering and input handling
|
||||
class Win32Window { |
||||
public: |
||||
struct Point { |
||||
unsigned int x; |
||||
unsigned int y; |
||||
Point(unsigned int x, unsigned int y) : x(x), y(y) {} |
||||
}; |
||||
|
||||
struct Size { |
||||
unsigned int width; |
||||
unsigned int height; |
||||
Size(unsigned int width, unsigned int height) |
||||
: width(width), height(height) {} |
||||
}; |
||||
|
||||
Win32Window(); |
||||
virtual ~Win32Window(); |
||||
|
||||
// Creates and shows a win32 window with |title| and position and size using
|
||||
// |origin| and |size|. New windows are created on the default monitor. Window
|
||||
// sizes are specified to the OS in physical pixels, hence to ensure a
|
||||
// consistent size to will treat the width height passed in to this function
|
||||
// as logical pixels and scale to appropriate for the default monitor. Returns
|
||||
// true if the window was created successfully.
|
||||
bool CreateAndShow(const std::wstring& title, |
||||
const Point& origin, |
||||
const Size& size); |
||||
|
||||
// Release OS resources associated with window.
|
||||
void Destroy(); |
||||
|
||||
// Inserts |content| into the window tree.
|
||||
void SetChildContent(HWND content); |
||||
|
||||
// Returns the backing Window handle to enable clients to set icon and other
|
||||
// window properties. Returns nullptr if the window has been destroyed.
|
||||
HWND GetHandle(); |
||||
|
||||
// If true, closing this window will quit the application.
|
||||
void SetQuitOnClose(bool quit_on_close); |
||||
|
||||
// Return a RECT representing the bounds of the current client area.
|
||||
RECT GetClientArea(); |
||||
|
||||
protected: |
||||
// Processes and route salient window messages for mouse handling,
|
||||
// size change and DPI. Delegates handling of these to member overloads that
|
||||
// inheriting classes can handle.
|
||||
virtual LRESULT MessageHandler(HWND window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept; |
||||
|
||||
// Called when CreateAndShow is called, allowing subclass window-related
|
||||
// setup. Subclasses should return false if setup fails.
|
||||
virtual bool OnCreate(); |
||||
|
||||
// Called when Destroy is called.
|
||||
virtual void OnDestroy(); |
||||
|
||||
private: |
||||
friend class WindowClassRegistrar; |
||||
|
||||
// OS callback called by message pump. Handles the WM_NCCREATE message which
|
||||
// is passed when the non-client area is being created and enables automatic
|
||||
// non-client DPI scaling so that the non-client area automatically
|
||||
// responsponds to changes in DPI. All other messages are handled by
|
||||
// MessageHandler.
|
||||
static LRESULT CALLBACK WndProc(HWND const window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept; |
||||
|
||||
// Retrieves a class instance pointer for |window|
|
||||
static Win32Window* GetThisFromHandle(HWND const window) noexcept; |
||||
|
||||
bool quit_on_close_ = false; |
||||
|
||||
// window handle for top level window.
|
||||
HWND window_handle_ = nullptr; |
||||
|
||||
// window handle for hosted content.
|
||||
HWND child_content_ = nullptr; |
||||
}; |
||||
|
||||
#endif // RUNNER_WIN32_WINDOW_H_
|
||||
#ifndef RUNNER_WIN32_WINDOW_H_ |
||||
#define RUNNER_WIN32_WINDOW_H_ |
||||
|
||||
#include <windows.h> |
||||
|
||||
#include <functional> |
||||
#include <memory> |
||||
#include <string> |
||||
|
||||
// A class abstraction for a high DPI-aware Win32 Window. Intended to be
|
||||
// inherited from by classes that wish to specialize with custom
|
||||
// rendering and input handling
|
||||
class Win32Window { |
||||
public: |
||||
struct Point { |
||||
unsigned int x; |
||||
unsigned int y; |
||||
Point(unsigned int x, unsigned int y) : x(x), y(y) {} |
||||
}; |
||||
|
||||
struct Size { |
||||
unsigned int width; |
||||
unsigned int height; |
||||
Size(unsigned int width, unsigned int height) |
||||
: width(width), height(height) {} |
||||
}; |
||||
|
||||
Win32Window(); |
||||
virtual ~Win32Window(); |
||||
|
||||
// Creates a win32 window with |title| that is positioned and sized using
|
||||
// |origin| and |size|. New windows are created on the default monitor. Window
|
||||
// sizes are specified to the OS in physical pixels, hence to ensure a
|
||||
// consistent size this function will scale the inputted width and height as
|
||||
// as appropriate for the default monitor. The window is invisible until
|
||||
// |Show| is called. Returns true if the window was created successfully.
|
||||
bool Create(const std::wstring& title, const Point& origin, const Size& size); |
||||
|
||||
// Show the current window. Returns true if the window was successfully shown.
|
||||
bool Show(); |
||||
|
||||
// Release OS resources associated with window.
|
||||
void Destroy(); |
||||
|
||||
// Inserts |content| into the window tree.
|
||||
void SetChildContent(HWND content); |
||||
|
||||
// Returns the backing Window handle to enable clients to set icon and other
|
||||
// window properties. Returns nullptr if the window has been destroyed.
|
||||
HWND GetHandle(); |
||||
|
||||
// If true, closing this window will quit the application.
|
||||
void SetQuitOnClose(bool quit_on_close); |
||||
|
||||
// Return a RECT representing the bounds of the current client area.
|
||||
RECT GetClientArea(); |
||||
|
||||
protected: |
||||
// Processes and route salient window messages for mouse handling,
|
||||
// size change and DPI. Delegates handling of these to member overloads that
|
||||
// inheriting classes can handle.
|
||||
virtual LRESULT MessageHandler(HWND window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept; |
||||
|
||||
// Called when CreateAndShow is called, allowing subclass window-related
|
||||
// setup. Subclasses should return false if setup fails.
|
||||
virtual bool OnCreate(); |
||||
|
||||
// Called when Destroy is called.
|
||||
virtual void OnDestroy(); |
||||
|
||||
private: |
||||
friend class WindowClassRegistrar; |
||||
|
||||
// OS callback called by message pump. Handles the WM_NCCREATE message which
|
||||
// is passed when the non-client area is being created and enables automatic
|
||||
// non-client DPI scaling so that the non-client area automatically
|
||||
// responds to changes in DPI. All other messages are handled by
|
||||
// MessageHandler.
|
||||
static LRESULT CALLBACK WndProc(HWND const window, |
||||
UINT const message, |
||||
WPARAM const wparam, |
||||
LPARAM const lparam) noexcept; |
||||
|
||||
// Retrieves a class instance pointer for |window|
|
||||
static Win32Window* GetThisFromHandle(HWND const window) noexcept; |
||||
|
||||
// Update the window frame's theme to match the system theme.
|
||||
static void UpdateTheme(HWND const window); |
||||
|
||||
bool quit_on_close_ = false; |
||||
|
||||
// window handle for top level window.
|
||||
HWND window_handle_ = nullptr; |
||||
|
||||
// window handle for hosted content.
|
||||
HWND child_content_ = nullptr; |
||||
}; |
||||
|
||||
#endif // RUNNER_WIN32_WINDOW_H_
|
||||
|