Statistics
| Branch: | Tag: | Revision:

hlrc / client / python / hlrc_client / textgrid_hlrc.py @ 61d23251

History | View | Annotate | Download (18.409 KB)

1
# Natural Language Toolkit: TextGrid analysis
2
#
3
# Copyright (C) 2001-2011 NLTK Project
4
# Author: Margaret Mitchell <itallow@gmail.com>
5
#         Steven Bird <sb@csse.unimelb.edu.au> (revisions)
6
# URL: <http://www.nltk.org>
7
# For license information, see LICENSE.TXT
8
#
9

    
10
"""
11
Tools for reading TextGrid files, the format used by Praat.
12

13
Module contents
14
===============
15

16
The textgrid corpus reader provides 4 data items and 1 function
17
for each textgrid file.  For each tier in the file, the reader
18
provides 10 data items and 2 functions.
19
 
20
For the full textgrid file: 
21

22
  - size
23
    The number of tiers in the file.
24

25
  - xmin
26
    First marked time of the file.
27

28
  - xmax
29
    Last marked time of the file.
30

31
  - t_time
32
    xmax - xmin.
33

34
  - text_type
35
    The style of TextGrid format:
36
        - ooTextFile:  Organized by tier.
37
        - ChronTextFile:  Organized by time.
38
        - OldooTextFile:  Similar to ooTextFile.
39

40
  - to_chron()
41
    Convert given file to a ChronTextFile format.
42

43
  - to_oo()
44
    Convert given file to an ooTextFile format.
45

46
For each tier:
47

48
  - text_type
49
    The style of TextGrid format, as above.
50

51
  - classid
52
    The style of transcription on this tier:
53
        - IntervalTier:  Transcription is marked as intervals.
54
        - TextTier:  Transcription is marked as single points.
55

56
  - nameid
57
    The name of the tier.
58

59
  - xmin
60
    First marked time of the tier.
61

62
  - xmax
63
    Last marked time of the tier.
64

65
  - size
66
    Number of entries in the tier.
67

68
  - transcript
69
    The raw transcript for the tier.
70

71
  - simple_transcript
72
    The transcript formatted as a list of tuples: (time1, time2, utterance).
73

74
  - tier_info
75
    List of (classid, nameid, xmin, xmax, size, transcript).
76

77
  - min_max()
78
    A tuple of (xmin, xmax).  
79

80
  - time(non_speech_marker)
81
    Returns the utterance time of a given tier.
82
    Excludes entries that begin with a non-speech marker.
83

84
"""
85

    
86
# needs more cleanup, subclassing, epydoc docstrings
87

    
88
import sys
89
import re
90

    
91
TEXTTIER = "TextTier"
92
INTERVALTIER = "IntervalTier"
93

    
94
OOTEXTFILE = re.compile(r"""(?x)
95
            xmin\ =\ (.*)[\r\n]+
96
            xmax\ =\ (.*)[\r\n]+
97
            [\s\S]+?size\ =\ (.*)[\r\n]+ 
98
""")
99

    
100
CHRONTEXTFILE = re.compile(r"""(?x)
101
            [\r\n]+(\S+)\ 
102
            (\S+)\ +!\ Time\ domain.\ *[\r\n]+
103
            (\S+)\ +!\ Number\ of\ tiers.\ *[\r\n]+"
104
""")
105

    
106
OLDOOTEXTFILE = re.compile(r"""(?x)
107
            [\r\n]+(\S+)
108
            [\r\n]+(\S+)
109
            [\r\n]+.+[\r\n]+(\S+)
110
""")
111

    
112

    
113

    
114
#################################################################
115
# TextGrid Class
116
#################################################################
117

    
118
class TextGrid(object):
119
    """
120
    Class to manipulate the TextGrid format used by Praat.
121
    Separates each tier within this file into its own Tier
122
    object.  Each TextGrid object has
123
    a number of tiers (size), xmin, xmax, a text type to help
124
    with the different styles of TextGrid format, and tiers with their
125
    own attributes.
126
    """
127

    
128
    def __init__(self, read_file):
129
        """
130
        Takes open read file as input, initializes attributes 
131
        of the TextGrid file.
132
        @type read_file: An open TextGrid file, mode "r".
133
        @param size:  Number of tiers.
134
        @param xmin: xmin.
135
        @param xmax: xmax.
136
        @param t_time:  Total time of TextGrid file.
137
        @param text_type:  TextGrid format.
138
        @type tiers:  A list of tier objects.
139
        """
140

    
141
        self.read_file = read_file
142
        self.size = 0
143
        self.xmin = 0
144
        self.xmax = 0
145
        self.t_time = 0
146
        self.text_type = self._check_type()
147
        self.tiers = self._find_tiers()
148

    
149
    def __iter__(self):
150
        for tier in self.tiers:
151
            yield tier
152

    
153
    def next(self):
154
        if self.idx == (self.size - 1):
155
            raise StopIteration
156
        self.idx += 1
157
        return self.tiers[self.idx]
158

    
159
    @staticmethod
160
    def load(file):
161
        """
162
        @param file: a file in TextGrid format
163
        """
164

    
165
        return TextGrid(open(file).read())
166

    
167
    def _load_tiers(self, header):
168
        """
169
        Iterates over each tier and grabs tier information.
170
        """ 
171

    
172
        tiers = []
173
        if self.text_type == "ChronTextFile":
174
            m = re.compile(header)
175
            tier_headers = m.findall(self.read_file)
176
            tier_re = " \d+.?\d* \d+.?\d*[\r\n]+\"[^\"]*\""
177
            for i in range(0, self.size):
178
                tier_info = [tier_headers[i]] + \
179
                re.findall(str(i + 1) + tier_re, self.read_file)
180
                tier_info = "\n".join(tier_info)
181
                tiers.append(Tier(tier_info, self.text_type, self.t_time))
182
            return tiers
183
        
184
        tier_re = header + "\d+[\s\S]+?(?=" + header + "|$$)"
185
        m = re.compile(tier_re)
186
        tier_iter = m.finditer(self.read_file)
187
        for iterator in tier_iter:
188
            (begin, end) = iterator.span()
189
            tier_info = self.read_file[begin:end]
190
            tiers.append(Tier(tier_info, self.text_type, self.t_time))
191
        return tiers
192
    
193
    def _check_type(self):
194
        """
195
        Figures out the TextGrid format.
196
        """
197

    
198
        m = re.match("(.*)[\r\n](.*)[\r\n](.*)[\r\n](.*)", self.read_file)
199
        try:
200
            type_id = m.group(1).strip()
201
        except AttributeError:
202
            raise TypeError("Cannot read file -- try TextGrid.load()")
203
        xmin = m.group(4)
204
        if type_id == "File type = \"ooTextFile\"":
205
            if "xmin" not in xmin:
206
                xmin = m.group(3)
207
                if "xmin" not in xmin:
208
                    text_type = "OldooTextFile"
209
                else:
210
                    text_type = "ooTextFile"
211
            else:
212
                text_type = "ooTextFile"
213
        elif type_id == "\"Praat chronological TextGrid text file\"":
214
            text_type = "ChronTextFile"
215
        else: 
216
            raise TypeError("Unknown format '(%s)'", (type_id))
217
        return text_type
218
        
219
    def _find_tiers(self):
220
        """
221
        Splits the textgrid file into substrings corresponding to tiers. 
222
        """
223

    
224
        if self.text_type == "ooTextFile":
225
            m = OOTEXTFILE
226
            header = " *item \["
227
        elif self.text_type == "ChronTextFile":
228
            m = CHRONTEXTFILE
229
            header = "\"\S+\" \".*\" \d+\.?\d* \d+\.?\d*"
230
        elif self.text_type == "OldooTextFile":
231
            m = OLDOOTEXTFILE
232
            header = "\".*\"[\r\n]+\".*\""
233

    
234
        file_info = m.findall(self.read_file)[0]
235
        self.xmin = float(file_info[0])
236
        self.xmax = float(file_info[1])
237
        self.t_time = self.xmax - self.xmin
238
        self.size = int(file_info[2])
239
        tiers = self._load_tiers(header)
240
        return tiers
241

    
242
    def to_chron(self):
243
        """ 
244
        @return:  String in Chronological TextGrid file format.
245
        """
246

    
247
        chron_file = ""
248
        chron_file += "\"Praat chronological TextGrid text file\"\n"
249
        chron_file += str(self.xmin) + " " + str(self.xmax)
250
        chron_file += "   ! Time domain.\n"
251
        chron_file += str(self.size) + "   ! Number of tiers.\n"
252
        for tier in self.tiers:
253
            idx = (self.tiers.index(tier)) + 1
254
            tier_header = "\"" + tier.classid + "\" \"" \
255
                          + tier.nameid + "\" " + str(tier.xmin) \
256
                          + " " + str(tier.xmax)
257
            chron_file += tier_header + "\n"
258
            transcript = tier.simple_transcript
259
            for (xmin, xmax, utt) in transcript:
260
                chron_file += str(idx) + " " + str(xmin) 
261
                chron_file += " " + str(xmax) +"\n"
262
                chron_file += "\"" + utt + "\"\n"
263
        return chron_file
264

    
265
    def to_oo(self):
266
        """ 
267
        @return:  A string in OoTextGrid file format.
268
        """
269
   
270
        oo_file = ""
271
        oo_file += "File type = \"ooTextFile\"\n"
272
        oo_file += "Object class = \"TextGrid\"\n\n"
273
        oo_file += "xmin = ", self.xmin, "\n"
274
        oo_file += "xmax = ", self.xmax, "\n"
275
        oo_file += "tiers? <exists>\n"
276
        oo_file += "size = ", self.size, "\n"
277
        oo_file += "item []:\n"
278
        for i in range(len(self.tiers)):
279
            oo_file += "%4s%s [%s]" % ("", "item", i + 1)
280
            _curr_tier = self.tiers[i]
281
            for (x, y) in _curr_tier.header:
282
                oo_file += "%8s%s = \"%s\"" % ("", x, y)
283
            if _curr_tier.classid != TEXTTIER:
284
                for (xmin, xmax, text) in _curr_tier.simple_transcript:
285
                    oo_file += "%12s%s = %s" % ("", "xmin", xmin)
286
                    oo_file += "%12s%s = %s" % ("", "xmax", xmax)
287
                    oo_file += "%12s%s = \"%s\"" % ("", "text", text)
288
            else:
289
                for (time, mark) in _curr_tier.simple_transcript:
290
                    oo_file += "%12s%s = %s" % ("", "time", time)
291
                    oo_file += "%12s%s = %s" % ("", "mark", mark)
292
        return oo_file
293

    
294

    
295
#################################################################
296
# Tier Class
297
#################################################################
298

    
299
class Tier(object):
300
    """ 
301
    A container for each tier.
302
    """
303

    
304
    def __init__(self, tier, text_type, t_time):
305
        """
306
        Initializes attributes of the tier: class, name, xmin, xmax
307
        size, transcript, total time.  
308
        Utilizes text_type to guide how to parse the file.
309
        @type tier: a tier object; single item in the TextGrid list.
310
        @param text_type:  TextGrid format
311
        @param t_time:  Total time of TextGrid file.
312
        @param classid:  Type of tier (point or interval).
313
        @param nameid:  Name of tier.
314
        @param xmin:  xmin of the tier.
315
        @param xmax:  xmax of the tier.
316
        @param size:  Number of entries in the tier
317
        @param transcript:  The raw transcript for the tier.
318
        """
319

    
320
        self.tier = tier
321
        self.text_type = text_type
322
        self.t_time = t_time
323
        self.classid = ""
324
        self.nameid = ""
325
        self.xmin = 0
326
        self.xmax = 0
327
        self.size = 0
328
        self.transcript = ""
329
        self.tier_info = ""
330
        self._make_info()
331
        self.simple_transcript = self.make_simple_transcript()
332
        if self.classid != TEXTTIER:
333
            self.mark_type = "intervals"
334
        else:
335
            self.mark_type = "points"
336
            self.header = [("class", self.classid), ("name", self.nameid), \
337
            ("xmin", self.xmin), ("xmax", self.xmax), ("size", self.size)]
338

    
339
    def __iter__(self):
340
        return self
341
  
342
    def _make_info(self):
343
        """
344
        Figures out most attributes of the tier object:
345
        class, name, xmin, xmax, transcript.
346
        """
347

    
348
        trans = "([\S\s]*)"
349
        if self.text_type == "ChronTextFile":
350
            classid = "\"(.*)\" +"
351
            nameid = "\"(.*)\" +"
352
            xmin = "(\d+\.?\d*) +"
353
            xmax = "(\d+\.?\d*) *[\r\n]+"
354
            # No size values are given in the Chronological Text File format.
355
            self.size = None
356
            size = ""
357
        elif self.text_type == "ooTextFile":
358
            classid = " *class = \"(.*)\" *[\r\n]+"
359
            nameid = " *name = \"(.*)\" *[\r\n]+"
360
            xmin = " *xmin = (\d+\.?\d*) *[\r\n]+"
361
            xmax = " *xmax = (\d+\.?\d*) *[\r\n]+"
362
            size = " *\S+: size = (\d+) *[\r\n]+"
363
        elif self.text_type == "OldooTextFile":
364
            classid = "\"(.*)\" *[\r\n]+"
365
            nameid = "\"(.*)\" *[\r\n]+"
366
            xmin = "(\d+\.?\d*) *[\r\n]+"
367
            xmax = "(\d+\.?\d*) *[\r\n]+"
368
            size = "(\d+) *[\r\n]+"
369
        m = re.compile(classid + nameid + xmin + xmax + size + trans)
370
        self.tier_info = m.findall(self.tier)[0]
371
        self.classid = self.tier_info[0]
372
        self.nameid = self.tier_info[1]
373
        self.xmin = float(self.tier_info[2])
374
        self.xmax = float(self.tier_info[3])
375
        if self.size != None:
376
            self.size = int(self.tier_info[4])
377
        self.transcript = self.tier_info[-1]
378
            
379
    def make_simple_transcript(self):
380
        """ 
381
        @return:  Transcript of the tier, in form [(start_time end_time label)]
382
        """
383

    
384
        if self.text_type == "ChronTextFile":
385
            trans_head = ""
386
            trans_xmin = " (\S+)"
387
            trans_xmax = " (\S+)[\r\n]+"
388
            trans_text = "\"([\S\s]*?)\""
389
        elif self.text_type == "ooTextFile":
390
            trans_head = " *\S+ \[\d+\]: *[\r\n]+"
391
            trans_xmin = " *\S+ = (\S+) *[\r\n]+"
392
            trans_xmax = " *\S+ = (\S+) *[\r\n]+"
393
            trans_text = " *\S+ = \"([^\"]*?)\""    
394
        elif self.text_type == "OldooTextFile":
395
            trans_head = ""
396
            trans_xmin = "(.*)[\r\n]+"
397
            trans_xmax = "(.*)[\r\n]+"
398
            trans_text = "\"([\S\s]*?)\""
399
        if self.classid == TEXTTIER:
400
            trans_xmin = ""
401
        trans_m = re.compile(trans_head + trans_xmin + trans_xmax + trans_text)
402
        self.simple_transcript = trans_m.findall(self.transcript)
403
        return self.simple_transcript
404

    
405
    def transcript(self):
406
        """
407
        @return:  Transcript of the tier, as it appears in the file.
408
        """
409
       
410
        return self.transcript
411

    
412
    def time(self, non_speech_char="."):
413
        """
414
        @return: Utterance time of a given tier.
415
        Screens out entries that begin with a non-speech marker.        
416
        """
417

    
418
        total = 0.0
419
        if self.classid != TEXTTIER:
420
            for (time1, time2, utt) in self.simple_transcript:
421
                utt = utt.strip()
422
                if utt and not utt[0] == ".":
423
                    total += (float(time2) - float(time1))
424
        return total
425
                    
426
    def tier_name(self):
427
        """
428
        @return:  Tier name of a given tier.
429
        """
430

    
431
        return self.nameid
432

    
433
    def classid(self):
434
        """
435
        @return:  Type of transcription on tier.
436
        """
437

    
438
        return self.classid
439

    
440
    def min_max(self):
441
        """
442
        @return:  (xmin, xmax) tuple for a given tier.
443
        """
444

    
445
        return (self.xmin, self.xmax)
446

    
447
    def __repr__(self):
448
        return "<%s \"%s\" (%.2f, %.2f) %.2f%%>" % (self.classid, self.nameid, self.xmin, self.xmax, 100*self.time()/self.t_time)
449

    
450
    def __str__(self):
451
        return self.__repr__() + "\n  " + "\n  ".join(" ".join(row) for row in self.simple_transcript)
452

    
453
def demo_TextGrid(demo_data):
454
    print "** Demo of the TextGrid class. **"
455

    
456
    fid = TextGrid(demo_data)
457
    print "Tiers:", fid.size
458

    
459
    for i, tier in enumerate(fid):
460
        print "\n***"
461
        print "Tier:", i + 1
462
        print tier
463

    
464
def demo():
465
    # Each demo demonstrates different TextGrid formats.
466
    print "Format 1"
467
    demo_TextGrid(demo_data1)
468
    print "\nFormat 2"
469
    demo_TextGrid(demo_data2)
470
    print "\nFormat 3"
471
    demo_TextGrid(demo_data3)
472

    
473

    
474
demo_data1 = """File type = "ooTextFile"
475
Object class = "TextGrid"
476

477
xmin = 0 
478
xmax = 2045.144149659864
479
tiers? <exists> 
480
size = 3 
481
item []: 
482
    item [1]:
483
        class = "IntervalTier" 
484
        name = "utterances" 
485
        xmin = 0 
486
        xmax = 2045.144149659864 
487
        intervals: size = 5 
488
        intervals [1]:
489
            xmin = 0 
490
            xmax = 2041.4217474125382 
491
            text = "" 
492
        intervals [2]:
493
            xmin = 2041.4217474125382 
494
            xmax = 2041.968276643991 
495
            text = "this" 
496
        intervals [3]:
497
            xmin = 2041.968276643991 
498
            xmax = 2042.5281632653062 
499
            text = "is" 
500
        intervals [4]:
501
            xmin = 2042.5281632653062 
502
            xmax = 2044.0487352585324 
503
            text = "a" 
504
        intervals [5]:
505
            xmin = 2044.0487352585324 
506
            xmax = 2045.144149659864 
507
            text = "demo" 
508
    item [2]:
509
        class = "TextTier" 
510
        name = "notes" 
511
        xmin = 0 
512
        xmax = 2045.144149659864 
513
        points: size = 3 
514
        points [1]:
515
            time = 2041.4217474125382 
516
            mark = ".begin_demo"
517
        points [2]:
518
            time = 2043.8338291031832
519
            mark = "voice gets quiet here" 
520
        points [3]:
521
            time = 2045.144149659864
522
            mark = ".end_demo" 
523
    item [3]:
524
        class = "IntervalTier" 
525
        name = "phones" 
526
        xmin = 0 
527
        xmax = 2045.144149659864
528
        intervals: size = 12
529
        intervals [1]:
530
            xmin = 0 
531
            xmax = 2041.4217474125382 
532
            text = "" 
533
        intervals [2]:
534
            xmin = 2041.4217474125382 
535
            xmax = 2041.5438290324326 
536
            text = "D"
537
        intervals [3]:
538
            xmin = 2041.5438290324326
539
            xmax = 2041.7321032910372
540
            text = "I"
541
        intervals [4]:
542
            xmin = 2041.7321032910372            
543
            xmax = 2041.968276643991 
544
            text = "s" 
545
        intervals [5]:
546
            xmin = 2041.968276643991 
547
            xmax = 2042.232189031843
548
            text = "I"
549
        intervals [6]:
550
            xmin = 2042.232189031843
551
            xmax = 2042.5281632653062 
552
            text = "z" 
553
        intervals [7]:
554
            xmin = 2042.5281632653062 
555
            xmax = 2044.0487352585324 
556
            text = "eI" 
557
        intervals [8]:
558
            xmin = 2044.0487352585324 
559
            xmax = 2044.2487352585324
560
            text = "dc"
561
        intervals [9]:
562
            xmin = 2044.2487352585324
563
            xmax = 2044.3102321849011
564
            text = "d"
565
        intervals [10]:
566
            xmin = 2044.3102321849011
567
            xmax = 2044.5748932104329
568
            text = "E"
569
        intervals [11]:
570
            xmin = 2044.5748932104329
571
            xmax = 2044.8329108578437
572
            text = "m"
573
        intervals [12]:
574
            xmin = 2044.8329108578437
575
            xmax = 2045.144149659864 
576
            text = "oU" 
577
"""
578

    
579
demo_data2 = """File type = "ooTextFile"
580
Object class = "TextGrid"
581

582
0
583
2.8
584
<exists>
585
2
586
"IntervalTier"
587
"utterances"
588
0
589
2.8
590
3
591
0
592
1.6229213249309031
593
""
594
1.6229213249309031
595
2.341428074708195
596
"demo"
597
2.341428074708195
598
2.8
599
""
600
"IntervalTier"
601
"phones"
602
0
603
2.8
604
6
605
0
606
1.6229213249309031
607
""
608
1.6229213249309031
609
1.6428291382019483
610
"dc"
611
1.6428291382019483
612
1.65372183721983721
613
"d"
614
1.65372183721983721
615
1.94372874328943728
616
"E"
617
1.94372874328943728
618
2.13821938291038210
619
"m"
620
2.13821938291038210
621
2.341428074708195
622
"oU"
623
2.341428074708195
624
2.8
625
""
626
"""
627

    
628
demo_data3 = """"Praat chronological TextGrid text file"
629
0 2.8   ! Time domain.
630
2   ! Number of tiers.
631
"IntervalTier" "utterances" 0 2.8
632
"IntervalTier" "utterances" 0 2.8
633
1 0 1.6229213249309031
634
""
635
2 0 1.6229213249309031
636
""
637
2 1.6229213249309031 1.6428291382019483
638
"dc"
639
2 1.6428291382019483 1.65372183721983721
640
"d"
641
2 1.65372183721983721 1.94372874328943728
642
"E"
643
2 1.94372874328943728 2.13821938291038210
644
"m"
645
2 2.13821938291038210 2.341428074708195
646
"oU"
647
1 1.6229213249309031 2.341428074708195
648
"demo"
649
1 2.341428074708195 2.8
650
""
651
2 2.341428074708195 2.8
652
""
653
"""
654

    
655
if __name__ == "__main__":
656
    demo()
657