diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 113549d7bbdd4e8529b787a6f375ff65eca3f8bd..dae6f391d6be34b13edc0a18e512ddabc47f4548 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -38,3 +38,11 @@ python3.9:
   stage: test
   script:
   - python -m unittest
+
+python3.10:
+  image: python:3.10
+  tags:
+  - passoft
+  stage: test
+  script:
+  - python -m unittest
diff --git a/conda.recipe/meta.yaml b/conda.recipe/meta.yaml
index 3fcee5f663548ab7e0a5977f6da00552348a47a5..4278af9234b716c9cf08a63e7d54a6b5d838efd5 100644
--- a/conda.recipe/meta.yaml
+++ b/conda.recipe/meta.yaml
@@ -1,6 +1,6 @@
 package:
   name: mseedpeek
-  version: 2023.3.0.0
+  version: 2023.3.1.0
 
 source:
   path: ../
diff --git a/docs/Makefile b/docs/Makefile
deleted file mode 100644
index fa5a7fd0033469d07fd1571f4e3b42169fa67600..0000000000000000000000000000000000000000
--- a/docs/Makefile
+++ /dev/null
@@ -1,20 +0,0 @@
-# Minimal makefile for Sphinx documentation
-#
-
-# You can set these variables from the command line.
-SPHINXOPTS    =
-SPHINXBUILD   = python -msphinx
-SPHINXPROJ    = mseedpeek
-SOURCEDIR     = .
-BUILDDIR      = _build
-
-# Put it first so that "make" without argument is like "make help".
-help:
-	@$(SPHINXBUILD) -M help "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
-
-.PHONY: help Makefile
-
-# Catch-all target: route all unknown targets to Sphinx using the new
-# "make mode" option.  $(O) is meant as a shortcut for $(SPHINXOPTS).
-%: Makefile
-	@$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
diff --git a/docs/authors.rst b/docs/authors.rst
deleted file mode 100644
index e122f914a87b277e565fc9567af1a7545ec9872b..0000000000000000000000000000000000000000
--- a/docs/authors.rst
+++ /dev/null
@@ -1 +0,0 @@
-.. include:: ../AUTHORS.rst
diff --git a/docs/conf.py b/docs/conf.py
deleted file mode 100755
index 7571621bb85a6280146838fa2d33858fc7034144..0000000000000000000000000000000000000000
--- a/docs/conf.py
+++ /dev/null
@@ -1,163 +0,0 @@
-#!/usr/bin/env python
-# -*- coding: utf-8 -*-
-#
-# mseedpeek documentation build configuration file, created by
-# sphinx-quickstart on Fri Jun  9 13:47:02 2017.
-#
-# This file is execfile()d with the current directory set to its
-# containing dir.
-#
-# Note that not all possible configuration values are present in this
-# autogenerated file.
-#
-# All configuration values have a default; values that are commented out
-# serve to show the default.
-
-# If extensions (or modules to document with autodoc) are in another
-# directory, add these directories to sys.path here. If the directory is
-# relative to the documentation root, use os.path.abspath to make it
-# absolute, like shown here.
-#
-import os
-import sys
-sys.path.insert(0, os.path.abspath('..'))
-
-import mseedpeek
-
-# -- General configuration ---------------------------------------------
-
-# If your documentation needs a minimal Sphinx version, state it here.
-#
-# needs_sphinx = '1.0'
-
-# Add any Sphinx extension module names here, as strings. They can be
-# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom ones.
-extensions = ['sphinx.ext.autodoc', 'sphinx.ext.viewcode']
-
-# Add any paths that contain templates here, relative to this directory.
-templates_path = ['_templates']
-
-# The suffix(es) of source filenames.
-# You can specify multiple suffix as a list of string:
-#
-# source_suffix = ['.rst', '.md']
-source_suffix = '.rst'
-
-# The master toctree document.
-master_doc = 'index'
-
-# General information about the project.
-project = 'mseedpeek'
-copyright = "2018, IRIS PASSCAL"
-author = "IRIS PASSCAL"
-
-# The version info for the project you're documenting, acts as replacement
-# for |version| and |release|, also used in various other places throughout
-# the built documents.
-#
-# The short X.Y version.
-version = mseedpeek.__version__
-# The full version, including alpha/beta/rc tags.
-release = mseedpeek.__version__
-
-# The language for content autogenerated by Sphinx. Refer to documentation
-# for a list of supported languages.
-#
-# This is also used if you do content translation via gettext catalogs.
-# Usually you set "language" from the command line for these cases.
-language = None
-
-# List of patterns, relative to source directory, that match files and
-# directories to ignore when looking for source files.
-# This patterns also effect to html_static_path and html_extra_path
-exclude_patterns = ['_build', 'Thumbs.db', '.DS_Store']
-
-# The name of the Pygments (syntax highlighting) style to use.
-pygments_style = 'sphinx'
-
-# If true, `todo` and `todoList` produce output, else they produce nothing.
-todo_include_todos = False
-
-
-# -- Options for HTML output -------------------------------------------
-
-# The theme to use for HTML and HTML Help pages.  See the documentation for
-# a list of builtin themes.
-#
-html_theme = 'alabaster'
-
-# Theme options are theme-specific and customize the look and feel of a
-# theme further.  For a list of options available for each theme, see the
-# documentation.
-#
-# html_theme_options = {}
-
-# Add any paths that contain custom static files (such as style sheets) here,
-# relative to this directory. They are copied after the builtin static files,
-# so a file named "default.css" will overwrite the builtin "default.css".
-html_static_path = ['_static']
-
-
-# -- Options for HTMLHelp output ---------------------------------------
-
-# Output file base name for HTML help builder.
-htmlhelp_basename = 'mseedpeekdoc'
-
-
-# -- Options for LaTeX output ------------------------------------------
-
-latex_elements = {
-    # The paper size ('letterpaper' or 'a4paper').
-    #
-    # 'papersize': 'letterpaper',
-
-    # The font size ('10pt', '11pt' or '12pt').
-    #
-    # 'pointsize': '10pt',
-
-    # Additional stuff for the LaTeX preamble.
-    #
-    # 'preamble': '',
-
-    # Latex figure (float) alignment
-    #
-    # 'figure_align': 'htbp',
-}
-
-# Grouping the document tree into LaTeX files. List of tuples
-# (source start file, target name, title, author, documentclass
-# [howto, manual, or own class]).
-latex_documents = [
-    (master_doc, 'mseedpeek.tex',
-     'mseedpeek Documentation',
-     'IRIS PASSCAL', 'manual'),
-]
-
-
-# -- Options for manual page output ------------------------------------
-
-# One entry per manual page. List of tuples
-# (source start file, name, description, authors, manual section).
-man_pages = [
-    (master_doc, 'mseedpeek',
-     'mseedpeek Documentation',
-     [author], 1)
-]
-
-
-# -- Options for Texinfo output ----------------------------------------
-
-# Grouping the document tree into Texinfo files. List of tuples
-# (source start file, target name, title, author,
-#  dir menu entry, description, category)
-texinfo_documents = [
-    (master_doc, 'mseedpeek',
-     'mseedpeek Documentation',
-     author,
-     'mseedpeek',
-     'One line description of project.',
-     'Miscellaneous'),
-]
-
-
-
diff --git a/docs/contributing.rst b/docs/contributing.rst
deleted file mode 100644
index e582053ea018c369be05aae96cf730744f1dc616..0000000000000000000000000000000000000000
--- a/docs/contributing.rst
+++ /dev/null
@@ -1 +0,0 @@
-.. include:: ../CONTRIBUTING.rst
diff --git a/docs/history.rst b/docs/history.rst
deleted file mode 100644
index 250649964bbc36f4bec2942f69238aa6f7c02c1a..0000000000000000000000000000000000000000
--- a/docs/history.rst
+++ /dev/null
@@ -1 +0,0 @@
-.. include:: ../HISTORY.rst
diff --git a/docs/index.rst b/docs/index.rst
deleted file mode 100644
index 602ab19aee92b7a96c3eeb95ef310e9b32ce7eaa..0000000000000000000000000000000000000000
--- a/docs/index.rst
+++ /dev/null
@@ -1,20 +0,0 @@
-Welcome to mseedpeek's documentation!
-======================================
-
-.. toctree::
-   :maxdepth: 2
-   :caption: Contents:
-
-   readme
-   installation
-   usage
-   modules
-   contributing
-   authors
-   history
-
-Indices and tables
-==================
-* :ref:`genindex`
-* :ref:`modindex`
-* :ref:`search`
diff --git a/docs/installation.rst b/docs/installation.rst
deleted file mode 100644
index 8a3665d9cf174768b8912cb057ae7b7d1543eec0..0000000000000000000000000000000000000000
--- a/docs/installation.rst
+++ /dev/null
@@ -1,32 +0,0 @@
-.. highlight:: shell
-
-============
-Installation
-============
-
-Fom sources
-------------
-
-The sources for mseedpeek can be downloaded from the `Github repo`_.
-
-You can either clone the public repository:
-
-.. code-block:: console
-
-    $ git clone https://git.passcal.nmt.edu/passoft/mseedpeek
-
-Or download the `tarball`_:
-
-.. code-block:: console
-
-    $ curl  -OL https://git.passcal.nmt.edu/passoft/mseedpeek/tarball/master
-
-Once you have a copy of the source, you can install it with:
-
-.. code-block:: console
-
-    $ python setup.py install
-
-
-.. _Github repo: https://git.passcal.nmt.edu/passoft/mseedpeek
-.. _tarball: https://git.passcal.nmt.edu/passoft/mseedpeek/tarball/master
diff --git a/docs/make.bat b/docs/make.bat
deleted file mode 100644
index d5cc6b3803c487069dd8ef99a260fcc60333daea..0000000000000000000000000000000000000000
--- a/docs/make.bat
+++ /dev/null
@@ -1,36 +0,0 @@
-@ECHO OFF
-
-pushd %~dp0
-
-REM Command file for Sphinx documentation
-
-if "%SPHINXBUILD%" == "" (
-	set SPHINXBUILD=python -msphinx
-)
-set SOURCEDIR=.
-set BUILDDIR=_build
-set SPHINXPROJ=mseedpeek
-
-if "%1" == "" goto help
-
-%SPHINXBUILD% >NUL 2>NUL
-if errorlevel 9009 (
-	echo.
-	echo.The Sphinx module was not found. Make sure you have Sphinx installed,
-	echo.then set the SPHINXBUILD environment variable to point to the full
-	echo.path of the 'sphinx-build' executable. Alternatively you may add the
-	echo.Sphinx directory to PATH.
-	echo.
-	echo.If you don't have Sphinx installed, grab it from
-	echo.http://sphinx-doc.org/
-	exit /b 1
-)
-
-%SPHINXBUILD% -M %1 %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
-goto end
-
-:help
-%SPHINXBUILD% -M help %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
-
-:end
-popd
diff --git a/docs/readme.rst b/docs/readme.rst
deleted file mode 100644
index 72a33558153fb57def85612b021ec596ef2a51b9..0000000000000000000000000000000000000000
--- a/docs/readme.rst
+++ /dev/null
@@ -1 +0,0 @@
-.. include:: ../README.rst
diff --git a/docs/usage.rst b/docs/usage.rst
deleted file mode 100644
index ec9ecaa6ad92b0fc559c8a76e5b6c5af2de0f16f..0000000000000000000000000000000000000000
--- a/docs/usage.rst
+++ /dev/null
@@ -1,7 +0,0 @@
-=====
-Usage
-=====
-
-To use mseedpeek in a project::
-
-    import mseedpeek
diff --git a/mseedpeek/__init__.py b/mseedpeek/__init__.py
index 67272133110e2b720ee9ba2fc3d0fb940e177e20..1c06f1d4ee0e2d9b6bb7c5aa068e9644b2f2e487 100644
--- a/mseedpeek/__init__.py
+++ b/mseedpeek/__init__.py
@@ -4,4 +4,4 @@
 
 __author__ = """IRIS PASSCAL"""
 __email__ = 'software-support@passcal.nmt.edu'
-__version__ = '2023.3.0.0'
+__version__ = '2023.3.1.0'
diff --git a/mseedpeek/mseedpeek.py b/mseedpeek/mseedpeek.py
index ab77ac912ff39530d5b081045bbc24369bca38e2..613ac0cde2eb53bc9f8a23302f07baa80e03a0ce 100644
--- a/mseedpeek/mseedpeek.py
+++ b/mseedpeek/mseedpeek.py
@@ -129,6 +129,7 @@
 # GUI updated to PySide2
 # Pmw dependency dropped
 # Version number updated
+# Run build trace to thread
 """
 
 import sys
@@ -138,17 +139,26 @@ import itertools
 from getopt import getopt
 from glob import glob
 from operator import mod
-from PySide2 import QtCore, QtGui, QtWidgets
+from PySide2.QtWidgets import (QApplication, QWidget, QTabWidget,
+                               QLineEdit, QVBoxLayout, QHBoxLayout,
+                               QTextEdit, QGroupBox, QLabel,
+                               QComboBox, QPushButton, QSizePolicy,
+                               QFormLayout, QRadioButton, QSlider,
+                               QSpacerItem, QStackedWidget, QFileDialog,
+                               QMessageBox, QProgressDialog, QTreeWidget,
+                               QTreeWidgetItem)
+from PySide2.QtCore import (Qt, QObject, Signal, QThread, QTimer, QEventLoop)
+from PySide2.QtGui import QColor
 
 from mseedpeek.libtrace import *
 from mseedpeek.mseedInfo import *
 
-VERSION = "2023.3.0.0"
+VERSION = "2023.3.1.0"
 
 
 def main():
     """
-    Main class for mseedpeek
+    Get commandline args and launch mseedpeek
     """
     # return version number if -# command line option
     file_list = []
@@ -169,7 +179,7 @@ def main():
         sys.exit(1)
     print("\n", os.path.basename(sys.argv[0]), VERSION)
 
-    app = QtWidgets.QApplication(sys.argv)
+    app = QApplication(sys.argv)
     # mc = main class
     if file_list:
         mc = MainWindow("mseedpeek %s" % VERSION, file_list)
@@ -179,7 +189,7 @@ def main():
     sys.exit(app.exec_())
 
 
-class MainWindow(QtWidgets.QWidget):
+class MainWindow(QWidget):
     """
     Main class for mseedpeek
     """
@@ -259,22 +269,22 @@ class MainWindow(QtWidgets.QWidget):
         Build tabs and info bar for root window
         """
         # tab widget
-        self.tabwidget = QtWidgets.QTabWidget()
-        self.trace_headers_tab = QtWidgets.QWidget()
-        self.blockettes_tab = QtWidgets.QWidget()
-        self.help_tab = QtWidgets.QWidget()
+        self.tabwidget = QTabWidget()
+        self.trace_headers_tab = QWidget()
+        self.blockettes_tab = QWidget()
+        self.help_tab = QWidget()
         self.tabwidget.addTab(self.trace_headers_tab, "Trace Headers")
         self.tabwidget.addTab(self.blockettes_tab, "Blockettes")
         self.tabwidget.addTab(self.help_tab, "Help")
 
         # info bar widget
-        self.infobar = QtWidgets.QLineEdit()
+        self.infobar = QLineEdit()
         self.infobar.setStyleSheet("background-color:yellow")
         self.infobar.setReadOnly(True)
-        self.infobar.setAlignment(QtGui.Qt.AlignCenter)
+        self.infobar.setAlignment(Qt.AlignCenter)
 
         # main window layout
-        self.window_layout = QtWidgets.QVBoxLayout()
+        self.window_layout = QVBoxLayout()
         self.setLayout(self.window_layout)
 
         # add tab widget and info bar to main window layout
@@ -288,18 +298,18 @@ class MainWindow(QtWidgets.QWidget):
         Build Help tab
         """
         # init tab layout
-        layout = QtWidgets.QVBoxLayout(self.help_tab)
+        layout = QVBoxLayout(self.help_tab)
 
         # Help textbox
-        Blue = QtGui.QColor(0, 0, 153)
-        Green = QtGui.QColor(0, 100, 0)
-        Red = QtGui.QColor(136, 8, 8)
-        Black = QtGui.QColor(0, 0, 0)
+        Blue = QColor(0, 0, 153)
+        Green = QColor(0, 100, 0)
+        Red = QColor(136, 8, 8)
+        Black = QColor(0, 0, 0)
 
-        help_text = QtWidgets.QTextEdit()
+        help_text = QTextEdit()
         layout.addWidget(help_text)
         help_text.setVerticalScrollBarPolicy(
-            QtCore.Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
+            Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
         help_text.setAcceptRichText(False)
 
         # Name
@@ -480,35 +490,39 @@ class MainWindow(QtWidgets.QWidget):
         Create initial widgets for Blockettes tab
         """
         # data boxes
-        self.blktype_box = QtWidgets.QGroupBox()
+        self.blktype_box = QGroupBox()
         self.blktype_box.setObjectName("Box1")
 
-        self.blkinfo_box = QtWidgets.QGroupBox()
-        self.blkinfo_box.setLayout(QtWidgets.QHBoxLayout())
+        self.blkinfo_box = QGroupBox()
+        self.blkinfo_box.setLayout(QHBoxLayout())
         self.blkinfo_box.setObjectName("Box2")
 
-        self.blk_vars_box = QtWidgets.QGroupBox()
-        self.blk_vars_box.setLayout(QtWidgets.QVBoxLayout())
+        self.blk_vars_box = QGroupBox()
+        self.blk_vars_box.setLayout(QVBoxLayout())
         self.blk_vars_box.setObjectName("Box3")
 
         # widgets
-        blk_label = QtWidgets.QLabel("Blockette:")
-        blk_menu = QtWidgets.QComboBox()
+        blk_label = QLabel("Blockette:")
+        blk_menu = QComboBox()
+        blk_menu.currentIndexChanged.connect(
+            lambda: self.fill_blockettes_tab(
+                0,
+                blk_menu.currentText()))
         blk_menu.setObjectName("type_menu")
 
-        self.blkinfo_btn = QtWidgets.QPushButton("Blockette Info")
+        self.blkinfo_btn = QPushButton("Blockette Info")
         self.blkinfo_btn.setStyleSheet("QPushButton{background-color:lightblue;}\
                                   QPushButton::hover{background-color:green;}")
         self.blkinfo_btn.clicked.connect(self.blkinfo_window)
 
         # box layouts
-        l1 = QtWidgets.QHBoxLayout(self.blktype_box)
+        l1 = QHBoxLayout(self.blktype_box)
         l1.addWidget(blk_label)
         l1.addWidget(blk_menu)
         l1.addStretch()
 
         # add to tab's main layout
-        main_layout = QtWidgets.QVBoxLayout()
+        main_layout = QVBoxLayout()
         main_layout.addWidget(self.blktype_box)
         main_layout.addWidget(self.blkinfo_box)
         main_layout.addWidget(self.blk_vars_box)
@@ -520,56 +534,56 @@ class MainWindow(QtWidgets.QWidget):
         Build widgets for Trace Headers tab
         """
         # tab layout
-        self.trace_headers_layout = QtWidgets.QVBoxLayout(
+        self.trace_headers_layout = QVBoxLayout(
             self.trace_headers_tab)
 
         # groupboxes for data fields
-        self.datadir_box = QtWidgets.QGroupBox()
+        self.datadir_box = QGroupBox()
         self.datadir_box.setCheckable(False)
         sp = self.datadir_box.sizePolicy()
-        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        sp.setVerticalPolicy(QSizePolicy.Fixed)
         self.datadir_box.setSizePolicy(sp)
 
-        self.stations_box = QtWidgets.QGroupBox()
+        self.stations_box = QGroupBox()
         self.stations_box.setCheckable(False)
         sp = self.stations_box.sizePolicy()
-        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        sp.setVerticalPolicy(QSizePolicy.Fixed)
         self.stations_box.setSizePolicy(sp)
 
-        self.radio_box = QtWidgets.QGroupBox()
+        self.radio_box = QGroupBox()
         self.radio_box.setCheckable(False)
         sp = self.radio_box.sizePolicy()
-        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        sp.setVerticalPolicy(QSizePolicy.Fixed)
         self.radio_box.setSizePolicy(sp)
 
-        self.dir_trace_box = QtWidgets.QGroupBox()
+        self.dir_trace_box = QGroupBox()
         self.dir_trace_box.setCheckable(False)
         sp = self.dir_trace_box.sizePolicy()
-        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        sp.setVerticalPolicy(QSizePolicy.Fixed)
         self.dir_trace_box.setSizePolicy(sp)
 
-        self.flush_exit_box = QtWidgets.QGroupBox()
+        self.flush_exit_box = QGroupBox()
         self.flush_exit_box.setCheckable(False)
         sp = self.flush_exit_box.sizePolicy()
-        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        sp.setVerticalPolicy(QSizePolicy.Fixed)
         self.flush_exit_box.setSizePolicy(sp)
         self.window_layout.addWidget(self.flush_exit_box)
 
         # layouts for groupboxes
-        datadir_layout = QtWidgets.QHBoxLayout(self.datadir_box)
-        stations_layout = QtWidgets.QHBoxLayout(self.stations_box)
-        rbuttons_layout = QtWidgets.QHBoxLayout(self.radio_box)
-        flush_exit_layout = QtWidgets.QHBoxLayout(self.flush_exit_box)
+        datadir_layout = QHBoxLayout(self.datadir_box)
+        stations_layout = QHBoxLayout(self.stations_box)
+        rbuttons_layout = QHBoxLayout(self.radio_box)
+        flush_exit_layout = QHBoxLayout(self.flush_exit_box)
         flush_exit_layout.setMargin(0)
         flush_exit_layout.setSpacing(450)
-        dir_trace_layout = QtWidgets.QFormLayout(self.dir_trace_box)
+        dir_trace_layout = QFormLayout(self.dir_trace_box)
 
         # fill flush / exit groubpx
-        self.exit_btn = QtWidgets.QPushButton("Exit")
+        self.exit_btn = QPushButton("Exit")
         self.exit_btn.setStyleSheet(
             "QPushButton::hover{background-color:darkred;}")
         self.exit_btn.clicked.connect(lambda: quit())
-        self.flush_btn = QtWidgets.QPushButton("Flush Directories")
+        self.flush_btn = QPushButton("Flush Directories")
         self.flush_btn.clicked.connect(self.flush_dict)
         self.flush_btn.setStyleSheet(
             "QPushButton::hover{background-color:orange;}")
@@ -577,18 +591,18 @@ class MainWindow(QtWidgets.QWidget):
         flush_exit_layout.addWidget(self.exit_btn)
 
         # fill data dir groupbox
-        self.dd_label = QtWidgets.QLabel("Data Directories:")
-        self.dd_text = QtWidgets.QLineEdit()
-        self.build_trace_btn = QtWidgets.QPushButton("Build Trace db")
+        self.dd_label = QLabel("Data Directories:")
+        self.dd_text = QLineEdit()
+        self.build_trace_btn = QPushButton("Build Trace db")
         self.build_trace_btn.setStyleSheet(
             """
             QPushButton{background-color:lightblue;}
             QPushButton::hover {background-color:green;}
             """)
         self.build_trace_btn.clicked.connect(self.clicked_build_trace)
-        self.find_btn = QtWidgets.QPushButton("Find")
+        self.find_btn = QPushButton("Find")
         self.find_btn.clicked.connect(self.clicked_find)
-        self.clear_btn = QtWidgets.QPushButton("Clear")
+        self.clear_btn = QPushButton("Clear")
         self.clear_btn.clicked.connect(lambda: self.dd_text.clear())
         datadir_layout.addWidget(self.dd_label)
         datadir_layout.addWidget(self.dd_text)
@@ -597,10 +611,10 @@ class MainWindow(QtWidgets.QWidget):
         datadir_layout.addWidget(self.clear_btn)
 
         # stations widgets
-        self.stations_label = QtWidgets.QLabel(
+        self.stations_label = QLabel(
             "Find only stations (colon separated list):")
-        self.stations_text = QtWidgets.QLineEdit()
-        self.stations_clear_btn = QtWidgets.QPushButton("Clear")
+        self.stations_text = QLineEdit()
+        self.stations_clear_btn = QPushButton("Clear")
         self.stations_clear_btn.clicked.connect(
             lambda: self.stations_text.clear())
         stations_layout.addWidget(self.stations_label)
@@ -608,27 +622,27 @@ class MainWindow(QtWidgets.QWidget):
         stations_layout.addWidget(self.stations_clear_btn)
 
         # fill stations groupbox
-        self.standard_rbtn = QtWidgets.QRadioButton("Standard")
+        self.standard_rbtn = QRadioButton("Standard")
         self.standard_rbtn.setChecked(True)
         self.standard_rbtn.clicked.connect(lambda: self.clicked_scan_type(0))
-        self.verbose_rbtn = QtWidgets.QRadioButton("Verbose")
+        self.verbose_rbtn = QRadioButton("Verbose")
         self.verbose_rbtn.setChecked(False)
         self.verbose_rbtn.clicked.connect(lambda: self.clicked_scan_type(1))
-        self.vv_rbtn = QtWidgets.QRadioButton("Very Verbose")
+        self.vv_rbtn = QRadioButton("Very Verbose")
         self.vv_rbtn.setChecked(False)
         self.vv_rbtn.clicked.connect(lambda: self.clicked_scan_type(2))
-        self.unique_rbtn = QtWidgets.QRadioButton("Unique")
+        self.unique_rbtn = QRadioButton("Unique")
         self.unique_rbtn.setChecked(False)
         self.unique_rbtn.clicked.connect(lambda: self.clicked_scan_type(3))
 
         # spacer item
         # adds distance between rbuttons and "Header Endianess"
-        spacerItem = QtWidgets.QSpacerItem(250, 0,
-                                           QtWidgets.QSizePolicy.Expanding,
-                                           QtWidgets.QSizePolicy.Minimum)
+        spacerItem = QSpacerItem(250, 0,
+                                 QSizePolicy.Expanding,
+                                 QSizePolicy.Minimum)
 
-        self.header_endianess_label = QtWidgets.QLabel("Header Endianess")
-        self.header_endianess_text = QtWidgets.QLineEdit()
+        self.header_endianess_label = QLabel("Header Endianess")
+        self.header_endianess_text = QLineEdit()
         self.header_endianess_text.setStyleSheet("background-color:yellow")
 
         rbuttons_layout.addWidget(self.standard_rbtn)
@@ -640,11 +654,11 @@ class MainWindow(QtWidgets.QWidget):
         rbuttons_layout.addWidget(self.header_endianess_text)
 
         # fill dir / trace groupbox
-        self.dir_label = QtWidgets.QLabel("Directory:")
-        self.dir_menu = QtWidgets.QComboBox()
+        self.dir_label = QLabel("Directory:")
+        self.dir_menu = QComboBox()
         self.dir_menu.activated.connect(lambda: self.select_dir_trace("dir"))
-        self.trace_label = QtWidgets.QLabel("Trace:")
-        self.trace_menu = QtWidgets.QComboBox()
+        self.trace_label = QLabel("Trace:")
+        self.trace_menu = QComboBox()
         self.trace_menu.activated.connect(
             lambda: self.select_dir_trace("trace"))
         dir_trace_layout.addRow(self.dir_label, self.dir_menu)
@@ -660,7 +674,7 @@ class MainWindow(QtWidgets.QWidget):
             self.first_vv_vars, self.standard_vars, self.v_vars, self.vv_vars)
         self.unique_box = UniqueBox()
 
-        self.jump_box = QtWidgets.QStackedWidget()
+        self.jump_box = QStackedWidget()
         self.jump_box.setStyleSheet("QGroupBox{border:0;}")
         self.jump_box.setMinimumSize(0, 200)
         self.jump_box.addWidget(self.standard_box)
@@ -676,6 +690,8 @@ class MainWindow(QtWidgets.QWidget):
             lambda: self.update_slider("menu"))
         self.unique_box.unique_jump.currentIndexChanged.connect(
             lambda: self.update_slider("unique"))
+        self.unique_box.keys_menu.currentIndexChanged.connect(
+            self.select_keys)
 
         # add everything to trace headers layout
         self.trace_headers_layout.addWidget(self.datadir_box)
@@ -697,13 +713,19 @@ class MainWindow(QtWidgets.QWidget):
         self.jump_box.setCurrentIndex(self.verb_var)
         if self.trace_menu.isHidden():
             self.jump_box.hide()
+        else:
+            self.jump_box.show()
+            if self.verb_var == 3:
+                self.slider_box.hide()
+            else:
+                self.slider_box.show()
         self.maybe_read_hdrs()
 
     def clicked_find(self):
         """
         Open file dialogue to search for mseed files
         """
-        search = QtWidgets.QFileDialog.getExistingDirectory()
+        search = QFileDialog.getExistingDirectory()
         directories = self.dd_text.text()
 
         if search:
@@ -719,11 +741,11 @@ class MainWindow(QtWidgets.QWidget):
         directories = self.dd_text.text()
         if directories:
             self.dir_list.clear()
-            self.build_trace_list(directories)
             self.hide_show("hide")
+            self.build_trace_list(directories)
         else:
-            error_msg = QtWidgets.QMessageBox()
-            error_msg.setIcon(QtWidgets.QMessageBox.Critical)
+            error_msg = QMessageBox()
+            error_msg.setIcon(QMessageBox.Critical)
             error_msg.setText("Error")
             error_msg.setInformativeText("No directories listed.")
             error_msg.setWindowTitle("Error")
@@ -737,7 +759,6 @@ class MainWindow(QtWidgets.QWidget):
             if not os.path.isdir(dir):
                 err = "***WARNING*** Directory " + dir + " not found."
                 self.infobar(err, "red")
-                QtWidgets.QApplication.beep()
             else:
                 self.dir_list.append(dir)
 
@@ -748,10 +769,7 @@ class MainWindow(QtWidgets.QWidget):
                 self.stat_sel_list.append(statsel)
 
         if self.dir_list:
-            (num_files, err_files) = self.find_trace()
-            text = ("Done. " + str(num_files) + " mseed files found. ***")
-            text += (str(err_files) + " files with errors.")
-            self.update_infobar(text, "green")
+            self.launch_find_trace()
 
     def hide_show(self, action):
         """
@@ -759,13 +777,13 @@ class MainWindow(QtWidgets.QWidget):
         """
         if action == "hide":
             widget = self.jump_box.currentWidget()
-            for child in widget.findChildren(QtWidgets.QWidget):
+            for child in widget.findChildren(QWidget):
                 if not child.isHidden():
                     child.hide()
-            for child in self.slider_box.findChildren(QtWidgets.QWidget):
+            for child in self.slider_box.findChildren(QWidget):
                 if not child.isHidden():
                     child.hide()
-            widgets = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+            widgets = self.blockettes_tab.findChildren(QGroupBox)
             for widget in widgets:
                 if not widget.isHidden():
                     widget.hide()
@@ -774,15 +792,16 @@ class MainWindow(QtWidgets.QWidget):
             self.blktype_box.hide()
             self.blkinfo_box.hide()
             self.blk_vars_box.hide()
+            self.slider_box.hide()
         else:
             widget = self.jump_box.currentWidget()
-            for child in widget.findChildren(QtWidgets.QWidget):
+            for child in widget.findChildren(QWidget):
                 if child.isHidden():
                     child.show()
-            for child in self.slider_box.findChildren(QtWidgets.QWidget):
+            for child in self.slider_box.findChildren(QWidget):
                 if child.isHidden():
                     child.show()
-            widgets = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+            widgets = self.blockettes_tab.findChildren(QGroupBox)
             for widget in widgets:
                 if widget.isHidden():
                     widget.show()
@@ -790,10 +809,22 @@ class MainWindow(QtWidgets.QWidget):
             self.blkinfo_box.show()
             self.blk_vars_box.show()
 
+    def launch_find_trace(self):
+        """
+        Separated out to run build trace list
+        as a thread
+        """
+
+        self.run_find_trace = 1
+        self.begin_thread("Find Trace",
+                          self.find_trace,
+                          self.after_find_trace)
+
     def find_trace(self):
         """
         based on traverse routine in "python standard library", Lundh pg 34
         """
+
         stack = []
         for k in range(len(self.dir_list)):
             stack.append(self.dir_list[k])
@@ -810,10 +841,13 @@ class MainWindow(QtWidgets.QWidget):
                 print("Directory Read Error: %s" % e)
 
             for file in listfiles:
-                if mod(cnt, 25):
+                if not self.run_find_trace:
+                    break
+                if mod(cnt, 5):
                     pass
                 else:
-                    self.wait("Examining File: ", cnt)
+                    self.func_worker.update.emit(cnt)
+                    # self.wait("Examining File: ", cnt)
                 fullname = os.path.join(directory, file)
                 if os.path.isfile(fullname):
                     if not os.access(fullname, 6):
@@ -863,8 +897,19 @@ class MainWindow(QtWidgets.QWidget):
 
         self.dir_trace_dict = {}
         self.dir_trace_dict = file_list
+        self.num_files = num_mseed_files
+        self.err_files = rw_error
+
+    def after_find_trace(self):
+        """
+        Function to run after find trace
+        to update some info
+        """
+
         self.update_dir_list()
-        return num_mseed_files, rw_error
+        text = ("Done. " + str(self.num_files) + " mseed files found. ***")
+        text += (str(self.err_files) + " files with errors.")
+        self.update_infobar(text, "green")
 
     def select_dir_trace(self, menu):
         """
@@ -896,6 +941,7 @@ class MainWindow(QtWidgets.QWidget):
         self.hide_show("show")
         self.header_endianess_text.setText(self.byte_order)
         self.jump_box.setCurrentIndex(self.verb_var)
+        self.jump_box.show()
         self.update_infobar("", "yellow")
 
         # hack, widget was behaving weird for some reason
@@ -956,7 +1002,7 @@ class MainWindow(QtWidgets.QWidget):
                     lastkey = key
                     if key not in self.unique_select_list:
                         self.unique_select_list.append(key)
-                    key = str(blk) + ":" + key
+                    key = str(n) + ":" + key
                     self.unique_list.append(key)
 
                 # build Blockette dictionary keyed to block number
@@ -1065,17 +1111,22 @@ class MainWindow(QtWidgets.QWidget):
         """
         clears/initializes entry fields in Blockettes tab
         """
+
         for block in self.blockettes_dict[key]:
-            if block[0] == blktype:
+            if int(block[0]) == int(blktype):
                 blocktuple = block
         for key, values in BlkVars.items():
-            if blktype == key:
-                boxes = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+            if int(blktype) == int(key):
+                boxes = self.blockettes_tab.findChildren(QGroupBox)
                 for box in boxes:
                     if box.objectName() == "Box1":
-                        menu = box.findChild(QtWidgets.QComboBox, "type_menu")
+                        menu = box.findChild(QComboBox, "type_menu")
+                        menu.blockSignals(True)
                         menu.clear()
-                        menu.insertItem(0, str(blktype))
+                        for blk in self.blockettes_list:
+                            menu.addItem(str(blk))
+                        menu.setCurrentText(str(blktype))
+                        menu.blockSignals(False)
                     elif box.objectName() == "Box2":
                         layout = box.layout()
                         # delete previous labels if they exist
@@ -1086,7 +1137,7 @@ class MainWindow(QtWidgets.QWidget):
                                 if widget is not None:
                                     widget.setParent(None)
                         # create new labels
-                        label = QtWidgets.QLabel()
+                        label = QLabel()
                         label.setText(str(values[0]))
                         layout.addWidget(label)
                         layout.addWidget(self.blkinfo_btn)
@@ -1102,7 +1153,7 @@ class MainWindow(QtWidgets.QWidget):
                                     widget.setParent(None)
                         # create new labels
                         for x in range(1, len(values)):
-                            label = QtWidgets.QLabel()
+                            label = QLabel()
                             label.setText(str(values[x]) + " " +
                                           str(blocktuple[x - 1]))
                             layout.addWidget(label)
@@ -1111,29 +1162,29 @@ class MainWindow(QtWidgets.QWidget):
         """
         Popup window for blk info button
         """
-        self.info_window = QtWidgets.QWidget()
+        self.info_window = QWidget()
         self.info_window.resize(650, 400)
         self.info_window.setWindowTitle("Blockette Information")
 
         # widgets
-        blk_label = QtWidgets.QLabel("Blockette:")
-        blk_menu = QtWidgets.QComboBox()
+        blk_label = QLabel("Blockette:")
+        blk_menu = QComboBox()
         blk_menu.activated.connect(
             lambda: info_box.setText(BlkInfoDict[int(blk_menu.currentText())])
             if blk_menu.currentText().isdigit()
             else info_box.setText(BlkInfoDict[blk_menu.currentText()]))
-        info_box = QtWidgets.QTextEdit()
-        done_btn = QtWidgets.QPushButton("Done")
+        info_box = QTextEdit()
+        done_btn = QPushButton("Done")
         done_btn.setStyleSheet("QPushButton{background-color:lightblue;}\
                                 QPushButton::hover{background-color:red;}")
         done_btn.clicked.connect(lambda: self.info_window.close())
 
-        top_layout = QtWidgets.QHBoxLayout()
+        top_layout = QHBoxLayout()
         top_layout.addWidget(blk_label)
         top_layout.addWidget(blk_menu)
         top_layout.addStretch()
 
-        main_layout = QtWidgets.QVBoxLayout(self.info_window)
+        main_layout = QVBoxLayout(self.info_window)
         main_layout.addLayout(top_layout)
         main_layout.addWidget(info_box)
         main_layout.addWidget(done_btn)
@@ -1152,7 +1203,7 @@ class MainWindow(QtWidgets.QWidget):
         """
         try:
             standard_widgets = self.standard_box.findChildren(
-                QtWidgets.QLineEdit)
+                QLineEdit)
             rate = self.rate_dict[key]
             vars = []
             # SeqNum, DHQual, res, Stat, Loc, Chan, Net
@@ -1180,7 +1231,7 @@ class MainWindow(QtWidgets.QWidget):
         """
         try:
             verbose_widgets = self.verbose_box.findChildren(
-                QtWidgets.QLineEdit)
+                QLineEdit)
             rate = self.rate_dict[key]
             vars = []
             # SeqNum, DHQual, res, Stat, Loc, Chan, Net
@@ -1210,7 +1261,7 @@ class MainWindow(QtWidgets.QWidget):
         Fills very verbose info on HdrDisplay
         """
         try:
-            vv_widgets = self.vv_box.findChildren(QtWidgets.QLineEdit)
+            vv_widgets = self.vv_box.findChildren(QLineEdit)
             rate = self.rate_dict[key]
             vars = []
             # SeqNum, DHQual, res, Stat, Loc, Chan, Net
@@ -1238,18 +1289,52 @@ class MainWindow(QtWidgets.QWidget):
         except Exception:
             return
 
+    # def update_unique_block(self):
+    #     """
+    #     Update selected item in Unique's tree widget
+    #     """
+
+    #     row_num = int(self.unique_box.unique_jump.currentText())
+    #     items = self.unique_box.unique_info_tree.findItems(
+    #         str(row_num), Qt.MatchExactly, 0)
+    #     self.unique_box.unique_info_tree.clearSelection()
+    #     for i in items:
+    #         self.unique_box.unique_info_tree.setItemSelected(i, True)
+    #         self.unique_box.unique_info_tree.scrollToItem(i)
+
+    def select_keys(self):
+        """
+        """
+
+        selectkey = self.unique_box.keys_menu.currentText().split(":")
+        if selectkey[0] == "*":
+            self.fill_unique()
+            return
+        unique_item = QTreeWidgetItem()
+        line = []
+        self.unique_box.unique_info_tree.clear()
+        for key in self.unique_list:
+            if selectkey == key.split(":")[1:]:
+                for var in key.split(":"):
+                    line.append(var)
+                for i in range(len(line)):
+                    unique_item.setText(i, line[i])
+                    unique_item.setBackgroundColor(i, Qt.white)
+                self.unique_box.unique_info_tree.addTopLevelItem(unique_item)
+                return
+
     def fill_unique(self):
         """
         Fills unique info on HdrDisplay
         """
-        # text colors
-        Blue = QtGui.QColor(0, 0, 153)
-        Black = QtGui.QColor(0, 0, 0)
 
         # fill drop down menus
+        self.unique_box.keys_menu.blockSignals(True)
         self.unique_box.keys_menu.clear()
         self.unique_box.keys_menu.addItem("*")
-        self.unique_box.keys_menu.addItem(f'{self.unique_select_list[0]}')
+        for key in self.unique_select_list:
+            self.unique_box.keys_menu.addItem(key)
+        self.unique_box.keys_menu.blockSignals(False)
 
         self.unique_box.unique_jump.blockSignals(True)
         i = 0
@@ -1258,16 +1343,26 @@ class MainWindow(QtWidgets.QWidget):
             i += 1
         self.unique_box.unique_jump.blockSignals(False)
 
-        # fill text box
-        text = "Block\tStat\tChan\tLoc\tNet\tRate"
-        self.unique_box.unique_infobox.setTextColor(Blue)
-        self.unique_box.unique_infobox.setText(text)
-        self.unique_box.unique_infobox.setTextColor(Black)
-        data = []
-        data.append(self.unique_list[0].split(":"))
-        data = list(itertools.chain.from_iterable(data))
-        data = '\t'.join(str(i) for i in data)
-        self.unique_box.unique_infobox.append(data)
+        # fill tree
+        self.unique_box.unique_info_tree.clear()
+        c = 0
+        for key in self.unique_list:
+            unique_item = QTreeWidgetItem()
+            line = []
+            for var in key.split(":"):
+                line.append(var)
+            if c:
+                for i in range(len(line)):
+                    unique_item.setText(i, line[i])
+                    unique_item.setBackgroundColor(i, Qt.cyan)
+            else:
+                for i in range(len(line)):
+                    unique_item.setText(i, line[i])
+            self.unique_box.unique_info_tree.addTopLevelItem(unique_item)
+            if c:
+                c = 0
+            else:
+                c += 1
 
     def wait(self, words, cnt):
         """
@@ -1305,7 +1400,8 @@ class MainWindow(QtWidgets.QWidget):
         """
         self.infobar.setText(text)
         self.infobar.setStyleSheet("background-color:" + color)
-        QtWidgets.QApplication.beep()
+        if color != "yellow":
+            QApplication.beep()
 
     def update_slider(self, widget):
         """
@@ -1365,10 +1461,90 @@ class MainWindow(QtWidgets.QWidget):
         self.clear_slider()
         self.header_endianess_text.clear()
         self.hide_show("hide")
-        QtWidgets.QApplication.beep()
+        QApplication.beep()
+
+    def begin_thread(self, title, func, after_func=''):
+        """
+        Create thread to run function
+        """
 
+        # init thread and worker
+        self.thread = QThread(self)
+        self.func_worker = Worker(func)
 
-class SliderBox(QtWidgets.QGroupBox):
+        # set signals/slots
+        self.func_worker.update.connect(self.update_progress_window)
+        self.thread.started.connect(self.func_worker.run)
+
+        # function to run after thread finishes
+        if after_func:
+            self.func_worker.finished.connect(after_func)
+
+        # delete worker and thread when done
+        self.func_worker.finished.connect(self.thread.quit)
+        self.func_worker.finished.connect(self.func_worker.deleteLater)
+        self.thread.finished.connect(self.thread.deleteLater)
+
+        # get file num for progress window
+        numfiles = 0
+        text = ""
+
+        if title == "Find Trace":
+            for dir in self.dir_list:
+                numfiles += (
+                    sum([len(files) for r, d, files in os.walk(dir)]))
+            text = title + " is active. Please wait."
+
+        # launch progress window
+        self.build_progress_window(title, text, numfiles)
+        self.func_worker.finished.connect(
+            lambda: self.progress_window.done(0))
+
+        # give progress window a small
+        # amount of time to finish
+        # loading before
+        # starting thread
+        loop = QEventLoop(self)
+        QTimer.singleShot(100, loop.quit)
+        loop.exec_()
+
+        self.thread.start()
+
+    def build_progress_window(self, title, text, max):
+        """
+        Create progress window to update user
+        on status of current process
+        """
+
+        self.progress_window = QProgressDialog(
+            labelText=text, minimum=0,
+            maximum=max, parent=self)
+
+        cancel_b = QPushButton("Cancel")
+        cancel_b.setStyleSheet(
+            "QPushButton::hover{background-color:red;}")
+        cancel_b.clicked.connect(
+            lambda: self.stop_thread(title))
+        self.progress_window.setCancelButton(cancel_b)
+        self.progress_window.open()
+
+    def update_progress_window(self, val):
+        """
+        Update progress window progress bar
+        """
+
+        self.progress_window.setValue(val)
+
+    def stop_thread(self, title):
+        """
+        Stop the currently running thread
+        """
+
+        if title == "Find Trace":
+            self.run_find_trace = 0
+
+
+class SliderBox(QGroupBox):
     """
     Box that holds all the blockette slider widgets
     """
@@ -1379,21 +1555,21 @@ class SliderBox(QtWidgets.QGroupBox):
         super().__init__()
         self.setStyleSheet("QGroupBox{border:0;}")
         # widgets
-        label1 = QtWidgets.QLabel("Jump To Block #:")
-        self.jump_menu = QtWidgets.QComboBox()
+        label1 = QLabel("Jump To Block #:")
+        self.jump_menu = QComboBox()
         self.jump_menu.setMinimumWidth(100)
         self.jump_menu.setEditable(True)
         self.jump_menu.insertItem(0, "0")
-        label2 = QtWidgets.QLabel("Block Number")
-        self.slider = QtWidgets.QSlider(QtCore.Qt.Horizontal)
+        label2 = QLabel("Block Number")
+        self.slider = QSlider(Qt.Horizontal)
         self.slider.setTickPosition(self.slider.NoTicks)
 
         # layouts
-        main_layout = QtWidgets.QVBoxLayout(self)
-        menu_layout = QtWidgets.QVBoxLayout()
-        slider_layout = QtWidgets.QHBoxLayout()
-        menu_layout.addWidget(label1, alignment=QtCore.Qt.AlignCenter)
-        menu_layout.addWidget(self.jump_menu, alignment=QtCore.Qt.AlignCenter)
+        main_layout = QVBoxLayout(self)
+        menu_layout = QVBoxLayout()
+        slider_layout = QHBoxLayout()
+        menu_layout.addWidget(label1, alignment=Qt.AlignCenter)
+        menu_layout.addWidget(self.jump_menu, alignment=Qt.AlignCenter)
         slider_layout.addWidget(label2)
         slider_layout.addWidget(self.slider)
 
@@ -1402,7 +1578,7 @@ class SliderBox(QtWidgets.QGroupBox):
         main_layout.addLayout(slider_layout)
 
 
-class StandardBox(QtWidgets.QGroupBox):
+class StandardBox(QGroupBox):
     """
     Box that holds all the standard info widgets
     """
@@ -1413,15 +1589,15 @@ class StandardBox(QtWidgets.QGroupBox):
         super().__init__()
 
         # layout
-        v_layout = QtWidgets.QVBoxLayout(self)
-        h_layout = QtWidgets.QHBoxLayout()
-        col1 = QtWidgets.QVBoxLayout()
-        col2 = QtWidgets.QVBoxLayout()
+        v_layout = QVBoxLayout(self)
+        h_layout = QHBoxLayout()
+        col1 = QVBoxLayout()
+        col2 = QVBoxLayout()
 
         # widgets
         for x in range(len(standard_vars)):
-            label = QtWidgets.QLabel(str(standard_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(standard_vars[x]))
+            line_edit = QLineEdit()
 
             col1.addWidget(label)
             col2.addWidget(line_edit)
@@ -1434,7 +1610,7 @@ class StandardBox(QtWidgets.QGroupBox):
         v_layout.addStretch()
 
 
-class VerboseBox(QtWidgets.QGroupBox):
+class VerboseBox(QGroupBox):
     """
     Box that holds all the verbose info widgets
     """
@@ -1444,24 +1620,24 @@ class VerboseBox(QtWidgets.QGroupBox):
         """
         super().__init__()
         # layout
-        v_layout = QtWidgets.QVBoxLayout(self)
-        h_layout = QtWidgets.QHBoxLayout()
-        col1 = QtWidgets.QVBoxLayout()
-        col2 = QtWidgets.QVBoxLayout()
-        col3 = QtWidgets.QVBoxLayout()
-        col4 = QtWidgets.QVBoxLayout()
+        v_layout = QVBoxLayout(self)
+        h_layout = QHBoxLayout()
+        col1 = QVBoxLayout()
+        col2 = QVBoxLayout()
+        col3 = QVBoxLayout()
+        col4 = QVBoxLayout()
 
         # widgets
         for x in range(len(standard_vars)):
-            label = QtWidgets.QLabel(str(standard_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(standard_vars[x]))
+            line_edit = QLineEdit()
 
             col1.addWidget(label)
             col2.addWidget(line_edit)
 
         for x in range(len(v_vars)):
-            label = QtWidgets.QLabel(str(v_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(v_vars[x]))
+            line_edit = QLineEdit()
 
             if x == 0:
                 col1.addWidget(label)
@@ -1482,7 +1658,7 @@ class VerboseBox(QtWidgets.QGroupBox):
         v_layout.addStretch()
 
 
-class VVBox(QtWidgets.QGroupBox):
+class VVBox(QGroupBox):
     """
     Box that holds all the very verbose info widgets
     """
@@ -1492,28 +1668,28 @@ class VVBox(QtWidgets.QGroupBox):
         """
         super().__init__()
         # layout
-        v_layout = QtWidgets.QVBoxLayout(self)
-        h_layout = QtWidgets.QHBoxLayout()
-        col1 = QtWidgets.QVBoxLayout()
-        col2 = QtWidgets.QVBoxLayout()
-        col3 = QtWidgets.QVBoxLayout()
-        col4 = QtWidgets.QVBoxLayout()
-        col5 = QtWidgets.QVBoxLayout()
-        col6 = QtWidgets.QVBoxLayout()
-        col7 = QtWidgets.QVBoxLayout()
-        col8 = QtWidgets.QVBoxLayout()
+        v_layout = QVBoxLayout(self)
+        h_layout = QHBoxLayout()
+        col1 = QVBoxLayout()
+        col2 = QVBoxLayout()
+        col3 = QVBoxLayout()
+        col4 = QVBoxLayout()
+        col5 = QVBoxLayout()
+        col6 = QVBoxLayout()
+        col7 = QVBoxLayout()
+        col8 = QVBoxLayout()
 
         # widgets
         for x in range(len(first_vv_vars)):
-            label = QtWidgets.QLabel(str(first_vv_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(first_vv_vars[x]))
+            line_edit = QLineEdit()
 
             col1.addWidget(label)
             col2.addWidget(line_edit)
 
         for x in range(len(standard_vars)):
-            label = QtWidgets.QLabel(str(standard_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(standard_vars[x]))
+            line_edit = QLineEdit()
 
             if x <= 2:
                 col1.addWidget(label)
@@ -1523,8 +1699,8 @@ class VVBox(QtWidgets.QGroupBox):
                 col4.addWidget(line_edit)
 
         for x in range(len(v_vars)):
-            label = QtWidgets.QLabel(str(v_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(v_vars[x]))
+            line_edit = QLineEdit()
 
             if x <= 3:
                 col3.addWidget(label)
@@ -1534,8 +1710,8 @@ class VVBox(QtWidgets.QGroupBox):
                 col6.addWidget(line_edit)
 
         for x in range(len(vv_vars)):
-            label = QtWidgets.QLabel(str(vv_vars[x]))
-            line_edit = QtWidgets.QLineEdit()
+            label = QLabel(str(vv_vars[x]))
+            line_edit = QLineEdit()
 
             if x <= 3:
                 col5.addWidget(label)
@@ -1560,7 +1736,7 @@ class VVBox(QtWidgets.QGroupBox):
         v_layout.addStretch()
 
 
-class UniqueBox(QtWidgets.QGroupBox):
+class UniqueBox(QGroupBox):
     """
     Box that holds all the unique info widgets
     """
@@ -1569,22 +1745,39 @@ class UniqueBox(QtWidgets.QGroupBox):
         Create unique widgets
         """
         super().__init__()
-        layout = QtWidgets.QVBoxLayout(self)
-        select_keys_label = QtWidgets.QLabel("Select Keys:")
-        self.keys_menu = QtWidgets.QComboBox()
-        self.keys_menu.setEditable(True)
-        jump_label = QtWidgets.QLabel("Jump To Block #:")
-        self.unique_jump = QtWidgets.QComboBox()
+        layout = QVBoxLayout(self)
+        select_keys_label = QLabel("Select Keys:")
+        self.keys_menu = QComboBox()
+        # self.keys_menu.setEditable(True)
+        jump_label = QLabel("Jump To Block #:")
+        self.unique_jump = QComboBox()
         self.unique_jump.setEditable(True)
-        self.unique_infobox = QtWidgets.QTextEdit()
+        self.unique_info_tree = QTreeWidget()
+        self.unique_info_tree.setRootIsDecorated(False)
+        self.unique_info_tree.setColumnCount(6)
+        self.unique_info_tree.setHeaderLabels(
+            ['Block', 'Stat', 'Chan', 'Loc', 'Net', 'Rate'])
 
         layout.addWidget(select_keys_label)
         layout.addWidget(self.keys_menu)
         layout.addWidget(jump_label)
         layout.addWidget(self.unique_jump)
-        layout.addWidget(self.unique_infobox)
+        layout.addWidget(self.unique_info_tree)
         layout.addStretch()
 
 
+class Worker(QObject):
+    update = Signal(int)
+    finished = Signal()
+
+    def __init__(self, func):
+        super().__init__()
+        self.func = func
+
+    def run(self):
+        self.func()
+        self.finished.emit()
+
+
 if __name__ == "__main__":
     main()
diff --git a/setup.py b/setup.py
index 908ab3c4479d8d91261ce07b7aed0132d1a86d6b..b9b080e8714f1d23665a5f84598bf79e0f1f6383 100644
--- a/setup.py
+++ b/setup.py
@@ -44,6 +44,6 @@ setup(
     name='mseedpeek',
     packages=find_packages(include=['mseedpeek']),
     url='https://git.passcal.nmt.edu/software_public/passoft/mseedpeek',
-    version='2023.3.0.0',
+    version='2023.3.1.0',
     zip_safe=False,
 )
diff --git a/tox.ini b/tox.ini
index 2f1ef0ccee9f1d8c231993850923ea7e5f597020..10760b0e3ae453f9ae4d704e9d5672e6b22c3390 100644
--- a/tox.ini
+++ b/tox.ini
@@ -1,5 +1,5 @@
 [tox]
-envlist = py39, flake8
+envlist = py39, py310, flake8
 
 [testenv:flake8]
 basepython = python