Skip to content

Commit 41bd790

Browse files
authored
Merge pull request #4 from actiontech/feature/add-water-mark
Feature/add water mark
2 parents 4bea11f + 39a9d33 commit 41bd790

File tree

3 files changed

+95
-1
lines changed

3 files changed

+95
-1
lines changed

webapp/packages/core-app/src/Body.tsx

+5-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,8 @@ import { observer } from 'mobx-react-lite';
99
import { useLayoutEffect, useRef } from 'react';
1010
import styled, { css } from 'reshadow';
1111

12-
import { Loader, useResource, useStyles } from '@cloudbeaver/core-blocks';
12+
import { AuthInfoService } from '@cloudbeaver/core-authentication';
13+
import { Loader, useResource, useStyles, Watermark } from '@cloudbeaver/core-blocks';
1314
import { useService } from '@cloudbeaver/core-di';
1415
import { DialogsPortal } from '@cloudbeaver/core-dialogs';
1516
import { Notifications } from '@cloudbeaver/core-notifications';
@@ -44,6 +45,8 @@ export const Body = observer(function Body() {
4445
const screenService = useService(ScreenService);
4546
const Screen = screenService.screen?.component;
4647
const { backendVersion } = useAppVersion();
48+
const authInfoService = useService(AuthInfoService);
49+
const userInfo = authInfoService.userInfo;
4750

4851
// TODO: must be loaded in place where it is used
4952
useResource(Body, ProjectInfoResource, CachedMapAllKey, { silent: true });
@@ -64,6 +67,7 @@ export const Body = observer(function Body() {
6467
<DialogsPortal />
6568
<Notifications />
6669
</theme>
70+
{userInfo && <Watermark theme={userInfo.configurationParameters?.['app.theme']} text={userInfo.displayName || userInfo.userId} />}
6771
</Loader>
6872
</DNDProvider>,
6973
);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
import React, { useEffect, useMemo, useState } from 'react';
2+
3+
interface WatermarkProps {
4+
text: string;
5+
fontSize?: number;
6+
fontFamily?: string;
7+
fontWeight?: string;
8+
textColor?: string;
9+
rotate?: number;
10+
zIndex?: number;
11+
theme?: 'light' | 'dark';
12+
density?: number;
13+
}
14+
15+
const Watermark: React.FC<WatermarkProps> = ({
16+
text,
17+
fontSize = 20,
18+
fontFamily = 'Arial',
19+
textColor = 'rgba(0, 0, 0, 0.1)',
20+
rotate = -45,
21+
zIndex = 9999,
22+
theme = 'light',
23+
fontWeight = 'bold',
24+
density = 1,
25+
}) => {
26+
const [dimensions, setDimensions] = useState({ width: window.innerWidth, height: window.innerHeight });
27+
28+
useEffect(() => {
29+
const handleResize = () => {
30+
setDimensions({
31+
width: window.innerWidth,
32+
height: window.innerHeight,
33+
});
34+
};
35+
36+
window.addEventListener('resize', handleResize);
37+
return () => window.removeEventListener('resize', handleResize);
38+
}, []);
39+
40+
const watermarkDataUrl = useMemo(() => {
41+
const canvas = document.createElement('canvas');
42+
const context = canvas.getContext('2d');
43+
44+
if (context) {
45+
const scale = window.devicePixelRatio;
46+
canvas.width = dimensions.width * scale;
47+
canvas.height = dimensions.height * scale;
48+
context.scale(scale, scale);
49+
50+
context.font = `${fontWeight} ${fontSize}px ${fontFamily}`;
51+
context.fillStyle = theme === 'light' ? textColor : 'rgba(255, 255, 255, 0.1)';
52+
context.rotate((rotate * Math.PI) / 180);
53+
context.fillText(text, 0, 100);
54+
55+
const metrics = context.measureText(text);
56+
const textWidth = metrics.width;
57+
const textHeight = fontSize;
58+
59+
const spacingX = (textWidth + 180) / density;
60+
const spacingY = (textHeight + 180) / density;
61+
62+
for (let y = -canvas.height; y < canvas.height * 2; y += spacingY) {
63+
for (let x = -canvas.width; x < canvas.width * 2; x += spacingX) {
64+
context.fillText(text, x, y);
65+
}
66+
}
67+
return canvas.toDataURL();
68+
}
69+
return '';
70+
}, [density, dimensions.height, dimensions.width, fontFamily, fontSize, fontWeight, rotate, text, textColor, theme]);
71+
72+
return (
73+
<div
74+
style={{
75+
position: 'fixed',
76+
top: 0,
77+
left: 0,
78+
width: '100%',
79+
height: '100%',
80+
zIndex,
81+
pointerEvents: 'none',
82+
backgroundImage: `url(${watermarkDataUrl})`,
83+
backgroundRepeat: 'repeat',
84+
}}
85+
/>
86+
);
87+
};
88+
89+
export default Watermark;

webapp/packages/core-blocks/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -120,6 +120,7 @@ export * from './FormControls/BASE_DROPDOWN_STYLES';
120120
export * from './FormControls/FormBox';
121121
export * from './FormControls/Filter';
122122
export * from './Fill';
123+
export { default as Watermark } from './Watermark/Watermark';
123124

124125
export * from './Containers/Container';
125126
export * from './Containers/Group';

0 commit comments

Comments
 (0)