@@ -11,80 +11,103 @@ export class HNParser extends BaseParser {
11
11
'Hacker News Discussion' ;
12
12
}
13
13
14
- getPageContent ( ) {
15
- if ( ! this . isDiscussionPage ( ) ) {
16
- return null ;
17
- }
18
-
19
- // Get main post content
20
- const mainPost = {
21
- title : this . getTitle ( ) ,
22
- url : document . querySelector ( '.titleline a' ) ?. href ,
23
- text : document . querySelector ( '.toptext' ) ?. textContent ,
24
- by : document . querySelector ( '.hnuser' ) ?. textContent ,
25
- time : document . querySelector ( '.age' ) ?. textContent
26
- } ;
27
-
28
- // Get all comments
29
- const threads = this . getTopLevelComments ( ) . map ( comment =>
30
- this . parseCommentThread ( comment )
31
- ) ;
32
-
33
- return {
34
- mainPost,
35
- threads
36
- } ;
37
- }
38
-
39
14
getTopLevelComments ( ) {
40
- // Get top-level comments
41
- return Array . from (
42
- document . querySelectorAll ( '.comment-tree > tbody > tr.athing.comtr' )
43
- ) ;
15
+ return Array . from ( document . querySelectorAll ( '.comment-tree > tbody > tr.athing.comtr' ) ) ;
44
16
}
45
17
46
18
parseCommentThread ( rootComment ) {
47
19
if ( ! rootComment ?. classList ?. contains ( 'comtr' ) ) {
48
- console . warn ( 'Invalid comment element provided' ) ;
20
+ log . warn ( 'Invalid comment element provided' ) ;
49
21
return null ;
50
22
}
51
-
23
+
52
24
const thread = {
53
25
id : rootComment . id ,
54
26
root : this . parseComment ( rootComment ) ,
55
27
replies : [ ] ,
56
- level : 0
28
+ level : this . getCommentIndent ( rootComment ) ,
57
29
} ;
58
-
59
- // Get all replies
30
+
60
31
let currentElement = rootComment . nextElementSibling ;
61
- const rootIndent = this . getCommentIndent ( rootComment ) ;
62
-
32
+ const rootIndent = thread . level ;
33
+
63
34
while ( currentElement ) {
64
35
const isComment = currentElement . classList . contains ( 'comtr' ) ;
65
36
if ( ! isComment ) {
66
37
currentElement = currentElement . nextElementSibling ;
67
38
continue ;
68
39
}
69
-
40
+
70
41
const currentIndent = this . getCommentIndent ( currentElement ) ;
71
-
72
- // If indentation is less than or equal to root, it's a new thread
42
+
73
43
if ( currentIndent <= rootIndent ) {
74
44
break ;
75
45
}
76
-
77
- // Add reply to thread
46
+
78
47
const reply = this . parseComment ( currentElement ) ;
79
48
if ( reply ) {
80
49
reply . level = currentIndent - rootIndent ;
81
50
thread . replies . push ( reply ) ;
82
51
}
83
-
52
+
84
53
currentElement = currentElement . nextElementSibling ;
85
54
}
86
-
55
+
87
56
return thread ;
57
+ }
58
+
59
+ getPageContent ( ) {
60
+ if ( ! this . isDiscussionPage ( ) ) {
61
+ return null ;
62
+ }
63
+
64
+ // Get main post content
65
+ const mainPost = {
66
+ title : this . getTitle ( ) ,
67
+ url : document . querySelector ( '.titleline a' ) ?. href ,
68
+ text : document . querySelector ( '.toptext' ) ?. textContent ,
69
+ by : document . querySelector ( '.hnuser' ) ?. textContent ,
70
+ time : document . querySelector ( '.age' ) ?. textContent
71
+ } ;
72
+
73
+ // Get and process all comments in a single pass
74
+ const threads = this . processComments ( ) ;
75
+
76
+ return {
77
+ mainPost,
78
+ threads
79
+ } ;
80
+ }
81
+
82
+ processComments ( ) {
83
+ // Process top-level comments and nested threads in a single pass
84
+ const comments = [ ] ;
85
+ const commentElements = Array . from ( document . querySelectorAll ( '.comment-tree > tbody > tr.athing.comtr' ) ) ;
86
+
87
+ let currentThread = null ;
88
+ commentElements . forEach ( ( commentElement ) => {
89
+ const currentIndent = this . getCommentIndent ( commentElement ) ;
90
+
91
+ if ( ! currentThread || currentIndent === 0 ) {
92
+ // Start a new thread if we are at top-level or after ending a previous thread
93
+ currentThread = {
94
+ id : commentElement . id ,
95
+ root : this . parseComment ( commentElement ) ,
96
+ replies : [ ] ,
97
+ level : 0
98
+ } ;
99
+ comments . push ( currentThread ) ;
100
+ } else if ( currentThread && currentIndent > currentThread . level ) {
101
+ // Add nested comments to the current thread based on indentation level
102
+ const reply = this . parseComment ( commentElement ) ;
103
+ if ( reply ) {
104
+ reply . level = currentIndent - currentThread . level ;
105
+ currentThread . replies . push ( reply ) ;
106
+ }
107
+ }
108
+ } ) ;
109
+
110
+ return comments ;
88
111
}
89
112
90
113
parseComment ( commentElement ) {
@@ -101,7 +124,7 @@ export class HNParser extends BaseParser {
101
124
by : userElement ?. textContent || '' ,
102
125
time : ageElement ?. getAttribute ( 'title' ) || ageElement ?. textContent || '' ,
103
126
score : scoreElement ?. textContent || '' ,
104
- element : commentElement , // Keep DOM reference for later use
127
+ element : commentElement ,
105
128
level : this . getCommentIndent ( commentElement )
106
129
} ;
107
130
}
@@ -120,6 +143,21 @@ export class HNParser extends BaseParser {
120
143
if ( comment ) {
121
144
comment . scrollIntoView ( { behavior : 'smooth' , block : 'center' } ) ;
122
145
this . highlightComment ( comment ) ;
146
+
147
+ // Add specific HN highlight style
148
+ comment . style . backgroundColor = '#f6f6ef' ;
149
+ comment . style . borderLeft = '3px solid #ff6600' ;
150
+
151
+ // Remove highlight after 3 seconds
152
+ setTimeout ( ( ) => {
153
+ comment . style . backgroundColor = '' ;
154
+ comment . style . borderLeft = '' ;
155
+ } , 3000 ) ;
123
156
}
124
157
}
158
+
159
+ destroy ( ) {
160
+ // Remove any added styles or elements
161
+ document . getElementById ( 'highlight-styles' ) ?. remove ( ) ;
162
+ }
125
163
}
0 commit comments