1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
|
/* Copyright (c) 2008-2025 the MRtrix3 contributors.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* Covered Software is provided under this License on an "as is"
* basis, without warranty of any kind, either expressed, implied, or
* statutory, including, without limitation, warranties that the
* Covered Software is free of defects, merchantable, fit for a
* particular purpose or non-infringing.
* See the Mozilla Public License v. 2.0 for more details.
*
* For more details, see http://www.mrtrix.org/.
*/
#include "command.h"
#include "progressbar.h"
#include "algo/loop.h"
#include "image.h"
#include "fixel/helpers.h"
#include "fixel/keys.h"
#include "fixel/types.h"
using namespace MR;
using namespace App;
using Fixel::index_type;
#define DEFAULT_ANGLE_THRESHOLD 45.0
void usage ()
{
AUTHOR = "David Raffelt (david.raffelt@florey.edu.au)";
SYNOPSIS = "Obtain fixel-fixel correpondence between a subject fixel image and a template fixel mask";
DESCRIPTION
+ "It is assumed that the subject image has already been spatially normalised and is aligned with the template. "
"The output fixel image will have the same fixels (and directions) of the template.";
ARGUMENTS
+ Argument ("subject_data", "the input subject fixel data file. This should be a file inside the fixel directory").type_image_in ()
+ Argument ("template_directory", "the input template fixel directory.").type_directory_in()
+ Argument ("output_directory", "the fixel directory where the output file will be written.").type_text()
+ Argument ("output_data", "the name of the output fixel data file. This will be placed in the output fixel directory").type_text();
OPTIONS
+ Option ("angle", "the max angle threshold for computing inter-subject fixel correspondence (Default: " + str(DEFAULT_ANGLE_THRESHOLD, 2) + " degrees)")
+ Argument ("value").type_float (0.0, 90.0);
}
void run ()
{
const float angular_threshold = get_option_value ("angle", DEFAULT_ANGLE_THRESHOLD);
const float angular_threshold_dp = cos (angular_threshold * (Math::pi/180.0));
const std::string input_file (argument[0]);
if (Path::is_dir (input_file))
throw Exception ("please input the specific fixel data file to be converted (not the fixel directory)");
auto subject_index = Fixel::find_index_header (Fixel::get_fixel_directory (input_file)).get_image<index_type>();
auto subject_directions = Fixel::find_directions_header (Fixel::get_fixel_directory (input_file)).get_image<float>().with_direct_io();
if (input_file == subject_directions.name())
throw Exception ("input fixel data file cannot be the directions file");
auto subject_data = Image<float>::open (input_file);
Fixel::check_fixel_size (subject_index, subject_data);
auto template_index = Fixel::find_index_header (argument[1]).get_image<index_type>();
auto template_directions = Fixel::find_directions_header (argument[1]).get_image<float>().with_direct_io();
check_dimensions (subject_index, template_index);
std::string output_fixel_directory = argument[2];
Fixel::copy_index_and_directions_file (argument[1], output_fixel_directory);
Header output_data_header (template_directions);
output_data_header.size(1) = 1;
auto output_data = Image<float>::create (Path::join (output_fixel_directory, argument[3]), output_data_header);
for (auto i = Loop ("mapping subject fixel data to template fixels", template_index, 0, 3)(template_index, subject_index); i; ++i) {
template_index.index(3) = 0;
index_type nfixels_template = template_index.value();
template_index.index(3) = 1;
index_type template_offset = template_index.value();
for (index_type t = 0; t < nfixels_template; ++t) {
float largest_dp = 0.0;
int index_of_closest_fixel = -1;
subject_index.index(3) = 0;
index_type nfixels_subject = subject_index.value();
subject_index.index(3) = 1;
index_type subject_offset = subject_index.value();
template_directions.index(0) = template_offset + t;
for (index_type s = 0; s < nfixels_subject; ++s) {
subject_directions.index(0) = subject_offset + s;
Eigen::Vector3f templatedir = template_directions.row(1);
templatedir.normalize();
Eigen::Vector3f subjectdir = subject_directions.row(1);
subjectdir.normalize();
float dp = abs (templatedir.dot (subjectdir));
if (dp > largest_dp) {
largest_dp = dp;
index_of_closest_fixel = s;
}
}
if (largest_dp > angular_threshold_dp) {
output_data.index(0) = template_offset + t;
subject_data.index(0) = subject_offset + index_of_closest_fixel;
output_data.value() = subject_data.value();
}
}
}
}
|