bookpage.js 10.6 KB
Newer Older
wanglch's avatar
wanglch committed
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
//import React from 'react';

const BookPageTemplate = () => {
  // Only three state variables as requested
  const [title, setTitle] = React.useState("ADVENTURES OF DON QUIXOTE");
  const [pageNumber, setPageNumber] = React.useState("289");
  const [text, setText] = React.useState(
    "deed,\" said Don Quixote, \"thou hast hit the point, Sancho, which can alone shake my resolution; I neither can, nor ought to, draw my sword, as I have often told thee, against those who are not dubbed knights. To thee which I had premeditated, thy share of the booty would have been at least the emperor's crown of gold and Cupid's painted wings; for I would have plucked them off perforce, and delivered them into thy hands.\" \"The"
  );

  // Styles for heavily degraded scan effect
  const heavilyDegradedStyles = {
    filter: 'grayscale(30%) contrast(120%) brightness(85%) sepia(20%)',
    position: 'relative',
    backgroundColor: '#e6ddc6', // More yellowed aged paper
    backgroundImage: 'url("data:image/svg+xml,%3Csvg viewBox=\'0 0 200 200\' xmlns=\'http://www.w3.org/2000/svg\'%3E%3Cfilter id=\'noiseFilter\'%3E%3CfeTurbulence type=\'fractalNoise\' baseFrequency=\'0.85\' numOctaves=\'3\' stitchTiles=\'stitch\'/%3E%3C/filter%3E%3Crect width=\'100%25\' height=\'100%25\' filter=\'url(%23noiseFilter)\' opacity=\'0.25\'/%3E%3C/svg%3E")',
    boxShadow: 'inset 0 0 70px rgba(0, 0, 0, 0.3), 0 0 5px rgba(0,0,0,0.1)',
    padding: '32px',
    borderRadius: '2px',
    overflow: 'hidden',
    transform: 'rotate(0.3deg)', // Slightly askew scan
  };

  // Heavily degraded text
  const badScanTextStyle = {
    fontFamily: '"Times New Roman", serif',
    letterSpacing: '-0.01em',
    wordSpacing: '0.02em',
    fontWeight: '500',
    color: '#222222',
    textShadow: '0 0 1px rgba(0, 0, 0, 0.5)',
    transform: 'scale(1.01, 0.99) rotate(-0.4deg)', // Distorted proportions
  };

  // Random coffee stain effect
  const coffeeStain = {
    position: 'absolute',
    width: '100px',
    height: '80px',
    top: '25%',
    right: '15%',
    borderRadius: '50%',
    background: 'radial-gradient(ellipse at center, rgba(139,69,19,0.15) 0%, rgba(139,69,19,0.1) 50%, rgba(139,69,19,0.05) 70%, rgba(139,69,19,0) 100%)',
    transform: 'rotate(30deg) scale(1.5, 1)',
    pointerEvents: 'none',
    zIndex: 1,
  };

  // Water damage effect
  const waterDamage = {
    position: 'absolute',
    width: '70%',
    height: '40%',
    bottom: '10%',
    left: '5%',
    opacity: 0.07,
    background: 'radial-gradient(ellipse at center, rgba(0,0,0,0.2) 0%, rgba(0,0,0,0.1) 40%, rgba(0,0,0,0) 70%)',
    borderRadius: '40% 60% 70% 30% / 40% 50% 60% 50%',
    pointerEvents: 'none',
    zIndex: 1,
  };

  // Add fold lines
  const foldLine = {
    position: 'absolute',
    width: '100%',
    height: '3px',
    top: '30%',
    left: 0,
    background: 'linear-gradient(to right, rgba(0,0,0,0) 0%, rgba(0,0,0,0.03) 20%, rgba(0,0,0,0.08) 50%, rgba(0,0,0,0.03) 80%, rgba(0,0,0,0) 100%)',
    boxShadow: '0 1px 3px rgba(255,255,255,0.2)',
    pointerEvents: 'none',
    zIndex: 2,
  };

  // Torn edge effect
  const tornEdge = {
    position: 'absolute',
    top: 0,
    right: 0,
    width: '100%',
    height: '100%',
    background: 'linear-gradient(135deg, transparent 97%, #e6ddc6 97%, #e6ddc6 100%)',
    pointerEvents: 'none',
  };

  return (
    <div style={{
      maxWidth: '800px',
      margin: '0 auto',
      padding: '16px',
    }}>
      {/* Heavily degraded scan container */}
      <div style={heavilyDegradedStyles}>
        {/* Noise overlay */}
        <div style={{
          position: 'absolute',
          top: 0,
          left: 0,
          right: 0,
          bottom: 0,
          background: 'url("") repeat',
          opacity: 0.15,
          pointerEvents: 'none',
        }}></div>
        
        {/* Scan lines effect */}
        <div style={{
          position: 'absolute',
          top: 0,
          left: 0,
          right: 0,
          bottom: 0,
          background: 'linear-gradient(to bottom, rgba(0,0,0,0.03) 1px, transparent 1px)',
          backgroundSize: '100% 2px',
          opacity: 0.5,
          pointerEvents: 'none',
        }}></div>
        
        {/* Add coffee stain */}
        <div style={coffeeStain}></div>
        
        {/* Add water damage */}
        <div style={waterDamage}></div>
        
        {/* Add fold line */}
        <div style={foldLine}></div>
        
        {/* Add torn edge */}
        <div style={tornEdge}></div>
        
        {/* Header with skewed alignment */}
        <div style={{
          display: 'flex',
          justifyContent: 'space-between',
          alignItems: 'center',
          borderBottom: '2px solid #000',
          paddingBottom: '4px',
          marginBottom: '24px',
          position: 'relative',
          opacity: 0.8,
          transform: 'skew(-0.5deg, 0.3deg)',
        }}>
          <div style={{width: '48px'}}></div>
          <h1 style={{
            ...badScanTextStyle,
            fontSize: '20px',
            fontWeight: 'bold',
            textAlign: 'center',
            textTransform: 'uppercase',
            letterSpacing: '1px',
            opacity: 0.8,
          }}>{title}</h1>
          <div style={{
            ...badScanTextStyle,
            fontSize: '20px', 
            fontWeight: 'bold',
            opacity: 0.85,
          }}>{pageNumber}</div>
        </div>
        
        {/* Horizontal divider with uneven quality */}
        <div style={{
          borderBottom: '1px solid #444',
          marginBottom: '24px',
          opacity: 0.6,
          filter: 'blur(0.3px)',
          transform: 'scaleY(1.5) skew(0.7deg)',
        }}></div>
        
        {/* Text content with severely degraded appearance */}
        <div style={{
          columnCount: 2,
          columnGap: '20px',
          columnRule: '1px solid rgba(0,0,0,0.1)',
          textAlign: 'justify',
          ...badScanTextStyle,
          fontSize: '16px',
          lineHeight: '1.5',
          opacity: 0.78,
          // Very uneven ink distribution with blurry and faded parts
          WebkitMaskImage: 'linear-gradient(to bottom, rgba(0,0,0,0.9), rgba(0,0,0,0.75) 50%, rgba(0,0,0,0.85))',
          // Text distortion
          filter: 'blur(0.2px)',
        }}>
          {/* Bad scan text with random character fading */}
          <p>{text.split('').map((char, index) => {
            const opacity = Math.random() > 0.8 ? 0.4 + Math.random() * 0.5 : 0.9 + Math.random() * 0.1;
            const blur = Math.random() > 0.95 ? 1 : 0;
            return <span key={index} style={{opacity, filter: `blur(${blur}px)`}}>{char}</span>;
          })}</p>
        </div>
        
        {/* Extra random ink spill */}
        <div style={{
          position: 'absolute',
          width: '10px',
          height: '20px',
          top: '60%',
          left: '25%',
          background: 'rgba(0,0,0,0.3)',
          borderRadius: '50%',
          transform: 'rotate(30deg)',
          filter: 'blur(1px)',
          zIndex: 3,
        }}></div>
      </div>
      
    </div>
  );
};

//export default BookPageTemplate;
window.BookPageTemplate = BookPageTemplate;