Source code for simba.ui.pop_ups.distance_plot_pop_up

__author__ = "Simon Nilsson; sronilsson@gmail.com"

import os
import threading
from collections import defaultdict
from tkinter import *
from typing import Union

import numpy as np

from simba.mixins.config_reader import ConfigReader
from simba.mixins.pop_up_mixin import PopUpMixin
from simba.plotting.distance_plotter import DistancePlotterSingleCore
from simba.plotting.distance_plotter_mp import DistancePlotterMultiCore
from simba.ui.tkinter_functions import (CreateLabelFrameWithIcon, DropDownMenu,
                                        Entry_Box, SimbaButton, SimbaCheckbox,
                                        SimBADropDown)
from simba.utils.checks import check_if_filepath_list_is_empty, check_int
from simba.utils.enums import Formats, Keys, Links
from simba.utils.errors import DuplicationError
from simba.utils.read_write import get_file_name_info_in_directory

AUTO = 'AUTO'

[docs]class DistancePlotterPopUp(PopUpMixin, ConfigReader): """ :example: >>> _ = DistancePlotterPopUp(config_path=r'C:\troubleshooting\RAT_NOR\project_folder\project_config.ini') """ def __init__(self, config_path: Union[str, os.PathLike]): ConfigReader.__init__(self, config_path=config_path, read_video_info=False) self.max_y_lst = list(range(10, 1010, 10)) self.max_y_lst.insert(0, AUTO) font_size_options = list(range(1, 33)) opacity_options = list(np.round(np.arange(0.0, 1.1, 0.1), 1)) self.number_of_distances_options = list(range(1, len(self.body_parts_lst) * 2)) self.files_found_dict = get_file_name_info_in_directory(directory=self.outlier_corrected_dir, file_type=self.file_type) check_if_filepath_list_is_empty(filepaths=list(self.files_found_dict.keys()), error_msg="SIMBA ERROR: Zero files found in the project_folder/csv/outlier_corrected_movement_location directory. ",) PopUpMixin.__init__(self, title="CREATE DISTANCE PLOTS", icon='distance') self.style_settings_frm = CreateLabelFrameWithIcon( parent=self.main_frm, header="STYLE SETTINGS", icon_name='style', icon_link=Links.DISTANCE_PLOTS.value,) self.resolution_dropdown = SimBADropDown(parent=self.style_settings_frm, dropdown_options=self.resolutions, label='RESOLUTION: ', label_width=25, dropdown_width=20, value=self.resolutions[1], img='monitor') self.font_size_dropdown = SimBADropDown(parent=self.style_settings_frm, dropdown_options=font_size_options, label='FONT SIZE: ', label_width=25, dropdown_width=20, value=8, img='font_size') self.line_width_dropdown = SimBADropDown(parent=self.style_settings_frm, dropdown_options=font_size_options, label='LINE WIDTH: ', label_width=25, dropdown_width=20, value=6, img='line') self.opacity_dropdown = SimBADropDown(parent=self.style_settings_frm, dropdown_options=opacity_options, label='LINE OPACITY: ', label_width=25, dropdown_width=20, value=0.5, img='opacity') self.max_y_dropdown = SimBADropDown(parent=self.style_settings_frm, dropdown_options=self.max_y_lst, label='MAX Y-AXIS: ', label_width=25, dropdown_width=20, value=AUTO, img='y') self.distances_frm = CreateLabelFrameWithIcon(parent=self.main_frm, header="CHOOSE DISTANCES", icon_name='distance', icon_link=Links.DISTANCE_PLOTS.value) self.number_of_distances_dropdown = SimBADropDown(parent=self.distances_frm, dropdown_options=self.number_of_distances_options, label='# DISTANCES: ', label_width=25, dropdown_width=20, value=1, command=self.__populate_distances_menu, img='abacus') self.__populate_distances_menu(1) self.settings_frm = CreateLabelFrameWithIcon(parent=self.main_frm, header="VISUALIZATION SETTING", icon_name='eye', icon_link=Links.DISTANCE_PLOTS.value) distance_frames_cb, self.distance_frames_var = SimbaCheckbox(parent=self.settings_frm, txt='CREATE FRAMES', txt_img='frames', val=False) distance_videos_cb, self.distance_videos_var = SimbaCheckbox(parent=self.settings_frm, txt='CREATE VIDEOS', txt_img='video', val=False) distance_final_img_cb, self.distance_final_img_var = SimbaCheckbox(parent=self.settings_frm, txt='CREATE LAST FRAME', txt_img='finish', val=True) last_frame_as_svg_cb, self.last_frame_as_svg_var = SimbaCheckbox(parent=self.settings_frm, txt='LAST FRAME AS SVG', txt_img='svg', val=False, tooltip_key='LAST_FRAME_AS_SVG') self.multiprocess_dropdown = SimBADropDown(parent=self.settings_frm, dropdown_options=list(range(1, self.cpu_cnt)), label='CPU CORE COUNT: ', label_width=25, dropdown_width=20, value=int(self.cpu_cnt/2), img='cpu_small') self.run_frm = CreateLabelFrameWithIcon(parent=self.main_frm, header="RUN", icon_name='run', icon_link=Links.DISTANCE_PLOTS.value) self.run_single_video_frm = LabelFrame(self.run_frm, text="SINGLE VIDEO", font=Formats.FONT_HEADER.value, pady=5, padx=5, fg="black") self.run_single_video_btn = SimbaButton(parent=self.run_single_video_frm, txt="Create single video", txt_clr="blue", font=Formats.FONT_REGULAR.value, cmd=self.__create_distance_plots, cmd_kwargs={'multiple_videos': False}) self.single_video_dropdown = DropDownMenu( self.run_single_video_frm, "Video:", list(self.files_found_dict.keys()), "12") self.single_video_dropdown.setChoices(list(self.files_found_dict.keys())[0]) self.run_multiple_videos = LabelFrame(self.run_frm, text="MULTIPLE VIDEO", font=Formats.FONT_HEADER.value, pady=5, padx=5, fg="black",) self.run_multiple_video_btn = SimbaButton(parent=self.run_multiple_videos, txt="Create multiple videos ({} video(s) found)".format(str(len(list(self.files_found_dict.keys())))), txt_clr="blue", font=Formats.FONT_REGULAR.value, cmd=self.__create_distance_plots, cmd_kwargs={'multiple_videos': True}) self.style_settings_frm.grid(row=0, sticky=NW) self.resolution_dropdown.grid(row=0, sticky=NW) self.font_size_dropdown.grid(row=1, sticky=NW) self.line_width_dropdown.grid(row=2, sticky=NW) self.opacity_dropdown.grid(row=3, sticky=NW) self.max_y_dropdown.grid(row=4, sticky=NW) self.distances_frm.grid(row=1, sticky=NW) self.number_of_distances_dropdown.grid(row=0, sticky=NW) self.settings_frm.grid(row=2, sticky=NW) self.multiprocess_dropdown.grid(row=0, column=0, sticky=NW) distance_frames_cb.grid(row=1, sticky=NW) distance_videos_cb.grid(row=2, sticky=NW) distance_final_img_cb.grid(row=3, sticky=NW) last_frame_as_svg_cb.grid(row=4, sticky=NW) self.run_frm.grid(row=3, sticky=NW) self.run_single_video_frm.grid(row=0, sticky=NW) self.run_single_video_btn.grid(row=0, column=0, sticky=NW) self.single_video_dropdown.grid(row=0, column=1, sticky=NW) self.run_multiple_videos.grid(row=1, sticky=NW) self.run_multiple_video_btn.grid(row=0, sticky=NW) self.main_frm.mainloop() def __populate_distances_menu(self, choice): if hasattr(self, "bp_1"): for k, v in self.bp_1.items(): self.bp_1[k].destroy() self.bp_2[k].destroy() self.distance_clrs[k].destroy() self.bp_1, self.bp_2, self.distance_clrs = {}, {}, {} for distance_cnt in range(int(self.number_of_distances_dropdown.getChoices())): self.bp_1[distance_cnt] = SimBADropDown(parent=self.distances_frm, dropdown_options=self.body_parts_lst, label=f'DISTANCE {distance_cnt + 1}: ', label_width=25, dropdown_width=20, value=self.body_parts_lst[distance_cnt], img='circle_black') self.bp_1[distance_cnt].grid(row=distance_cnt + 1, column=0, sticky=NW) self.bp_2[distance_cnt] = SimBADropDown(parent=self.distances_frm, dropdown_options=self.body_parts_lst, label='', label_width=0, dropdown_width=20, value=self.body_parts_lst[distance_cnt]) self.bp_2[distance_cnt].grid(row=distance_cnt + 1, column=1, sticky=NW) self.distance_clrs[distance_cnt] = SimBADropDown(parent=self.distances_frm, dropdown_options=list(self.colors_dict.keys()), label='', label_width=0, dropdown_width=20, value=list(self.colors_dict.keys())[distance_cnt]) self.distance_clrs[distance_cnt].grid(row=distance_cnt + 1, column=3, sticky=NW) def __create_distance_plots(self, multiple_videos: bool): if multiple_videos: data_paths = list(self.files_found_dict.values()) else: data_paths = [self.files_found_dict[self.single_video_dropdown.getChoices()]] line_attr = defaultdict(list) for attr in (self.bp_1, self.bp_2, self.distance_clrs): for key, value in attr.items(): line_attr[key].append(value.getChoices()) line_attr = list(line_attr.values()) for cnt, v in enumerate(line_attr): if v[0] == v[1]: raise DuplicationError(msg=f"DISTANCE LINE {cnt+1} ERROR: The two body-parts cannot be the same body-part.",source=self.__class__.__name__,) width = int(self.resolution_dropdown.getChoices().split("×")[0]) height = int(self.resolution_dropdown.getChoices().split("×")[1]) font_size = int(self.font_size_dropdown.get_value()) line_width = int(self.line_width_dropdown.get_value()) opacity = float(self.opacity_dropdown.getChoices()) core_cnt = int(self.multiprocess_dropdown.get_value()) as_svg = self.last_frame_as_svg_var.get() max_y = -1 if self.max_y_dropdown.getChoices() == AUTO else int(self.max_y_dropdown.getChoices()) style_attr = {"width": width, "height": height, "line width": line_width, "font size": font_size, "opacity": opacity, "y_max": max_y} if core_cnt == 1: distance_plotter = DistancePlotterSingleCore(config_path=self.config_path, frame_setting=self.distance_frames_var.get(), video_setting=self.distance_videos_var.get(), final_img=self.distance_final_img_var.get(), style_attr=style_attr, last_frame_as_svg=as_svg, data_paths=data_paths, line_attr=line_attr) else: distance_plotter = DistancePlotterMultiCore(config_path=self.config_path, frame_setting=self.distance_frames_var.get(), video_setting=self.distance_videos_var.get(), final_img=self.distance_final_img_var.get(), style_attr=style_attr, last_frame_as_svg=as_svg, data_paths=data_paths, line_attr=line_attr, core_cnt=core_cnt) threading.Thread(distance_plotter.run()).start()
#_ = DistancePlotterPopUp(config_path=r"D:\troubleshooting\maplight_ri\project_folder\project_config.ini") # _ = DistancePlotterPopUp(config_path='/Users/simon/Desktop/envs/troubleshooting/two_black_animals_14bp/project_folder/project_config.ini') # _ = DistancePlotterPopUp(config_path='/Users/simon/Desktop/envs/troubleshooting/Termites_5/project_folder/project_config.ini')