1
- import { useCallback , useRef , useState } from 'react' ;
1
+ import { forwardRef , useCallback , useState } from 'react' ;
2
2
import {
3
3
FeedbackContainer ,
4
4
VoteButton ,
5
+ VoteButtonAfter ,
5
6
VoteButtonsContainer ,
6
- Toggle ,
7
- FeedbackMobileContainer ,
8
- ThankYouContainer
7
+ FeedbackText ,
8
+ FeedbackTextAfter ,
9
+ ButtonStyles
9
10
} from './styles' ;
10
- import { useEffect } from 'react' ;
11
11
import { trackFeedbackSubmission } from '../../utils/track' ;
12
+ import {
13
+ ThumbsUpIcon ,
14
+ ThumbsDownIcon ,
15
+ ThumbsUpFilledIcon ,
16
+ ThumbsDownFilledIcon
17
+ } from '../Icons' ;
18
+ import ExternalLink from '../ExternalLink' ;
12
19
13
20
enum FeedbackState {
14
21
START = 'START' ,
15
- END = 'END' ,
22
+ UP = 'UP' ,
23
+ DOWN = 'DOWN' ,
16
24
HIDDEN = 'HIDDEN'
17
25
}
18
26
@@ -23,99 +31,173 @@ type Feedback = {
23
31
comment ?: string ;
24
32
} ;
25
33
26
- export default function Feedback ( ) {
27
- const [ state , setState ] = useState < FeedbackState > ( FeedbackState . START ) ;
28
- const feedbackQuestion = 'Was this page helpful?' ;
29
- const feedbackAppreciation = 'Thank you for your feedback!' ;
34
+ // eslint-disable-next-line no-empty-pattern
35
+ const Feedback = forwardRef ( function Feedback ( { } , ref ) {
36
+ const [ state , setState ] = useState ( FeedbackState . START ) ;
37
+
38
+ // Feedback Component Customizations
39
+ const c = {
40
+ feedbackQuestion : 'Was this page helpful?' ,
41
+ yesVoteResponse : 'Thanks for your feedback!' ,
42
+ noVoteResponse : 'Thanks for your feedback!' ,
43
+ noVoteCTA : 'Can you provide more details?' ,
44
+ noVoteCTAButton : 'File an issue on GitHub' ,
45
+ ctaIcon : 'external' ,
46
+ iconPosition : 'right' ,
47
+ buttonLink : 'https://github.com/aws-amplify/docs/issues/new/choose'
48
+ } ;
30
49
31
- const onYesVote = useCallback ( ( ) => {
32
- setState ( FeedbackState . END ) ;
50
+ let currentState = state ;
33
51
52
+ const onYesVote = useCallback ( ( e ) => {
34
53
trackFeedbackSubmission ( true ) ;
35
- } , [ ] ) ;
54
+ const yesButton = e . currentTarget ;
55
+ const noButton = yesButton . nextSibling ;
56
+ const feedbackComponent = yesButton . parentElement . parentElement ;
57
+ const feedbackText = feedbackComponent . getElementsByTagName ( 'p' ) [ 0 ] ;
58
+ const feedbackTextWidth = feedbackText . offsetWidth ;
59
+
60
+ const transitionUpButton = [
61
+ {
62
+ maxWidth : yesButton . offsetWidth + 'px' ,
63
+ overflow : 'visible'
64
+ } ,
65
+ {
66
+ maxWidth : '40px' ,
67
+ overflow : 'hidden' ,
68
+ color : 'green' ,
69
+ border : '1px solid green' ,
70
+ transform : `translateX(-${ feedbackTextWidth } px)` ,
71
+ marginLeft : '0px'
72
+ }
73
+ ] ;
74
+
75
+ const transitionDownButton = [
76
+ {
77
+ maxWidth : noButton . offsetWidth + 'px' ,
78
+ overflow : 'visible'
79
+ } ,
80
+ {
81
+ maxWidth : 0 ,
82
+ overflow : 'hidden' ,
83
+ margin : 0 ,
84
+ padding : 0 ,
85
+ border : 'none'
86
+ }
87
+ ] ;
88
+
89
+ const transitionFeedbackText = [
90
+ { transform : 'translateX(-40px)' , opacity : 0 }
91
+ ] ;
92
+
93
+ const animationTiming = {
94
+ duration : 300 ,
95
+ iterations : 1 ,
96
+ fill : 'forwards'
97
+ } ;
98
+
99
+ yesButton . animate ( transitionUpButton , animationTiming ) ;
100
+ noButton . animate ( transitionDownButton , animationTiming ) ;
101
+ feedbackText . animate ( transitionFeedbackText , animationTiming ) ;
36
102
37
- const onNoVote = useCallback ( ( ) => {
38
- setState ( FeedbackState . END ) ;
103
+ setTimeout ( function ( ) {
104
+ currentState = FeedbackState . UP ;
105
+ setState ( currentState ) ;
106
+ } , 300 ) ;
107
+ } , [ ] ) ;
39
108
109
+ const onNoVote = useCallback ( ( e ) => {
40
110
trackFeedbackSubmission ( false ) ;
111
+ const feedbackContent = e . currentTarget . parentNode . parentNode ;
112
+
113
+ feedbackContent . classList . add ( 'fadeOut' ) ;
114
+
115
+ setTimeout ( function ( ) {
116
+ currentState = FeedbackState . DOWN ;
117
+ feedbackContent . classList . remove ( 'fadeOut' ) ;
118
+ feedbackContent . classList . add ( 'fadeIn' ) ;
119
+ setState ( currentState ) ;
120
+ } , 300 ) ;
41
121
} , [ ] ) ;
42
122
43
123
return (
44
124
< FeedbackContainer
45
- style = { state === FeedbackState . HIDDEN ? { display : 'none' } : { } }
125
+ id = "feedback-container"
126
+ ref = { ref }
127
+ aria-hidden = { state == FeedbackState . UP ? true : false }
46
128
>
47
- { state == FeedbackState . START ? (
48
- < >
49
- < p > { feedbackQuestion } </ p >
50
- < VoteButtonsContainer >
51
- < VoteButton onClick = { onYesVote } >
52
- < img src = "/assets/thumbs-up.svg" alt = "Thumbs up" />
53
- Yes
54
- </ VoteButton >
55
- < VoteButton onClick = { onNoVote } >
56
- < img src = "/assets/thumbs-down.svg" alt = "Thumbs down" />
57
- No
58
- </ VoteButton >
59
- </ VoteButtonsContainer >
60
- </ >
61
- ) : (
62
- < ThankYouContainer >
63
- < p > { feedbackAppreciation } </ p >
64
- </ ThankYouContainer >
65
- ) }
129
+ { ( ( ) => {
130
+ switch ( state ) {
131
+ case 'START' :
132
+ return (
133
+ < div
134
+ id = "start-state"
135
+ aria-label = { c . feedbackQuestion }
136
+ tabIndex = { 0 }
137
+ >
138
+ < FeedbackText > { c . feedbackQuestion } </ FeedbackText >
139
+ < VoteButtonsContainer >
140
+ < VoteButton
141
+ onClick = { onYesVote }
142
+ aria-label = "Yes"
143
+ role = "button"
144
+ tabIndex = { 0 }
145
+ >
146
+ < ThumbsUpIcon />
147
+ < FeedbackText > Yes</ FeedbackText >
148
+ </ VoteButton >
149
+ < VoteButton
150
+ onClick = { onNoVote }
151
+ aria-label = "No"
152
+ role = "button"
153
+ tabIndex = { 0 }
154
+ >
155
+ < ThumbsDownIcon />
156
+ < FeedbackText > No</ FeedbackText >
157
+ </ VoteButton >
158
+ </ VoteButtonsContainer >
159
+ </ div >
160
+ ) ;
161
+ case 'UP' :
162
+ return (
163
+ < div className = "up" >
164
+ < VoteButtonsContainer className = "up-response" >
165
+ < VoteButtonAfter className = "up-response" >
166
+ < ThumbsUpFilledIcon />
167
+ </ VoteButtonAfter >
168
+ < FeedbackTextAfter className = "up-response" >
169
+ { c . yesVoteResponse }
170
+ </ FeedbackTextAfter >
171
+ </ VoteButtonsContainer >
172
+ </ div >
173
+ ) ;
174
+ case 'DOWN' :
175
+ return (
176
+ < div className = "down" >
177
+ < VoteButtonsContainer className = "down-response" >
178
+ < VoteButtonAfter className = "down-response" >
179
+ < ThumbsDownFilledIcon />
180
+ </ VoteButtonAfter >
181
+ < FeedbackTextAfter className = "down-response" >
182
+ { c . noVoteResponse }
183
+ </ FeedbackTextAfter >
184
+ </ VoteButtonsContainer >
185
+ < FeedbackTextAfter className = "cta" >
186
+ { c . noVoteCTA }
187
+ </ FeedbackTextAfter >
188
+ < ButtonStyles >
189
+ < ExternalLink href = { c . buttonLink } icon = { true } >
190
+ { c . noVoteCTAButton }
191
+ </ ExternalLink >
192
+ </ ButtonStyles >
193
+ </ div >
194
+ ) ;
195
+ default :
196
+ return < div > </ div > ;
197
+ }
198
+ } ) ( ) }
66
199
</ FeedbackContainer >
67
200
) ;
68
- }
201
+ } ) ;
69
202
70
- export function FeedbackToggle ( ) {
71
- const [ inView , setInView ] = useState ( false ) ;
72
- const feedbackContainer = useRef ( null ) ;
73
-
74
- function toggleView ( ) {
75
- if ( inView ) {
76
- setInView ( false ) ;
77
- } else {
78
- setInView ( true ) ;
79
- }
80
- }
81
-
82
- function handleClickOutside ( e ) {
83
- if (
84
- feedbackContainer . current &&
85
- feedbackContainer . current . contains ( e . target )
86
- ) {
87
- // inside click
88
- return ;
89
- }
90
- // outside click
91
- setInView ( false ) ;
92
- }
93
-
94
- useEffect ( ( ) => {
95
- if ( inView ) {
96
- document . addEventListener ( 'mousedown' , handleClickOutside ) ;
97
- } else {
98
- document . removeEventListener ( 'mousedown' , handleClickOutside ) ;
99
- }
100
-
101
- return ( ) => {
102
- document . removeEventListener ( 'mousedown' , handleClickOutside ) ;
103
- } ;
104
- } , [ inView ] ) ;
105
-
106
- return (
107
- < div ref = { feedbackContainer } >
108
- < FeedbackMobileContainer style = { inView ? { } : { display : 'none' } } >
109
- < Feedback > </ Feedback >
110
- </ FeedbackMobileContainer >
111
- < Toggle
112
- onClick = { ( ) => {
113
- toggleView ( ) ;
114
- } }
115
- >
116
- < img src = "/assets/thumbs-up.svg" alt = "Thumbs up" />
117
- < img src = "/assets/thumbs-down.svg" alt = "Thumbs down" />
118
- </ Toggle >
119
- </ div >
120
- ) ;
121
- }
203
+ export default Feedback ;
0 commit comments