diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index ed0077b62ea6c680cdc8a5103af9bde82ec873c5..57dce3e715ba8db1071766feb94f496194629a22 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -18,6 +18,8 @@ stages:
 - test
 
 before_script:
+- uname -a
+- apt-get update && apt install -y libgl1-mesa-glx
 - pip install -e .[dev]
 
 linting:
diff --git a/HISTORY.rst b/HISTORY.rst
index 16e6d57b26a314f5d4b34ccaf7a10581f785de2d..6d97ddb7c00f7a80ab0b5a5a0ef4daa7bd5fc1f8 100644
--- a/HISTORY.rst
+++ b/HISTORY.rst
@@ -22,3 +22,9 @@ History
 2022.1.0.0 (2022-01-11)
 ------------------
 * New versioning scheme
+
+2022.2.0.0 (2023-04-03)
+------------------
+* Gui updated to PySide2
+* Pmw dependency dropped
+* Version number updated
diff --git a/conda.recipe/meta.yaml b/conda.recipe/meta.yaml
index 381116cd7fed15468bf2790021325841a15070c4..40ab1da3a74f8387cbbb31d7cd2ed8e6c3aade5c 100644
--- a/conda.recipe/meta.yaml
+++ b/conda.recipe/meta.yaml
@@ -1,6 +1,6 @@
 package:
   name: mseedpeek
-  version: 2022.1.0.0
+  version: 2022.2.0.0
 
 source:
   path: ../
@@ -11,11 +11,11 @@ build:
 
 requirements:
   host:
-    - python >=3.6
+    - python >=3.8
     - pip
   run:
-    - python >=3.6
-    - pmw
+    - python >=3.8
+    - pyside2
 
 test:
   source_files:
diff --git a/mseedpeek/__init__.py b/mseedpeek/__init__.py
index b2d7faa954d0db0333beb2eb12c08783165eb1b9..ded7e1b562dc5ac23371c364a3e8a7b802d45e81 100644
--- a/mseedpeek/__init__.py
+++ b/mseedpeek/__init__.py
@@ -4,4 +4,4 @@
 
 __author__ = """IRIS PASSCAL"""
 __email__ = 'software-support@passcal.nmt.edu'
-__version__ = '2022.1.0.0'
+__version__ = '2022.2.0.0'
diff --git a/mseedpeek/mseedInfo.py b/mseedpeek/mseedInfo.py
index 3df5a218a672be8591892668de75de3c10084763..ea0a95fe30136933db7a5d551f0055bf27edc711 100755
--- a/mseedpeek/mseedInfo.py
+++ b/mseedpeek/mseedInfo.py
@@ -182,33 +182,30 @@ BlkVars[2000] = [
 BlkInfoDict = {}
 
 BlkInfoDict["Data Fields"] = """
-
-Field type   Number of bits   Field description
-UBYTE         8               Unsigned quantity
-BYTE          8               Twos complement signed quantity
-UWORD        16               Unsigned quantity
-WORD         16               Twos complement signed quantity
-ULONG        32               Unsigned quantity
-LONG         32               Twos complement signed quantity
-CHAR*n       n*8              n characters, each 8 bits and each with a
-                              7-bit ASCII character (high bit always 0)
-FLOAT        32               IEEE Floating point number
+Field type\tNumber of bits\tField description
+UBYTE\t8\t\tUnsigned quantity
+BYTE\t8\t\tTwos complement signed quantity
+UWORD\t16\t\tUnsigned quantity
+WORD\t16\t\tTwos complement signed quantity
+ULONG\t32\t\tUnsigned quantity
+LONG\t32\t\tTwos complement signed quantity
+CHAR*n\tn*8\t\tn characters, each 8 bits and each with a
+\t\t\t7-bit ASCII character (high bit always 0)
+FLOAT\t32\t\tIEEE Floating point number
 """
 
 BlkInfoDict["BTime"] = """
-
-Field type   Number of bits   Field description
-UWORD        16               Year (e.g., 1987)
-UWORD        16               Day of Year (Jan 1 is 1)
-UBYTE         8               Hours of day (023)
-UBYTE         8               Minutes of day (059)
-UBYTE         8               Seconds of day (059, 60 for leap seconds)
-UBYTE         8               Unused for data (required for alignment)
-UWORD        16               .0001 seconds (09999)
+Field type\tNumber of bits\tField description
+UWORD\t16\t\tYear (e.g., 1987)
+UWORD\t16\t\tDay of Year (Jan 1 is 1)
+UBYTE\t8\t\tHours of day (023)
+UBYTE\t8\t\tMinutes of day (059)
+UBYTE\t8\t\tSeconds of day (059, 60 for leap seconds)
+UBYTE\t8\t\tUnused for data (required for alignment)
+UWORD\t16\t\t.0001 seconds (09999)
 """
 
 BlkInfoDict["Fixed Header"] = """
-
 Fixed Section of Data Header (48 bytes)
 
 The data record header starts at the first byte. The next eight bytes follow
@@ -218,97 +215,98 @@ an ASCII space  shown here as a "delta"). The next ten bytes contain the
 station, location, and channel identity of the record. The rest of the header
 section is binary.
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Sequence number               A       6       "######"
-2      Data header/quality indicator
-       (D|R|Q)                       A       1
-3      Reserved byte ("delta")       A       1
-4      Station identifier code       A       5       [UN]
-5      Location identifier           A       2       [UN]
-6      Channel identifier            A       3       [UN]
-7      Network Code                  A       2
-8      Record start time             B      10
-9      Number of samples             B       2
-10      Sample rate factor           B       2
-11      Sample rate multiplier       B       2
-12      Activity flags               B       1
-13      I/O and clock flags          B       1
-14      Data quality flags           B       1
-15      Number of blockettes that
-        follow                       B       1
-16      Time correction              B       4
-17      Beginning of data            B       2
-18      First blockette              B       2
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tSequence number\t\tA\t6\t"######"
+2\tData header/quality indicator
+\t(D|R|Q)\t\t\tA\t1
+3\tReserved byte ("delta")\t\tA\t1
+4\tStation identifier code\t\tA\t5\t[UN]
+5\tLocation identifier\t\tA\t2\t[UN]
+6\tChannel identifier\t\tA\t3\t[UN]
+7\tNetwork Code\t\tA\t2
+8\tRecord start time\t\tB\t10
+9\tNumber of samples\t\tB\t2
+10\tSample rate factor\t\tB\t2
+11\tSample rate multiplier\t\tB\t2
+12\tActivity flags\t\tB\t1
+13\tI/O and clock flagst\t\tB\t1
+14\tData quality flags\t\tB\t1
+15\tNumber of blockettes that
+\tfollow\t\t\tB\t1
+16\tTime correction\t\tB\t4
+17\tBeginning of data\t\tB\t2
+18\tFirst blockette\t\tB\t2
 
 Notes for fields: * indicates mandatory information
 1 * Data record sequence number (Format "######").
 2 * "D" or "R" or "Q"  Data header/quality indicator. Previously, this
-    field was only allowed to be "D" and was only used to indicate that
-    this is a data header. As of SEED version 2.4 the meaning of this
-    field has been extended to also indicate the level of quality control
-    that has been applied to the record.
-      D  The state of quality control of the data is indeterminate.
-      R  Raw Waveform Data with no Quality Control
-      Q  Quality Controlled Data, some processes have been applied to the data.
+      field was only allowed to be "D" and was only used to indicate that
+      this is a data header. As of SEED version 2.4 the meaning of this
+      field has been extended to also indicate the level of quality control
+      that has been applied to the record.
+        D  The state of quality control of the data is indeterminate.
+        R  Raw Waveform Data with no Quality Control
+        Q  Quality Controlled Data, some processes have been applied to the
+             data.
 3   Space (ASCII 32)  Reserved; do not use this byte.
 4 * Station identifier designation (see Appendix G). Left justify and pad with
-    spaces.
+      spaces.
 5 * Location identifier designation. Left justify and pad with spaces.
 6 * Channel identifier designation (see Appendix A). Left justify and pad
-    with spaces.
+      with spaces.
 7 * A two character alphanumeric identifier that uniquely identifies the
-    network operator responsible for the data logger. This identifier is
-    assigned by the IRIS Data Management Center in consultation with the
-    FDSN working group on the SEED format.
+      network operator responsible for the data logger. This identifier is
+      assigned by the IRIS Data Management Center in consultation with the
+      FDSN working group on the SEED format.
 8 * BTIME: Start time of record.
 9 * UWORD: Number of samples in record.
 10 * WORD: Sample rate factor:
-           > 0  Samples/second
-           < 0  Seconds/sample
+             > 0  Samples/second
+             < 0  Seconds/sample
 11 * WORD: Sample rate multiplier:
-           >0  Multiplication factor
-           <0  Division factor
+             > 0  Multiplication factor
+             < 0  Division factor
 12 UBYTE: Activity flags:
-          [Bit 0]  Calibration signals present
-   *      [Bit 1]  Time correction applied. Set this bit to 1 if the time
-                    correction in field 16 has been applied to field 8. Set
-                    this bit to 0 if the time correction in field 16 has not
-                    been applied to field 8.
-          [Bit 2]  Beginning of an event, station trigger
-          [Bit 3]  End of the event, station detriggers
-          [Bit 4]  A positive leap second happened during this record (A 61
-                    second minute).
-          [Bit 5]  A negative leap second happened during this record (A 59
-                    second minute). A negative leap second clock correction has
-                    not yet been used, but the U.S. National Bureau of
-                    Standards has said that it might be necessary someday.
-          [Bit 6]  Event in progress
+             [Bit 0]  Calibration signals present
+   *        [Bit 1]  Time correction applied. Set this bit to 1 if the time
+                      correction in field 16 has been applied to field 8. Set
+                      this bit to 0 if the time correction in field 16 has not
+                      been applied to field 8.
+            [Bit 2]  Beginning of an event, station trigger
+            [Bit 3]  End of the event, station detriggers
+            [Bit 4]  A positive leap second happened during this record (A 61
+                      second minute).
+            [Bit 5]  A negative leap second happened during this record (A 59
+                      second minute). A negative leap second clock correction
+                      has not yet been used, but the U.S. National Bureau of
+                      Standards has said that it might be necessary someday.
+            [Bit 6]  Event in progress
 13 UBYTE: I/O flags and clock flags:
-          [Bit 0]  Station volume parity error possibly present
-          [Bit 1]  Long record read (possibly no problem)
-          [Bit 2]  Short record read (record padded)
-          [Bit 3]  Start of time series
-          [Bit 4]  End of time series
-          [Bit 5]  Clock locked
+            [Bit 0]  Station volume parity error possibly present
+            [Bit 1]  Long record read (possibly no problem)
+            [Bit 2]  Short record read (record padded)
+            [Bit 3]  Start of time series
+            [Bit 4]  End of time series
+            [Bit 5]  Clock locked
 14 UBYTE: Data quality flags
-          [Bit 0]  Amplifier saturation detected (station dependent)
-          [Bit 1]  Digitizer clipping detected
-          [Bit 2]  Spikes detected
-          [Bit 3]  Glitches detected
-          [Bit 4]  Missing/padded data present
-          [Bit 5]  Telemetry synchronization error
-          [Bit 6]  A digital filter may be charging
-          [Bit 7]  Time tag is questionable
+            [Bit 0]  Amplifier saturation detected (station dependent)
+            [Bit 1]  Digitizer clipping detected
+            [Bit 2]  Spikes detected
+            [Bit 3]  Glitches detected
+            [Bit 4]  Missing/padded data present
+            [Bit 5]  Telemetry synchronization error
+            [Bit 6]  A digital filter may be charging
+            [Bit 7]  Time tag is questionable
 15 * UBYTE: Total number of blockettes that follow.
 16 * LONG: Time correction. This field contains a value that may modify the
-           field 8 record start time. Depending on the setting of bit 1 in
-           field 12, the record start time may have already been adjusted.
-           The units are in 0.0001 seconds.
+             field 8 record start time. Depending on the setting of bit 1 in
+             field 12, the record start time may have already been adjusted.
+             The units are in 0.0001 seconds.
 17 * UWORD: Offset in bytes to the beginning of data. The first byte of the
-            data records is byte 0.
+              data records is byte 0.
 18 * UWORD: Offset in bytes to the first data blockette in this data record.
-            Enter 0 if there are no data blockettes. The first byte in the
-            data record is byte offset 0.
+              Enter 0 if there are no data blockettes. The first byte in the
+              data record is byte offset 0.
 
 NOTE: All unused bits in the flag bytes are reserved and must be set to zero.
 The last word defines the length of the fixed header. The next-to-last word
@@ -322,113 +320,111 @@ Here is an algorithm and some sample rate combinations that describe how the
 sample rate factors and multipliers work:
 
 If Sample rate factor > 0 and Sample rate Multiplier > 0,
-   Then nominal Sample rate = Sample rate factor X Sample rate multiplier
+    Then nominal Sample rate = Sample rate factor X Sample rate multiplier
 If Sample rate factor > 0 and Sample rate Multiplier < 0,
-   Then nominal Sample rate = -1 X Sample rate factor / Sample rate multiplier
+    Then nominal Sample rate = -1 X Sample rate factor / Sample rate multiplier
 If Sample rate factor < 0 and Sample rate Multiplier > 0,
-   Then nominal Sample rate = -1 X Sample rate multiplier / Sample rate factor
+    Then nominal Sample rate = -1 X Sample rate multiplier / Sample rate factor
 If Sample rate factor < 0 and Sample rate Multiplier < 0,
-   Then nominal Sample rate = 1/ (Sample rate factor X Sample rate multiplier)
-
-Sample rate   Sample rate factor   Sample rate multiplier
-330 SPS         33                  10
-330              1
-330.6 SPS     3306                 -10
-1 SP Min       -60                   1
-0.1 SPS          1                 -10
--10              1
--1             -10
+    Then nominal Sample rate = 1/ (Sample rate factor X Sample rate multiplier)
+
+Sample rate\t\tSample rate factor\tSample rate multiplier
+330 SPS\t\t33\t\t10
+330\t\t1
+330.6 SPS\t\t3306\t\t-10
+1 SP Min\t\t-60\t\t1
+0.1 SPS\t\t1\t\t-10
+-10\t\t1
+-1\t\t-10
 """
 
 BlkInfoDict[100] = """
-
 Sample Rate Blockette 100 (12 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 100            B      2
-2      Next blockettes byte number   B      2
-3      Actual Sample Rate            B      4
-4      Flags (to be defined)         B      1
-5      Reserved byte                 B      3
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 100\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tActual Sample Rate\t\tB\t4
+4\tFlags (to be defined)\t\tB\t1
+5\tReserved byte\t\tB\t3
 
 Notes for fields:
-1 UWORD: Blockette type [100]: sample rate.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 FLOAT: Actual sample rate of this data block.
-4 BYTE:  Flags (to be defined)
-5 UBYTE: Reserved; do not use.
+1 UWORD:\tBlockette type [100]: sample rate.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 FLOAT:\tActual sample rate of this data block.
+4 BYTE:\tFlags (to be defined)
+5 UBYTE:\tReserved; do not use.
 """
 BlkInfoDict[200] = """
 Generic Event Detection Blockette 200 (52 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 200            B       2
-2      Next blockettes byte number   B       2
-3      Signal amplitude              B       4
-4      Signal period                 B       4
-5      Background estimate           B       4
-6      Event detection flags         B       1
-7      Reserved byte                 B       1
-8      Signal onset time             B      10
-9      Detector Name                 A      24
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 200\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tSignal amplitude\t\tB\t4
+4\tSignal period\t\tB\t4
+5\tBackground estimate\t\tB\t4
+6\tEvent detection flags\t\tB\t1
+7\tReserved byte\t\tB\t1
+8\tSignal onset time\t\tB\t10
+9\tDetector Name\t\tA\t24
 
 Notes for fields:
-1 UWORD: Blockette type [200]: event detection information.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 FLOAT: Amplitude of signal (for units, see event detection flags, below;
-         0 if unknown).
-4 FLOAT: Period of signal, in seconds (0 if unknown).
-5 FLOAT: Background estimate (for units, see event detection flags, below;
-         0 if unknown).
-6 UBYTE: Event detection flags:
-         [Bit 0]  If set: dilatation wave; if unset: compression
-         [Bit 1]  If set: units above are after deconvolution
-                   (see Channel Identifier Blockette [52], field 8);
-                   if unset: digital counts
-         [Bit 2]  When set, bit 0 is undetermined
-         [Other bits reserved and must be zero.]
-7 UBYTE: Reserved; do not use.
-8 BTIME: Time of the onset of the signal.
+1 UWORD:\tBlockette type [200]: event detection information.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 FLOAT:\tAmplitude of signal (for units, see event detection flags,
+\tbelow; 0 if unknown).
+4 FLOAT:\tPeriod of signal, in seconds (0 if unknown).
+5 FLOAT:\tBackground estimate (for units, see event detection flags,
+\tbelow; 0 if unknown).
+6 UBYTE:\tEvent detection flags:
+\t[Bit 0]  If set: dilatation wave; if unset: compression
+\t[Bit 1]  If set: units above are after deconvolution
+\t            (see Channel Identifier Blockette [52], field 8);
+\t            if unset: digital counts
+\t[Bit 2]  When set, bit 0 is undetermined
+\t[Other bits reserved and must be zero.]
+7 UBYTE:\tReserved; do not use.
+8 BTIME:\tTime of the onset of the signal.
 9 CHAR*24: The name of the event detector.
 """
 BlkInfoDict[201] = """
-
 Murdock Event Detection Blockette 201 (60 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 201            B       2
-2      Next blockettes byte number   B       2
-3      Signal amplitude              B       4
-4      Signal period                 B       4
-5      Background estimate           B       4
-6      Event detection flags         B       1
-7      Reserved byte                 B       1
-8      Signal onset time             B      10
-9      Signal-to-noise ratio values  B       6
-10      Lookback value               B       1
-11      Pick algorithm               B       1
-12      Detector name                A      24
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 201\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tSignal amplitude\t\tB\t4
+4\tSignal period\t\tB\t4
+5\tBackground estimate\t\tB\t4
+6\tEvent detection flags\t\tB\t1
+7\tReserved byte\t\tB\t1
+8\tSignal onset time\t\tB\t10
+9\tSignal-to-noise ratio values\tB\t6
+10\tLookback value\t\tB\t1
+11\tPick algorithm\t\tB\t1
+12\tDetector name\t\tA\t24
 
 Notes for fields:
-1 UWORD: Blockette type [201]: event detection information.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 FLOAT: Amplitude of signal (in counts).
-4 FLOAT: Period of signal (in seconds).
-5 FLOAT: Background estimate (in counts).
-6 UBYTE: Event detection flags:
-         [Bit 0]  If set: dilatation wave; if unset: compression
-         [Other bits reserved and must be zero.]
-7 UBYTE: Reserved; do not use.
-8 BTIME: Onset time of the signal.
-9 UBYTE*6: Signal-to-noise ratio values.
-10 UBYTE: Lookback value (0,1,2).
-11 UBYTE: Pick algorithm (0,1).
+1 UWORD:\tBlockette type [201]: event detection information.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 FLOAT:\tAmplitude of signal (in counts).
+4 FLOAT:\tPeriod of signal (in seconds).
+5 FLOAT:\tBackground estimate (in counts).
+6 UBYTE:\tEvent detection flags:
+\t[Bit 0]  If set: dilatation wave; if unset: compression
+\t[Other bits reserved and must be zero.]
+7 UBYTE:\tReserved; do not use.
+8 BTIME:\tOnset time of the signal.
+9 UBYTE*6:\tSignal-to-noise ratio values.
+10 UBYTE:\tLookback value (0,1,2).
+11 UBYTE:\tPick algorithm (0,1).
 12 CHAR*24: The name of the event detector.
 
 NOTE: See Murdock (1983) and Murdock (1987) for more information on this
@@ -436,224 +432,218 @@ type of eventdetector, and on what the fields listed above should contain.
 """
 
 BlkInfoDict[300] = """
-
 Step Calibration Blockette 300 (60 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 300            B       2
-2      Next blockettes byte number   B       2
-3      Beginning of calibration time B      10
-4      Number of step calibrations   B       1
-5      Calibration flags             B       1
-6      Step duration                 B       4
-7      Interval duration             B       4
-8      Calibration signal amplitude  B       4
-9      Channel w/ calibration input  A       3
-10     Reserved byte                 B       1
-11     Reference amplitude           B       4
-12     Coupling                      A      12
-13     Rolloff                       A      12
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 300\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tBeginning of calibration time\tB\t10
+4\tNumber of step calibrations\tB\t1
+5\tCalibration flags\t\tB\t1
+6\tStep duration\t\tB\t4
+7\tInterval duration\t\tB\t4
+8\tCalibration signal amplitude\tB\t4
+9\tChannel w/ calibration input\tA\t3
+10\tReserved byte\t\tB\t1
+11\tReference amplitude\t\tB\t4
+12\tCoupling\t\t\tA\t12
+13\tRolloff\t\t\tA\t12
 
 Notes for fields:
-1 UWORD: Blockette type[300]: step calibration.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 BTIME: Beginning time of calibration.
-4 UBYTE: Number of step calibrations in sequence.
-5 UBYTE: Calibration flags:
-         [Bit 0]  If set: first pulse is positive
-         [Bit 1]  If set: calibrations alternate sign
-         [Bit 2]  If set: calibration was automatic; if unset: manual
-         [Bit 3]  If set: calibration continued from previous record(s)
-         [Other bits reserved and must be zero.]
-6 ULONG: Number of .0001 second ticks for the duration of the step.
-7 ULONG: Number of .0001 second ticks for the interval between times the
-         calibration step is on.
-8 FLOAT: Amplitude of calibration signal in units (see Channel Identifier
-         Blockette [52), field 9).
-9 CHAR*3: Channel containing calibration input (blank means none).
-          SEED assumes that the calibration output is on the current
-          channel, identified in the fixed header.
-10 UBYTE: Reserved; do not use.
-11 ULONG: Reference amplitude. This is a user defined value that indicates
-          either the voltage or amperage of the calibration signal when the
-          calibrator is set to 0dB. If this value is zero, then no units are
-          specified, and the amplitude (Note 4) will be reported in "binary
-          decibels" from to -96.
+1 UWORD:\tBlockette type[300]: step calibration.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 BTIME:\tBeginning time of calibration.
+4 UBYTE:\tNumber of step calibrations in sequence.
+5 UBYTE:\tCalibration flags:
+\t[Bit 0]  If set: first pulse is positive
+\t[Bit 1]  If set: calibrations alternate sign
+\t[Bit 2]  If set: calibration was automatic; if unset: manual
+\t[Bit 3]  If set: calibration continued from previous record(s)
+\t[Other bits reserved and must be zero.]
+6 ULONG:\tNumber of .0001 second ticks for the duration of the step.
+7 ULONG:\tNumber of .0001 second ticks for the interval between times the
+\tcalibration step is on.
+8 FLOAT:\tAmplitude of calibration signal in units (see Channel Identifier
+\tBlockette [52), field 9).
+9 CHAR*3:\tChannel containing calibration input (blank means none).
+\tSEED assumes that the calibration output is on the current
+\tchannel, identified in the fixed header.
+10 UBYTE:\tReserved; do not use.
+11 ULONG:\tReference amplitude. This is a user defined value that indicates
+\teither the voltage or amperage of the calibration signal when the
+\tcalibrator is set to 0dB. If this value is zero, then no units are
+\tspecified, and the amplitude (Note 4) will be reported in "binary
+\tdecibels" from to -96.
 12 CHAR*12: Coupling of calibration signal, such as "Resistive " or
-            "Capacitive".
+\t"Capacitive".
 13 CHAR*12: Rolloff characteristics for any filters used on the calibrator,
-            such as "3dB@10Hz".
+\tsuch as "3dB@10Hz".
 """
 
 BlkInfoDict[310] = """
-
 Sine Calibration Blockette 310 (60 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 310            B       2
-2      Next blockettes byte number   B       2
-3      Beginning of calibration time B      10
-4      Reserved byte                 B       1
-5      Calibration flags             B       1
-6      Calibration duration          B       4
-7      Period of signal (seconds)    B       4
-8      Amplitude of signal           B       4
-9      Channel w/ calibration input  A       3
-10      Reserved byte                B       1
-11      Reference amplitude          B       4
-12      Coupling                     A      12
-13      Rolloff                      A      12
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 310\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tBeginning of calibration time\tB\t10
+4\tReserved byte\t\tB\t1
+5\tCalibration flags\t\tB\t1
+6\tCalibration duration\t\tB\t4
+7\tPeriod of signal (seconds)\tB\t4
+8\tAmplitude of signal\t\tB\t4
+9\tChannel w/ calibration input\tA\t3
+10\tReserved byte\t\tB\t1
+11\tReference amplitude\t\tB\t4
+12\tCoupling\t\t\tA\t12
+13\tRolloff\t\t\tA\t12
 
 Notes for fields:
-1 UWORD: Blockette type [310]: sine calibration.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 BTIME: Beginning time of calibration.
-4 UBYTE: Reserved; do not use.
-5 UBYTE : Calibration flags:
-[Bit 2]  If set: calibration was automatic; otherwise: manual
-[Bit 3]  If set: calibration continued from previous record(s)
-[Bit 4]  If set: peak-to-peak amplitude
-[Bit 5]  If set: zero-to-peak amplitude
-[Bit 6]  If set: RMS amplitude
-[Other bits reserved and must be zero.]
-6 ULONG: Number of .0001 second ticks for the duration of calibration.
-7 FLOAT: Period of signal in seconds.
-8 FLOAT: Amplitude of signal in units (see Channel Identifier Blockette [52),
-         field 9).
-9 CHAR*3: Channel containing calibration input (blank means none).
-10 UBYTE: Reserved; do not use.
-11 ULONG: Reference amplitude. This is a user defined value that indicates
-          either the voltage or amperage of the calibration signal when the
-          calibrator is set to 0dB. If this value is zero, then no units are
-          specified, and the amplitude (Note 4) will be reported in "binary
-          decibels" from 0 to -96.
+1 UWORD:\tBlockette type [310]: sine calibration.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 BTIME:\tBeginning time of calibration.
+4 UBYTE:\tReserved; do not use.
+5 UBYTE:\tCalibration flags:
+\t[Bit 2]  If set: calibration was automatic; otherwise: manual
+\t[Bit 3]  If set: calibration continued from previous record(s)
+\t[Bit 4]  If set: peak-to-peak amplitude
+\t[Bit 5]  If set: zero-to-peak amplitude
+\t[Bit 6]  If set: RMS amplitude
+\t[Other bits reserved and must be zero.]
+6 ULONG:\tNumber of .0001 second ticks for the duration of calibration.
+7 FLOAT:\tPeriod of signal in seconds.
+8 FLOAT:\tAmplitude of signal in units (see Channel Identifier Blockette [52),
+\tfield 9).
+9 CHAR*3:\tChannel containing calibration input (blank means none).
+10 UBYTE:\tReserved; do not use.
+11 ULONG:\tReference amplitude. This is a user defined value that indicates
+\teither the voltage or amperage of the calibration signal when the
+\tcalibrator is set to 0dB. If this value is zero, then no units are
+\tspecified, and the amplitude (Note 4) will be reported in "binary
+\tdecibels" from 0 to -96.
 12 CHAR*12: Coupling of calibration signal such as "Resistive or "Capacitive".
 13 CHAR*12: Rolloff characteristics for any filters used on the calibration,
-            such as "3dB@10Hz".
+\tsuch as "3dB@10Hz".
 
 NOTE: Only one of flag bits 4, 5, and 6 can be set at one time, but one of
 them must be set.
 """
 
 BlkInfoDict[320] = """
-
 Pseudo-random Calibraton Blockette 320 (64 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 320            B       2
-2      Next blockettes byte number   B       2
-3      Beginning of calibration time B      10
-4      Reserved byte                 B       1
-5      Calibration flags             B       1
-6      Calibration duration          B       4
-7      Peak-to-peak amplitude of
-       steps                         B       4
-8      Channel w/ calibration input  A       3
-9      Reserved byte                 B       1
-10      Reference amplitude          B       4
-11      Coupling                     A      12
-12      Rolloff                      A      12
-13      Noise type                   A       8
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 320\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tBeginning of calibration time\tB\t10
+4\tReserved byte\t\tB\t1
+5\tCalibration flags\t\tB\t1
+6\tCalibration duration\t\tB\t4
+7\tPeak-to-peak amplitude of
+\tsteps\t\t\tB\t4
+8\tChannel w/ calibration input\tA\t3
+9\tReserved byte\t\tB\t1
+10\tReference amplitude\t\tB\t4
+11\tCoupling\t\t\tA\t12
+12\tRolloff\t\t\tA\t12
+13\tNoise type\t\t\tA\t8
 
 Notes for fields:
-1 UWORD: Blockette type [320]: pseudo-random binary sequence.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 BTIME: Beginning time of calibration.
-4 UBYTE: Reserved; do not use.
-5 UBYTE: Calibration flags:
-         [Bit 2]  If set: calibration was automatic; otherwise: manual
-         [Bit 3]  If set: calibration continued from previous record(s)
-         [Bit 4]  If set: random amplitudes
-         (must have a calibration in channel)
-         [Other bits reserved and must be zero.]
-6 ULONG: Number of .0001 second ticks for the duration of calibration.
-7 FLOAT: Peak-to-peak amplitude of steps in units (see Channel Identifier
-         Blockette [52], field 9).
-8 CHAR*3: Channel containing calibration input (blank if none).
-9 UBYTE: Reserved; do not use.
-10 ULONG: Reference amplitude. This is a user defined value that indicates
-          either the voltage or amperage of the calibration signal when the
-          calibrator is set to 0dB. If this value is zero, then no units are
-          specified, and the amplitude (Note 4) will be reported in "binary
-          decibels" from 0 to -96.
+1 UWORD:\tBlockette type [320]: pseudo-random binary sequence.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+        \tfrom the beginning of the logical record  including the fixed
+        \tsection of the data header; use 0 if no more blockettes will follow.)
+3 BTIME:\tBeginning time of calibration.
+4 UBYTE:\tReserved; do not use.
+5 UBYTE:\tCalibration flags:
+        \t[Bit 2]  If set: calibration was automatic; otherwise: manual
+        \t[Bit 3]  If set: calibration continued from previous record(s)
+        \t[Bit 4]  If set: random amplitudes
+        \t(must have a calibration in channel)
+        \t[Other bits reserved and must be zero.]
+6 ULONG:\tNumber of .0001 second ticks for the duration of calibration.
+7 FLOAT:\tPeak-to-peak amplitude of steps in units (see Channel Identifier
+        \tBlockette [52], field 9).
+8 CHAR*3:\tChannel containing calibration input (blank if none).
+9 UBYTE:\tReserved; do not use.
+10 ULONG:\tReference amplitude. This is a user defined value that indicates
+        \teither the voltage or amperage of the calibration signal when the
+        \tcalibrator is set to 0dB. If this value is zero, then no units are
+        \tspecified, and the amplitude (Note 4) will be reported in "binary
+        \tdecibels" from 0 to -96.
 11 CHAR*12: Coupling of calibration signal such as "Resistive or "Capacitive".
 12 CHAR*12: Rolloff characteristics for any filters used on the calibration,
-            such as "3dB@10Hz".
-13 CHAR*8: Noise characteristics, such as "White" or "Red".
+        \tsuch as "3dB@10Hz".
+13 CHAR*8:\tNoise characteristics, such as "White" or "Red".
 
 NOTE: When you set calibration flag bit 4, the amplitude value contains the
 maximum peakto-peak amplitude.
 """
 
 BlkInfoDict[390] = """
-
 Generic Calibraton Blockette 390 (28 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 390            B       2
-2      Next blockettes byte number   B       2
-3      Beginning of calibration time B      10
-4      Reserved byte                 B       1
-5      Calibration flags             B       1
-6      Calibration duration          B       4
-7      Calibration signal amplitude  B       4
-8      Channel w/ calibration input  A       3
-9      Reserved byte                 B       1
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 390\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tBeginning of calibration time\tB\t10
+4\tReserved byte\t\tB\t1
+5\tCalibration flags\t\tB\t1
+6\tCalibration duration\t\tB\t4
+7\tCalibration signal amplitude\tB\t4
+8\tChannel w/ calibration input\tA\t3
+9\tReserved byte\t\tB\t1
 
 Notes for fields:
-1 UWORD: Blockette type [390]: generic calibration.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 BTIME: Beginning time of calibration.
-4 UBYTE: Reserved; do not use.
-5 UBYTE: Calibration flags:
-         [Bit 2]  If set: calibration was automatic; otherwise: manual
-         [Bit 3]  If set: calibration continued from previous record(s)
-         [Other bits reserved and must be zero.]
-6 ULONG: Number of .0001 second ticks for the duration of calibration.
-7 FLOAT: Amplitude of calibration in units, if known (see Channel Identifier
-         Blockette [52], field 9).
-8 CHAR*3: Channel containing calibration input (must be specified).
-9 UBYTE: Reserved; do not use.
+1 UWORD:\tBlockette type [390]: generic calibration.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 BTIME:\tBeginning time of calibration.
+4 UBYTE:\tReserved; do not use.
+5 UBYTE:\tCalibration flags:
+\t[Bit 2]  If set: calibration was automatic; otherwise: manual
+\t[Bit 3]  If set: calibration continued from previous record(s)
+\t[Other bits reserved and must be zero.]
+6 ULONG:\tNumber of .0001 second ticks for the duration of calibration.
+7 FLOAT:\tAmplitude of calibration in units, if known (see Channel Identifier
+\tBlockette [52], field 9).
+8 CHAR*3:\tChannel containing calibration input (must be specified).
+9 UBYTE:\tReserved; do not use.
 """
 
 BlkInfoDict[395] = """
-
 Calibraton Abort Blockette 395 (16 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 395            B       2
-2      Next blockettes byte number   B       2
-3      End of calibration time       B      10
-4      Reserved bytes                B       2
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 395\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tEnd of calibration time\t\tB\t10
+4\tReserved bytes\t\tB\t2
 
 Notes for fields:
-1 UWORD: Blockette type [395]: calibration abort.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 BTIME: Time calibration ends.
-4 UWORD: Reserved; do not use.
+1 UWORD:\tBlockette type [395]: calibration abort.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 BTIME:\tTime calibration ends.
+4 UWORD:\tReserved; do not use.
 """
 
 BlkInfoDict[400] = """
-
 Beam Blockette 400 (16 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 400            B       2
-2      Next blockettes byte number   B       2
-3      Beam azimuth (degrees)        B       4
-4      Beam slowness (sec/degree)    B       4
-5      Beam configuration            B       2
-6      Reserved bytes                B       2
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 400\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tBeam azimuth (degrees)\tB\t4
+4\tBeam slowness (sec/degree)\tB\t4
+5\tBeam configuration\t\tB\t2
+6\tReserved bytes\t\tB\t2
 
 This blockette is used to specify how the beam indicated by the corresponding
 Beam Configuration Blockette [35] was formed for this data record. For beams
@@ -662,29 +652,28 @@ determine the beam delay for each component referred to in the Beam
 Configuration Blockette [35].
 
 Notes for fields:
-1 UWORD: Blockette type [400]: beam forming.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 FLOAT: Azimuth of beam (degrees clockwise from north).
-4 FLOAT: Beam slowness (sec/degree).
-5 UWORD: Beam configuration (see field 3 of the Beam Configuration Blockette
-         [35] abbreviation dictionary).
-         NOTE: This field is a binary equivalent of the ASCII formatted
-         dictionary key entry number in the Beam Configuration Blockette [35].
-         This is the only place in SEED Version 2.1 where this ASCII-to-binary
-         conversion needs to be made.
-6 UWORD: Reserved; do not use.
+1 UWORD:\tBlockette type [400]: beam forming.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 FLOAT:\tAzimuth of beam (degrees clockwise from north).
+4 FLOAT:\tBeam slowness (sec/degree).
+5 UWORD:\tBeam configuration (see field 3 of the Beam Configuration Blockette
+\t[35] abbreviation dictionary).
+\tNOTE: This field is a binary equivalent of the ASCII formatted
+\tdictionary key entry number in the Beam Configuration Blockette [35].
+\tThis is the only place in SEED Version 2.1 where this ASCII-to-binary
+\tconversion needs to be made.
+6 UWORD:\tReserved; do not use.
 """
 
 BlkInfoDict[405] = """
-
 Beam Delay Blockette 405 (6 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 405            B       2
-2      Next blockettes byte number   B       2
-3      Array of delay values         B       2
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 405\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tArray of delay values\t\tB\t2
 
 Use this blockette to define beams that do not travel as plane waves at
 constant velocities across arrays. This blockette, if used, will always
@@ -695,165 +684,159 @@ component in the Beam Configuration Blockette [35] of the abbreviation
 dictionary control headers, indexed by the Beam Blockette [400].
 
 Notes for fields:
-1 UWORD: Blockette type [405]: beam delay.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 UWORD: Array of delay values (one for each entry of the Beam Configuration
-         Blockette [35]. The array values are in .0001 second ticks.
+1 UWORD:\tBlockette type [405]: beam delay.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 UWORD:\tArray of delay values (one for each entry of the Beam Configuration
+\tBlockette [35]. The array values are in .0001 second ticks.
 """
 
 BlkInfoDict[500] = """
-
 Timing Blockette 500 (200 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 500            B        2
-2      Next blockettes byte number   B        2
-3      VCO correction                B        4
-4      Time of exception             B       10
-5      microsec                      B        1
-6      Reception Quality             B        1
-7      Exception count               B        4
-8      Exception type                A       16
-9      Clock model                   A       32
-10      Clock status                 A      128
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 500\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tVCO correction\t\tB\t4
+4\tTime of exception\t\tB\t10
+5\tmicrosec\t\t\tB\t1
+6\tReception Quality\t\tB\t1
+7\tException count\t\tB\t4
+8\tException type\t\tA\t16
+9\tClock model\t\tA\t32
+10\tClock status\t\t\tA\t128
 
 Notes for fields:
-1 UWORD: Blockette type [500]: Timing blockette.
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 FLOAT: VCO correction is a floating point percentage from 0.0 to 100.0% of
-         VCO control value, where 0.0 is slowest , and 100.0% is fastest.
-4 BTIME: Time of exception, same format as record start time.
-5 UBYTE: microsec has the clock time down to the microsecond. The SEED format
-         handles down to 100microsecs. This field is an offset from that value.
-         The recommended value is from -50 to +49microsecs. At the users
-         option, this value may be from 0 to +99microsecs.
-6 UBYTE: Reception quality is a number from 0 to 100% of maximum clock accuracy
-         based only on information from the clock.
-7 ULONG: Exception count is an integer count, with its meaning based on the
-         type of exception, such as 15 missing timemarks.
-8 CHAR*16: Exception type describes the type of clock exception, such as
-           "Missing" or "Unexpected".
-9 CHAR*32: Clock model is an optional description of the clock, such as
-           "Quanterra GPS1/QTS".
+1 UWORD:\tBlockette type [500]: Timing blockette.
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 FLOAT:\tVCO correction is a floating point percentage from 0.0 to 100.0% of
+\tVCO control value, where 0.0 is slowest , and 100.0% is fastest.
+4 BTIME:\tTime of exception, same format as record start time.
+5 UBYTE:\tmicrosec has the clock time down to the microsecond. The SEED format
+\thandles down to 100microsecs. This field is an offset from that value.
+\tThe recommended value is from -50 to +49microsecs. At the users
+\toption, this value may be from 0 to +99microsecs.
+6 UBYTE:\tReception quality is a number from 0 to 100% of maximum clock
+\taccuracy based only on information from the clock.
+7 ULONG:\tException count is an integer count, with its meaning based on the
+\ttype of exception, such as 15 missing timemarks.
+8 CHAR*16:\tException type describes the type of clock exception, such as
+\t"Missing" or "Unexpected".
+9 CHAR*32:\tClock model is an optional description of the clock, such as
+\t"Quanterra GPS1/QTS".
 10 CHAR*128: Clock status is an optional description of clock specific
-             parameters, such as the station for an Omega clock, or satellite
-             signal to noise ratios for GPS clocks.
+\tparameters, such as the station for an Omega clock, or satellite
+\tsignal to noise ratios for GPS clocks.
 """
 
 BlkInfoDict[1000] = """
-
 Data Only SEED Blockette 1000 (8 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 1000           B       2
-2      Next blockettes byte number   B       2
-3      Encoding Format               B       1
-4      Word order                    B       1
-5      Data Record Length            B       1
-6      Reserved                      B       1
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 1000\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tEncoding Format\t\tB\t1
+4\tWord order\t\t\tB\t1
+5\tData Record Length\t\tB\t1
+6\tReserved\t\t\tB\t1
 
 Notes for fields:
-1 UWORD: Blockette type [1000]: Data Only SEED
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3. BYTE : A code indicating the encoding format. This number is assigned by
-          the FDSN Data Exchange Working Group. To request that a new format
-          be included contact the FDSN through the FDSN Archive at the IRIS
-          Data Management Center. To be supported in Data Only SEED, the dat
-          format must be expressible in SEED DDL. A list of valid codes at the
-          time of publication follows.
-
-               CODES 0-9 GENERAL
-               0        ASCII text, byte order as specified in field 4
-               1        16 bit integers
-               2        24 bit integers
-               3        32 bit integers
-               4        IEEE floating point
-               5        IEEE double precision floating point
-
-               CODES 10 - 29 FDSN Networks
-               10        STEIM (1) Compression
-               11        STEIM (2) Compression
-               12        GEOSCOPE Multiplexed Format 24 bit integer
-               13        GEOSCOPE Multiplexed Format 16 bit gain ranged,
-                         3 bit exponent
-               14        GEOSCOPE Multiplexed Format 16 bit gain ranged,
-                         4 bit exponent
-               15        US National Network compression
-               16        CDSN 16 bit gain ranged
-               17        Graefenberg 16 bit gain ranged
-               18        IPG - Strasbourg 16 bit gain ranged
-               19        STEIM (3) Compression
-
-               CODES 30 - 39 OLDER NETWORKS
-               30        SRO Format
-               31        HGLP Format
-               32        DWWSSN Gain Ranged Format
-               33        RSTN 16 bit gain ranged
+1 UWORD:\tBlockette type [1000]: Data Only SEED
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3. BYTE:\tA code indicating the encoding format. This number is assigned by
+\tthe FDSN Data Exchange Working Group. To request that a new format
+\tbe included contact the FDSN through the FDSN Archive at the IRIS
+\tData Management Center. To be supported in Data Only SEED, the dat
+\tformat must be expressible in SEED DDL. A list of valid codes at the
+\ttime of publication follows.
+
+\tCODES 0-9 GENERAL
+\t0\tASCII text, byte order as specified in field 4
+\t1\t16 bit integers
+\t2\t24 bit integers
+\t3\t32 bit integers
+\t4\tIEEE floating point
+\t5\tIEEE double precision floating point
+
+\tCODES 10 - 29 FDSN Networks
+\t10\tSTEIM (1) Compression
+\t11\tSTEIM (2) Compression
+\t12\tGEOSCOPE Multiplexed Format 24 bit integer
+\t13\tGEOSCOPE Multiplexed Format 16 bit gain ranged, 3 bit exponent
+\t14\tGEOSCOPE Multiplexed Format 16 bit gain ranged, 4 bit exponent
+\t15\tUS National Network compression
+\t16\tCDSN 16 bit gain ranged
+\t17\tGraefenberg 16 bit gain ranged
+\t18\tIPG - Strasbourg 16 bit gain ranged
+\t19\tSTEIM (3) Compression
+
+\tCODES 30 - 39 OLDER NETWORKS
+\t30\tSRO Format
+\t31\tHGLP Format
+\t32\tDWWSSN Gain Ranged Format
+\t33\tRSTN 16 bit gain ranged
 
 4. The byte swapping order for 16 bit and 32 bit words. A 0 indicates VAX or
-   8086 order and a 1 indicates 68000 or SPARC word order. See fields 11 and
-   12 of blockette 50.
+    8086 order and a 1 indicates 68000 or SPARC word order. See fields 11 and
+    12 of blockette 50.
 5. The exponent (as a power of two) of the record length for these data. The
-   data record can be as small as 256 bytes and, in Data Only SEED format as
-   large as 2 raised to the 256 power.
+    data record can be as small as 256 bytes and, in Data Only SEED format as
+    large as 2 raised to the 256 power.
 """
 
 BlkInfoDict[1001] = """
-
 Data Extension Blockette 1001 (8 bytes)
 
-Note   Field Name                    Type   Length   Mask or Flags
-1      Blockette type 1001           B       2
-2      Next blockettes byte number   B       2
-3      Timing quality                B       1
-4      microsec                      B       1
-5      Reserved                      B       1
-6      Frame count                   B       1
+Note\tField Name\t\t\tType\tLength\tMask or Flags
+1\tBlockette type 1001\t\tB\t2
+2\tNext blockettes byte number\tB\t2
+3\tTiming quality\t\tB\t1
+4\tmicrosec \t\t\tB\t1
+5\tReserved\t\t\tB\t1
+6\tFrame count\t\tB\t1
 
 Notes for fields:
-1 UWORD: Blockette type [1001]: Data extension blockette
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3. UBYTE: Timing quality is a vendor specific value from 0 to 100% of maxium
-          accuracy, taking into account both clock quality and data flags.
-4. UBYTE: microsec has the data start time down to the microsecond. The SEED
-          format handles down to 100microsecs. This field is an offset from
-          that value. The recommended value is from -50 to +49microsecs. At
-          the users option, this value may be from 0 to +99microsecs.
+1 UWORD:\tBlockette type [1001]: Data extension blockette
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3. UBYTE:\tTiming quality is a vendor specific value from 0 to 100% of maxium
+\taccuracy, taking into account both clock quality and data flags.
+4. UBYTE:\tmicrosec has the data start time down to the microsecond. The SEED
+\tformat handles down to 100microsecs. This field is an offset from
+\tthat value. The recommended value is from -50 to +49microsecs. At
+\tthe users option, this value may be from 0 to +99microsecs.
 5. Reserved byte.
-6. UBYTE: Frame count is the number of 64 byte compressed data frames in the
-          4K record (maximum of 63). Note that the user may specify fewer than
-          the maxium allowable frames in a 4K record to reduce latency
+6. UBYTE:\tFrame count is the number of 64 byte compressed data frames in the
+\t4K record (maximum of 63). Note that the user may specify fewer than
+\tthe maxium allowable frames in a 4K record to reduce latency
 """
 
 BlkInfoDict[2000] = """
-
 Variable Length Opaque Data Blockette
 
-Note   Field Name                    Type   Length   Offset
-1      Blockette type 2000           B       2        0
-2      Next blockettes byte number   B       2        2
-3      Total blockette length in
-       bytes                         B       2        4
-4      Offset to Opaque Data         B       2        6
-5      Record number                 B       4        8
-6      Data Word order               B       1       12
-7      Opaque Data flags             B       1       13
-8      Number Opaque Header fields   B       1       14
-9      Opaque Data Header fields     V       V       15
-          a Record type
-          b Vendor type
-          c Model type
-          d Software
-          e Firmware
-10      Opaque Data                  Opaque
+Note\tField Name\t\t\tType\tLength\tOffset
+1\tBlockette type 2000\t\tB\t2\t0
+2\tNext blockettes byte number\tB\t2\t2
+3\tTotal blockette length in
+\tbytes\t\t\tB\t2\t4
+4\tOffset to Opaque Data\t\tB\t2\t6
+5\tRecord number\t\tB\t4\t8
+6\tData Word order\t\tB\t1\t12
+7\tOpaque Data flags\t\tB\t1\t13
+8\tNumber Opaque Header fields\tB\t1\t14
+9\tOpaque Data Header fields\tV\tV\t15
+\ta Record type
+\tb Vendor type
+\tc Model type
+\td Software
+\te Firmware
+10\tOpaque Data\t\tOpaque
 
 More than one blockette 2000 may be stored in a SEED data record if the
 SEED data record timetag is not required for precise timing of the data
@@ -865,67 +848,68 @@ to provide additional information
 for a normal timeseries data channel.
 
 Notes for fields:
-1 UWORD: Blockette type [2000]: Opaque Data blockette
-2 UWORD: Byte number of next blockette. (Calculate this as the byte offset
-         from the beginning of the logical record  including the fixed
-         section of the data header; use 0 if no more blockettes will follow.)
-3 UWORD: Blockette length. The total number of bytes in this blockette,
-         including the 6 bytes of header. The only restriction is that the
-         blockette must fit within a single SEED data record for the channel.
-         Otherwise, the blockette must be partitioned into multiple blockettes.
-4 UWORD: Offset to Opaque Data. Byte offset from beginning of blockette to
-         Opaque Data.
-5 ULONG: Record Number. The record number may be used for sequence
-         identification of stream, record, or file oriented data. If a record
-         is partitioned into multiple opaque blockettes, each blockette
-         containing a portion of the record should contain the identical record
-         number. It is strongly recommended that the record number be used to
-         aid in the detection of missing data and in merging data from
-         different telemetry streams. Use 0 if data is not record oriented, or
-         if record number is not required.
-6 UBYTE: Word order of binary opaque data. See field 4 of blockette 1000, and
-         fields 11 and 12 of blockette 50.
-              0 = little endian (VAX or 80x86 byte order).
-              1 = big endian (68000 or SPARC byte order).
-7 UBYTE: Opaque Data flags.
-         [bit 0] Opaque blockette orientation.
-              0 = record oriented.
-              1 = stream oriented.
-         [bit 1] Packaging bit.
-              0 = Blockette 2000s from multiple SEED data records with
-                  different timetags may be packaged into a single SEED data
-                  record. The exact original timetag in each SEED Fixed Data
-                  Header is not required for each blockette 2000.
-              1 = Blockette 2000s from multiple SEED data records with
-                  differing timetags may NOT be repackaged into a single SEED
-                  data record. Set this bit if the timetag in the SEED Fixed
-                  Data Header is required to properly interpret the opaque
-                  data.
-         [bits 2-3] Opaque blockette fragmentation flags.
-              00 = opaque record identified by record number is completely
-                   contained in this opaque blockette.
-              01 = first opaque blockette for record spanning multiple
-                   blockettes.
-              11 = continuation blockette 2...N-1 of record spanning N
-                   blockettes.
-              10 = final blockette for record spanning N blockettes.
-         [bits 4-5] File blockette information.
-              00 = not file oriented.
-              01 = first blockette of file.
-              10 = continuation of file.
-              11 = last blockette of file.
-8 UBYTE: Number of Opaque Header fields. Each opaque header field is a variable
-         length ascii string, terminated by the character ~.
-9 VAR: Opaque Data Header string, which contains the ascii variable length
-       fields. Each field is terminated by a "~". The definition of the fields
-       may be defined by the originator and receiver, but the following are
-       recommended.
-       Any of the fields may be empty.
-         a    Record Type - name of the type of record (eg "GPS", "GPS MBEN").
-         b    Vendor Type - name of equipment vendor (eg "ASHTECH").
-         c    Model type - model type of equipment (eg "Z12").
-         d   Software Version - software version number (eg "").
-         e   Firmware Version - firmware version number (eg "1G0C").
+1 UWORD:\tBlockette type [2000]: Opaque Data blockette
+2 UWORD:\tByte number of next blockette. (Calculate this as the byte offset
+\tfrom the beginning of the logical record  including the fixed
+\tsection of the data header; use 0 if no more blockettes will follow.)
+3 UWORD:\tBlockette length. The total number of bytes in this blockette,
+\tincluding the 6 bytes of header. The only restriction is that the
+\tblockette must fit within a single SEED data record for the channel.
+\tOtherwise, the blockette must be partitioned into multiple blockettes.
+4 UWORD:\tOffset to Opaque Data. Byte offset from beginning of blockette to
+\tOpaque Data.
+5 ULONG:\tRecord Number. The record number may be used for sequence
+\tidentification of stream, record, or file oriented data. If a record
+\tis partitioned into multiple opaque blockettes, each blockette
+\tcontaining a portion of the record should contain the identical record
+\tnumber. It is strongly recommended that the record number be used to
+\taid in the detection of missing data and in merging data from
+\tdifferent telemetry streams. Use 0 if data is not record oriented, or
+\tif record number is not required.
+6 UBYTE:\tWord order of binary opaque data. See field 4 of blockette 1000, and
+\tfields 11 and 12 of blockette 50.
+\t     0 = little endian (VAX or 80x86 byte order).
+\t     1 = big endian (68000 or SPARC byte order).
+7 UBYTE:\tOpaque Data flags.
+\t[bit 0] Opaque blockette orientation.
+\t     0 = record oriented.
+\t     1 = stream oriented.
+\t[bit 1] Packaging bit.
+\t     0 = Blockette 2000s from multiple SEED data records with
+\t          different timetags may be packaged into a single SEED data
+\t          record. The exact original timetag in each SEED Fixed Data
+\t          Header is not required for each blockette 2000.
+\t     1 = Blockette 2000s from multiple SEED data records with
+\t          differing timetags may NOT be repackaged into a single SEED
+\t          data record. Set this bit if the timetag in the SEED Fixed
+\t          Data Header is required to properly interpret the opaque
+\t          data.
+\t[bits 2-3] Opaque blockette fragmentation flags.
+\t     00 = opaque record identified by record number is completely
+\t           contained in this opaque blockette.
+\t     01 = first opaque blockette for record spanning multiple
+\t           blockettes.
+\t     11 = continuation blockette 2...N-1 of record spanning N
+\t           blockettes.
+\t     10 = final blockette for record spanning N blockettes.
+\t[bits 4-5] File blockette information.
+\t     00 = not file oriented.
+\t     01 = first blockette of file.
+\t     10 = continuation of file.
+\t     11 = last blockette of file.
+8 UBYTE:\tNumber of Opaque Header fields. Each opaque header field is a
+\tvariable
+\tlength ascii string, terminated by the character ~.
+9 VAR:\tOpaque Data Header string, which contains the ascii variable length
+\tfields. Each field is terminated by a "~". The definition of the fields
+\tmay be defined by the originator and receiver, but the following are
+\trecommended.
+\tAny of the fields may be empty.
+\t     a    Record Type - name of the type of record (eg "GPS", "GPS MBEN").
+\t     b    Vendor Type - name of equipment vendor (eg "ASHTECH").
+\t     c    Model type - model type of equipment (eg "Z12").
+\t     d   Software Version - software version number (eg "").
+\t     e   Firmware Version - firmware version number (eg "1G0C").
 10 OPAQUE: Opaque Data - bytes of opaque data. Total length of opaque data in
-           bytes is blockette_length - 15 - length (opaque_data_header_string)
+\tbytes is blockette_length - 15 - length (opaque_data_header_string)
 """
diff --git a/mseedpeek/mseedpeek.py b/mseedpeek/mseedpeek.py
old mode 100755
new mode 100644
index 7383dd085068ad74b6251ac77bc5f2db85530ed2..732e39a1eeee872f37acfa3aa7a0718e68ea8797
--- a/mseedpeek/mseedpeek.py
+++ b/mseedpeek/mseedpeek.py
@@ -1,4 +1,5 @@
-#!/usr/bin/env python
+"""
+!/usr/bin/env python
 
 
 # gui for viewing mseed headers
@@ -120,27 +121,37 @@
 #
 # Update versioning scheme
 ################################################################
+#
+# modification
+# version: 2022.2.0.0
+# author: Destiny Kuehn
+#
+# GUI updated to PySide2
+# Pmw dependency dropped
+# Version number updated
+"""
 
-import Pmw
 import sys
+import os
+import itertools
 
 from getopt import getopt
 from glob import glob
 from operator import mod
-from os import *
-from tkinter import *
-from tkinter.filedialog import *
+from PySide2 import QtCore, QtGui, QtWidgets
 
 from mseedpeek.libtrace import *
 from mseedpeek.mseedInfo import *
 
-VERSION = "2022.1.0.0"
-SPACE = " "
+VERSION = "2022.2.0.0"
 
 
 def main():
+    """
+    Main class for mseedpeek
+    """
     # return version number if -# command line option
-    ListInFiles = []
+    file_list = []
     try:
         opts, pargs = getopt(sys.argv[1:], 'f#')
         for flag, arg in opts:
@@ -148,7 +159,7 @@ def main():
                 InFiles = pargs
                 for fl in InFiles:
                     for f in glob(fl):
-                        ListInFiles.append(f)
+                        file_list.append(f)
             if flag == "-#":
                 print(VERSION)
                 sys.exit(0)
@@ -156,638 +167,671 @@ def main():
         print("\nInvalid command line usage\n")
         print("Usage:\nmseedpeek\nmseedpeek -#\nmseedpeek -f file(s)\n")
         sys.exit(1)
-
     print("\n", os.path.basename(sys.argv[0]), VERSION)
 
-    if ListInFiles:
-        mw = MainWindow("mseedpeek %s" % VERSION, ListInFiles)
-        mw.root.geometry("775x545")
-        mw.root.mainloop()
+    app = QtWidgets.QApplication(sys.argv)
+    # mc = main class
+    if file_list:
+        mc = MainWindow("mseedpeek %s" % VERSION, file_list)
     else:
-        mw = MainWindow("mseedpeek %s" % VERSION)
-        mw.root.geometry("775x545")
-        mw.root.mainloop()
+        mc = MainWindow("mseedpeek %s" % VERSION, file_list=None)
+    mc.show()
+    sys.exit(app.exec_())
 
 
-# From "Python and Tkinter Programming", J.E. Grayson, pg.103
-class Command:
-    def __init__(self, func, *args, **kw):
-        self.func = func
-        self.args = args
-        self.kw = kw
+class MainWindow(QtWidgets.QWidget):
+    """
+    Main class for mseedpeek
+    """
+    def __init__(self, title, file_list):
+        """
+        Init everything for startup
+        """
+        super().__init__()
 
-    def __call__(self, *args, **kw):
-        args = self.args + args
-        kw.update(self.kw)
-        self.func(*args, **kw)
+        # window settings
+        self.resize(self.minimumSize())
+        self.setWindowTitle(title)
 
-#
-# Over ride some methods in FileDialog so we only select directories
-# From TraceBuilder.py, auth Steve Azevedo & Lloyd Carothers
-#
+        # init vars
+        self.data_init()
 
+        # set up main window
+        self.create_main_window()
+        self.build_blockettes()
+        self.build_help()
+        self.hide_show("hide")
 
-class DirectoryDialog(FileDialog):
-
-    title = "Directory Select"
-
-    def __init__(self, root):
-        FileDialog.__init__(self, root)
-
-    #   Only allow selection of directories (single click)
-    def files_select_event(self, e):
-        pass
-    #   Double click
-
-    def files_double_event(self, e):
-        pass
-
-    #   Make sure it's a directory and accessable
-    def ok_command(self):
-        ifile = self.get_selection()
-        if not path.isdir(ifile):
-            if not access(ifile, R_OK):
-                self.root.bell()
-                ifile = None
-        self.quit(ifile)
-
-
-class MainWindow:
-    def __init__(self, title='', InFiles=''):
-
-        self.root = Tk()
-        Pmw.initialise(self.root)
-        self.root.title(title)
-
-        self.e_width = 8
-        self.InfoString = StringVar()
-
-        # if files entered on commandline set self.ListInFiles
-        self.ListInFiles = {}
-        if InFiles:
-            self.ListInFiles = InFiles
-        # some standard vars
-        self.DataDirs = StringVar()
-        self.StatSel = StringVar()
-        self.StatSelList = []
-        self.Trace = StringVar()
-        self.SelDir = StringVar()
-        self.SelDir.set("")
-        self.Station = StringVar()
-        self.NetCode = StringVar()
-        self.SampleRate = IntVar()
-        self.Channel = StringVar()
-        self.LocCode = StringVar()
-        self.ByteOrder = StringVar()
-
-        # positional vars
-        self.ByteOffset = IntVar()
-        self.BlkSize = IntVar()
-
-        # dictionaries, lists and misc vars
-        self.BlockNum = IntVar()
-        self.JumpNum = IntVar()
-        self.NumBlocks = IntVar()
-        self.FixedHdrDict = {}
-        self.RateDict = {}
-        self.Blockettes = {}
-        self.BlockettesList = []
-        self.UniqueList = []
-        self.UniqueSelectList = []
-        self.UniqueSelect = StringVar()
-        self.TraceList = []
-        self.TraceListDict = {}
-
-        # file and verbosity vars
-        self.MseedFile = StringVar()
-        self.Blockette = StringVar()
-        self.BlocketteType = StringVar()
-        self.OldMseedFile = StringVar()
-        self.VerbVar = IntVar()
-        self.OldVerbVar = IntVar()
-        self.VerbList = [
-            ["Standard", 0],
-            ["Verbose", 1],
-            ["Very Verbose", 2],
-            ["Unique", 3]
-        ]
-        self.VerbVar.set(0)
-        self.OldVerbVar.set(0)
-
-        # verbosity lists - each builds on the previous
-        self.StandardVars = [
-            ["Station Name:", self.Station],
-            ["Location Code:", self.LocCode],
-            ["Channel:", self.Channel],
-            ["Net Code:", self.NetCode],
-            ["Sps (nominal):", self.SampleRate]
-        ]
+        # if files entered on commandline call build trace
+        if file_list:
+            self.dd_text.setText(':'.join(str(file) for file in file_list))
+            self.build_trace_list(self.dd_text.text())
 
-        self.Year = StringVar()
-        self.Jday = StringVar()
-        self.Hour = StringVar()
-        self.Min = StringVar()
-        self.Sec = StringVar()
-        self.Micro = StringVar()
-        self.VerboseVars = [
-            ["Year:", self.Year],
-            ["Day:", self.Jday],
-            ["Hour:", self.Hour],
-            ["Min:", self.Min],
-            ["Sec:", self.Sec],
-            ["0.0001s:", self.Micro]
-        ]
-        self.VerboseVars = self.StandardVars + self.VerboseVars
-
-        self.SeqNum = StringVar()
-        self.DQual = StringVar()
-        self.Resv = StringVar()
-        self.AddVVVars = [
-            ["Sequence Number:", self.SeqNum],
-            ["Data Hdr/Qaul:", self.DQual],
-            ["Reserved:", self.Resv]
+    def data_init(self):
+        """
+        Init starting vars
+        """
+        self.verb_var = 0
+        self.blk_size = 0
+        self.num_blocks = 0
+        self.dir_list = []
+        self.stat_sel_list = []
+        self.old_mseed_file = ""
+
+        self.standard_vars = [
+            "Station Name:",
+            "Location Code:",
+            "Channel:",
+            "Net Code:",
+            "Sps (nominal):",
         ]
 
-        self.NumSamp = IntVar()
-        self.SampFact = IntVar()
-        self.SampMult = IntVar()
-        self.ActFlag = StringVar()
-        self.IOFlag = StringVar()
-        self.DQFlag = StringVar()
-        self.NumBlockettes = IntVar()
-        self.TimeCorr = IntVar()
-        self.BeginData = IntVar()
-        self.FstBlkett = IntVar()
-        self.VeryVerboseVars = [
-            ["Number Samples:", self.NumSamp],
-            ["Sample Factor:", self.SampFact],
-            ["Sample Multiplier:", self.SampMult],
-            ["Activity Flags:", self.ActFlag],
-            ["I/O & Clk Flags:", self.IOFlag],
-            ["Data Qaul Flags:", self.DQFlag],
-            ["Number Blockettes:", self.NumBlockettes],
-            ["Time Corr:", self.TimeCorr],
-            ["Offet Data:", self.BeginData],
-            ["Offset Blockette:", self.FstBlkett]
+        self.v_vars = [
+            "Year:",
+            "Day:",
+            "Hour:",
+            "Min:",
+            "Sec:",
+            "0.0001s:"
         ]
-        self.VeryVerboseVars = (self.AddVVVars + self.VerboseVars +
-                                self.VeryVerboseVars)
-
-        # set up notebooks
-        self.createMainButtons(self.root)
-        self.createNoteBooks(self.root)
-
-########################################
-    def createMainButtons(self, master):
-        """
-        Build info bar and buttons for root window
-        """
-        self.InfoString_l = Label(master,
-                                  bg="yellow",
-                                  relief="ridge")
-        self.InfoString_l.pack(side='bottom', fill='x')
-
-        self.Button_fm = Frame(master, relief='groove', borderwidth=2)
-        self.Button_fm.pack(side='bottom', pady=5, fill='x')
-
-        self.exit_b = Button(self.Button_fm,
-                             text="Exit",
-                             relief="ridge",
-                             cursor='pirate',
-                             command=Command(self.Exit),
-                             activebackground='red',
-                             activeforeground='black')
-        self.exit_b.pack(side='right', anchor='e')
-
-        self.flush_b = Button(self.Button_fm,
-                              text="Flush Dictionaries",
-                              relief="ridge",
-                              command=Command(self.FlushDict),
-                              activebackground='orange',
-                              activeforeground='black')
-        self.flush_b.pack(side='left', anchor='w')
-
-########################################
-    def createNoteBooks(self, master):
-        """
-        Set up notebooks in root window
-        """
-        nb = Pmw.NoteBook(master)
-        self.buildHdrDisplay(nb)
-        self.buildBlockette(nb)
-        self.buildHelp(nb)
-        nb.pack(padx=5, pady=5, fill='both', expand=1)
-        # If commandline trace input
-        if self.ListInFiles:
-            (self.TraceListDict, NumFiles) = self.IndexFiles(self.ListInFiles)
-            self.UpdateDirList(self.HdrSelect_fm)
-            if NumFiles == 1:
-                vdir = list(self.TraceListDict.keys())[0]
-                self.SelDir.set(vdir)
-                self.UpdateTrcList()
-                trace = self.TraceListDict[vdir][0]
-                self.MseedFile.set(trace)
-                self.ReadHdrs()
-
-            # clear any lingering updates
-            # self.MseedFile.set("")
-            # self.ClearAll(self.VeryVerboseVars)
-            # self.Hdrs_fm.destroy()
-            # self.Blks_fm.destroy()
-
-            self.root.bell()
-            text = "Done. " + str(NumFiles) + " mseed files found."
-            self.addTextInfoBar(self.InfoString_l, text, "green")
-##################################################################
-
-    def buildHdrDisplay(self, master):
-        """
-        Populate HdrDisplay NoteBook
-        """
-        self.FixedHdr_nb = master.add('Trace Headers')
-
-        # data selection frame
-        self.DataDirs_fm = Frame(self.FixedHdr_nb, relief='groove',
-                                 borderwidth=2)
-        self.DataDirs_fm.pack(side='top', pady=5, fill='x')
-        self.DataDirs_l = Label(self.DataDirs_fm, text='Data Directories: ')
-        self.DataDirs_l.pack(side='left')
-        self.DataDirs_e = Entry(self.DataDirs_fm,
-                                selectbackground='yellow',
-                                textvariable=self.DataDirs)
-        self.DataDirs_e.pack(side='left', anchor='w', expand=1, fill='x')
-        self.ClearDataDir_b = Button(self.DataDirs_fm,
-                                     activebackground='orange',
-                                     relief="ridge",
-                                     text="Clear",
-                                     command=Command(self.setValue,
-                                                     self.DataDirs))
-        self.ClearDataDir_b.pack(side='right')
-        self.FindDataDir_b = Button(self.DataDirs_fm,
-                                    activebackground='green',
-                                    relief="ridge",
-                                    text="Find",
-                                    command=Command(self.getPath,
-                                                    self.DataDirs,
-                                                    None))
-        self.FindDataDir_b.pack(side='right')
-        self.DataDirs.set(getcwd())
-
-        self.BuildTrcList_b = Button(self.DataDirs_fm,
-                                     activebackground='green',
-                                     relief="ridge",
-                                     background='lightblue',
-                                     text="Build Trace db",
-                                     command=Command(self.BuildTrcList)
-                                     )
-        self.BuildTrcList_b.pack(side='right')
-
-        # station selection frame and entry box
-        self.StatSel_fm = Frame(self.FixedHdr_nb, relief='groove',
-                                borderwidth=2)
-        self.StatSel_fm.pack(side='top', pady=5, fill='x')
-        self.StatSel_l = Label(self.StatSel_fm,
-                               text='Find only stations (colon separated '
-                                    'list): ')
-        self.StatSel_l.pack(side='left')
-        self.StatSel_e = Entry(self.StatSel_fm,
-                               selectbackground='yellow',
-                               textvariable=self.StatSel)
-        self.StatSel_e.pack(side='left', anchor='w', expand=1, fill='x')
-
-        self.ClearStatSel_b = Button(self.StatSel_fm,
-                                     activebackground='orange',
-                                     relief="ridge",
-                                     text="Clear",
-                                     command=Command(self.setValue,
-                                                     self.StatSel))
-        self.ClearStatSel_b.pack(side='right')
-
-        # verbosity buttons
-        self.RadButtons_fm = Frame(
-            self.FixedHdr_nb, relief='groove', borderwidth=2)
-        self.RadButtons_fm.pack(side='top', pady=5, fill='x')
-
-        for text, value in self.VerbList:
-            Radiobutton(self.RadButtons_fm,
-                        text=text,
-                        value=value,
-                        command=Command(self.ReadHdrs, None),
-                        variable=self.VerbVar).pack(side='left', anchor='e')
-
-        Entry(self.RadButtons_fm,
-              width=self.e_width,
-              background='yellow',
-              relief='ridge',
-              textvariable=self.ByteOrder).pack(side='right', anchor='e')
-        Label(self.RadButtons_fm,
-              text="Header Endianess").pack(side='right', anchor='e')
-
-        # set up frames that get filled when traces are selected
-        self.HdrScale_fm = Frame(self.FixedHdr_nb, borderwidth=2)
-        self.HdrScale_fm.pack(side='bottom', padx=5, pady=5, fill='x')
-
-        self.HdrSelect_fm = Frame(
-            self.FixedHdr_nb, relief='groove', borderwidth=2)
-        self.HdrSelect_fm.pack(side='top', pady=5, fill='x')
-
-        self.Hdrs_fm = Frame(self.FixedHdr_nb, relief='groove', borderwidth=2)
-        self.Hdrs_fm.pack(side='top', pady=5, fill='x')
-
-##################################################################
-    def buildBlockette(self, master):
-        """
-        Populate Blockettes NoteBook
-        """
-        self.Blockette_nb = master.add('Blockettes')
-
-        # these get filled when traces are selected
-        self.BlkScale_fm = Frame(self.Blockette_nb, borderwidth=2)
-        self.BlkScale_fm.pack(side='bottom', padx=5, pady=5, fill='x')
-
-        self.BlksSelect_fm = Frame(
-            self.Blockette_nb, relief='groove', borderwidth=2)
-        self.BlksSelect_fm.pack(side='top', pady=5, fill='x')
-
-        self.BlksTitle_fm = Frame(
-            self.Blockette_nb, relief='groove', borderwidth=2)
-        self.BlksTitle_fm.pack(side='top', pady=5, fill='x')
-
-        self.Blks_fm = Frame(self.Blockette_nb, relief='groove', borderwidth=2)
-        self.Blks_fm.pack(side='top', pady=5, fill='x')
-
-##################################################################
-    def buildBlkInfo(self, blktype, master):
-        """
-        Provide information window for blockettes
-        """
 
-        try:
-            self.tl_State = self.BlkInfo_tl.winfo_exists()
-        except Exception:
-            self.tl_State = 0
+        self.first_vv_vars = [
+            "Sequence Number:",
+            "Data Hdr/Qaul:",
+            "Reserved:"
+        ]
 
-        if self.tl_State == 1:
-            self.BlkInfo_tl.tkraise()
-            # display info on selected block
-            self.BlocketteType.set(blktype)
-            self.displayBlkInfo(blktype)
+        self.vv_vars = [
+            "Number Samples:",
+            "Sample Factor:",
+            "Sample Multiplier:",
+            "Activity Flags:",
+            "I/O & Clk Flags:",
+            "Data Qaul Flags:",
+            "Number Blockettes:",
+            "Time Corr:",
+            "Offet Data:",
+            "Offset Blockette:"
+        ]
 
-        else:
-            self.BlkInfo_tl = Toplevel(master)
-            self.BlkInfo_tl.title("Blockette Information")
-
-            self.BlkInfoSelect_fm = Frame(
-                self.BlkInfo_tl, relief='groove', borderwidth=2)
-            self.BlkInfoSelect_fm.pack(side='top', pady=5, fill='x')
-
-            List = ["Fixed Header", "Data Fields", "BTime", 100, 200, 201,
-                    300, 310, 320, 390, 395, 400, 405, 500, 1000, 1001, 2000]
-            Label(self.BlkInfoSelect_fm, text="Blockette:").grid(row=1,
-                                                                 sticky=W)
-            Pmw.ComboBox(self.BlkInfoSelect_fm,
-                         history=0,
-                         entry_width=self.e_width + 15,
-                         entry_textvariable=self.BlocketteType,
-                         selectioncommand=Command(self.displayBlkInfo),
-                         scrolledlist_items=(List)
-                         ).grid(row=1, column=1, sticky=W)
-
-            self.ExitBlkInfo_b = Button(self.BlkInfo_tl,
-                                        activebackground='red',
-                                        cursor='pirate',
-                                        background='lightblue',
-                                        text="Done",
-                                        command=Command(self.killWindow,
-                                                        self.BlkInfo_tl))
-            self.ExitBlkInfo_b.pack(side='bottom', fill='x', expand=1)
-
-            self.BlkInfoText = Pmw.ScrolledText(self.BlkInfo_tl,
-                                                borderframe=1)
-            self.BlkInfoText.pack(side='bottom', fill='both', expand=1)
-
-            # display info on selected block
-            self.BlocketteType.set(blktype)
-            self.displayBlkInfo(blktype)
-
-##################################################################
-
-    def buildUniqueSelectBox(self, master):
-        """
-        In unique view allows user to select block to jump to
-        """
-        if "*" not in self.UniqueSelectList:
-            self.UniqueSelectList.append("*")
-            self.UniqueSelectList.sort()
-        Label(master, text="Select Keys:").pack(side='top', anchor='w')
-
-        Pmw.ComboBox(master,
-                     history=1,
-                     entry_width=self.e_width + 15,
-                     entry_textvariable=self.UniqueSelect,
-                     selectioncommand=Command(self.SelectKeys),
-                     scrolledlist_items=self.UniqueSelectList
-                     ).pack(side='top', anchor='w')
-        self.UniqueSelect.set("*")
-
-##################################################################
-
-    def buildJumpBox(self, master, numblocks=0):
-        """
-        In unique view allows user to select block to jump to
-        """
-        if self.VerbVar.get() == 3:
-            Label(master, text="Jump To Block #:").pack(side='top', anchor='w')
-        else:
-            Label(master, text="Jump To Block #:").pack(side='top', anchor='n')
-
-        jumpbox_cb = Pmw.ComboBox(master,
-                                  history=1,
-                                  entry_width=self.e_width,
-                                  entry_textvariable=self.JumpNum,
-                                  selectioncommand=Command(self.Jump),
-                                  scrolledlist_items=(list(range(numblocks))))
-        if self.VerbVar.get() == 3:
-            jumpbox_cb.pack(side='top', anchor='w')
-        else:
-            jumpbox_cb.pack(side='top', anchor='n')
-
-##################################################################
-    def Jump(self, value):
-        """
-        Fills hdr values based on block selected in JumpBox
-        """
-        if int(value) < 0:
-            value = 0
-        if int(value) > (self.NumBlocks.get() - 1):
-            value = self.NumBlocks.get() - 1
-        self.BlockNum.set(value)
-        if self.VerbVar.get() == 3:
-            self.VerbVar.set(2)
-        self.ReadHdrs()
-        self.FillHdr(value)
-
-##################################################################
-    def SelectKeys(self, value):
-        """
-        Fills hdr values based on block selected in JumpBox
-        """
-        self.UniqueText.clear()
-        self.UniqueText.tag_config("list", foreground="blue")
-        self.UniqueText.tag_config("odd", backgroun="lightblue")
-
-        self.UniqueText.insert(
-            "end", "Block\tStat\tChan\tLoc\tNet\tRate\n", "list")
-
-        selectkey = self.UniqueSelect.get()
-        c = 0
-        for key in self.UniqueList:
-            testkey = key.split(":")[1:].join(":")
-            if selectkey != "*":
-                if selectkey != testkey:
-                    continue
-            for var in key.split(":"):
-                if c:
-                    self.UniqueText.insert("end", """%s\t""" % var, "odd")
-                else:
-                    self.UniqueText.insert("end", """%s\t""" % var)
-            self.UniqueText.insert("end", """\n""")
-            if c:
-                c = 0
+    def create_main_window(self):
+        """
+        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.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.setStyleSheet("background-color:yellow")
+        self.infobar.setReadOnly(True)
+        self.infobar.setAlignment(QtGui.Qt.AlignCenter)
+
+        # main window layout
+        self.window_layout = QtWidgets.QVBoxLayout()
+        self.setLayout(self.window_layout)
+
+        # add tab widget and info bar to main window layout
+        self.window_layout.addWidget(self.tabwidget)
+        self.build_trace_headers()  # build widgets for Trace Headers tab
+        self.window_layout.addWidget(self.infobar)
+        self.window_layout.setMargin(0)  # set spacing
+
+    def build_help(self):
+        """
+        Build Help tab
+        """
+        # init tab layout
+        layout = QtWidgets.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)
+
+        help_text = QtWidgets.QTextEdit()
+        layout.addWidget(help_text)
+        help_text.setVerticalScrollBarPolicy(
+            QtCore.Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
+        help_text.setAcceptRichText(False)
+
+        # Name
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("NAME")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        mseedpeek - GUI for displaying mseed file headers.\n\n""")
+
+        # Version
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("VERSION")
+        help_text.setTextColor(Black)
+        text = "\n" + SPACE + str(VERSION) + "\n\n"
+        help_text.insertPlainText("""
+        %s\n\n""" % VERSION)
+
+        # Synopsis
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("SYNOPSIS")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        mseedpeek
+        mseedpeek -#
+        mseedpeek -f file_names\n\n""")
+
+        # Options
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("OPTIONS")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        -# returns version number
+        -f file_name(s) - accepts command line input for files to inspect (wildcards accepted)\n\n""")  # noqa: E501
+
+        # Description
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("DESCRIPTION")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        mseedpeek has three notebooks: """)
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("[Trace Headers][Blockettes]")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(", and ")
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("[Help]")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(".")
+        help_text.insertPlainText("""
+        The """)
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("[Trace Headers]")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(
+            """ notebook displays various the fixed header in""" +
+            """four levels of verbosity.
+        The """)
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("[Blockettes]")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""" notebook displays blockette information.\n""")  # noqa: E501)
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("""
+        [Trace Headers]""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        General:
+            >> specify/load directories for building trace db
+            >> select trace for header display
+            >> select level of verbosity for display
+
+        Buttons:""")
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Build Trace db>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Searchs the directories listed in the "Data-
+\tDirectories" entry box (a colon separated list) and builds a list of mseed
+\tfiles found indexing them on unique values of <dir>:<file_name>. It then
+\tcreates a dropdown menu with entries for each data directory found.""")  # noqa: E501
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Find>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Launches a file browser allowing the user to
+\tadd directories to the "Data Directories" entry box. Double clicking selects
+\tthe new directory.""")  # noqa: E501
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Clear>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Clears the \"Data Directories\"""" +
+                                  """ entry box.\n""")
+        help_text.insertPlainText("""
+                By selecting a data directory a dropdown menu is created for
+                trace selection. Selecting a trace from this menu will create
+                a header display.
+
+        Radio Buttons:""")
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Standard>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": This is the default. It displays Station, Channel, Location Code,
+\tNet Code, and the nominal Sample Rate.""")  # noqa: E501
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Verbose>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Standard display plus time of block.""")
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Very Verbose>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Verbose plus the rest of the mseed fixed header fields.""")  # noqa: E501
+        help_text.setTextColor(Red)
+        help_text.insertPlainText(("""
+            <Unique>"""))
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": This displays changes in the """ +
+                                  """Standard fields within a mseed file.""")
+        help_text.insertPlainText("""
+
+        Slider Scale and """)
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("<Jump To Block #>")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""" Button:
+                At the bottom of the page is a scale bar showing block numbers. Sliding
+                the bar scans through all fixed headers for the selected mseed file.\n""")  # noqa: E501
+        help_text.setTextColor(Red)
+
+        help_text.insertPlainText("""
+        <Flush Dictionaries>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(
+            """: Clears cached trace header data. As you view mseed header""" +
+            """s,\n\tmseedpeek maintains a cache of previously viewed""" +
+            """ headers for\n\tquick, future access. If you wish to """ +
+            """ monitor changes to mseed files you\n\tneed to flush the """ +
+            """dictionaries so that the new changes will be displayed.""")
+        help_text.setTextColor(Green)
+        help_text.insertPlainText("""
+
+        [Blockettes]""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""":
+                The 'Blockettes' notebook displays all blockettes found for the select
+                mseed file in a dropdown menu. By selecting a blockette from the dropdown
+                menu, the contents of the blockette will be displayed.""")  # noqa: E501
+        help_text.setTextColor(Red)
+        help_text.insertPlainText("""
+            <Blockette Info>""")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText(""": Displays a pop-up window with a definition of the SEED Blockette.
+
+                At the bottom of the page is a scale bar showing block numbers and
+                Sliding the bar scans through all blockettes for the selected mseed file.\n""")  # noqa: E501
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("\nKEYWORDS")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        mseed; header information\n""")
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("\nSEE ALSO")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        SEED manual (pdf), fixhdr, mseedhdr\n""")
+        help_text.setTextColor(Blue)
+        help_text.insertPlainText("\nAUTHOR")
+        help_text.setTextColor(Black)
+        help_text.insertPlainText("""
+        Bruce Beaudoin <bruce@passcal.nmt.edu>
+        """)
+
+    def build_blockettes(self):
+        """
+        Create initial widgets for Blockettes tab
+        """
+        # data boxes
+        self.blktype_box = QtWidgets.QGroupBox()
+        self.blktype_box.setObjectName("Box1")
+
+        self.blkinfo_box = QtWidgets.QGroupBox()
+        self.blkinfo_box.setLayout(QtWidgets.QHBoxLayout())
+        self.blkinfo_box.setObjectName("Box2")
+
+        self.blk_vars_box = QtWidgets.QGroupBox()
+        self.blk_vars_box.setLayout(QtWidgets.QVBoxLayout())
+        self.blk_vars_box.setObjectName("Box3")
+
+        # widgets
+        blk_label = QtWidgets.QLabel("Blockette:")
+        blk_menu = QtWidgets.QComboBox()
+        blk_menu.setObjectName("type_menu")
+
+        self.blkinfo_btn = QtWidgets.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.addWidget(blk_label)
+        l1.addWidget(blk_menu)
+        l1.addStretch()
+
+        # add to tab's main layout
+        main_layout = QtWidgets.QVBoxLayout()
+        main_layout.addWidget(self.blktype_box)
+        main_layout.addWidget(self.blkinfo_box)
+        main_layout.addWidget(self.blk_vars_box)
+        main_layout.addStretch()
+        self.blockettes_tab.setLayout(main_layout)
+
+    def build_trace_headers(self):
+        """
+        Build widgets for Trace Headers tab
+        """
+        # tab layout
+        self.trace_headers_layout = QtWidgets.QVBoxLayout(
+            self.trace_headers_tab)
+
+        # groupboxes for data fields
+        self.datadir_box = QtWidgets.QGroupBox()
+        self.datadir_box.setCheckable(False)
+        sp = self.datadir_box.sizePolicy()
+        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        self.datadir_box.setSizePolicy(sp)
+
+        self.stations_box = QtWidgets.QGroupBox()
+        self.stations_box.setCheckable(False)
+        sp = self.stations_box.sizePolicy()
+        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        self.stations_box.setSizePolicy(sp)
+
+        self.radio_box = QtWidgets.QGroupBox()
+        self.radio_box.setCheckable(False)
+        sp = self.radio_box.sizePolicy()
+        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        self.radio_box.setSizePolicy(sp)
+
+        self.dir_trace_box = QtWidgets.QGroupBox()
+        self.dir_trace_box.setCheckable(False)
+        sp = self.dir_trace_box.sizePolicy()
+        sp.setVerticalPolicy(QtWidgets.QSizePolicy.Fixed)
+        self.dir_trace_box.setSizePolicy(sp)
+
+        self.flush_exit_box = QtWidgets.QGroupBox()
+        self.flush_exit_box.setCheckable(False)
+        sp = self.flush_exit_box.sizePolicy()
+        sp.setVerticalPolicy(QtWidgets.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)
+        flush_exit_layout.setMargin(0)
+        flush_exit_layout.setSpacing(450)
+        dir_trace_layout = QtWidgets.QFormLayout(self.dir_trace_box)
+
+        # fill flush / exit groubpx
+        self.exit_btn = QtWidgets.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.clicked.connect(self.flush_dict)
+        self.flush_btn.setStyleSheet(
+            "QPushButton::hover{background-color:orange;}")
+        flush_exit_layout.addWidget(self.flush_btn)
+        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.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.clicked.connect(self.clicked_find)
+        self.clear_btn = QtWidgets.QPushButton("Clear")
+        self.clear_btn.clicked.connect(lambda: self.dd_text.clear())
+        datadir_layout.addWidget(self.dd_label)
+        datadir_layout.addWidget(self.dd_text)
+        datadir_layout.addWidget(self.build_trace_btn)
+        datadir_layout.addWidget(self.find_btn)
+        datadir_layout.addWidget(self.clear_btn)
+
+        # stations widgets
+        self.stations_label = QtWidgets.QLabel(
+            "Find only stations (colon separated list):")
+        self.stations_text = QtWidgets.QLineEdit()
+        self.stations_clear_btn = QtWidgets.QPushButton("Clear")
+        self.stations_clear_btn.clicked.connect(
+            lambda: self.stations_text.clear())
+        stations_layout.addWidget(self.stations_label)
+        stations_layout.addWidget(self.stations_text)
+        stations_layout.addWidget(self.stations_clear_btn)
+
+        # fill stations groupbox
+        self.standard_rbtn = QtWidgets.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.setChecked(False)
+        self.verbose_rbtn.clicked.connect(lambda: self.clicked_scan_type(1))
+        self.vv_rbtn = QtWidgets.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.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)
+
+        self.header_endianess_label = QtWidgets.QLabel("Header Endianess")
+        self.header_endianess_text = QtWidgets.QLineEdit()
+        self.header_endianess_text.setStyleSheet("background-color:yellow")
+
+        rbuttons_layout.addWidget(self.standard_rbtn)
+        rbuttons_layout.addWidget(self.verbose_rbtn)
+        rbuttons_layout.addWidget(self.vv_rbtn)
+        rbuttons_layout.addWidget(self.unique_rbtn)
+        rbuttons_layout.addItem(spacerItem)
+        rbuttons_layout.addWidget(self.header_endianess_label)
+        rbuttons_layout.addWidget(self.header_endianess_text)
+
+        # fill dir / trace groupbox
+        self.dir_label = QtWidgets.QLabel("Directory:")
+        self.dir_menu = QtWidgets.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_menu.activated.connect(
+            lambda: self.select_dir_trace("trace"))
+        dir_trace_layout.addRow(self.dir_label, self.dir_menu)
+        dir_trace_layout.addRow(self.trace_label, self.trace_menu)
+        self.dir_trace_box.hide()
+        self.trace_label.hide()
+        self.trace_menu.hide()
+
+        # stacked widget (jump box)
+        self.standard_box = StandardBox(self.standard_vars)
+        self.verbose_box = VerboseBox(self.standard_vars, self.v_vars)
+        self.vv_box = VVBox(
+            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.setStyleSheet("QGroupBox{border:0;}")
+        self.jump_box.setMinimumSize(0, 200)
+        self.jump_box.addWidget(self.standard_box)
+        self.jump_box.addWidget(self.verbose_box)
+        self.jump_box.addWidget(self.vv_box)
+        self.jump_box.addWidget(self.unique_box)
+
+        # slider
+        self.slider_box = SliderBox()
+        self.slider_box.slider.valueChanged.connect(
+            lambda: self.update_slider("slider"))
+        self.slider_box.jump_menu.currentTextChanged.connect(
+            lambda: self.update_slider("menu"))
+        self.unique_box.unique_jump.currentIndexChanged.connect(
+            lambda: self.update_slider("unique"))
+
+        # add everything to trace headers layout
+        self.trace_headers_layout.addWidget(self.datadir_box)
+        self.trace_headers_layout.addWidget(self.stations_box)
+        self.trace_headers_layout.addWidget(self.radio_box)
+        self.trace_headers_layout.addWidget(self.dir_trace_box)
+        self.trace_headers_layout.addWidget(self.jump_box)
+        self.trace_headers_layout.addWidget(self.slider_box)
+
+        # set data directory widget to display cwd
+        default_dir = os.getcwd()
+        self.dd_text.setText(default_dir)
+
+    def clicked_scan_type(self, index):
+        """
+        Show data related to selected scan type
+        """
+        self.verb_var = index
+        self.jump_box.setCurrentIndex(self.verb_var)
+        if self.trace_menu.isHidden():
+            self.jump_box.hide()
+        self.maybe_read_hdrs()
+
+    def clicked_find(self):
+        """
+        Open file dialogue to search for mseed files
+        """
+        search = QtWidgets.QFileDialog.getExistingDirectory()
+        directories = self.dd_text.text()
+
+        if search:
+            if directories:
+                self.dd_text.insert(":" + search)
             else:
-                c += 1
-##########################################
+                self.dd_text.insert(search)
 
-    def UpdateDirList(self, fm):
+    def clicked_build_trace(self):
         """
-        build dropdown for selecting data filled directories
+        Check directories have been provided before building trace list
         """
-        self.DirList = list(self.TraceListDict.keys())
-
-        self.DirList.sort()
-        self.SelDir.set("")
-        if not self.DirList:
-            self.root.bell()
-            self.addTextInfoBar(self.InfoString_l, "No Data Found", 'orange')
-
+        directories = self.dd_text.text()
+        if directories:
+            self.dir_list.clear()
+            self.build_trace_list(directories)
+            self.hide_show("hide")
         else:
-            Label(fm, text="Directory:").grid(row=0, sticky=W)
-            Pmw.ComboBox(fm,
-                         history=0,
-                         entry_width=self.e_width + 40,
-                         entry_textvariable=self.SelDir,
-                         selectioncommand=Command(self.UpdateTrcList),
-                         scrolledlist_items=(self.DirList)
-                         ).grid(row=0, column=1, sticky=W)
-
-##################################################################
-
-    def UpdateTrcList(self, name=""):
-        """
-        create dropdown file list based on selected directory
-        """
-        self.MseedFile.set("")
-        self.TraceList = self.TraceListDict[self.SelDir.get()]
-        self.TraceList.sort()
-        self.ClearAll(self.VeryVerboseVars)
-        if not self.TraceList:
-            self.root.bell()
-            self.addTextInfoBar(self.InfoString_l, "No Data Found", 'orange')
-
-        Label(self.HdrSelect_fm, text="Trace:").grid(row=1, sticky=W)
-        Pmw.ComboBox(self.HdrSelect_fm,
-                     history=0,
-                     entry_width=self.e_width + 40,
-                     entry_textvariable=self.MseedFile,
-                     selectioncommand=Command(self.ReadHdrs),
-                     scrolledlist_items=(self.TraceList)
-                     ).grid(row=1, column=1, sticky=W)
-
-#######################################
-
-    def BuildTrcList(self):
-        """
-        build a trace list using DataDirList as base directories
-        """
-        self.addTextInfoBar(self.InfoString_l)
-
-        self.ByteOrder.set("")
-        DataDirList = []
-        self.StatSelList = []
-        for idir in self.DataDirs.get().split(":"):
-            dirlist = glob(idir)
-            for newdir in dirlist:
-                if not path.isdir(newdir):
-                    err = "***WARNING*** Directory " + newdir + " not found."
-                    self.root.bell()
-                    self.addTextInfoBar(self.InfoString_l, err, "red")
-                    return
-                DataDirList.append(newdir)
-
-        # split station select list if it exists
-        if self.StatSel.get() == "*" or self.StatSel.get() == "":
-            pass
+            error_msg = QtWidgets.QMessageBox()
+            error_msg.setIcon(QtWidgets.QMessageBox.Critical)
+            error_msg.setText("Error")
+            error_msg.setInformativeText("No directories listed.")
+            error_msg.setWindowTitle("Error")
+            error_msg.exec_()
+
+    def build_trace_list(self, directories):
+        """
+        Build trace list from given directories
+        """
+        for dir in str(directories).split(":"):
+            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)
+
+        stat_sel = self.stations_text.text()
+        if stat_sel:
+            for stat in str(stat_sel).split(":"):
+                statsel = stat.strip()
+                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")
+
+    def hide_show(self, action):
+        """
+        Hide/show widgets between scans
+        """
+        if action == "hide":
+            widget = self.jump_box.currentWidget()
+            for child in widget.findChildren(QtWidgets.QWidget):
+                if not child.isHidden():
+                    child.hide()
+            for child in self.slider_box.findChildren(QtWidgets.QWidget):
+                if not child.isHidden():
+                    child.hide()
+            widgets = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+            for widget in widgets:
+                if not widget.isHidden():
+                    widget.hide()
+            self.trace_label.hide()
+            self.trace_menu.hide()
+            self.blktype_box.hide()
+            self.blkinfo_box.hide()
+            self.blk_vars_box.hide()
         else:
-            for statsel in str(self.StatSel.get()).split(":"):
-                statsel = statsel.strip()
-                self.StatSelList.append(statsel)
-
-        (self.TraceListDict, NumFiles, errfiles) = self.FindTrace(DataDirList)
-        self.UpdateDirList(self.HdrSelect_fm)
-
-        # clear any lingering updates
-        self.MseedFile.set("")
-        self.ClearAll(self.VeryVerboseVars)
-        self.Hdrs_fm.destroy()
-        self.Blks_fm.destroy()
-
-        self.root.bell()
-        text = "Done. " + str(NumFiles) + " mseed files found. ***" \
-            + str(errfiles) + " files with errors."
-        self.addTextInfoBar(self.InfoString_l, text, "green")
-
-#######################################
-
-    def FindTrace(self, DataDir):
+            widget = self.jump_box.currentWidget()
+            for child in widget.findChildren(QtWidgets.QWidget):
+                if child.isHidden():
+                    child.show()
+            for child in self.slider_box.findChildren(QtWidgets.QWidget):
+                if child.isHidden():
+                    child.show()
+            widgets = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+            for widget in widgets:
+                if widget.isHidden():
+                    widget.show()
+            self.blktype_box.show()
+            self.blkinfo_box.show()
+            self.blk_vars_box.show()
+
+    def find_trace(self):
         """
         based on traverse routine in "python standard library", Lundh pg 34
         """
         stack = []
-        for k in range(len(DataDir)):
-            stack.append(DataDir[k])
+        for k in range(len(self.dir_list)):
+            stack.append(self.dir_list[k])
+
         file_list = {}
-        NumMseedFiles = 0
-        rwError = 0
+        num_mseed_files = 0
+        rw_error = 0
         cnt = 1
         while stack:
             directory = stack.pop()
-            if not path.isdir(directory):
-                print("\n***WARNING*** Directory \"%s\" not found.\n" %
-                      directory)
-                continue
-
             try:
-                listfiles = listdir(directory)
+                listfiles = os.listdir(directory)
             except Exception as e:
                 print("Directory Read Error: %s" % e)
 
-            for ifile in listfiles:
+            for file in listfiles:
                 if mod(cnt, 25):
                     pass
                 else:
                     self.wait("Examining File: ", cnt)
-                fullname = path.join(directory, ifile)
-                if path.isfile(fullname):
+                fullname = os.path.join(directory, file)
+                if os.path.isfile(fullname):
                     if not os.access(fullname, 6):
                         err = "ERROR: Read/Write permission denied."
                         err1 = "\t File: " + fullname
                         print(err)
                         print(err1)
-                        rwError += 1
+                        rw_error += 1
                         continue
                     try:
                         msfile = Mseed(fullname)
                         if msfile.isMseed():
+                            num_mseed_files += 1
+                            if directory in file_list:
+                                file_list[directory].append(file)
+                            else:
+                                file_list[directory] = []
+                                file_list[directory].append(file)
                             try:
                                 # simple test to determine if correct size file
                                 filesize = msfile.filesize
@@ -800,146 +844,106 @@ class MainWindow:
                                             str(blksize) + "). \n\t File: " +
                                             fullname)
                                     print(warn)
-                                    rwError += 1
+                                    rw_error += 1
                                     continue
                             except Exception:
                                 err = ("ERROR: Cannot determine file and "
                                        "block sizes. \n\t File:" + fullname)
                                 print(err)
-                                rwError += 1
+                                rw_error += 1
                                 continue
-                            NumMseedFiles += 1
-                            stat = msfile.FH.Stat.strip()
-
-                            if self.StatSelList:
-                                if stat not in self.StatSelList:
-                                    continue
-
-                            if directory in file_list:
-                                file_list[directory].append(ifile)
-                            else:
-                                file_list[directory] = []
-                                file_list[directory].append(ifile)
                         msfile.close()
                     except Exception as e:
-                        rwError += 1
+                        rw_error += 1
                         print("File Open Error: %s" % fullname, e)
                 cnt += 1
-
-                if (path.isdir(fullname) or (path.islink(fullname) and
-                                             not path.isfile(fullname))):
+                if (os.path.isdir(fullname) or (os.path.islink(fullname) and
+                                                not os.path.isfile(fullname))):
                     stack.append(fullname)
 
-        return file_list, NumMseedFiles, rwError
-
-#######################################
+        self.dir_trace_dict = {}
+        self.dir_trace_dict = file_list
+        self.update_dir_list()
+        return num_mseed_files, rw_error
 
-    def IndexFiles(self, ListInFiles):
+    def select_dir_trace(self, menu):
         """
-        based on traverse routine in "python standard library", Lundh pg 34
+        If directory menu selected, show trace menu
+        If trace menu selected, read hdrs
         """
-        file_list = {}
-        cnt = 1
-        NumMseedFiles = 0
-        directory = getcwd()
-        self.DataDirs.set(getcwd())
-        for ifile in ListInFiles:
-            if mod(cnt, 25):
-                pass
+        if menu == "dir":
+            dir = self.dir_menu.currentText()
+            if dir:
+                self.update_trace_list(dir)
+                self.trace_label.show()
+                self.trace_menu.show()
             else:
-                self.wait("Examining File: ", cnt)
-            fullname = path.join(directory, ifile)
-            if path.isfile(fullname):
-                try:
-                    newfile = Mseed(fullname)
-                    if newfile.isMseed():
-                        stat = newfile.FH.Stat.strip()
-                        if self.StatSelList:
-                            if stat not in self.StatSelList:
-                                continue
-
-                        NumMseedFiles += 1
-                        if directory in file_list:
-                            file_list[directory].append(ifile)
-                        else:
-                            file_list[directory] = []
-                            file_list[directory].append(ifile)
-                    newfile.close()
-                except Exception as e:
-                    print("File Open Error: %s" % e)
-            cnt += 1
-
-        return file_list, NumMseedFiles
-#######################################
+                self.trace_label.hide()
+                self.trace_menu.hide()
+        else:
+            trace = self.trace_menu.currentText()
+            if trace:
+                dir = self.dir_menu.currentText()
+                self.next_scan(dir, trace)
 
-    def ReadHdrs(self, ifile=""):
+    def next_scan(self, dir, trace):
         """
-        read headers of a given file build dictionary for quick access
+        Prepare for next scan, i.e. reset/clear/hide some widgets
         """
+        self.clear_slider()
+        self.read_hdrs(dir, trace, 0)
+        self.fill_slider()
+        self.hide_show("show")
+        self.header_endianess_text.setText(self.byte_order)
+        self.jump_box.setCurrentIndex(self.verb_var)
+        self.update_infobar("", "yellow")
 
-        # this is a bit of a hack, but I am unable to pass self.MseedFile.get()
-        # from buildHdrDisplay:Radiobutton:command
-        if not ifile:
-            ifile = self.MseedFile.get()
-            # if someone changes VerbVar before selecting file do nothing
-            if not ifile:
-                return
+        # hack, widget was behaving weird for some reason
+        # after showing it
+        self.slider_box.jump_menu.setEnabled(False)
+        self.slider_box.jump_menu.setEnabled(True)
 
-        FileAbsolutePath = path.join(self.SelDir.get(), ifile)
+    def read_hdrs(self, dir, trace, blk):
+        """
+        Read headers of a given file, build dictionary for quick access
+        """
+        file_absolute_path = os.path.join(dir, trace)
 
-        # create object (we've already tested all files for mseed)
-        # and get some base info
-        rdfile = Mseed(FileAbsolutePath)
+        rdfile = Mseed(file_absolute_path)
         if rdfile.isMseed():
             try:
                 filesize = rdfile.filesize
                 blksize = rdfile.blksize
             except Exception:
-                self.ClearAll(self.VeryVerboseVars)
-                self.Hdrs_fm.destroy()
-                self.HdrScale_fm.destroy()
-                self.BlkScale_fm.destroy()
-                self.BlksSelect_fm.destroy()
-                self.BlksTitle_fm.destroy()
-                self.Blks_fm.destroy()
-                err = "Cannot determine file and block sizes. File: " + ifile
-                self.addTextInfoBar(self.InfoString_l, err, 'red')
+                err = "Cannot determine file and block sizes. File: " + trace
+                self.update_infobar(err, "red")
                 rdfile.close()
                 return
-        else:
-            return
 
-        # initialize variables, lists, and dicts for later
-        self.BlkSize.set(blksize)
-        (numblocks, self.odd_size) = divmod(filesize, blksize)
-        self.NumBlocks.set(numblocks)
-        verbose = self.VerbVar.get()
-        if not self.OldMseedFile.get() or \
-           self.OldMseedFile.get() != FileAbsolutePath:
-            self.FixedHdrDict = {}
-            self.RateDict = {}
-            self.Blockettes = {}
-            self.keyList = []
-            self.UniqueList = []
-            self.UniqueSelectList = []
+            self.blk_size = blksize
+            (numblocks, self.odd_size) = divmod(filesize, blksize)
+            self.num_blocks = numblocks
+            verbose = self.verb_var
+
+            self.fixed_hdr_dict = {}
+            self.rate_dict = {}
+            self.blockettes_dict = {}
+            self.key_list = []
+            self.unique_list = []
+            self.unique_select_list = []
             n = 0
 
             # now build a dictionary of all fixed header info keyed to block
             # number looping over total number of blocks in files
             lastkey = -1
             while n < numblocks:
-
                 hdrs = rdfile.fixedhdr(n * blksize)
-                self.FixedHdrDict[n] = hdrs
-                # fact=hdrs[2][1]
-                # mult=hdrs[2][2]
-                # self.RateDict[n]=rdfile.calcrate(mult,fact)
-                self.RateDict[n] = rdfile.rate
+                self.fixed_hdr_dict[n] = hdrs
+                self.rate_dict[n] = rdfile.rate
 
                 # if unique selected build unique list
-                # if verbose == 3:
                 (SeqNum, DHQual, res, Stat, Loc, Chan, Net) = hdrs[0]
-                Rate = str(self.RateDict[n])
+                Rate = str(self.rate_dict[n])
                 Stat = Stat.strip().decode()
                 Chan = Chan.strip().decode()
                 Loc = Loc.strip().decode()
@@ -949,27 +953,25 @@ class MainWindow:
                 if key == lastkey:
                     pass
                 else:
-                    # self.keyList.append(key)
                     lastkey = key
-                    if key not in self.UniqueSelectList:
-                        self.UniqueSelectList.append(key)
-                    key = str(n) + ":" + key
-                    self.UniqueList.append(key)
+                    if key not in self.unique_select_list:
+                        self.unique_select_list.append(key)
+                    key = str(blk) + ":" + key
+                    self.unique_list.append(key)
 
                 # build Blockette dictionary keyed to block number
                 numblk = hdrs[3][3]
                 addseek = hdrs[3][6]
                 b = 0
+
                 # loop over number of blockettes following fixed header at
                 # block n
                 while b < numblk:
                     seekval = (n * blksize) + addseek
                     (blktype, next) = rdfile.typenxt(seekval)
                     # builds command to read specific blockette
-                    # getBlkCmd="blklist=rdfile.blk" + str(blktype) + \
-                    #           "(" + str(seekval) + ")"
-                    blklist = eval("rdfile.blk" + str(blktype) +
-                                   "(" + str(seekval) + ")")
+                    blklist = eval("rdfile.blk" + str(blktype) + "\
+                                   (" + str(seekval) + ")")
                     if blklist:
                         # handle awkward lists w/ muli-byte reserve, array, or
                         # calib fields
@@ -991,581 +993,598 @@ class MainWindow:
                             tmplist.append(tup)
                             blklist = tmplist
                         # build Blockettes
-                        if n in self.Blockettes:
-                            self.Blockettes[n].append(blklist)
+                        if n in self.blockettes_dict:
+                            self.blockettes_dict[n].append(blklist)
                         else:
-                            self.Blockettes[n] = []
-                            self.Blockettes[n].append(blklist)
-
+                            self.blockettes_dict[n] = []
+                            self.blockettes_dict[n].append(blklist)
                     addseek = next
                     b += 1
                 n += 1
-        # get endianess before closing
-        self.ByteOrder.set(rdfile.byteorder)
-        rdfile.close()
+            # get endianess before closing
+            self.byte_order = rdfile.byteorder
+            rdfile.close()
 
-        # if unique proceed, else set up scales and entry fields.
-        if verbose == 3:
-            self.Unique(numblocks)
-        else:
-            try:
-                self.HdrScale_fm.destroy()
-                self.HdrScale_fm = Frame(self.FixedHdr_nb, borderwidth=2)
-                self.HdrScale_fm.pack(side='bottom', padx=5, pady=5, fill='x')
-                self.BlkScale_fm.destroy()
-                self.BlkScale_fm = Frame(self.Blockette_nb, borderwidth=2)
-                self.BlkScale_fm.pack(side='bottom', padx=5, pady=5, fill='x')
-            except Exception:
-                pass
-            self.buildJumpBox(self.HdrScale_fm, numblocks)
-
-            if FileAbsolutePath != self.OldMseedFile.get():
-                self.BlockNum.set(0)
-                self.JumpNum.set(0)
-
-            Label(self.HdrScale_fm, text='Block Number').pack(
-                side='left', anchor='w')
-
-            Scale(self.HdrScale_fm,
-                  variable=self.BlockNum,
-                  length=625,
-                  orient='horizontal',
-                  from_=0,
-                  to=numblocks - 1,
-                  tickinterval=numblocks / 6,
-                  command=Command(self.FillHdr)
-                  ).pack(side='left', anchor='w')
-
-            Label(self.BlkScale_fm, text='Block Number').pack(
-                side='left', anchor='w')
-            Scale(self.BlkScale_fm,
-                  variable=self.BlockNum,
-                  length=625,
-                  orient='horizontal',
-                  from_=0,
-                  to=numblocks - 1,
-                  tickinterval=numblocks / 6,
-                  command=Command(self.FillHdr)
-                  ).pack(side='left', anchor='w')
-
-            self.SetBlockettes(0)
+            self.set_blockettes(0)
             if not verbose:
-                self.Standard(0, 1)
+                self.fill_standard(int(blk))
             elif verbose == 1:
-                self.Verbose(0, 1)
+                self.fill_verbose(int(blk))
             elif verbose == 2:
-                self.VeryVerbose(0, 1)
-
-        self.OldMseedFile.set(FileAbsolutePath)
-
-#######################################
-    def SetBlockettes(self, key):
-        """
-        setup blockette frames for new key entry
-        """
-        self.BlockettesList = []
-        for blk in self.Blockettes[key]:
-            self.BlockettesList.append(blk[0])
-        self.BlockettesList.sort()
-        self.Blockette.set(self.BlockettesList[0])
-
-        self.BlksSelect_fm.destroy()
-        self.BlksSelect_fm = Frame(
-            self.Blockette_nb, relief='groove', borderwidth=2)
-        self.BlksSelect_fm.pack(side='top', pady=5, fill='x')
-
-        Label(self.BlksSelect_fm, text="Blockette:").grid(row=1, sticky=W)
-        Pmw.ComboBox(self.BlksSelect_fm,
-                     history=0,
-                     entry_width=self.e_width,
-                     entry_textvariable=self.Blockette,
-                     selectioncommand=Command(self.FillSelectBlk, key),
-                     scrolledlist_items=(self.BlockettesList)
-                     ).grid(row=1, column=1, sticky=W)
-
-# fill blockette info for first blockette found
-        self.FillSelectBlk(key, self.BlockettesList[0])
-
-#######################################
-    def VarInit(self, Vars):
-        """
-        clears/initializes entry fields on HdrDisplay
-        """
-
-        self.Hdrs_fm.destroy()
-        self.Hdrs_fm = Frame(self.FixedHdr_nb, relief='groove', borderwidth=2)
-        self.Hdrs_fm.pack(side='top', pady=5, fill='x')
-
-        c = r = n = 0
-        for var in Vars:
-            Label(self.Hdrs_fm, text=var[0]).grid(row=r, column=c, sticky=W)
-            Entry(self.Hdrs_fm,
-                  width=self.e_width,
-                  selectbackground='yellow',
-                  textvariable=var[1]).grid(row=r, column=c + 1, sticky=W)
-            if n == 5:
-                n = r = 0
-                c += 2
+                self.fill_vv(int(blk))
             else:
-                n += 1
-                r += 1
-
-#######################################
-    def Unique(self, numblocks=0):
-        """
-        setup hdr frames for new unique display
-        """
-        self.ClearAll(self.VeryVerboseVars)
-        self.Hdrs_fm.destroy()
-        self.Hdrs_fm = Frame(self.FixedHdr_nb, relief='groove', borderwidth=2)
-        self.Hdrs_fm.pack(side='top', pady=5, fill='x')
-        self.HdrScale_fm.destroy()
-        self.HdrScale_fm = Frame(self.FixedHdr_nb, borderwidth=2)
-        self.HdrScale_fm.pack(side='bottom', padx=5, pady=5, fill='x')
-        self.buildUniqueSelectBox(self.Hdrs_fm)
-        self.buildJumpBox(self.Hdrs_fm, numblocks)
-
-        self.FillUnique()
+                self.fill_unique()
 
-#######################################
-    def Standard(self, key, init=0):
+    def clear_slider(self):
         """
-        setup standard vars for new key entry
+        Reset blk slider between scans
         """
-        if init:
-            self.VarInit(self.StandardVars)
+        # block/unblock widget signals
+        self.slider_box.slider.blockSignals(True)
+        self.slider_box.jump_menu.blockSignals(True)
+        self.slider_box.slider.setValue(0)
+        self.slider_box.jump_menu.clear()
+        self.slider_box.jump_menu.insertItem(0, "0")
+        self.slider_box.jump_menu.setCurrentIndex(0)
+        self.slider_box.jump_menu.blockSignals(False)
+        self.slider_box.slider.blockSignals(False)
 
-        self.FillStandard(key)
-
-#######################################
-    def Verbose(self, key, init=0):
+    def fill_slider(self):
         """
-        setup verbose vars for new key entry
+        Set slider range for chosen trace
         """
+        # block/unblock widget signals
+        self.slider_box.slider.blockSignals(True)
+        self.slider_box.jump_menu.blockSignals(True)
+        self.slider_box.slider.setTickPosition(
+            self.slider_box.slider.TicksBelow)
+        self.slider_box.slider.setRange(0, self.num_blocks - 1)
+        i = 0
+        self.slider_box.jump_menu.clear()
+        while i < self.num_blocks:
+            self.slider_box.jump_menu.insertItem(i, str(i))
+            i += 1
+        self.slider_box.slider.setTickInterval(
+            self.slider_box.slider.maximum() / 5)
+        self.slider_box.jump_menu.blockSignals(False)
+        self.slider_box.slider.blockSignals(False)
 
-        if init:
-            self.VarInit(self.VerboseVars)
-
-        self.FillVerbose(key)
-
-#######################################
-    def VeryVerbose(self, key, init=0):
+    def set_blockettes(self, key):
         """
-        setup very verbose vars for new key entry
+        Setup blockette frames for new key entry
         """
+        self.blockettes_list = []
+        for blk in self.blockettes_dict[key]:
+            self.blockettes_list.append(blk[0])
+        self.blockettes_list.sort()
 
-        if init:
-            self.VarInit(self.VeryVerboseVars)
-
-        self.FillVeryVerbose(key)
+        # fill blockette info for first blockette found
+        self.fill_blockettes_tab(key, self.blockettes_list[0])
 
-#######################################
-    def FillHdr(self, value):
+    def fill_blockettes_tab(self, key, blktype):
         """
-        fill header field on page based on block number
+        clears/initializes entry fields in Blockettes tab
         """
-        key = int(value)
-        self.JumpNum.set(value)
-        self.ByteOffset.set(self.BlockNum.get() * self.BlkSize.get())
-        verbose = self.VerbVar.get()
-        self.SetBlockettes(key)
-        if not verbose:
-            self.Standard(key)
-        elif verbose == 1:
-            self.Verbose(key)
-        elif verbose == 2:
-            self.VeryVerbose(key)
-
-        if self.odd_size:
-            info = ("WARNING: File size is not an integer number of block "
-                    "size (" + str(self.BlkSize.get()) + ")")
-            self.addTextInfoBar(self.InfoString_l, info, "orange")
-            self.odd_size = 0
-        else:
-            info = "Byte Offset: " + \
-                   str(self.BlockNum.get() * self.BlkSize.get())
-            self.addTextInfoBar(self.InfoString_l, info)
-
-#######################################
-    def FillSelectBlk(self, key, blktype):
-        """
-        clears/initializes entry fields on Blockette Display
-        """
-        self.BlksTitle_fm.destroy()
-        self.BlksTitle_fm = Frame(self.Blockette_nb, relief='groove',
-                                  borderwidth=2)
-        self.BlksTitle_fm.pack(side='top', pady=5, fill='x')
-
-        self.BlkInfo_b = Button(self.BlksTitle_fm,
-                                activebackground='green',
-                                background='lightblue',
-                                text="Blockette Info",
-                                activeforeground='black',
-                                command=Command(self.buildBlkInfo, blktype,
-                                                self.Blockette_nb))
-        self.BlkInfo_b.grid(row=0, column=1, padx=10)
-
-        self.Blks_fm.destroy()
-        self.Blks_fm = Frame(self.Blockette_nb, relief='groove', borderwidth=2)
-        self.Blks_fm.pack(side='top', pady=5, fill='x')
-
-        for block in self.Blockettes[key]:
+        for block in self.blockettes_dict[key]:
             if block[0] == blktype:
                 blocktuple = block
-
-        Label(self.BlksTitle_fm, text=BlkVars[blktype][0]).grid(
-            row=0, column=0)
-
-        c = r = n = 0
-        for var in range(len(BlkVars[blktype]) - 1):
-            Label(self.Blks_fm,
-                  text=BlkVars[blktype][var + 1]).grid(row=r, column=c,
-                                                       sticky=W)
-            Label(self.Blks_fm,
-                  text=blocktuple[var]).grid(row=r, column=c + 1, sticky=W)
-            if n == 5:
-                n = r = 0
-                c += 2
-            else:
-                n += 1
-                r += 1
-
-#######################################
-    def FillUnique(self):
-        """
-        Fills unique info on HdrDisplay
-        """
-        self.UniqueText = Pmw.ScrolledText(self.Hdrs_fm,
-                                           borderframe=1)
-        self.UniqueText.pack(side='bottom', fill='both', expand=1)
-
-        self.UniqueText.tag_config("list", foreground="blue")
-        self.UniqueText.tag_config("odd", backgroun="lightblue")
-
-        self.UniqueText.insert("end", "Block\tStat\tChan\tLoc\tNet\tRate\n",
-                               "list")
-
-        c = 0
-        for key in self.UniqueList:
-            for var in key.split(":"):
-                if c:
-                    self.UniqueText.insert("end", """%s\t""" % var, "odd")
-                else:
-                    self.UniqueText.insert("end", """%s\t""" % var)
-            self.UniqueText.insert("end", """\n""")
-            if c:
-                c = 0
-            else:
-                c += 1
-
-#######################################
-    def FillStandard(self, key):
+        for key, values in BlkVars.items():
+            if blktype == key:
+                boxes = self.blockettes_tab.findChildren(QtWidgets.QGroupBox)
+                for box in boxes:
+                    if box.objectName() == "Box1":
+                        menu = box.findChild(QtWidgets.QComboBox, "type_menu")
+                        menu.clear()
+                        menu.insertItem(0, str(blktype))
+                    elif box.objectName() == "Box2":
+                        layout = box.layout()
+                        # delete previous labels if they exist
+                        if layout is not None:
+                            while layout.count():
+                                item = layout.takeAt(0)
+                                widget = item.widget()
+                                if widget is not None:
+                                    widget.setParent(None)
+                        # create new labels
+                        label = QtWidgets.QLabel()
+                        label.setText(str(values[0]))
+                        layout.addWidget(label)
+                        layout.addWidget(self.blkinfo_btn)
+                        layout.addStretch()
+                    else:
+                        layout = box.layout()
+                        # delete previous labels if they exist
+                        if layout is not None:
+                            while layout.count():
+                                item = layout.takeAt(0)
+                                widget = item.widget()
+                                if widget is not None:
+                                    widget.setParent(None)
+                        # create new labels
+                        for x in range(1, len(values)):
+                            label = QtWidgets.QLabel()
+                            label.setText(str(values[x]) + " " +
+                                          str(blocktuple[x - 1]))
+                            layout.addWidget(label)
+
+    def blkinfo_window(self):
+        """
+        Popup window for blk info button
+        """
+        self.info_window = QtWidgets.QWidget()
+        self.info_window.resize(650, 400)
+        self.info_window.setWindowTitle("Blockette Information")
+
+        # widgets
+        blk_label = QtWidgets.QLabel("Blockette:")
+        blk_menu = QtWidgets.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")
+        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.addWidget(blk_label)
+        top_layout.addWidget(blk_menu)
+        top_layout.addStretch()
+
+        main_layout = QtWidgets.QVBoxLayout(self.info_window)
+        main_layout.addLayout(top_layout)
+        main_layout.addWidget(info_box)
+        main_layout.addWidget(done_btn)
+
+        for key in BlkInfoDict.keys():
+            blk_menu.addItem(str(key))
+        blk_menu.setCurrentText(str(self.blockettes_list[0]))
+
+        info_box.setText(BlkInfoDict[self.blockettes_list[0]])
+
+        self.info_window.show()
+
+    def fill_standard(self, key):
         """
         Fills standard info on HdrDisplay
         """
+        try:
+            standard_widgets = self.standard_box.findChildren(
+                QtWidgets.QLineEdit)
+            rate = self.rate_dict[key]
+            vars = []
+            # SeqNum, DHQual, res, Stat, Loc, Chan, Net
+            vars.append(self.fixed_hdr_dict[key][0])
+            vars = list(itertools.chain.from_iterable(vars))
+            vars = vars[3:]  # remove SeqNum, DHQual, res
+            vars.append(rate)
+
+            for x in range(len(vars)):
+                widget = standard_widgets[x]
+                var = vars[x]
+                try:
+                    var = var.strip().decode()
+                except Exception:
+                    pass
+                widget.setText(str(var))
 
-        # cleanup existing vars and reset to new key location
-        self.ClearAll(self.StandardVars)
-        (SeqNum, DHQual, res, Stat, Loc, Chan, Net) = self.FixedHdrDict[key][0]
-        self.SeqNum.set(SeqNum)
-        self.DQual.set(DHQual)
-        self.Resv.set(res)
-        self.Station.set(Stat)
-        self.Channel.set(Chan)
-        self.LocCode.set(Loc)
-        self.NetCode.set(Net)
-        (numsamp, sampfact, sampmult) = self.FixedHdrDict[key][2]
-        rate = self.RateDict[key]
-        self.SampleRate.set(rate)
+        # if user inputs invalid blk num
+        except Exception:
+            return
 
-#######################################
-    def FillVerbose(self, key):
+    def fill_verbose(self, key):
         """
         Fills verbose info on HdrDisplay
         """
+        try:
+            verbose_widgets = self.verbose_box.findChildren(
+                QtWidgets.QLineEdit)
+            rate = self.rate_dict[key]
+            vars = []
+            # SeqNum, DHQual, res, Stat, Loc, Chan, Net
+            vars.append(self.fixed_hdr_dict[key][0])
+            vars.append([str(rate)])
+            # Year, Day, Hour, Min, Sec, junk, Micro
+            vars.append(self.fixed_hdr_dict[key][1])
+            vars = list(itertools.chain.from_iterable(vars))
+            vars = vars[3:]  # remove SeqNum, DHQual, res
+            del vars[-2]  # remove junk
+
+            for x in range(len(vars)):
+                widget = verbose_widgets[x]
+                var = vars[x]
+                try:
+                    var = var.strip().decode()
+                except Exception:
+                    pass
+                widget.setText(str(var))
 
-        # cleanup existing vars and reset to new key location
-        self.ClearAll(self.VerboseVars)
-        self.FillStandard(key)
-        (Year, Day, Hour, Min, Sec, junk, Micro) = self.FixedHdrDict[key][1]
-        self.Year.set(Year)
-        self.Jday.set(Day)
-        self.Hour.set(Hour)
-        self.Min.set(Min)
-        self.Sec.set(Sec)
-        self.Micro.set(Micro)
+        # if user inputs invalid blk num
+        except Exception:
+            return
 
-#######################################
-    def FillVeryVerbose(self, key):
+    def fill_vv(self, key):
         """
         Fills very verbose info on HdrDisplay
         """
+        try:
+            vv_widgets = self.vv_box.findChildren(QtWidgets.QLineEdit)
+            rate = self.rate_dict[key]
+            vars = []
+            # SeqNum, DHQual, res, Stat, Loc, Chan, Net
+            vars.append(self.fixed_hdr_dict[key][0])
+            vars.append([str(rate)])
+            # Year, Day, Hour, Min, Sec, junk, Micro
+            vars.append(self.fixed_hdr_dict[key][1])
+            # NumSamp, SampFact, SampMult
+            vars.append(self.fixed_hdr_dict[key][2])
+            # act, io, DQual, numblk, timcorr, bdata, bblock
+            vars.append(self.fixed_hdr_dict[key][3])
+            vars = list(itertools.chain.from_iterable(vars))
+            del vars[-12]  # remove junk
+
+            for x in range(len(vars)):
+                widget = vv_widgets[x]
+                var = vars[x]
+                try:
+                    var = var.strip().decode()
+                except Exception:
+                    pass
+                widget.setText(str(var))
 
-        # cleanup existing vars and reset to new key location
-        self.ClearAll(self.VeryVerboseVars)
-        self.FillVerbose(key)
-        (NumSamp, SampFact, SampMult) = self.FixedHdrDict[key][2]
-        (act, io, DQual, numblk, timcorr, bdata,
-         bblock) = self.FixedHdrDict[key][3]
-        self.NumSamp.set(NumSamp)
-        self.SampFact.set(SampFact)
-        self.SampMult.set(SampMult)
-        self.ActFlag.set(act)
-        self.IOFlag.set(io)
-        self.DQFlag.set(DQual)
-        self.NumBlockettes.set(numblk)
-        self.TimeCorr.set(timcorr)
-        self.BeginData.set(bdata)
-        self.FstBlkett.set(bblock)
-
-##########################################
-    def FlushDict(self):
-        """
-        clears dictionaries and OldMseedFile. Allows user to apply a reset and
-        dynamically monitor changes to mseed files.
-        """
-
-        # clear selected directory and file
-        self.MseedFile.set("")
-        self.SelDir.set("")
-
-        # clear variables and destroy frames
-        self.ClearAll(self.VeryVerboseVars)
-        self.Hdrs_fm.destroy()
-        self.HdrScale_fm.destroy()
-        self.BlkScale_fm.destroy()
-        self.BlksSelect_fm.destroy()
-        self.BlksTitle_fm.destroy()
-        self.Blks_fm.destroy()
-
-        # now clear dictionaries and set OldMseedFile to blank
-        self.OldMseedFile.set("")
-        self.FixedHdrDict = {}
-        self.RateDict = {}
-        self.Blockettes = {}
-        self.keyList = []
-        self.UniqueList = []
-
-##########################################
-
-    def getPath(self, var, clear=1):
-        """
-        Concatonate paths
-        """
-        self.var = var
-        if clear:
-            self.var.set('')
-        newpath_dd = DirectoryDialog(self.root)
-        self.path = newpath_dd.go(dir_or_file=getcwd())
-        if self.path is not None:
-            if self.var.get():
-                self.var.set(self.var.get() + ':')
-            if len(self.path) == 1 and self.path[0] == "/":
-                self.var.set(self.var.get() + self.path)
-            elif self.path[-1] == "/":
-                self.var.set(self.var.get() + self.path[:-1])
-            else:
-                self.var.set(self.var.get() + self.path)
-
-##################################################################
+        # if user inputs invalid blk num
+        except Exception:
+            return
 
-    def ClearTxt(self, var):
+    def fill_unique(self):
         """
-        clear text
+        Fills unique info on HdrDisplay
         """
-        var.clear()
-
-##################################################################
+        # text colors
+        Blue = QtGui.QColor(0, 0, 153)
+        Black = QtGui.QColor(0, 0, 0)
+
+        # fill drop down menus
+        self.unique_box.keys_menu.clear()
+        self.unique_box.keys_menu.addItem("*")
+        self.unique_box.keys_menu.addItem(f'{self.unique_select_list[0]}')
+
+        self.unique_box.unique_jump.blockSignals(True)
+        i = 0
+        while i < self.num_blocks:
+            self.unique_box.unique_jump.addItem(str(i))
+            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)
 
-    def ClearAll(self, varlist):
+    def wait(self, words, cnt):
         """
-        clear a variable list
+        routine to put words on info bar, used to count traces
+        being examined
         """
-        self.addTextInfoBar(self.InfoString_l)
-        for var in varlist:
-            self.setValue(var[1])
-
-#######################################
+        text = words + str(cnt)
+        self.update_infobar(text, "lightblue")
 
-    def setValue(self, var, value=''):
+    def update_dir_list(self):
         """
-        Sets Value of var
+        Update directory list for each scan
         """
-        var.set(value)
+        self.dir_menu.clear()
+        self.dir_menu.insertItem(0, "")
+        for key in self.dir_trace_dict:
+            self.dir_menu.addItem(key)
+        self.dir_trace_box.show()
 
-#######################################
-
-    def addTextInfoBar(self, widget, str='', bg='yellow'):
+    def update_trace_list(self, dir):
         """
-        Adds Text To InfoBar
+        Update trace list for each scan
         """
-        widget.configure(background=bg, text=str)
+        self.trace_menu.clear()
+        self.trace_menu.insertItem(0, "")
+        for key, values in self.dir_trace_dict.items():
+            if dir == key:
+                values.sort()
+                for trace in values:
+                    self.trace_menu.addItem(trace)
 
-##################################################################
+    def update_infobar(self, text, color):
+        """
+        Update info bar when certain actions happen
+        """
+        self.infobar.setText(text)
+        self.infobar.setStyleSheet("background-color:" + color)
+        QtWidgets.QApplication.beep()
 
-    def killWindow(self, widget):
+    def update_slider(self, widget):
         """
-        Destroy Widget and Set InfoBar to Default
+        Get new blk number if any of the slider related widgets
+        are toggled, and update hdr info
         """
-        widget.destroy()
-        self.addTextInfoBar(self.InfoString_l)
+        try:
+            if widget == "slider":
+                value = self.slider_box.slider.value()
+                self.slider_box.jump_menu.setCurrentText(str(value))
+                self.unique_box.unique_jump.setCurrentText(str(value))
+            elif widget == "menu":
+                value = self.slider_box.jump_menu.currentText()
+                self.slider_box.slider.setValue(int(value))
+                self.unique_box.unique_jump.setCurrentText(str(value))
+            else:
+                value = self.unique_box.unique_jump.currentText()
+                self.slider_box.slider.setValue(int(value))
+                self.slider_box.jump_menu.setCurrentText(str(value))
+                self.vv_rbtn.setChecked(True)
+                self.verb_var = 2
+                self.jump_box.setCurrentIndex(2)
+
+            # update hdr info & info bar
+            self.maybe_read_hdrs()
+            byte_offset = int(value) * int(self.blk_size)
+            text = "Byte Offset: " + str(byte_offset)
+            self.update_infobar(text, "yellow")
+
+        # if user inputs invalid blk num
+        except Exception:
+            return
 
-#######################################
+    def maybe_read_hdrs(self):
+        """
+        If certain conditions are met, call read hdrs
+        """
+        if not self.dir_menu.isHidden():
+            dir = self.dir_menu.currentText()
+            if dir:
+                if not self.trace_menu.isHidden():
+                    trace = self.trace_menu.currentText()
+                    if trace:
+                        blk = self.slider_box.jump_menu.currentText()
+                        self.read_hdrs(dir, trace, blk)
 
-    def wait(self, words, cnt):
+    def flush_dict(self):
         """
-        routine to put words on info bar, used to count traces
-        being examined
+        clears dictionaries. Allows user to apply a reset and
+        dynamically monitor changes to mseed files.
         """
+        self.fixed_hdr_dict = {}
+        self.rate_dict = {}
+        self.blockettes_dict = {}
+        self.key_list = []
+        self.unique_list = []
+        self.clear_slider()
+        self.header_endianess_text.clear()
+        self.hide_show("hide")
+        QtWidgets.QApplication.beep()
+
+
+class SliderBox(QtWidgets.QGroupBox):
+    """
+    Box that holds all the blockette slider widgets
+    """
+    def __init__(self):
+        """
+        Init widgets for blockette parsing
+        """
+        super().__init__()
+        self.setStyleSheet("QGroupBox{border:0;}")
+        # widgets
+        label1 = QtWidgets.QLabel("Jump To Block #:")
+        self.jump_menu = QtWidgets.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)
+        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)
+        slider_layout.addWidget(label2)
+        slider_layout.addWidget(self.slider)
+
+        # add to main layout
+        main_layout.addLayout(menu_layout)
+        main_layout.addLayout(slider_layout)
+
+
+class StandardBox(QtWidgets.QGroupBox):
+    """
+    Box that holds all the standard info widgets
+    """
+    def __init__(self, standard_vars):
+        """
+        Create standard widgets
+        """
+        super().__init__()
+
+        # layout
+        v_layout = QtWidgets.QVBoxLayout(self)
+        h_layout = QtWidgets.QHBoxLayout()
+        col1 = QtWidgets.QVBoxLayout()
+        col2 = QtWidgets.QVBoxLayout()
+
+        # widgets
+        for x in range(len(standard_vars)):
+            label = QtWidgets.QLabel(str(standard_vars[x]))
+            line_edit = QtWidgets.QLineEdit()
+
+            col1.addWidget(label)
+            col2.addWidget(line_edit)
+
+        # add to layout
+        h_layout.addLayout(col1)
+        h_layout.addLayout(col2)
+        v_layout.addLayout(h_layout)
+
+        v_layout.addStretch()
+
+
+class VerboseBox(QtWidgets.QGroupBox):
+    """
+    Box that holds all the verbose info widgets
+    """
+    def __init__(self, standard_vars, v_vars):
+        """
+        Create verbose widgets
+        """
+        super().__init__()
+        # layout
+        v_layout = QtWidgets.QVBoxLayout(self)
+        h_layout = QtWidgets.QHBoxLayout()
+        col1 = QtWidgets.QVBoxLayout()
+        col2 = QtWidgets.QVBoxLayout()
+        col3 = QtWidgets.QVBoxLayout()
+        col4 = QtWidgets.QVBoxLayout()
+
+        # widgets
+        for x in range(len(standard_vars)):
+            label = QtWidgets.QLabel(str(standard_vars[x]))
+            line_edit = QtWidgets.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()
+
+            if x == 0:
+                col1.addWidget(label)
+                col2.addWidget(line_edit)
+            else:
+                col3.addWidget(label)
+                col4.addWidget(line_edit)
+
+        col3.addSpacing(25)
+        col4.addSpacing(25)
+
+        # add to layout
+        h_layout.addLayout(col1)
+        h_layout.addLayout(col2)
+        h_layout.addLayout(col3)
+        h_layout.addLayout(col4)
+        v_layout.addLayout(h_layout)
+        v_layout.addStretch()
+
+
+class VVBox(QtWidgets.QGroupBox):
+    """
+    Box that holds all the very verbose info widgets
+    """
+    def __init__(self, first_vv_vars, standard_vars, v_vars, vv_vars):
+        """
+        Create very verbose widgets
+        """
+        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()
+
+        # widgets
+        for x in range(len(first_vv_vars)):
+            label = QtWidgets.QLabel(str(first_vv_vars[x]))
+            line_edit = QtWidgets.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()
+
+            if x <= 2:
+                col1.addWidget(label)
+                col2.addWidget(line_edit)
+            else:
+                col3.addWidget(label)
+                col4.addWidget(line_edit)
+
+        for x in range(len(v_vars)):
+            label = QtWidgets.QLabel(str(v_vars[x]))
+            line_edit = QtWidgets.QLineEdit()
 
-        txt = words + str(cnt)
-        self.root.update()
-        self.addTextInfoBar(self.InfoString_l, txt, 'lightblue')
-
-##################################################################
-
-    def Exit(self):
-        """
-        a no questions asked exit
-        """
-        sys.exit(0)
-
-##################################################################
-
-    def displayBlkInfo(self, type):
-        text = BlkInfoDict[type]
-        self.BlkInfoText.settext(text)
-
-##################################################################
-    def buildHelp(self, master):
-        """
-        Populate Help NoteBook
-        """
-        self.Help_nb = master.add('Help')
-
-        self.HelpText = Pmw.ScrolledText(self.Help_nb,
-                                         borderframe=1)
-        self.HelpText.pack(side='bottom', fill='both', expand=1)
-
-        self.HelpText.tag_config("hdr1", foreground="blue")
-        self.HelpText.tag_config("hdr2", foreground="darkgreen")
-        self.HelpText.tag_config("bttn", foreground="red")
-
-        self.HelpText.insert("end", "NAME", "hdr1")
-        self.HelpText.insert("end", """
-   mseedpeek - GUI for displaying mseed file headers.\n""")
-
-        self.HelpText.insert("end", "\nVERSION", "hdr1")
-        self.HelpText.insert("end", """
-   %s\n""" % VERSION)
-
-        self.HelpText.insert("end", "\nSYNOPSIS", "hdr1")
-        self.HelpText.insert("end", """
-   mseedpeek
-   mseedpeek -#
-   mseedpeek -f file_names\n""")
-
-        self.HelpText.insert("end", "\nOPTIONS", "hdr1")
-        self.HelpText.insert("end", """
-   -# returns version number
-   -f file_name(s) - accepts command line input for files to inspect (wildcards accepted)\n""")  # noqa: E501
-
-        self.HelpText.insert("end", "\nDESCRIPTION", "hdr1")
-        self.HelpText.insert("end", """
-   mseedpeek has three notebooks: """)
-        self.HelpText.insert("end", "[Trace Headers]", "hdr2")
-        self.HelpText.insert("end", "[Blockettes]", "hdr2")
-        self.HelpText.insert("end", """, and """)
-        self.HelpText.insert("end", "[Help]", "hdr2")
-        self.HelpText.insert("end", """.""")
-
-        self.HelpText.insert("end", """
-   The """)
-        self.HelpText.insert("end", "[Trace Headers]", "hdr2")
-        self.HelpText.insert("end", """ notebook displays various the fixed
-   header in four levels of verbosity.\n   The """)
-        self.HelpText.insert("end", "[Blockettes]", "hdr2")
-        self.HelpText.insert("end", """ notebook displays blockette information.\n""")  # noqa: E501
-
-        self.HelpText.insert("end", "\n   [Trace Headers]", "hdr2")
-        self.HelpText.insert("end", """
-   General:
-      >> specify/load directories for building trace db
-      >> select trace for header display
-      >> select level of verbosity for display
-
-   Buttons:\n""")
-        self.HelpText.insert("end", "      <Build Trace db>", "bttn")
-        self.HelpText.insert("end", """: Searchs the directories listed in the "Data-
-         Directories" entry box (a colon separated list) and builds a list of mseed
-         files found indexing them on unique values of <dir>:<file_name>. It then
-         creates a dropdown menu with entries for each data directory found.\n""")  # noqa: E501
-        self.HelpText.insert("end", "      <Find>", "bttn")
-        self.HelpText.insert("end", """: Launches a file browser allowing the user to
-         add directories to the "Data Directories" entry box. Double clicking selects
-         the new directory.\n""")  # noqa: E501
-        self.HelpText.insert("end", "      <Clear>", "bttn")
-        self.HelpText.insert(
-            "end", """: Clears the "Data Directories" entry box.\n""")
-
-        self.HelpText.insert("end", """
-        By selecting a data directory a dropdown menu is created for
-        trace selection. Selecting a trace from this menu will create
-        a header display.
-
-   Radio Buttons:\n""")
-        self.HelpText.insert("end", "      <Standard>", "bttn")
-        self.HelpText.insert("end", """: This is the default. It displays Station, Channel, Location Code,
-        Net Code, and the nominal Sample Rate.\n""")  # noqa: E501
-        self.HelpText.insert("end", "      <Verbose>", "bttn")
-        self.HelpText.insert(
-            "end", """: Standard display plus time of block.\n""")
-        self.HelpText.insert("end", "      <Very Verbose>", "bttn")
-        self.HelpText.insert(
-            "end", """: Verbose plus the rest of the mseed fixed header fields.\n""")  # noqa: E501
-        self.HelpText.insert("end", "      <Unique>", "bttn")
-        self.HelpText.insert(
-            "end", """: This displays changes in the Standard fields within a mseed file.\n""")  # noqa: E501
-        self.HelpText.insert("end", """
-
-   Slider Scale and """)
-        self.HelpText.insert("end", "<Jump To Block #>", "bttn")
-        self.HelpText.insert("end", """ Button:
-        At the bottom of the page is a scale bar showing block numbers. Sliding
-        the bar scans through all fixed headers for the selected mseed file.\n""")  # noqa: E501
-
-        self.HelpText.insert("end", "\n   <Flush Dictionaries>", "bttn")
-        self.HelpText.insert("end", """: Clears cached trace header data.
-        As you view mseed headers, mseedpeek maintains a cache of previously
-        viewed headers for quick, future access. If you wish to monitor changes
-        to mseed files you need to flush the dictionaries so that the new
-        changes will be displayed.\n\n""")
-
-        self.HelpText.insert("end", "   [Blockettes]", "hdr2")
-        self.HelpText.insert("end", """:
-        The 'Blockettes' notebook displays all blockettes found for the select
-        mseed file in a dropdown menu. By selecting a blockette from the dropdown
-        menu, the contents of the blockette will be displayed.\n""")  # noqa: E501
-
-        self.HelpText.insert("end", "      <Blockette Info>", "bttn")
-        self.HelpText.insert("end", """: Displays a pop-up window with a definition of the SEED Blockette.
-
-        At the bottom of the page is a scale bar showing block numbers and
-        Sliding the bar scans through all blockettes for the selected mseed file.\n""")  # noqa: E501
-
-        self.HelpText.insert("end", "\nKEYWORDS", "hdr1")
-        self.HelpText.insert("end", """
-   mseed; header information\n""")
-
-        self.HelpText.insert("end", "\nSEE ALSO", "hdr1")
-        self.HelpText.insert("end", """
-   SEED manual (pdf), fixhdr, mseedhdr\n""")
-
-        self.HelpText.insert("end", "\nAUTHOR", "hdr1")
-        self.HelpText.insert("end", """
-   Bruce Beaudoin <bruce@passcal.nmt.edu>
-""")
-
-
-if __name__ == '__main__':
+            if x <= 3:
+                col3.addWidget(label)
+                col4.addWidget(line_edit)
+            else:
+                col5.addWidget(label)
+                col6.addWidget(line_edit)
+
+        for x in range(len(vv_vars)):
+            label = QtWidgets.QLabel(str(vv_vars[x]))
+            line_edit = QtWidgets.QLineEdit()
+
+            if x <= 3:
+                col5.addWidget(label)
+                col6.addWidget(line_edit)
+            elif x > 3 and x <= 6:
+                col7.addWidget(label)
+                col8.addWidget(line_edit)
+            else:
+                col7.addWidget(label)
+                col8.addWidget(line_edit)
+
+        # add to layout
+        h_layout.addLayout(col1)
+        h_layout.addLayout(col2)
+        h_layout.addLayout(col3)
+        h_layout.addLayout(col4)
+        h_layout.addLayout(col5)
+        h_layout.addLayout(col6)
+        h_layout.addLayout(col7)
+        h_layout.addLayout(col8)
+        v_layout.addLayout(h_layout)
+        v_layout.addStretch()
+
+
+class UniqueBox(QtWidgets.QGroupBox):
+    """
+    Box that holds all the unique info widgets
+    """
+    def __init__(self):
+        """
+        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()
+        self.unique_jump.setEditable(True)
+        self.unique_infobox = QtWidgets.QTextEdit()
+
+        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.addStretch()
+
+
+if __name__ == "__main__":
     main()
diff --git a/setup.py b/setup.py
index fb8a9dfaee4cca263abc99e1c54dd22fc51a6a1a..f8bfd07c9bbcc4d8f88c0de43148f7dc26af7273 100644
--- a/setup.py
+++ b/setup.py
@@ -20,9 +20,9 @@ setup(
         'Intended Audience :: Developers',
         'License :: OSI Approved ::  GNU General Public License v3 (GPLv3)',
         'Natural Language :: English',
-        'Programming Language :: Python :: 3.6',
-        'Programming Language :: Python :: 3.7',
         'Programming Language :: Python :: 3.8',
+        'Programming Language :: Python :: 3.9',
+        'Programming Language :: Python :: 3.10',
     ],
     description="View mseed headers",
     entry_points={
@@ -30,7 +30,7 @@ setup(
             'mseedpeek=mseedpeek.mseedpeek:main',
         ],
     },
-    install_requires=['Pmw @ https://github.com/schrodinger/pmw-patched/archive/master.tar.gz'],
+    install_requires=['PySide2'],
     setup_requires=[],
     extras_require={
         'dev': [
@@ -45,6 +45,6 @@ setup(
     name='mseedpeek',
     packages=find_packages(include=['mseedpeek']),
     url='https://git.passcal.nmt.edu/passoft/mseedpeek',
-    version='2022.1.0.0',
+    version='2022.2.0.0',
     zip_safe=False,
 )