From 880530839a9aebaf7715d5149f19823d0b87a3e7 Mon Sep 17 00:00:00 2001 From: Alexander Vieth Date: Wed, 18 Sep 2024 18:01:36 +0200 Subject: [PATCH] WIP: Add example project to manage plugin dependencies --- CMakeLists.txt | 7 + ExampleDependencies/CMakeLists.txt | 183 ++++++++++++++++++ ExampleDependencies/cmake/get_cpm.cmake | 24 +++ ExampleDependencies/cmake/highway.patch | 91 +++++++++ ExampleDependencies/src/.editorconfig | 10 + .../src/ExampleDependenciesPlugin.cpp | 169 ++++++++++++++++ .../src/ExampleDependenciesPlugin.h | 63 ++++++ .../src/ExampleDependenciesPlugin.json | 5 + ExampleDependencies/src/SettingsAction.cpp | 12 ++ ExampleDependencies/src/SettingsAction.h | 19 ++ vcpkg.json | 13 ++ 11 files changed, 596 insertions(+) create mode 100644 ExampleDependencies/CMakeLists.txt create mode 100644 ExampleDependencies/cmake/get_cpm.cmake create mode 100644 ExampleDependencies/cmake/highway.patch create mode 100644 ExampleDependencies/src/.editorconfig create mode 100644 ExampleDependencies/src/ExampleDependenciesPlugin.cpp create mode 100644 ExampleDependencies/src/ExampleDependenciesPlugin.h create mode 100644 ExampleDependencies/src/ExampleDependenciesPlugin.json create mode 100644 ExampleDependencies/src/SettingsAction.cpp create mode 100644 ExampleDependencies/src/SettingsAction.h create mode 100644 vcpkg.json diff --git a/CMakeLists.txt b/CMakeLists.txt index a54b9f1..b12d82b 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -1,5 +1,11 @@ cmake_minimum_required(VERSION 3.17) +set(VCPKG_LIBRARY_LINKAGE "dynamic" CACHE STRING "Link vcpkg libraries dynamically") + +if(NOT DEFINED CMAKE_TOOLCHAIN_FILE) + message(FATAL_ERROR "Define CMAKE_TOOLCHAIN_FILE poiting to vcpkg: PATH/vcpkg/scripts/buildsystems/vcpkg.cmake") +endif() + # ----------------------------------------------------------------------------- # ExamplePlugins # ----------------------------------------------------------------------------- @@ -14,3 +20,4 @@ add_subdirectory(ExampleTransformation) add_subdirectory(ExampleLoader) add_subdirectory(ExampleWriter) add_subdirectory(ExampleData) +add_subdirectory(ExampleDependencies) diff --git a/ExampleDependencies/CMakeLists.txt b/ExampleDependencies/CMakeLists.txt new file mode 100644 index 0000000..526fa37 --- /dev/null +++ b/ExampleDependencies/CMakeLists.txt @@ -0,0 +1,183 @@ +cmake_minimum_required(VERSION 3.21) + +option(MV_UNITY_BUILD "Combine target source files into batches for faster compilation" OFF) + +# ----------------------------------------------------------------------------- +# ExampleAnalysis Plugin +# ----------------------------------------------------------------------------- +PROJECT("ExampleDependenciesPlugin") + +# ----------------------------------------------------------------------------- +# CMake Options +# ----------------------------------------------------------------------------- +set(CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) +set(CMAKE_INCLUDE_CURRENT_DIR ON) +set(CMAKE_AUTOMOC ON) + +if(MSVC) + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /DWIN32 /EHsc /MP /permissive- /Zc:__cplusplus") + set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} /NODEFAULTLIB:LIBCMT") + set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MDd") + set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MD") +endif(MSVC) + +include(cmake/get_cpm.cmake) + +# ----------------------------------------------------------------------------- +# Dependencies +# ----------------------------------------------------------------------------- +find_package(Qt6 COMPONENTS Widgets WebEngineWidgets REQUIRED) + +find_package(ManiVault COMPONENTS Core PointData CONFIG) + +find_package(blake3 CONFIG REQUIRED) +find_package(faiss CONFIG REQUIRED) + +CPMAddPackage( + NAME highway + URL https://github.com/google/highway/archive/refs/tags/1.2.0.tar.gz + URL_HASH SHA256=7e0be78b8318e8bdbf6fa545d2ecb4c90f947df03f7aadc42c1967f019e63343 + PATCHES "cmake/highway.patch" # see https://github.com/conan-io/conan-center-index/pull/24197/files, fixes https://github.com/google/highway/issues/2225 + OPTIONS "HWY_ENABLE_EXAMPLES OFF" "HWY_ENABLE_INSTALL OFF" "HWY_ENABLE_TESTS OFF" "HWY_ENABLE_CONTRIB ON" "BUILD_SHARED_LIBS ON" +) + +# ----------------------------------------------------------------------------- +# Source files +# ----------------------------------------------------------------------------- +# Define the plugin sources +set(PLUGIN_SOURCES + src/ExampleDependenciesPlugin.h + src/ExampleDependenciesPlugin.cpp + src/SettingsAction.h + src/SettingsAction.cpp + src/ExampleDependenciesPlugin.json +) + +source_group( Plugin FILES ${PLUGIN_SOURCES}) + +# ----------------------------------------------------------------------------- +# CMake Target +# ----------------------------------------------------------------------------- +# Create dynamic library for the plugin +add_library(${PROJECT_NAME} SHARED ${PLUGIN_SOURCES}) + +# ----------------------------------------------------------------------------- +# Target include directories +# ----------------------------------------------------------------------------- +# Include ManiVault headers, including system data plugins +target_include_directories(${PROJECT_NAME} PRIVATE "${ManiVault_INCLUDE_DIR}") + +# ----------------------------------------------------------------------------- +# Target properties +# ----------------------------------------------------------------------------- +# Request C++17 +target_compile_features(${PROJECT_NAME} PRIVATE cxx_std_17) + +# Enable unity build +if(MV_UNITY_BUILD) + set_target_properties(${PROJECT_NAME} PROPERTIES UNITY_BUILD ON) +endif() + +# ----------------------------------------------------------------------------- +# Target library linking +# ----------------------------------------------------------------------------- +# Link to Qt libraries +target_link_libraries(${PROJECT_NAME} PRIVATE Qt6::Widgets) +target_link_libraries(${PROJECT_NAME} PRIVATE Qt6::WebEngineWidgets) + +# Link to ManiVault and data plugins +target_link_libraries(${PROJECT_NAME} PRIVATE ManiVault::Core) +target_link_libraries(${PROJECT_NAME} PRIVATE ManiVault::PointData) + +target_link_libraries(${PROJECT_NAME} PRIVATE BLAKE3::blake3) +target_link_libraries(${PROJECT_NAME} PRIVATE hwy hwy_contrib) +target_link_libraries(${PROJECT_NAME} PRIVATE faiss) + +# ----------------------------------------------------------------------------- +# Target installation +# ----------------------------------------------------------------------------- +# Install the shared plugin libary to the "Plugins" folder in the ManiVault install directory +install(TARGETS ${PROJECT_NAME} + RUNTIME DESTINATION Plugins COMPONENT PLUGINS # Windows .dll + LIBRARY DESTINATION Plugins COMPONENT PLUGINS # Linux/Mac .so +) + +if(WIN32) + set(PRE_EXCLUDE ".*[\\/]api-ms.*" ".*[\\/]ext-ms.*" ".*[\\/]hvsi.*" ".*[\\/]pdmutilities.*" ".*[\\/]wpaxholder.*" ".*[\\/]dxgi.*" ".*[\\/]uxtheme.*" ".*[\\/]d3d11.*" ".*[\\/]winmm.*" ".*[\\/]wldp.*") + set(POST_EXCLUDE ".*[\\/]WINDOWS[\\/]system32[\\/].*\\.dll") +elseif(APPLE) + set(PRE_EXCLUDE "/usr/lib" "/System/") + set(POST_EXCLUDE "") +else() + set(PRE_EXCLUDE ".*/ld-linux-.*" ".*/linux-vdso.*" ".*/libm\\..*" ".*/libc\\..*" ".*/libpthread\\..*" ".*/libdl\\..*") + set(POST_EXCLUDE "") +endif() + +set(deps_folder "bin/${PROJECT_NAME}") + +message(STATUS "Install deps:: ${ManiVault_INSTALL_DIR}/$/${deps_folder}") + +install(CODE [[ + function(install_deps LIBRARY) + file(INSTALL + DESTINATION "${CMAKE_INSTALL_PREFIX}/PluginDependencies/ExampleDependenciesPlugin" + TYPE SHARED_LIBRARY + FOLLOW_SYMLINK_CHAIN + FILES "${LIBRARY}" + ) + endfunction() + file(GET_RUNTIME_DEPENDENCIES + EXECUTABLES "$" + RESOLVED_DEPENDENCIES_VAR RESOLVED_DEPS + UNRESOLVED_DEPENDENCIES_VAR UNRESOLVED_DEPS + CONFLICTING_DEPENDENCIES_PREFIX CONFLICTING_DEPS + DIRECTORIES "$" "$" "$" + PRE_EXCLUDE_REGEXES ".*[\\/]api-ms.*" ".*[\\/]ext-ms.*" ".*[\\/]hvsi.*" ".*[\\/]pdmutilities.*" ".*[\\/]wpaxholder.*" ".*[\\/]dxgi.*" ".*[\\/]uxtheme.*" ".*[\\/]d3d11.*" ".*[\\/]winmm.*" ".*[\\/]wldp.*" + POST_EXCLUDE_REGEXES ".*[\\/]WINDOWS[\\/]system32[\\/].*\\.dll" + ) + + foreach(FILE ${RESOLVED_DEPS}) + message(STATUS "Resolved: ${FILE}") + install_deps(${FILE}) + endforeach() + + foreach(FILE ${UNRESOLVED_DEPS}) + message(STATUS "Unresolved: ${FILE}") + endforeach() + + if(CONFLICTING_DEPS) + message(WARNING "Conflicting dependencies!") + foreach(FILE ${CONFLICTING_DEPS}) + message(STATUS "Conflicting: ${FILE}") + endforeach() + endif() + + ]] + COMPONENT DEP_PLUGIN_PRINT +) + +add_custom_command(TARGET ${PROJECT_NAME} POST_BUILD + COMMAND ${CMAKE_COMMAND} -E echo "Installing: Create dir" + COMMAND ${CMAKE_COMMAND} -E make_directory ${ManiVault_INSTALL_DIR}/$/${deps_folder} + COMMAND "${CMAKE_COMMAND}" + --install ${CMAKE_CURRENT_BINARY_DIR} + --config $ + --component PLUGINS + --prefix ${ManiVault_INSTALL_DIR}/$ + --verbose + COMMAND "${CMAKE_COMMAND}" + --install ${CMAKE_CURRENT_BINARY_DIR} + --config $ + --component DEP_PLUGIN_PRINT + --prefix ${ManiVault_INSTALL_DIR}/$ + --verbose +) + +# ----------------------------------------------------------------------------- +# Miscellaneous +# ----------------------------------------------------------------------------- +# Automatically set the debug environment (command + working directory) for MSVC +if(MSVC) + set_property(TARGET ${PROJECT_NAME} PROPERTY VS_DEBUGGER_WORKING_DIRECTORY $,${ManiVault_INSTALL_DIR}/debug,${ManiVault_INSTALL_DIR}/release>) + set_property(TARGET ${PROJECT_NAME} PROPERTY VS_DEBUGGER_COMMAND $,${ManiVault_INSTALL_DIR}/debug/ManiVault\ Studio.exe,${ManiVault_INSTALL_DIR}/release/ManiVault\ Studio.exe>) +endif() diff --git a/ExampleDependencies/cmake/get_cpm.cmake b/ExampleDependencies/cmake/get_cpm.cmake new file mode 100644 index 0000000..baf2d8c --- /dev/null +++ b/ExampleDependencies/cmake/get_cpm.cmake @@ -0,0 +1,24 @@ +# SPDX-License-Identifier: MIT +# +# SPDX-FileCopyrightText: Copyright (c) 2019-2023 Lars Melchior and contributors + +set(CPM_DOWNLOAD_VERSION 0.40.2) +set(CPM_HASH_SUM "c8cdc32c03816538ce22781ed72964dc864b2a34a310d3b7104812a5ca2d835d") + +if(CPM_SOURCE_CACHE) + set(CPM_DOWNLOAD_LOCATION "${CPM_SOURCE_CACHE}/cpm/CPM_${CPM_DOWNLOAD_VERSION}.cmake") +elseif(DEFINED ENV{CPM_SOURCE_CACHE}) + set(CPM_DOWNLOAD_LOCATION "$ENV{CPM_SOURCE_CACHE}/cpm/CPM_${CPM_DOWNLOAD_VERSION}.cmake") +else() + set(CPM_DOWNLOAD_LOCATION "${CMAKE_BINARY_DIR}/cmake/CPM_${CPM_DOWNLOAD_VERSION}.cmake") +endif() + +# Expand relative path. This is important if the provided path contains a tilde (~) +get_filename_component(CPM_DOWNLOAD_LOCATION ${CPM_DOWNLOAD_LOCATION} ABSOLUTE) + +file(DOWNLOAD + https://github.com/cpm-cmake/CPM.cmake/releases/download/v${CPM_DOWNLOAD_VERSION}/CPM.cmake + ${CPM_DOWNLOAD_LOCATION} EXPECTED_HASH SHA256=${CPM_HASH_SUM} +) + +include(${CPM_DOWNLOAD_LOCATION}) diff --git a/ExampleDependencies/cmake/highway.patch b/ExampleDependencies/cmake/highway.patch new file mode 100644 index 0000000..647a64f --- /dev/null +++ b/ExampleDependencies/cmake/highway.patch @@ -0,0 +1,91 @@ +diff --git a/hwy/contrib/thread_pool/topology.cc b/hwy/contrib/thread_pool/topology.cc +index 3d24f4f..17f3563 100644 +--- a/hwy/contrib/thread_pool/topology.cc ++++ b/hwy/contrib/thread_pool/topology.cc +@@ -64,7 +64,7 @@ + + namespace hwy { + +-HWY_DLLEXPORT bool HaveThreadingSupport() { ++HWY_CONTRIB_DLLEXPORT bool HaveThreadingSupport() { + #if HWY_ARCH_WASM + return emscripten_has_threading_support() != 0; + #else +@@ -72,7 +72,7 @@ HWY_DLLEXPORT bool HaveThreadingSupport() { + #endif + } + +-HWY_DLLEXPORT size_t TotalLogicalProcessors() { ++HWY_CONTRIB_DLLEXPORT size_t TotalLogicalProcessors() { + size_t lp = 0; + #if HWY_ARCH_WASM + const int num_cores = emscripten_num_logical_cores(); +@@ -111,7 +111,7 @@ HWY_DLLEXPORT size_t TotalLogicalProcessors() { + #include + #endif + +-HWY_DLLEXPORT bool GetThreadAffinity(LogicalProcessorSet& lps) { ++HWY_CONTRIB_DLLEXPORT bool GetThreadAffinity(LogicalProcessorSet& lps) { + #if HWY_OS_WIN + // Only support the first 64 because WINE does not support processor groups. + const HANDLE hThread = GetCurrentThread(); +@@ -173,7 +173,7 @@ HWY_DLLEXPORT bool GetThreadAffinity(LogicalProcessorSet& lps) { + #endif + } + +-HWY_DLLEXPORT bool SetThreadAffinity(const LogicalProcessorSet& lps) { ++HWY_CONTRIB_DLLEXPORT bool SetThreadAffinity(const LogicalProcessorSet& lps) { + #if HWY_OS_WIN + const HANDLE hThread = GetCurrentThread(); + const DWORD_PTR prev = SetThreadAffinityMask(hThread, lps.Get64()); +@@ -385,7 +385,7 @@ std::vector DetectPackages(std::vector& lps) { + } // namespace + #endif // HWY_OS_LINUX + +-HWY_DLLEXPORT Topology::Topology() { ++HWY_CONTRIB_DLLEXPORT Topology::Topology() { + #if HWY_OS_LINUX + lps.resize(TotalLogicalProcessors()); + const std::vector& per_package = DetectPackages(lps); +diff --git a/hwy/contrib/thread_pool/topology.h b/hwy/contrib/thread_pool/topology.h +index 95b0835..f80fc47 100644 +--- a/hwy/contrib/thread_pool/topology.h ++++ b/hwy/contrib/thread_pool/topology.h +@@ -28,7 +28,7 @@ + namespace hwy { + + // Returns false if std::thread should not be used. +-HWY_DLLEXPORT bool HaveThreadingSupport(); ++HWY_CONTRIB_DLLEXPORT bool HaveThreadingSupport(); + + // Upper bound on logical processors, including hyperthreads. + static constexpr size_t kMaxLogicalProcessors = 1024; // matches glibc +@@ -38,12 +38,12 @@ using LogicalProcessorSet = BitSet4096; + + // Returns false, or sets `lps` to all logical processors which are online and + // available to the current thread. +-HWY_DLLEXPORT bool GetThreadAffinity(LogicalProcessorSet& lps); ++HWY_CONTRIB_DLLEXPORT bool GetThreadAffinity(LogicalProcessorSet& lps); + + // Ensures the current thread can only run on the logical processors in `lps`. + // Returns false if not supported (in particular on Apple), or if the + // intersection between `lps` and `GetThreadAffinity` is the empty set. +-HWY_DLLEXPORT bool SetThreadAffinity(const LogicalProcessorSet& lps); ++HWY_CONTRIB_DLLEXPORT bool SetThreadAffinity(const LogicalProcessorSet& lps); + + // Returns false, or ensures the current thread will only run on `lp`, which + // must not exceed `TotalLogicalProcessors`. Note that this merely calls +@@ -58,11 +58,11 @@ static inline bool PinThreadToLogicalProcessor(size_t lp) { + // provided by the hardware clamped to `kMaxLogicalProcessors`. + // These processors are not necessarily all usable; you can determine which are + // via GetThreadAffinity(). +-HWY_DLLEXPORT size_t TotalLogicalProcessors(); ++HWY_CONTRIB_DLLEXPORT size_t TotalLogicalProcessors(); + + struct Topology { + // Caller must check packages.empty(); if so, do not use any fields. +- HWY_DLLEXPORT Topology(); ++ HWY_CONTRIB_DLLEXPORT Topology(); + + // Clique of cores with lower latency to each other. On Apple M1 these are + // four cores sharing an L2. On Zen4 these 'CCX' are up to eight cores sharing \ No newline at end of file diff --git a/ExampleDependencies/src/.editorconfig b/ExampleDependencies/src/.editorconfig new file mode 100644 index 0000000..7eddc72 --- /dev/null +++ b/ExampleDependencies/src/.editorconfig @@ -0,0 +1,10 @@ +# To learn more about .editorconfig see https://aka.ms/editorconfigdocs +root = true + +# All files +[*] +indent_style = space + +# Cpp files +[*.{h,cpp}] +indent_size = 4 diff --git a/ExampleDependencies/src/ExampleDependenciesPlugin.cpp b/ExampleDependencies/src/ExampleDependenciesPlugin.cpp new file mode 100644 index 0000000..83cf471 --- /dev/null +++ b/ExampleDependencies/src/ExampleDependenciesPlugin.cpp @@ -0,0 +1,169 @@ +#include "ExampleDependenciesPlugin.h" + +#include + +#include +#include +#include +#include +#include +#include + +#include +#include + +#include + +#include + +// 64-bit int +using idx_t = faiss::idx_t; + +Q_PLUGIN_METADATA(IID "studio.manivault.ExampleDependenciesPlugin") + +using namespace mv; +using namespace mv::plugin; + +ExampleDependenciesPlugin::ExampleDependenciesPlugin(const PluginFactory* factory) : + AnalysisPlugin(factory), + _settingsAction() +{ +} + +void ExampleDependenciesPlugin::init() +{ + setOutputDataset(mv::data().createDerivedDataset("Output Data", getInputDataset(), getInputDataset())); + + const auto inputPoints = getInputDataset(); + + auto outputPoints = getOutputDataset(); + + outputPoints->addAction(_settingsAction); + + std::vector initialData; + const size_t numDims = inputPoints->getNumDimensions(); + const size_t numPoints = inputPoints->getNumPoints(); + initialData.resize(numDims * numPoints); + outputPoints->setData(initialData.data(), numPoints, numDims); + events().notifyDatasetDataChanged(outputPoints); + + // Start the analysis when the user clicks the start analysis push button + connect(&_settingsAction.getStartAnalysisAction(), &mv::gui::TriggerAction::triggered, this, &ExampleDependenciesPlugin::compute); +} + +void ExampleDependenciesPlugin::compute() +{ + std::cout << "ExampleDependenciesPlugin: Startin..." << std::endl; + + std::cout << "ExampleDependenciesPlugin: Using blake version: " << blake3_version() << std::endl; + + auto printData = [](const std::vector& vec, size_t dims, size_t points) { + for (size_t dim = 0; dim < dims; dim++) + { + for (size_t point = 0; point < points; point++) + std::cout << vec[point + points * dim] << " "; + std::cout << std::endl; + } + }; + + // Get input data + const auto inputPoints = getInputDataset(); + const size_t numDims = inputPoints->getNumDimensions(); + const size_t numPoints = inputPoints->getNumPoints(); + + std::vector data; + std::vector dimensionIndices(numDims); + std::iota(dimensionIndices.begin(), dimensionIndices.end(), 0); + data.resize(numDims * numPoints); + inputPoints->populateDataForDimensions(data, dimensionIndices); + + // Sort + std::cout << "ExampleDependenciesPlugin: Data before sorting" << std::endl; + printData(data, numDims, numPoints); + + for (size_t dim = 0; dim < numDims; dim++) + { + size_t offset = numPoints * dim; + hwy::VQSort(data.data() + offset, numPoints, hwy::SortAscending{}); + } + + std::cout << "ExampleDependenciesPlugin: Data after sorting" << std::endl; + printData(data, numDims, numPoints); + + // Save to output + auto outputPoints = getOutputDataset(); + outputPoints->setData(data.data(), numPoints, numDims); + events().notifyDatasetDataChanged(outputPoints); + + // Create hash + blake3_hasher hasher; + blake3_hasher_init(&hasher); + + blake3_hasher_update(&hasher, data.data(), data.size()); + + uint8_t output[BLAKE3_OUT_LEN]; + blake3_hasher_finalize(&hasher, output, BLAKE3_OUT_LEN); + + // Print the hash as hexadecimal + for (size_t i = 0; i < BLAKE3_OUT_LEN; i++) { + printf("%02x", output[i]); + } + printf("\n"); + + // find knn + faiss::IndexFlatL2 index(numDims); // call constructor + index.add(numPoints, data.data()); // add vectors to the index + + int k = std::max(static_cast(5), numPoints); + std::vector I(k * numPoints); + std::vector D(k * numPoints); + + index.search(numPoints, data.data(), k, D.data(), I.data()); + + printf("knn results\n"); + for (size_t i = 0; i < numPoints; i++) { + for (int j = 0; j < k; j++) + printf("%5zd ", I[i * k + j]); + printf("\n"); + } + + std::cout << "ExampleDependenciesPlugin: Finished." << std::endl; +} + +AnalysisPlugin* ExampleDependenciesPluginFactory::produce() +{ + // Return a new instance of the example analysis plugin + return new ExampleDependenciesPlugin(this); +} + +mv::DataTypes ExampleDependenciesPluginFactory::supportedDataTypes() const +{ + DataTypes supportedTypes; + + // This example analysis plugin is compatible with points datasets + supportedTypes.append(PointType); + + return supportedTypes; +} + +mv::gui::PluginTriggerActions ExampleDependenciesPluginFactory::getPluginTriggerActions(const mv::Datasets& datasets) const +{ + PluginTriggerActions pluginTriggerActions; + + const auto getPluginInstance = [this](const Dataset& dataset) -> ExampleDependenciesPlugin* { + return dynamic_cast(plugins().requestPlugin(getKind(), { dataset })); + }; + + const auto numberOfDatasets = datasets.count(); + + if (numberOfDatasets >= 1 && PluginFactory::areAllDatasetsOfTheSameType(datasets, PointType)) { + auto pluginTriggerAction = new PluginTriggerAction(const_cast(this), this, "Example dependencies", "Perform an Example Analysis", getIcon(), [this, getPluginInstance, datasets](PluginTriggerAction& pluginTriggerAction) -> void { + for (auto& dataset : datasets) + getPluginInstance(dataset); + }); + + pluginTriggerActions << pluginTriggerAction; + } + + return pluginTriggerActions; +} diff --git a/ExampleDependencies/src/ExampleDependenciesPlugin.h b/ExampleDependencies/src/ExampleDependenciesPlugin.h new file mode 100644 index 0000000..acb72e0 --- /dev/null +++ b/ExampleDependencies/src/ExampleDependenciesPlugin.h @@ -0,0 +1,63 @@ +#pragma once + +#include + +#include "SettingsAction.h" + +using namespace mv::plugin; +using namespace mv; + +/** + * Example dependencies plugin class + * + * + * @authors + */ +class ExampleDependenciesPlugin : public AnalysisPlugin +{ + Q_OBJECT + +public: + + ExampleDependenciesPlugin(const PluginFactory* factory); + ~ExampleDependenciesPlugin() override = default; + + void init() override; + +private: + void compute(); + +private: + SettingsAction _settingsAction; /** The place where settings are stored (more info in SettingsAction.h) */ + +}; + +/** + * Example analysis plugin factory class + * + * Note: Factory does not need to be altered (merely responsible for generating new plugins when requested) + */ +class ExampleDependenciesPluginFactory : public AnalysisPluginFactory +{ + Q_INTERFACES(mv::plugin::AnalysisPluginFactory mv::plugin::PluginFactory) + Q_OBJECT + Q_PLUGIN_METADATA(IID "studio.manivault.ExampleDependenciesPlugin" + FILE "ExampleDependenciesPlugin.json") + +public: + + /** Default constructor */ + ExampleDependenciesPluginFactory() {} + + /** Destructor */ + ~ExampleDependenciesPluginFactory() override {} + + /** Creates an instance of the example analysis plugin */ + AnalysisPlugin* produce() override; + + /** Returns the data types that are supported by the example analysis plugin */ + mv::DataTypes supportedDataTypes() const override; + + /** Enable right-click on data set to open analysis */ + PluginTriggerActions getPluginTriggerActions(const mv::Datasets& datasets) const override; +}; diff --git a/ExampleDependencies/src/ExampleDependenciesPlugin.json b/ExampleDependencies/src/ExampleDependenciesPlugin.json new file mode 100644 index 0000000..23a6f94 --- /dev/null +++ b/ExampleDependencies/src/ExampleDependenciesPlugin.json @@ -0,0 +1,5 @@ +{ + "name" : "Example Dependencies", + "version" : "1.0", + "dependencies" : ["Points"] +} diff --git a/ExampleDependencies/src/SettingsAction.cpp b/ExampleDependencies/src/SettingsAction.cpp new file mode 100644 index 0000000..dc4ffc3 --- /dev/null +++ b/ExampleDependencies/src/SettingsAction.cpp @@ -0,0 +1,12 @@ +#include "SettingsAction.h" + +SettingsAction::SettingsAction(QObject* parent) : + GroupAction(parent, "SettingsAction", true), + _startAnalysisAction(this, "Start analysis") +{ + setText("Example Settings"); + + _startAnalysisAction.setToolTip("Start the analysis"); + + addAction(&_startAnalysisAction); +} diff --git a/ExampleDependencies/src/SettingsAction.h b/ExampleDependencies/src/SettingsAction.h new file mode 100644 index 0000000..84d3743 --- /dev/null +++ b/ExampleDependencies/src/SettingsAction.h @@ -0,0 +1,19 @@ +#pragma once + +#include "actions/GroupAction.h" +#include "actions/TriggerAction.h" + +using namespace mv::gui; + +class SettingsAction : public GroupAction +{ +public: + + SettingsAction(QObject* parent = nullptr); + +public: + TriggerAction& getStartAnalysisAction() { return _startAnalysisAction; } + +public: + TriggerAction _startAnalysisAction; +}; diff --git a/vcpkg.json b/vcpkg.json new file mode 100644 index 0000000..bdb36a6 --- /dev/null +++ b/vcpkg.json @@ -0,0 +1,13 @@ +{ + "name": "example-plugins", + "version": "0.0.1", + "dependencies": [ + "blake3", + "faiss" + ], + "builtin-baseline":"73964f854d0d6424f335e9515b61b3a2dad5e15b", + "overrides": [ + { "name": "blake3", "version": "1.5.1" }, + { "name": "blake3", "version": "1.7.4#1" } + ] +} \ No newline at end of file