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
|
import * as React from 'react';
import ReactDOM from 'react-dom';
import styled from 'styled-components';
import { colors } from '../../config.json';
import { Scheduler } from '../../shared/scheduler';
import ImageView from './ImageView';
const MODAL_CONTAINER_ID = 'modalContainer';
const ModalContent = styled.div({
position: 'absolute',
display: 'flex',
flexDirection: 'column',
flex: 1,
top: 0,
left: 0,
right: 0,
bottom: 0,
});
const ModalBackground = styled.div({
backgroundColor: 'rgba(0,0,0,0.5)',
position: 'absolute',
display: 'flex',
flexDirection: 'column',
flex: 1,
top: 0,
left: 0,
right: 0,
bottom: 0,
});
export const StyledModalContainer = styled.div({
position: 'relative',
flex: 1,
});
interface IModalContainerProps {
children?: React.ReactNode;
}
export function ModalContainer(props: IModalContainerProps) {
return (
<StyledModalContainer id={MODAL_CONTAINER_ID}>
<ModalContent>{props.children}</ModalContent>
</StyledModalContainer>
);
}
export enum ModalAlertType {
Info = 1,
Warning,
}
const ModalAlertContainer = styled.div({
display: 'flex',
flexDirection: 'column',
flex: 1,
justifyContent: 'center',
padding: '26px 14px 14px',
});
const StyledModalAlert = styled.div({
display: 'flex',
flexDirection: 'column',
backgroundColor: colors.darkBlue,
borderRadius: '11px',
padding: '16px',
});
const ModalAlertIcon = styled.div({
display: 'flex',
justifyContent: 'center',
marginTop: '8px',
});
const ModalAlertButtonContainer = styled.div({
display: 'flex',
flexDirection: 'column',
marginTop: '18px',
});
interface IModalAlertProps {
type?: ModalAlertType;
message?: string;
buttons: React.ReactNode[];
children?: React.ReactNode;
}
export class ModalAlert extends React.Component<IModalAlertProps> {
private element = document.createElement('div');
private modalContainer?: Element;
private appendScheduler = new Scheduler();
public componentDidMount() {
const modalContainer = document.getElementById(MODAL_CONTAINER_ID);
if (modalContainer) {
this.modalContainer = modalContainer;
// Mounting the container element immediately results in a graphical issue with the dialog
// first rendering with the wrong proportions and then changing to the correct proportions.
// Postponing it to the next event cycle solves this issue.
this.appendScheduler.schedule(() => {
modalContainer.appendChild(this.element);
});
} else {
throw Error('Modal container not found when mounting modal');
}
}
public componentWillUnmount() {
this.appendScheduler.cancel();
if (this.modalContainer) {
this.modalContainer.removeChild(this.element);
}
}
public render() {
return ReactDOM.createPortal(this.renderModal(), this.element);
}
private renderModal() {
return (
<ModalBackground>
<ModalAlertContainer>
<StyledModalAlert>
{this.props.type && (
<ModalAlertIcon>{this.renderTypeIcon(this.props.type)}</ModalAlertIcon>
)}
{this.props.message && <ModalMessage>{this.props.message}</ModalMessage>}
{this.props.children}
{this.props.buttons.map((button, index) => (
<ModalAlertButtonContainer key={index}>{button}</ModalAlertButtonContainer>
))}
</StyledModalAlert>
</ModalAlertContainer>
</ModalBackground>
);
}
private renderTypeIcon(type: ModalAlertType) {
let source = '';
let color = '';
switch (type) {
case ModalAlertType.Info:
source = 'icon-alert';
color = colors.white;
break;
case ModalAlertType.Warning:
source = 'icon-alert';
color = colors.red;
break;
}
return <ImageView height={44} width={44} source={source} tintColor={color} />;
}
}
export const ModalMessage = styled.span({
fontFamily: 'Open Sans',
fontSize: '13px',
fontWeight: 500,
lineHeight: '20px',
color: colors.white80,
marginTop: '16px',
});
|