forked from BigWigsMods/Transcriptor
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTranscriptor_Vanilla.lua
2178 lines (2051 loc) · 88.7 KB
/
Transcriptor_Vanilla.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
local Transcriptor = {}
local playerSpellBlacklist
local badSourcelessPlayerSpellList
local specialEvents
local data
do
local n, tbl = ...
playerSpellBlacklist = tbl.blacklist or {}
specialEvents = tbl.TIMERS_SPECIAL_EVENTS or {} -- TimersSpecialEvents.lua
data = tbl.TIMERS_SPECIAL_EVENTS_DATA or {} -- TimersSpecialEvents.lua
end
local logName = nil
local currentLog = nil
local logStartTime = nil
local logging = nil
local compareSuccess = nil
local compareUnitSuccess = nil
local compareEmotes = nil
local compareStart = nil
local compareAuraApplied = nil
local compareStartTime = nil
local collectNameplates = nil
local collectPlayerAuras = nil
local hiddenUnitAuraCollector = nil
local playerSpellCollector = nil
local hiddenAuraPermList = {
[5384] = true, -- Feign Death
[209997] = true, -- Play Dead (Hunter Pet)
}
local previousSpecialEvent = nil
local hiddenAuraEngageList = nil
local shouldLogFlags = false
local inEncounter, blockingRelease, limitingRes = false, false, false
local mineOrPartyOrRaid = 7 -- COMBATLOG_OBJECT_AFFILIATION_MINE + COMBATLOG_OBJECT_AFFILIATION_PARTY + COMBATLOG_OBJECT_AFFILIATION_RAID
local band = bit.band
local tinsert = table.insert
local format, find, strjoin = string.format, string.find, string.join
local tostring, tostringall = tostring, tostringall
local type, next, print = type, next, print
local date = date
local debugprofilestop, wipe = debugprofilestop, wipe
local C_Scenario, C_DeathInfo_GetSelfResurrectOptions, Enum = C_Scenario, C_DeathInfo.GetSelfResurrectOptions, Enum
local IsEncounterInProgress, IsEncounterLimitingResurrections, IsEncounterSuppressingRelease = IsEncounterInProgress, IsEncounterLimitingResurrections, IsEncounterSuppressingRelease
local IsAltKeyDown, C_Map_GetMapInfo = IsAltKeyDown, C_Map.GetMapInfo
local UnitInRaid, UnitInParty, UnitIsFriend, UnitCastingInfo, UnitChannelInfo = UnitInRaid, UnitInParty, UnitIsFriend, UnitCastingInfo, UnitChannelInfo
local UnitCanAttack, UnitExists, UnitIsVisible, UnitGUID, UnitClassification = UnitCanAttack, UnitExists, UnitIsVisible, UnitGUID, UnitClassification
local UnitName, UnitPower, UnitPowerMax, UnitPowerType, UnitHealth = UnitName, UnitPower, UnitPowerMax, UnitPowerType, UnitHealth
local UnitLevel, UnitCreatureType = UnitLevel, UnitCreatureType
local GetInstanceInfo = GetInstanceInfo
local GetZoneText, GetRealZoneText, GetSubZoneText, GetSpellInfo = GetZoneText, GetRealZoneText, GetSubZoneText, GetSpellInfo
local GetBestMapForUnit = C_Map.GetBestMapForUnit
-- GLOBALS: TranscriptDB BigWigsLoader DBM CLOSE SlashCmdList SLASH_TRANSCRIPTOR1 SLASH_TRANSCRIPTOR2 SLASH_TRANSCRIPTOR3 EasyMenu CloseDropDownMenus
-- GLOBALS: GetMapID GetBossID GetSectionID
do
local origPrint = print
function print(msg, ...)
return origPrint(format("|cffffff00%s|r", tostring(msg)), tostringall(...))
end
local origUnitName = UnitName
function UnitName(name)
return origUnitName(name) or "??"
end
end
local function MobId(guid, extra)
if not guid then return 1 end
local _, _, _, _, _, strId, unq = strsplit("-", guid)
if extra then
local id = tonumber(strId)
if id then
return strId.."-"..unq
else
return 1
end
else
return tonumber(strId) or 1
end
end
local function InsertSpecialEvent(name)
if type(name) == "function" then
name = name()
end
if not name then return end
local t = debugprofilestop()
previousSpecialEvent = {t, name}
if compareSuccess then
for id,tbl in next, compareSuccess do
for npcId, list in next, tbl do
list[#list+1] = {t, name}
end
end
end
if compareStart then
for id,tbl in next, compareStart do
for npcId, list in next, tbl do
list[#list+1] = {t, name}
end
end
end
if compareAuraApplied then
for id,tbl in next, compareAuraApplied do
for npcId, list in next, tbl do
list[#list+1] = {t, name}
end
end
end
if compareUnitSuccess then
for id,tbl in next, compareUnitSuccess do
for npcId, list in next, tbl do
list[#list+1] = {t, name}
end
end
end
if compareEmotes then
for id,tbl in next, compareEmotes do
for npcName, list in next, tbl do
list[#list+1] = {t, name}
end
end
end
end
--------------------------------------------------------------------------------
-- Utility
--
function GetMapArtID(name)
name = name:lower()
for i=1,3000 do
local fetchedTbl = C_Map.GetMapInfo(i)
if fetchedTbl and fetchedTbl.name then
local lowerFetchedName = fetchedTbl.name:lower()
if find(lowerFetchedName, name, nil, true) then
print(fetchedTbl.name..": "..i)
end
end
end
end
function GetInstanceID(name)
name = name:lower()
for i=1,3000 do
local fetchedName = GetRealZoneText(i)
local lowerFetchedName = fetchedName:lower()
if find(lowerFetchedName, name, nil, true) then
print(fetchedName..": "..i)
end
end
end
--function GetBossID(name)
-- name = name:lower()
-- for i=1,3000 do
-- local fetchedName = EJ_GetEncounterInfo(i)
-- if fetchedName then
-- local lowerFetchedName = fetchedName:lower()
-- if find(lowerFetchedName, name, nil, true) then
-- print(fetchedName..": "..i)
-- end
-- end
-- end
--end
--function GetSectionID(name)
-- name = name:lower()
-- for i=1,15000 do
-- local tbl = C_EncounterJournal_GetSectionInfo(i)
-- if tbl then
-- local fetchedName = tbl.title
-- local lowerFetchedName = fetchedName:lower()
-- if find(lowerFetchedName, name, nil, true) then
-- print(fetchedName..": "..i)
-- end
-- end
-- end
--end
do
-- Create UI spell display, copied from BasicChatMods
local frame, editBox = {}, {}
for i = 1, 4 do
if BackdropTemplateMixin then -- XXX Shadowlands change
frame[i] = CreateFrame("Frame", nil, UIParent, "BackdropTemplate")
frame[i].backdropInfo = {bgFile = "Interface\\DialogFrame\\UI-DialogBox-Background",
edgeFile = "Interface\\DialogFrame\\UI-DialogBox-Border",
tile = true, tileSize = 16, edgeSize = 16,
insets = {left = 1, right = 1, top = 1, bottom = 1}}
frame[i]:ApplyBackdrop()
else
frame[i] = CreateFrame("Frame", nil, UIParent)
frame[i]:SetBackdrop({bgFile = "Interface\\DialogFrame\\UI-DialogBox-Background",
edgeFile = "Interface\\DialogFrame\\UI-DialogBox-Border",
tile = true, tileSize = 16, edgeSize = 16,
insets = {left = 1, right = 1, top = 1, bottom = 1}}
)
end
frame[i]:SetBackdropColor(0,0,0,1)
frame[i]:SetWidth(650)
frame[i]:SetHeight(450)
frame[i]:Hide()
frame[i]:SetFrameStrata("DIALOG")
local scrollArea = CreateFrame("ScrollFrame", "TranscriptorDevScrollArea"..i, frame[i], "UIPanelScrollFrameTemplate")
scrollArea:SetPoint("TOPLEFT", frame[i], "TOPLEFT", 8, -5)
scrollArea:SetPoint("BOTTOMRIGHT", frame[i], "BOTTOMRIGHT", -30, 5)
editBox[i] = CreateFrame("EditBox", nil, frame[i])
editBox[i]:SetMultiLine(true)
editBox[i]:EnableMouse(true)
editBox[i]:SetAutoFocus(false)
editBox[i]:SetFontObject(ChatFontNormal)
editBox[i]:SetWidth(620)
editBox[i]:SetHeight(450)
editBox[i]:SetScript("OnEscapePressed", function(f) f:GetParent():GetParent():Hide() f:SetText("") end)
if i % 2 ~= 0 then
editBox[i]:SetScript("OnHyperlinkLeave", GameTooltip_Hide)
editBox[i]:SetScript("OnHyperlinkEnter", function(self, link, text)
if link and find(link, "spell", nil, true) then
local spellId = link:match("(%d+)")
if spellId then
GameTooltip:SetOwner(frame[i], "ANCHOR_LEFT", 0, -500)
GameTooltip:SetSpellByID(spellId)
end
end
end)
editBox[i]:SetHyperlinksEnabled(true)
end
scrollArea:SetScrollChild(editBox[i])
local close = CreateFrame("Button", nil, frame[i], "UIPanelCloseButton")
close:SetPoint("TOPRIGHT", frame[i], "TOPRIGHT", 0, 25)
end
local function GetLogSpells()
if InCombatLockdown() or UnitAffectingCombat("player") or IsFalling() then return end
local total, totalSorted = {}, {}
local auraTbl, castTbl, summonTbl = {}, {}, {}
local aurasSorted, castsSorted, summonSorted = {}, {}, {}
local playerCastList = {}
local ignoreList = {
[43681] = true, -- Inactive (PvP)
[94028] = true, -- Inactive (PvP)
[66186] = true, -- Napalm (IoC PvP)
[66195] = true, -- Napalm (IoC PvP)
[66268] = true, -- Place Seaforium Bomb (IoC PvP)
[66271] = true, -- Carrying Seaforium (IoC PvP)
[66456] = true, -- Glaive Throw (IoC PvP)
[66518] = true, -- Airship Cannon (IoC PvP)
[66541] = true, -- Incendiary Rocket (IoC PvP)
[66542] = true, -- Incendiary Rocket (IoC PvP)
[66657] = true, -- Parachute (IoC PvP)
[66674] = true, -- Place Huge Seaforium Bomb (IoC PvP)
[67195] = true, -- Blade Salvo (IoC PvP)
[67200] = true, -- Blade Salvo (IoC PvP)
[67440] = true, -- Hurl Boulder (IoC PvP)
[67441] = true, -- Ram (IoC PvP)
[67452] = true, -- Rocket Blast (IoC PvP)
[67461] = true, -- Fire Cannon (IoC PvP)
[67796] = true, -- Ram (IoC PvP)
[67797] = true, -- Steam Rush (IoC PvP)
[68077] = true, -- Repair Cannon (IoC PvP)
[68298] = true, -- Parachute (IoC PvP)
[68377] = true, -- Carrying Huge Seaforium (IoC PvP)
[270620] = true, -- Psionic Blast (Zek'voz/Uldir || Mind Controlled player)
[272407] = true, -- Oblivion Sphere (Mythrax/Uldir || Orb spawning on player)
[263372] = true, -- Power Matrix (G'huun/Uldir || Holding the orb)
[263436] = true, -- Imperfect Physiology (G'huun/Uldir || After the orb)
[263373] = true, -- Deposit Power Matrix (G'huun/Uldir)
[263416] = true, -- Throw Power Matrix (G'huun/Uldir)
[269455] = true, -- Collect Power Matrix (G'huun/Uldir)
}
local npcIgnoreList = {
[154297] = true, -- Ankoan Bladesman
[154304] = true, -- Waveblade Shaman
[150202] = true, -- Waveblade Hunter
}
local events = {
"SPELL_AURA_[AR][^#]+#(%d+)#([^#]+%-[^#]+)#([^#]+)#([^#]*)#([^#]+)#(%d+)#([^#]+)#", -- SPELL_AURA_[AR] to filter _BROKEN
"SPELL_CAST_[^#]+#(%d+)#([^#]+%-[^#]+)#([^#]+)#([^#]*)#([^#]+)#(%d+)#([^#]+)#",
"SPELL_SUMMON#(%d+)#([^#]+%-[^#]+)#([^#]+)#([^#]*)#([^#]+)#(%d+)#([^#]+)#"
}
local tables = {
auraTbl,
castTbl,
summonTbl,
}
local sortedTables = {
aurasSorted,
castsSorted,
summonSorted,
}
for logName, logTbl in next, TranscriptDB do
if type(logTbl) == "table" then
if logTbl.total then
for i=1, #logTbl.total do
local text = logTbl.total[i]
for j = 1, 3 do
local flagsText, srcGUID, name, destGUID, tarName, idText = text:match(events[j])
local id = tonumber(idText)
local flags = tonumber(flagsText)
local tbl = tables[j]
local sortedTbl = sortedTables[j]
if id and flags and band(flags, mineOrPartyOrRaid) ~= 0 and not ignoreList[id] and not playerSpellBlacklist[id] and #sortedTbl < 15 then -- Check total to avoid duplicates and lock to a max of 15 for sanity
if not total[id] or destGUIDType ~= "" then
local srcGUIDType, _, _, _, _, npcIdStr = strsplit("-", srcGUID)
local npcId = tonumber(npcIdStr)
if not npcIgnoreList[npcId] then
local destGUIDType = strsplit("-", destGUID)
local trim = destGUID and find(destGUID, "^P[le][at]")
if srcGUID == destGUID then
tbl[id] = "|cFF81BEF7".. name:gsub("%-.+", "*") .."(".. srcGUIDType ..") >> ".. (trim and tarName:gsub("%-.+", "*") or tarName) .."(".. destGUIDType ..")|r"
else
tbl[id] = "|cFF3ADF00".. name:gsub("%-.+", "*") .."(".. srcGUIDType ..") >> ".. (trim and tarName:gsub("%-.+", "*") or tarName) .."(".. destGUIDType ..")|r"
end
if not total[id] then
total[id] = true
sortedTbl[#sortedTbl+1] = id
end
end
end
end
end
end
end
if logTbl.TIMERS and logTbl.TIMERS.PLAYER_SPELLS then
for i=1, #logTbl.TIMERS.PLAYER_SPELLS do
local text = logTbl.TIMERS.PLAYER_SPELLS[i]
local spellId, _, _, player = strsplit("#", text)
local id = tonumber(spellId)
if id and not playerSpellBlacklist[id] and not playerCastList[id] and not total[id] then
playerCastList[id] = player
total[id] = true
end
end
end
end
end
sort(aurasSorted)
local text = "-- AURAS\n"
for i = 1, #aurasSorted do
local id = aurasSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, auraTbl[id])
end
sort(castsSorted)
text = text.. "\n-- CASTS\n"
for i = 1, #castsSorted do
local id = castsSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, castTbl[id])
end
sort(summonSorted)
text = text.. "\n-- SUMMONS\n"
for i = 1, #summonSorted do
local id = summonSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, summonTbl[id])
end
text = text.. "\n-- PLAYER_CASTS\n"
for k, v in next, playerCastList do
local name = GetSpellInfo(k)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, k, k, name, v)
end
-- Display newly found spells for analysis
editBox[1]:SetText(text)
frame[1]:ClearAllPoints()
frame[1]:SetPoint("BOTTOMRIGHT", UIParent, "CENTER")
frame[1]:Show()
for k, v in next, playerSpellBlacklist do
if GetSpellInfo(k) then -- Filter out removed spells when a new patch hits
total[k] = true
end
end
for k, v in next, total do
totalSorted[#totalSorted+1] = k
end
sort(totalSorted)
text = "local n, tbl = ...\ntbl.blacklist = {\n"
for i = 1, #totalSorted do
local id = totalSorted[i]
local name = GetSpellInfo(id)
text = format("%s[%d] = true, -- %s\n", text, id, name)
end
text = text .. "}"
-- Display full blacklist for copying into Transcriptor
editBox[2]:SetText(text)
frame[2]:ClearAllPoints()
frame[2]:SetPoint("BOTTOMLEFT", UIParent, "CENTER")
frame[2]:Show()
---------------------------------------------------------------------------------
-- SOURCELESS
---------------------------------------------------------------------------------
total, totalSorted = {}, {}
auraTbl, castTbl, summonTbl = {}, {}, {}
aurasSorted, castsSorted, summonSorted = {}, {}, {}
ignoreList = {
[236283] = true, -- Belac's Prisoner
[236516] = true, -- Twilight Volley
[236519] = true, -- Moon Burn
[237351] = true, -- Lunar Barrage
[240706] = true, -- Arcane Ward
[241032] = true, -- Desolation of the Moon
[241169] = true, -- Umbra Destruction
[236011] = true, -- Tormented Cries
[236241] = true, -- Soul Rot
[236459] = true, -- Soulbind
[235534] = true, -- Creator's Grace
[235538] = true, -- Demon's Vigor
[236420] = true, -- Aegwynn's Ward
[240209] = true, -- Unstable Soul
[240249] = true, -- Molten Fel
[243267] = true, -- Velen's Light
[231768] = true, -- Drenching Waters
[231770] = true, -- Drenched
[232732] = true, -- Slicing Tornado
[232913] = true, -- Befouling Ink
[234621] = true, -- Devouring Maw
[236329] = true, -- Star Burn
[243294] = true, -- Fel Slicer
[238442] = true, -- Spear of Anguish
[241593] = true, -- Aegwynn's Ward
[236555] = true, -- Deceiver's Veil
[241721] = true, -- Illidan's Sightless Gaze
[241822] = true, -- Choking Shadow
[242696] = true, -- Deceiver's Veil
[230345] = true, -- Crashing Comet
[230348] = true, -- Fel Pool
[233901] = true, -- Suffocating Dark
}
local eventsNoSource = {
"SPELL_AURA_[AR][^#]+#(%d+)##([^#]+)#([^#]+%-[^#]+)#([^#]+)#(%d+)#([^#]+)#", -- SPELL_AURA_[AR] to filter _BROKEN
"SPELL_CAST_[^#]+#(%d+)##([^#]+)#([^#]+%-[^#]+)#([^#]+)#(%d+)#([^#]+)#",
"SPELL_SUMMON#(%d+)##([^#]+)#([^#]+%-[^#]+)#([^#]+)#(%d+)#([^#]+)#"
}
tables = {
auraTbl,
castTbl,
summonTbl,
}
sortedTables = {
aurasSorted,
castsSorted,
summonSorted,
}
for logName, logTbl in next, TranscriptDB do
if type(logTbl) == "table" and logTbl.total then
for i=1, #logTbl.total do
local text = logTbl.total[i]
for j = 1, 3 do
local flagsText, name, destGUID, tarName, idText = text:match(eventsNoSource[j])
local id = tonumber(idText)
local flags = tonumber(flagsText)
local tbl = tables[j]
local sortedTbl = sortedTables[j]
if name == "nil" and id and flags and band(flags, mineOrPartyOrRaid) ~= 0 and not ignoreList[id] and not badSourcelessPlayerSpellList[id] and not total[id] and #sortedTbl < 15 then -- Check total to avoid duplicates and lock to a max of 15 for sanity
tbl[id] = tarName:gsub("%-.+", "*")
total[id] = true
sortedTbl[#sortedTbl+1] = id
end
end
end
end
end
sort(aurasSorted)
local text = "-- AURAS\n"
for i = 1, #aurasSorted do
local id = aurasSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, auraTbl[id])
end
sort(castsSorted)
text = text.. "\n-- CASTS\n"
for i = 1, #castsSorted do
local id = castsSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, castTbl[id])
end
sort(summonSorted)
text = text.. "\n-- SUMMONS\n"
for i = 1, #summonSorted do
local id = summonSorted[i]
local name = GetSpellInfo(id)
text = format("%s%d || |cFFFFFF00|Hspell:%d|h%s|h|r || %s\n", text, id, id, name, summonTbl[id])
end
-- Display newly found spells for analysis
editBox[3]:SetText(text)
frame[3]:ClearAllPoints()
frame[3]:SetPoint("TOPRIGHT", UIParent, "CENTER")
frame[3]:Show()
for k, v in next, badSourcelessPlayerSpellList do
total[k] = true
end
for k, v in next, total do
totalSorted[#totalSorted+1] = k
end
sort(totalSorted)
text = ""
for i = 1, #totalSorted do
local id = totalSorted[i]
local name = GetSpellInfo(id)
text = format("%s[%d] = true, -- %s\n", text, id, name)
end
-- Display full blacklist for copying into Transcriptor
editBox[4]:SetText(text)
frame[4]:ClearAllPoints()
frame[4]:SetPoint("TOPLEFT", UIParent, "CENTER")
frame[4]:Show()
end
SlashCmdList["GETSPELLS"] = GetLogSpells
SLASH_GETSPELLS1 = "/getspells"
end
--------------------------------------------------------------------------------
-- Localization
--
local L = {}
L["Remember to stop and start Transcriptor between each wipe or boss kill to get the best logs."] = "Remember to stop and start Transcriptor between each wipe or boss kill to get the best logs."
L["You are already logging an encounter."] = "You are already logging an encounter."
L["Beginning Transcript: "] = "Beginning Transcript: "
L["You are not logging an encounter."] = "You are not logging an encounter."
L["Ending Transcript: "] = "Ending Transcript: "
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."
L["All transcripts cleared."] = "All transcripts cleared."
L["You can't clear your transcripts while logging an encounter."] = "You can't clear your transcripts while logging an encounter."
L["|cff696969Idle|r"] = "|cff696969Idle|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."
L["|cffFF0000Recording|r"] = "|cffFF0000Recording|r"
L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - Disabled Events"
do
local locale = GetLocale()
if locale == "deDE" then
L["Remember to stop and start Transcriptor between each wipe or boss kill to get the best logs."] = "Um die besten Logs zu bekommen, solltest du Transcriptor zwischen Wipes oder Bosskills stoppen bzw. starten."
L["You are already logging an encounter."] = "Du zeichnest bereits einen Begegnung auf."
L["Beginning Transcript: "] = "Beginne Aufzeichnung: "
L["You are not logging an encounter."] = "Du zeichnest keine Begegnung auf."
L["Ending Transcript: "] = "Beende Aufzeichnung: "
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "Aufzeichnungen werden gespeichert nach WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua sobald du reloggst oder das Interface neu lädst."
L["All transcripts cleared."] = "Alle Aufzeichnungen gelöscht."
L["You can't clear your transcripts while logging an encounter."] = "Du kannst deine Aufzeichnungen nicht löschen, während du eine Begegnung aufnimmst."
L["|cff696969Idle|r"] = "|cff696969Leerlauf|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55fKlicken|r, um eine Aufzeichnung zu starten oder zu stoppen. |cffeda55fRechts-Klicken|r, um Events zu konfigurieren. |cffeda55fAlt-Mittel-Klicken|r, um alle Aufzeichnungen zu löschen."
L["|cffFF0000Recording|r"] = "|cffFF0000Aufzeichnung|r"
--L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - Disabled Events"
elseif locale == "zhTW" then
L["You are already logging an encounter."] = "你已經準備記錄戰鬥"
L["Beginning Transcript: "] = "開始記錄於: "
L["You are not logging an encounter."] = "你不處於記錄狀態"
L["Ending Transcript: "] = "結束記錄於: "
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "記錄儲存於 WoW\\WTF\\Account\\<名字>\\SavedVariables\\Transcriptor.lua"
L["You are not logging an encounter."] = "你沒有記錄此次戰鬥"
L["All transcripts cleared."] = "所有記錄已清除"
L["You can't clear your transcripts while logging an encounter."] = "正在記錄中,你不能清除。"
L["|cffFF0000Recording|r: "] = "|cffFF0000記錄中|r: "
L["|cff696969Idle|r"] = "|cff696969閒置|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55f點擊|r開始/停止記錄戰鬥"
L["|cffFF0000Recording|r"] = "|cffFF0000記錄中|r"
--L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - Disabled Events"
elseif locale == "zhCN" then
L["You are already logging an encounter."] = "你已经准备记录战斗"
L["Beginning Transcript: "] = "开始记录于: "
L["You are not logging an encounter."] = "你不处于记录状态"
L["Ending Transcript: "] = "结束记录于:"
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "记录保存于WoW\\WTF\\Account\\<名字>\\SavedVariables\\Transcriptor.lua中,你可以上传于Cwowaddon.com论坛,提供最新的BOSS数据."
L["You are not logging an encounter."] = "你没有记录此次战斗"
L["Added Note: "] = "添加书签于: "
L["All transcripts cleared."] = "所有记录已清除"
L["You can't clear your transcripts while logging an encounter."] = "正在记录中,你不能清除."
L["|cffFF0000Recording|r: "] = "|cffFF0000记录中|r: "
L["|cff696969Idle|r"] = "|cff696969空闲|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55f点击|r开始/停止记录战斗."
L["|cffFF0000Recording|r"] = "|cffFF0000记录中|r"
--L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - Disabled Events"
elseif locale == "koKR" then
L["Remember to stop and start Transcriptor between each wipe or boss kill to get the best logs."] = "최상의 기록을 얻으려면 전멸이나 우두머리 처치 후에 Transcriptor를 중지하고 시작하는 걸 기억하세요."
L["You are already logging an encounter."] = "이미 우두머리 전투를 기록 중입니다."
L["Beginning Transcript: "] = "기록 시작: "
L["You are not logging an encounter."] = "우두머리 전투를 기록하고 있지 않습니다."
L["Ending Transcript: "] = "기록 종료: "
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "재기록하거나 사용자 인터페이스를 다시 불러오면 WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua에 기록이 저장됩니다."
L["All transcripts cleared."] = "모든 기록이 초기화되었습니다."
L["You can't clear your transcripts while logging an encounter."] = "우두머리 전투를 기록 중일 때는 기록을 초기화 할 수 없습니다."
L["|cff696969Idle|r"] = "|cff696969대기|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55f클릭|r - 기록을 시작하거나 중지합니다.\n|cffeda55f오른쪽-클릭|r - 이벤트를 구성합니다.\n|cffeda55fAlt-가운데 클릭|r - 저장된 모든 기록을 초기화합니다."
L["|cffFF0000Recording|r"] = "|cffFF0000기록 중|r"
L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - 비활성된 이벤트"
elseif locale == "ruRU" then
L["Remember to stop and start Transcriptor between each wipe or boss kill to get the best logs."] = "Чтобы получить лучшие записи боя, не забудьте остановить и запустить Transcriptor между вайпом или убийством босса."
L["You are already logging an encounter."] = "Вы уже записываете бой."
L["Beginning Transcript: "] = "Начало записи: "
L["You are not logging an encounter."] = "Вы не записываете бой."
L["Ending Transcript: "] = "Окончание записи: "
L["Logs will probably be saved to WoW\\WTF\\Account\\<name>\\SavedVariables\\Transcriptor.lua once you relog or reload the user interface."] = "Записи боя будут записаны в WoW\\WTF\\Account\\<название>\\SavedVariables\\Transcriptor.lua после того как вы перезайдете или перезагрузите пользовательский интерфейс."
L["All transcripts cleared."] = "Все записи очищены."
L["You can't clear your transcripts while logging an encounter."] = "Вы не можете очистить ваши записи пока идет запись боя."
L["|cff696969Idle|r"] = "|cff696969Ожидание|r"
L["|cffeda55fClick|r to start or stop transcribing. |cffeda55fRight-Click|r to configure events. |cffeda55fAlt-Middle Click|r to clear all stored transcripts."] = "|cffeda55fЛКМ|r - запустить или остановить запись.\n|cffeda55fПКМ|r - настройка событий.\n|cffeda55fAlt-СКМ|r - очистить все сохраненные записи."
L["|cffFF0000Recording|r"] = "|cffFF0000Запись|r"
--L["|cFFFFD200Transcriptor|r - Disabled Events"] = "|cFFFFD200Transcriptor|r - Disabled Events"
end
end
--------------------------------------------------------------------------------
-- Events
--
local eventFrame = CreateFrame("Frame")
eventFrame:Hide()
local sh = {}
--[[
UIWidget
see https://wow.gamepedia.com/UPDATE_UI_WIDGET
widgetID =
The Black Morass: 507 (health), 527 (waves)
The Violet Hold (WotLK): 565 (health), 566 (waves)
]]
do
local blackList = {
[1309] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1309)
[1311] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1311)
[1313] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1313)
[1315] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1315)
[1317] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1317)
[1319] = true, -- C_UIWidgetManager.GetStatusBarWidgetVisualizationInfo(1319)
--[[ Shadowlands - Castle Nathria ]]--
[2436] = true, -- Lady Inerva Darkvein, static icon (widgetType:13)
[2437] = true, -- Lady Inerva Darkvein, static icon (widgetType:13)
[2438] = true, -- Lady Inerva Darkvein, static icon (widgetType:13)
[2439] = true, -- Lady Inerva Darkvein, static icon (widgetType:13)
}
-- Each widgetType has a different function to get all data for the widget.
-- Those function names are consistent with the enum.
local widgetFuncs = {}
for name,n in pairs(Enum.UIWidgetVisualizationType) do
widgetFuncs[n] = C_UIWidgetManager["Get"..name.. "WidgetVisualizationInfo"] or C_UIWidgetManager["Get"..name.."VisualizationInfo"]
end
function sh.UPDATE_UI_WIDGET(tbl)
if blackList[tbl.widgetID] then return end
-- Store widget data, so we're able to only add changes to the log
if not currentLog.WIDGET then currentLog.WIDGET = {} end
local txt
if not currentLog.WIDGET[tbl.widgetID] then -- New widget in this log
currentLog.WIDGET[tbl.widgetID] = {}
for k, v in next, tbl do
if not txt then
txt = format("%s:%s", k, v)
else
txt = format("%s, %s:%s", txt, k, v)
end
end
else -- Shorter output with just the id
txt = format("%s:%s", "widgetID", tbl.widgetID)
end
if tbl.widgetType and widgetFuncs[tbl.widgetType] then
local t = widgetFuncs[tbl.widgetType](tbl.widgetID)
if t then
for k, v in next, t do
if type(v) == "table" then -- We don't care about data in tables
v = "table"
elseif type(v) == "boolean" then -- Needs to be done manually in Lua 5.1
v = tostring(v)
end
-- Only add data to log if it's new or it changed to reduce spam
if not currentLog.WIDGET[tbl.widgetID][k] or currentLog.WIDGET[tbl.widgetID][k] ~= v then
currentLog.WIDGET[tbl.widgetID][k] = v
txt = format("%s, %s:%s", txt, k, v)
end
end
end
end
return txt
end
end
do
badSourcelessPlayerSpellList = {
[81782] = true, -- Power Word: Barrier
}
local badPlayerFilteredEvents = {
["SPELL_CAST_SUCCESS"] = true,
["SPELL_AURA_APPLIED"] = true,
["SPELL_AURA_APPLIED_DOSE"] = true,
["SPELL_AURA_REFRESH"] = true,
["SPELL_AURA_REMOVED"] = true,
["SPELL_AURA_REMOVED_DOSE"] = true,
["SPELL_CAST_START"] = true,
["SPELL_SUMMON"] = true,
--"<87.10 17:55:03> [CLEU] SPELL_AURA_BROKEN_SPELL#Creature-0-3771-1676-28425-118022-000004A6B5#Infernal Chaosbringer#Player-XYZ#XYZ#115191#Stealth#242906#Immolation Aura", -- [148]
--"<498.56 22:02:38> [CLEU] SPELL_AURA_BROKEN_SPELL#Creature-0-3895-1676-10786-106551-00008631CC-TSGuardian#Hati#Creature-0-3895-1676-10786-120697-000086306F#Worshiper of Elune#206961#Tremble Before Me#118459#Beast Cleave", -- [8039]
--["SPELL_AURA_BROKEN_SPELL"] = true,
}
local badPlayerEvents = {
["SPELL_DAMAGE"] = true,
["SPELL_MISSED"] = true,
["SWING_DAMAGE"] = true,
["SWING_MISSED"] = true,
["RANGE_DAMAGE"] = true,
["RANGE_MISSED"] = true,
["SPELL_PERIODIC_DAMAGE"] = true,
["SPELL_PERIODIC_MISSED"] = true,
["DAMAGE_SPLIT"] = true,
["SPELL_HEAL"] = true,
["SPELL_PERIODIC_HEAL"] = true,
["SPELL_ENERGIZE"] = true,
["SPELL_PERIODIC_ENERGIZE"] = true,
}
local badEvents = {
["SPELL_ABSORBED"] = true,
["SPELL_CAST_FAILED"] = true,
}
local guardian = 8192 -- COMBATLOG_OBJECT_TYPE_GUARDIAN
local dmgCache, dmgPrdcCache = {}, {}
local CombatLogGetCurrentEventInfo = CombatLogGetCurrentEventInfo
-- Note some things we are trying to avoid filtering:
-- BRF/Kagraz - Player damage with no source "SPELL_DAMAGE##nil#Player-GUID#PLAYER#154938#Molten Torrent#"
-- HFC/Socrethar - Player cast on friendly vehicle "SPELL_CAST_SUCCESS#Player-GUID#PLAYER#Vehicle-0-3151-1448-8853-90296-00001D943C#Soulbound Construct#190466#Incomplete Binding"
-- HFC/Zakuun - Player boss debuff cast on self "SPELL_AURA_APPLIED#Player-GUID#PLAYER#Player-GUID#PLAYER#189030#Befouled#DEBUFF#"
-- ToS/Sisters - Boss pet marked as guardian "SPELL_CAST_SUCCESS#Creature-0-3895-1676-10786-119205-0000063360#Moontalon##nil#236697#Deathly Screech"
function sh.COMBAT_LOG_EVENT_UNFILTERED()
local timeStamp, event, caster, sourceGUID, sourceName, sourceFlags, sourceRaidFlags, destGUID, destName, destFlags, destRaidFlags, spellId, spellName, _, extraSpellId, amount = CombatLogGetCurrentEventInfo()
--if (event == "SPELL_AURA_APPLIED" or event == "SPELL_AURA_REMOVED") and not hiddenAuraPermList[spellId] then
-- hiddenAuraPermList[spellId] = true
--end
if badEvents[event] or
(event == "UNIT_DIED" and band(destFlags, mineOrPartyOrRaid) ~= 0 and band(destFlags, guardian) == guardian) or -- Guardian deaths, player deaths can explain debuff removal
(sourceName and badPlayerEvents[event] and band(sourceFlags, mineOrPartyOrRaid) ~= 0) or
(sourceName and badPlayerFilteredEvents[event] and playerSpellBlacklist[spellId] and band(sourceFlags, mineOrPartyOrRaid) ~= 0) or
(not sourceName and destName and badPlayerFilteredEvents[event] and badSourcelessPlayerSpellList[spellId] and band(destFlags, mineOrPartyOrRaid) ~= 0) or
-- Temporary (hopefully) hacks, due to srcFlags not correctly attributing as mineOrPartyOrRaid
(spellId == 120694 and sourceName == "Beast" and band(destFlags, mineOrPartyOrRaid) ~= 0) -- Dire Beast from summoned creature to player
then
return
else
--if (sourceName and badPlayerFilteredEvents[event] and playerSpellBlacklist[spellId] and band(sourceFlags, mineOrPartyOrRaid) == 0) then
-- print("Transcriptor:", sourceName..":"..MobId(sourceGUID), "used spell", spellName..":"..spellId, "in event", event, "but isn't in our group.")
--end
if event == "SPELL_CAST_SUCCESS" and (not sourceName or (band(sourceFlags, mineOrPartyOrRaid) == 0 and not find(sourceGUID, "Player", nil, true))) then
if not compareSuccess then compareSuccess = {} end
if not compareSuccess[spellName] then compareSuccess[spellName] = {} end
local npcId = MobId(sourceGUID, true)
if not compareSuccess[spellName][npcId] then
if previousSpecialEvent then
compareSuccess[spellName][npcId] = {{compareStartTime, previousSpecialEvent[1], previousSpecialEvent[2]}}
else
compareSuccess[spellName][npcId] = {compareStartTime}
end
end
compareSuccess[spellName][npcId][#compareSuccess[spellName][npcId]+1] = debugprofilestop()
end
if event == "SPELL_CAST_START" and (not sourceName or (band(sourceFlags, mineOrPartyOrRaid) == 0 and not find(sourceGUID, "Player", nil, true))) then
if not compareStart then compareStart = {} end
if not compareStart[spellName] then compareStart[spellName] = {} end
local npcId = MobId(sourceGUID, true)
if not compareStart[spellName][npcId] then
if previousSpecialEvent then
compareStart[spellName][npcId] = {{compareStartTime, previousSpecialEvent[1], previousSpecialEvent[2]}}
else
compareStart[spellName][npcId] = {compareStartTime}
end
end
compareStart[spellName][npcId][#compareStart[spellName][npcId]+1] = debugprofilestop()
end
if event == "SPELL_AURA_APPLIED" and (not sourceName or (band(sourceFlags, mineOrPartyOrRaid) == 0 and not find(sourceGUID, "Player", nil, true))) then
if not compareAuraApplied then compareAuraApplied = {} end
if not compareAuraApplied[spellName] then compareAuraApplied[spellName] = {} end
local npcId = MobId(sourceGUID, true)
if not compareAuraApplied[spellName][npcId] then
if previousSpecialEvent then
compareAuraApplied[spellName][npcId] = {{compareStartTime, previousSpecialEvent[1], previousSpecialEvent[2]}}
else
compareAuraApplied[spellName][npcId] = {compareStartTime}
end
end
compareAuraApplied[spellName][npcId][#compareAuraApplied[spellName][npcId]+1] = debugprofilestop()
end
if sourceName and badPlayerFilteredEvents[event] and band(sourceFlags, mineOrPartyOrRaid) ~= 0 then
--if not collectPlayerAuras then collectPlayerAuras = {} end
--if not collectPlayerAuras[spellId] then collectPlayerAuras[spellId] = {} end
--if not collectPlayerAuras[spellId][event] then collectPlayerAuras[spellId][event] = true end
end
if event == "UNIT_DIED" then
local name = specialEvents.UNIT_DIED[MobId(destGUID)]
if name then
InsertSpecialEvent(name)
end
elseif specialEvents[event] and specialEvents[event][spellId] then
local name = specialEvents[event][spellId][MobId(sourceGUID)]
if name then
InsertSpecialEvent(name)
end
end
if event == "SPELL_DAMAGE" or event == "SPELL_MISSED" then
if dmgPrdcCache.spellName then
if dmgPrdcCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.event, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.destGUID, dmgPrdcCache.destName, dmgPrdcCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_PERIODIC_DAMAGE[CONDENSED]#%s#%s Targets#%d#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.count, dmgPrdcCache.spellName)
end
dmgPrdcCache.spellName = nil
end
if spellName == dmgCache.spellName then
if timeStamp - dmgCache.timeStamp > 0.2 then
if dmgCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgCache.timeStop, dmgCache.time, dmgCache.event, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.destGUID, dmgCache.destName, dmgCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgCache.timeStop, dmgCache.time, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.count, dmgCache.spellName)
end
dmgCache.sourceGUID = sourceGUID
dmgCache.sourceName = sourceName or "nil"
dmgCache.spellName = spellName
dmgCache.timeStop = (debugprofilestop() / 1000) - logStartTime
dmgCache.time = date("%H:%M:%S")
dmgCache.timeStamp = timeStamp
dmgCache.count = 1
dmgCache.event = event
dmgCache.destGUID = destGUID
dmgCache.destName = destName
else
dmgCache.count = dmgCache.count + 1
end
else
if dmgCache.spellName then
if dmgCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgCache.timeStop, dmgCache.time, dmgCache.event, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.destGUID, dmgCache.destName, dmgCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgCache.timeStop, dmgCache.time, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.count, dmgCache.spellName)
end
end
dmgCache.sourceGUID = sourceGUID
dmgCache.sourceName = sourceName or "nil"
dmgCache.spellName = spellName
dmgCache.timeStop = (debugprofilestop() / 1000) - logStartTime
dmgCache.time = date("%H:%M:%S")
dmgCache.timeStamp = timeStamp
dmgCache.count = 1
dmgCache.event = event
dmgCache.destGUID = destGUID
dmgCache.destName = destName
end
elseif event == "SPELL_PERIODIC_DAMAGE" or event == "SPELL_PERIODIC_MISSED" then
if dmgCache.spellName then
if dmgCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgCache.timeStop, dmgCache.time, dmgCache.event, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.destGUID, dmgCache.destName, dmgCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgCache.timeStop, dmgCache.time, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.count, dmgCache.spellName)
end
dmgCache.spellName = nil
end
if spellName == dmgPrdcCache.spellName then
if timeStamp - dmgPrdcCache.timeStamp > 0.2 then
if dmgPrdcCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.event, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.destGUID, dmgPrdcCache.destName, dmgPrdcCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_PERIODIC_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.count, dmgPrdcCache.spellName)
end
dmgPrdcCache.sourceGUID = sourceGUID
dmgPrdcCache.sourceName = sourceName or "nil"
dmgPrdcCache.spellName = spellName
dmgPrdcCache.timeStop = (debugprofilestop() / 1000) - logStartTime
dmgPrdcCache.time = date("%H:%M:%S")
dmgPrdcCache.timeStamp = timeStamp
dmgPrdcCache.count = 1
dmgPrdcCache.event = event
dmgPrdcCache.destGUID = destGUID
dmgPrdcCache.destName = destName
else
dmgPrdcCache.count = dmgPrdcCache.count + 1
end
else
if dmgPrdcCache.spellName then
if dmgPrdcCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%d#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.event, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.destGUID, dmgPrdcCache.destName, dmgPrdcCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_PERIODIC_DAMAGE[CONDENSED]#%s#%s#%d Targets#%d#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.count, dmgPrdcCache.spellName)
end
end
dmgPrdcCache.sourceGUID = sourceGUID
dmgPrdcCache.sourceName = sourceName or "nil"
dmgPrdcCache.spellName = spellName
dmgPrdcCache.timeStop = (debugprofilestop() / 1000) - logStartTime
dmgPrdcCache.time = date("%H:%M:%S")
dmgPrdcCache.timeStamp = timeStamp
dmgPrdcCache.count = 1
dmgPrdcCache.event = event
dmgPrdcCache.destGUID = destGUID
dmgPrdcCache.destName = destName
end
else
if dmgCache.spellName then
if dmgCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgCache.timeStop, dmgCache.time, dmgCache.event, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.destGUID, dmgCache.destName, dmgCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgCache.timeStop, dmgCache.time, dmgCache.sourceGUID, dmgCache.sourceName, dmgCache.count, dmgCache.spellName)
end
dmgCache.spellName = nil
elseif dmgPrdcCache.spellName then
if dmgPrdcCache.count == 1 then
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] %s#%s#%s#%s#%s#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.event, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.destGUID, dmgPrdcCache.destName, dmgPrdcCache.spellName)
else
currentLog.total[#currentLog.total+1] = format("<%.2f %s> [CLEU] SPELL_PERIODIC_DAMAGE[CONDENSED]#%s#%s#%d Targets#%s", dmgPrdcCache.timeStop, dmgPrdcCache.time, dmgPrdcCache.sourceGUID, dmgPrdcCache.sourceName, dmgPrdcCache.count, dmgPrdcCache.spellName)
end
dmgPrdcCache.spellName = nil
end
if shouldLogFlags and (sourceName or destName) and badPlayerFilteredEvents[event] then
return strjoin("#", tostringall(event, sourceName and sourceFlags or destFlags, sourceGUID, sourceName, destGUID, destName, spellId, spellName, extraSpellId, amount))
else
return strjoin("#", tostringall(event, sourceGUID, sourceName, destGUID, destName, spellId, spellName, extraSpellId, amount))
end
end
end
end
end
function sh.PLAYER_REGEN_DISABLED()
return "+Entering combat!"
end
function sh.PLAYER_REGEN_ENABLED()
return "-Leaving combat!"
end
do
local UnitIsUnit = UnitIsUnit
local wantedUnits = {
target = true, focus = true,
nameplate1 = true, nameplate2 = true, nameplate3 = true, nameplate4 = true, nameplate5 = true, nameplate6 = true, nameplate7 = true, nameplate8 = true, nameplate9 = true, nameplate10 = true,
nameplate11 = true, nameplate12 = true, nameplate13 = true, nameplate14 = true, nameplate15 = true, nameplate16 = true, nameplate17 = true, nameplate18 = true, nameplate19 = true, nameplate20 = true,
nameplate21 = true, nameplate22 = true, nameplate23 = true, nameplate24 = true, nameplate25 = true, nameplate26 = true, nameplate27 = true, nameplate28 = true, nameplate29 = true, nameplate30 = true,
nameplate31 = true, nameplate32 = true, nameplate33 = true, nameplate34 = true, nameplate35 = true, nameplate36 = true, nameplate37 = true, nameplate38 = true, nameplate39 = true, nameplate40 = true,
}
local bossUnits = {
boss1 = true, boss2 = true, boss3 = true, boss4 = true, boss5 = true,
boss6 = true, boss7 = true, boss8 = true, boss9 = true, boss10 = true,
boss11 = true, boss12 = true, boss13 = true, boss14 = true, boss15 = true,
arena1 = true, arena2 = true, arena3 = true, arena4 = true, arena5 = true,
}
local raidList = {
raid1 = true, raid2 = true, raid3 = true, raid4 = true, raid5 = true, raid6 = true, raid7 = true, raid8 = true, raid9 = true, raid10 = true,
raid11 = true, raid12 = true, raid13 = true, raid14 = true, raid15 = true, raid16 = true, raid17 = true, raid18 = true, raid19 = true, raid20 = true,
raid21 = true, raid22 = true, raid23 = true, raid24 = true, raid25 = true, raid26 = true, raid27 = true, raid28 = true, raid29 = true, raid30 = true,
raid31 = true, raid32 = true, raid33 = true, raid34 = true, raid35 = true, raid36 = true, raid37 = true, raid38 = true, raid39 = true, raid40 = true
}
local function safeUnit(unit)
if bossUnits[unit] then -- Accept any boss unit
return true
elseif wantedUnits[unit] and not UnitIsUnit("player", unit) and not UnitInRaid(unit) and not UnitInParty(unit) then
for k in next, bossUnits do
if UnitIsUnit(unit, k) then -- Reject if the unit is also a boss unit
return false