forked from nonsalant/contract
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontract.php
342 lines (313 loc) · 26.8 KB
/
contract.php
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
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
<?php /*
contract-1448515317.php

###################################
1. If you got this file from GitHub and its name is still contract.php, you should RENAME the file to something harder to guess.
The 4th line is a timestamp outputed by the online generator and an example of a more secure filename.
2. The 2nd and 3rd lines (the emails) are read by this PHP script (from itself).
When a signature is submitted, this script will send both parties an email with a link to the signed contract.
3. The 5th line ($lines[4] below) is the data for $DEV_SIGNATURE.
If you got this file from GitHub, the default developer signature just says "John Doe".
You can generate a new signature @ http://cdpn.io/JYpjvE and replace the data above.
4. The $CLIENT_SIGNATURE is received by this script from itself when Client signs the contract.
*/
$CLIENT_SIGNATURE = isset($_POST['client_signature']) ? $_POST['client_signature'] : null;
if (substr( $CLIENT_SIGNATURE, 0, 22 ) === 'data:image/png;base64,') {
$CLIENT_SIGNATURE = '<img id="hk" src="' . htmlspecialchars($CLIENT_SIGNATURE) . '" >';
} else {
$CLIENT_SIGNATURE = null;
}
$lines = file(__FILE__);
$clientEmail = trim($lines[1]);
$devEmail = trim($lines[2]);
$DEV_SIGNATURE = trim($lines[4]);
$DEV_SIGNATURE = '<img id="dev_signature" src="' . $DEV_SIGNATURE . '" >';
$phpName = basename($_SERVER['PHP_SELF']) ? basename($_SERVER['PHP_SELF']) : 'index.php';
$fileName = substr($phpName , 0, -4);
$htmlName = $fileName.'.html';
// If the filename is (or starts with) "test" or "demo" the PHP file won't delete itself, nor will it redirect to the HTML contract (when one exists)
if ( substr($fileName,0,4) == 'test' || substr($fileName,0,4) == 'demo' ) { $selfDelete = 0; }
else { $selfDelete = 1; }
/**
The HTML code (and some PHP) is kept in PHP variables like $HEADER, $CONTRACT_HTML, $FOOTER_UNSIGNED, and $FOOTER_SIGNED_PHP.
**/
$HEADER ='<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Signed Contract</title>
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
<meta name="robots" content="noindex">
<style>
@import url(http://fonts.googleapis.com/css?family=Libre+Baskerville:400,700,400italic);
@import url(http://fonts.googleapis.com/css?family=Arapey);
@import url(http://fonts.googleapis.com/css?family=Open+Sans+Condensed:300,700);
@import url(http://fonts.googleapis.com/css?family=Open+Sans:300,400,600,700,800);
body {
font-family: "Libre Baskerville", serif;
font-size:16px;
line-height:1.5em;
color:#000;
margin:0;
background:#fff;
}
@media (max-width:600px) {
body { font-size:14px; }
}
@media print {
.noprint { display:none !important; }
}
.hidden {
display:none;
}
#content {
max-width: 600px;
margin: 0 auto;
margin-bottom: 3em;
padding: 0 2em;
background: #fff;
}
h1 {
font-size: 2em;
line-height: 1.2em;
letter-spacing: 0.15em;
font-family: "Arapey", serif;
font-weight: normal;
margin: 1em 0;
position: relative;
text-align: center;
text-transform: uppercase;
padding: .5em 0;
}
h2 {
font-size:1.2em;
line-height:1.2em;
letter-spacing:.05em;
font-family:"Open Sans Condensed",sans-serif;
font-weight:700;
}
#signature {
width:auto;
border:dashed 2px #53777A;
margin:0;
text-align:center;
}
#hk,
#dev_signature {
max-width:333px;
display:block;
}
#date-ip {
font-size:1.2em;
line-height:1.2em;
letter-spacing:.05em;
font-family:"Open Sans Condensed",sans-serif;
font-weight:400;
}
#print-pdf {
text-align:center;
padding:1.5em 0;
margin-top:2em;
border-top:solid 1px #ccc;
}
.buttons {
text-align:center;
margin:1.5em auto;
}
button {
margin: 0 .5em;
font-size:1.2em;
line-height:1.5em;
font-family: "Open Sans Condensed",sans-serif;
font-weight: 700;
text-transform:uppercase;
color: #0a3666;
}
button:hover {
color: #136fd2;
}
</style>
</head>
<body>
<div id="content">
';
$CONTRACT_HTML='<h1>Contract of work for website design and development</h1>
<p>Between <strong>Alice</strong>, referred to below as "Designer", and <strong>Bob</strong>, referred to below as "Customer".</p>
<h2>1. Agreement of parties</h2>
<p>Customer hires Designer to redesign the current website, <strong>bobswebsite.com</strong>, for the estimated total price of <strong>$PRICE</strong>. Designer agrees to provide quality service and to answer to the Customers requests in a timely manner.</p>
<p>The agreed payment plan is at the end of the document.</p>
<h2>2. Legal matters and copyrights</h2>
<p>The Customer will guarantee to the Designer that any elements of text, graphics, photos, trademarks or other artwork that the Customer provides for inclusion in the website are either owned by him or that he has the permission to use them. When the Designer receives the final payment, copyright is automatically assigned as follows: Customer will own the graphics, virtual elements, text content photographs and other data provided, unless someone else owns them. The Designer owns the XHTML markup, CSS and other code and he licenses it to the Customer for use on only this project. Designer can reserve the right to display, with Customer\'s consent, the work as part of the portfolio.</p>
<h2>3. Term and termination</h2>
<ol style="list-style-type: lower-alpha;">
<li>This contract shall commence upon the first payment, as outlined in the payment plan and shall remain effective until the services are completed and delivered.</li>
<li>This contract may be terminated at any time by either party effective immediately upon notice, or the mutual agreement of the parties.</li>
<li>In the event of termination, Designer shall be compensated for the service performed through the date of termination in the amount (a) any advanced payment, (b) a prorated portion of the fees due, or (c) hourly fees for work performed by the Designer at the time of the termination, whichever is greater, together with any additional costs incurred trough and up to the date of cancellation.</li>
<li>This Contract may be modified by parties; any modification must be in writing.</li>
<li>All notices to be given hereunder shall be transmitted in writing by electronic mail and will only be effective upon return confirmation.</li>
</ol>
<h2>4. Force majeure</h2>
<p>Designer shall not be deemed in breach of this contract if Designer is unable to complete the services or any portion by reason of fire, earthquake, labor dispute, illness, internet breaches or any technical issues that may appear beyond Designer\'s control. Upon occurrence of any Force Majeure Event, Designer shall give notice to the Customer of his inability to perform or of delay in completing the services and shall propose revisions to the schedule for completion of the services.</p>
<h2>5. Payment plan</h2>
<p>Payment shall be done as follows:</p>
<ul>
<li>50% of total estimated fee will be required to commence work, after this contract has been approved and signed by both parties.</li>
<li>50% upon project closure.</li>
</ul>
<p>Any extra time required outside the project timeline/services mentioned at point 1 of this contract, will be billed at a rate of <strong>$FEE</strong> per hour.</p>
<h2>Both parties agree to the terms stipulated above:</h2>';
$FOOTER_UNSIGNED = '
<form method="post" class="noprint" id="signature_form">
<div id="signature">
<!-- Client Signature Canvas -->
</div>
<div class="buttons">
<button id="reset" type="button">Reset</button>
<button id="submit" type="submit">Done →</button>
</div>
<input type="hidden" id="client_signature" name="client_signature" />
</form>
</div> <!-- #content -->
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script>
<!-- https://github.com/brinley/jSignature/blob/master/README.md -->
<script src="https://cdn.rawgit.com/willowsystems/jSignature/master/libs/jSignature.min.js"></script>
<script>
$(document).ready(function() {
$("#signature").jSignature();
var $sigdiv = $("#signature");
var datapair = $sigdiv.jSignature("getData", "svgbase64");
$("#signature").bind("change", function(e) {
var data = $("#signature").jSignature("getData");
$("#client_signature").val(data);
});
$("#reset").click(function(e){
$("#signature").jSignature("clear");
var data = $("#signature").jSignature("getData");
$("#client_signature").val("");
e.preventDefault();
});
$("#submit").click(function(e) {
$("#signature_form").slideUp(200);
//$(".buttons").slideUp(300);
$("#signature_form").after("<img id=\"hk\" class=\"hidden\" />");
var data = $("#signature").jSignature("getData");
$("#hk").attr("src", data );
$("#hk").slideDown(200);
// Loading text
$("#dev_signature").css("opacity",".625")
$("#content").css({"color":"#aaa","list-style-color":"#aaa !important"}).append("<div id=\"loading_area\"></div>");
$("#loading_area").append("<h2 id=\"loading\" style=\"text-align:center; color:green; display:none;\">Saving Contract…</h2>");
$("#loading_area").append("<p id=\"loading2\" style=\"text-align:center; color:#222; display:none;\">This shouldn\'t take more than a minute.</p>");
$("#loading").slideDown(300, function() {
$("#loading2").delay(2000).slideDown(300);
});
});
});
</script>
</body>
</html>';
if ($CLIENT_SIGNATURE) { $FOOTER_SIGNED_PHP ='
$phpName = basename($_SERVER["PHP_SELF"]) ? basename($_SERVER["PHP_SELF"]) : "index.php";
$fileName = substr($phpName , 0, -4);
$htmlName = $fileName.".html";
$pdfName = $fileName.".pdf";
?>
<div id="date-ip">
<strong>Signed on:</strong> <?php echo date("j F Y"); ?>
<br><strong>IP address:</strong>
<?php echo get_client_ip_env(); ?><br>
</div>
<?php // Function to get the client ip address
function get_client_ip_env() {
$ipaddress = "";
if (getenv("HTTP_CLIENT_IP"))
$ipaddress = getenv("HTTP_CLIENT_IP");
else if(getenv("HTTP_X_FORWARDED_FOR"))
$ipaddress = getenv("HTTP_X_FORWARDED_FOR");
else if(getenv("HTTP_X_FORWARDED"))
$ipaddress = getenv("HTTP_X_FORWARDED");
else if(getenv("HTTP_FORWARDED_FOR"))
$ipaddress = getenv("HTTP_FORWARDED_FOR");
else if(getenv("HTTP_FORWARDED"))
$ipaddress = getenv("HTTP_FORWARDED");
else if(getenv("REMOTE_ADDR"))
$ipaddress = getenv("REMOTE_ADDR");
else
$ipaddress = "UNKNOWN";
return $ipaddress;
} ?>
<div class="noprint" id="print-pdf">
<button id="print" type="button" class="button-secondary" onclick="printContract()">
Print contract
</button>
<button id="pdf" type="button" class="button-secondary" onclick="generatePdf()">
Download as PDF
</button>
</div>
</div> <!--#content-->
<!-- <script src="//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script> -->
<script>
function printContract() {
window.print();
}
function generatePdf() {
// http://pdfcrowd.com/save-to-pdf/
window.location.href = "//pdfcrowd.com/url_to_pdf/?use_print_media=1&height=-1&pdf_name=<?php echo $pdfName; ?>";
}
</script>
</body>
</html>';
} else $FOOTER_SIGNED_PHP = null;
if($CLIENT_SIGNATURE==null) {
if ( $selfDelete && file_exists($htmlName) ) {
header('Location: '.$htmlName.'#hk');
die();
}
/** Waiting for Client to sign: include signature elements and javascript **/
echo $HEADER;
echo $CONTRACT_HTML;
echo $DEV_SIGNATURE;
eval (' ?>'. $FOOTER_UNSIGNED .'<?php ');
}
else {
/** Contract was just signed: put $CLIENT_SIGNATURE and the other parts in the .html file **/
file_put_contents($htmlName, $HEADER);
file_put_contents($htmlName, $CONTRACT_HTML, FILE_APPEND | LOCK_EX);
file_put_contents($htmlName, $DEV_SIGNATURE, FILE_APPEND | LOCK_EX);
file_put_contents($htmlName, $CLIENT_SIGNATURE, FILE_APPEND | LOCK_EX);
ob_start();
eval($FOOTER_SIGNED_PHP); // https://cgd.io/2008/how-to-execute-php-code-in-a-php-string/
$FOOTER_SIGNED_COMPILED = ob_get_contents();
ob_end_clean();
file_put_contents($htmlName, $FOOTER_SIGNED_COMPILED, FILE_APPEND | LOCK_EX);
// Email client & dev, delete php, redirect to html
if ($clientEmail) {
$headers = "From: " . $devEmail . "\r\n";
$headers .= "MIME-Version: 1.0\r\n";
$headers .= "Content-Type: text/html; charset=ISO-8859-1\r\n";
$msg = 'The contract was signed. You can <a href="' .getUrl(). '">view or download this contract from here</a>.';
mail($clientEmail,'Contract signed', $msg, $headers);
}
if ($devEmail) {
$headers = "From: " . $clientEmail . "\r\n";
$headers .= "MIME-Version: 1.0\r\n";
$headers .= "Content-Type: text/html; charset=ISO-8859-1\r\n";
$msg = '<p>A new contract was signed. You can <a href="' .getUrl(). '">view or download this contract from here</a>.</p>';
$msg.= 'The contract was signed by: ' .$clientEmail;
mail($devEmail,'Contract signed!', $msg, $headers);
}
if ($selfDelete) unlink(__FILE__);
header('Location: '.$htmlName.'#hk');
die();
}
// Get the current file URL and replaces the .php extension with .html
function getUrl() {
$url = @( $_SERVER["HTTPS"] != 'on' ) ? 'http://'.$_SERVER["SERVER_NAME"] : 'https://'.$_SERVER["SERVER_NAME"];
$url .= ( $_SERVER["SERVER_PORT"] !== 80 ) ? ":".$_SERVER["SERVER_PORT"] : "";
$url .= $_SERVER["REQUEST_URI"];
$url = substr($url,0,-4) . '.html';
return $url;
}
?>