Skip to content

Commit c3c9ef8

Browse files
author
steve-ks
committed
Add mail support
* Allow sending emails with generated link * Updated language support * Added webbox config for storage lifetime * Updated responsive design for modal * Moved select storage lifetime to own vue component
1 parent 759887e commit c3c9ef8

22 files changed

+323
-112
lines changed

app/Console/Kernel.php

+2-2
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,9 @@ class Kernel extends ConsoleKernel
2525
protected function schedule(Schedule $schedule)
2626
{
2727
// Cleanup old upload directories
28-
$schedule->command('upload:cleanup')->hourly();
28+
$schedule->command('upload:cleanup')->everyFifteenMinutes();
2929
// Cleanup old storage directories
30-
$schedule->command('storage:cleanup')->hourly();
30+
$schedule->command('storage:cleanup')->everyFifteenMinutes();
3131
}
3232

3333
/**

app/Http/Controllers/UploadController.php

+17
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,13 @@
22

33
namespace App\Http\Controllers;
44

5+
use App\Mail\SharedLink;
56
use App\Traits\SessionLifetime;
67
use App\Traits\StorageTime;
78
use App\Traits\StringAdditions;
89
use Illuminate\Http\Request;
910
use Illuminate\Support\Facades\File;
11+
use Illuminate\Support\Facades\Mail;
1012
use Illuminate\Support\Facades\Storage;
1113
use Intervention\Image\Exception\NotReadableException;
1214
use Intervention\Image\Facades\Image;
@@ -118,4 +120,19 @@ public function store(Request $request)
118120
'url' => 'share/'.$sessId
119121
]);
120122
}
123+
124+
/**
125+
* Send an email with the given session id as link
126+
*/
127+
public function sendmail(Request $request)
128+
{
129+
$sessId = $request->session()->get('sessionid');
130+
Mail::to($request->input('email'))
131+
->send(new SharedLink(
132+
$sessId,
133+
$this->getStorageTime($sessId)->isoFormat('LLLL')
134+
));
135+
136+
return response()->json(['success' => true]);
137+
}
121138
}

app/Mail/SharedLink.php

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<?php
2+
3+
namespace App\Mail;
4+
5+
use Illuminate\Bus\Queueable;
6+
use Illuminate\Mail\Mailable;
7+
use Illuminate\Queue\SerializesModels;
8+
9+
class SharedLink extends Mailable
10+
{
11+
use Queueable, SerializesModels;
12+
13+
/**
14+
* This is the shared link ending date
15+
*/
16+
public $validUntil;
17+
/**
18+
* This is the shared link URL passed to the email
19+
*/
20+
public $url;
21+
22+
/**
23+
* Create a new message instance.
24+
*
25+
* @return void
26+
*/
27+
public function __construct(string $path, string $validUntil)
28+
{
29+
$this->validUntil = $validUntil;
30+
$this->url = preg_replace('/\/*$/', '', config('app.url')) . '/' . $path;
31+
}
32+
33+
/**
34+
* Build the message.
35+
*
36+
* @return $this
37+
*/
38+
public function build()
39+
{
40+
// Use from address as given by config/mail.php
41+
return $this->markdown('emails.shared.link');
42+
}
43+
}

app/User.php

-39
This file was deleted.

composer.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
"php": "^7.2.5",
1212
"fideloper/proxy": "^4.2",
1313
"fruitcake/laravel-cors": "^1.0",
14-
"guzzlehttp/guzzle": "^6.3",
14+
"guzzlehttp/guzzle": "^6.5",
1515
"illuminate/filesystem": "^7.5",
1616
"intervention/image": "^2.5",
1717
"laravel/framework": "^7.5",

composer.lock

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

config/mail.php

+3-3
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@
1313
|
1414
*/
1515

16-
'default' => env('MAIL_MAILER', 'smtp'),
16+
'default' => env('MAIL_MAILER', null),
1717

1818
/*
1919
|--------------------------------------------------------------------------
@@ -83,8 +83,8 @@
8383
*/
8484

8585
'from' => [
86-
'address' => env('MAIL_FROM_ADDRESS', '[email protected]'),
87-
'name' => env('MAIL_FROM_NAME', 'Example'),
86+
'address' => env('MAIL_FROM_ADDRESS', '[email protected]'),
87+
'name' => env('MAIL_FROM_NAME', 'Shared Webbox Link'),
8888
],
8989

9090
/*

config/webbox.php

+38
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,10 @@
1616
// Maximum allowed file size in megabytes (should fit with php & webserver upload limits)
1717
'max_filesize' => env('MAX_FILESIZE_MB', '256'),
1818

19+
// Footer text, can also be empty
20+
'footer_text' => env('FOOTER_TEXT', '&#169; 2020 powered by KingStarter GbR'),
21+
'footer_link' => env('FOOTER_LINK', 'https://kingstarter.de'),
22+
1923
/*
2024
|--------------------------------------------------------------------------
2125
| Honeypot configuration
@@ -30,4 +34,38 @@
3034

3135
// Honeypot field name for form input, should be a known field with some random chars
3236
'honeypot_field' => env('HONEYPOT_FIELD', 'phone_number_4f3dx'),
37+
38+
/*
39+
|--------------------------------------------------------------------------
40+
| Storage lifetime
41+
|--------------------------------------------------------------------------
42+
|
43+
| Set an array of possible storage lifetime durations when generating
44+
| data links. The labels for generic time periods are translated
45+
| automatically with singular and plural modifications added.
46+
|
47+
| NOTE: Cleanup schedules are set to everyFifteenMinutes, to provide
48+
| even shorter lifetimes, the app/Console/Kernel.php cleanup
49+
| schedules need to be called more frequently.
50+
|
51+
*/
52+
53+
// Value format: /[0-9]+ [a-z]+/
54+
'storage_lifetime' => [
55+
'1 hour',
56+
'6 hours',
57+
'1 day',
58+
'2 days',
59+
'3 days',
60+
'1 week',
61+
'2 weeks',
62+
'1 month',
63+
'2 months',
64+
'3 months',
65+
'6 months',
66+
'1 year',
67+
],
68+
69+
// The default lifetime should be one option within the storage duration array
70+
'default_lifetime' => '1 month',
3371
];

public/css/app.css

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

public/js/app.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

resources/js/components/Modal.vue

+50-8
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
<div class="modal fixed flex z-20 top-0 left-0 w-full h-full overflow-auto backdrop justify-around pt-6 sm:pt-12 md:pt-32"
55
v-if="show" @click.prevent="handleClick">
66

7-
<div class="modal-container bg-white w-full sm:w-4/5 md:w-3/5 max-w-md h-64 sm:h-56 md:h-48 shadow" @click.prevent="handleClick">
7+
<div class="modal-container bg-white w-full sm:w-4/5 md:w-3/5 max-w-md rounded" :class="modalHeight" @click.prevent="handleClick">
88

99
<!-- Add margin if you want to see some of the overlay behind the modal-->
1010
<div class="modal-content py-4 text-left px-6">
@@ -20,15 +20,25 @@
2020
</div>
2121

2222
<!--Body-->
23-
<div class="w-full p-3 border rounded shadow bg-gray-100 truncate mb-4 sm:mb-3 md:mb-0" v-text="url"></div>
23+
<div>
24+
<div class="w-full p-3 border rounded shadow bg-gray-100 truncate mb-4 sm:mb-3" v-text="url + 'a'"></div>
25+
26+
<!-- Mail input -->
27+
<div class="flex my-4" v-if="withmail">
28+
<input type="email" class="form-control text-left rounded-r-none" v-model="email" @keyup.enter="sendEmail" :placeholder="$trans('email.placeholder')" />
29+
<button class="btn btn-teal rounded-l-none" :title="$trans('link.email')" @click="sendEmail()" :class="btnClsDisabled" :disabled="!emailValid">
30+
<span class="fa fa-envelope"></span>
31+
</button>
32+
</div>
33+
</div>
2434

2535
<!-- footer -->
26-
<div class="w-full flex justify-around py-5 px-0 sm:px-2 md:px-5">
27-
<button class="btn btn-teal w-40 text-center mr-2" @click="copyToClipboard">
36+
<div class="w-full flex flex-col sm:flex-row justify-around items-center h-24 sm:h-20 pt-8 sm:pt-0">
37+
<button class="btn btn-teal text-center w-56 sm:w-40 my-2 sm:my-0 sm:mr-2" @click="copyToClipboard">
2838
<span class="fa fa-copy"></span>
2939
<span v-trans="'link.copy'"></span>
3040
</button>
31-
<button class="btn btn-teal w-40 text-center ml-2" @click="openPath">
41+
<button class="btn btn-teal text-center w-56 sm:w-40 my-2 sm:my-0 sm:ml-2" @click="openPath">
3242
<span class="fa fa-external-link-alt"></span>
3343
<span v-trans="'link.open'"></span>
3444
</button>
@@ -49,13 +59,33 @@ export default {
4959
type: String,
5060
default: 'Modal title'
5161
},
52-
url: String
62+
url: String,
63+
withmail: {
64+
type: Boolean,
65+
default: false
66+
}
67+
},
68+
69+
data () {
70+
return {
71+
email: '',
72+
sending: false
73+
}
5374
},
5475
5576
computed: {
5677
fullPath () {
5778
let basepath = window.location.href
5879
return basepath.endsWith('/') ? basepath + this.url : basepath + '/' + this.url
80+
},
81+
modalHeight () {
82+
return this.withmail ? 'h-84 sm:h-72' : 'h-72 sm:h-56'
83+
},
84+
emailValid () {
85+
return /\S+@\S+\.\S+/.test(this.email)
86+
},
87+
btnClsDisabled () {
88+
return !this.emailValid || this.sending ? 'opacity-50 cursor-not-allowed' : ''
5989
}
6090
},
6191
@@ -67,12 +97,24 @@ export default {
6797
emitModalClose () {
6898
this.$emit('close')
6999
},
100+
openPath () {
101+
window.open(this.fullPath)
102+
},
70103
async copyToClipboard () {
71104
await navigator.clipboard.writeText(this.fullPath);
72105
this.$toastr.s(this.$trans('link.copied'));
73106
},
74-
openPath () {
75-
window.open(this.fullPath)
107+
sendEmail () {
108+
if (this.emailValid || this.sending) {
109+
this.sending = true
110+
this.$http.post('/sendmail', { 'email': this.email }).then(reponse => {
111+
this.$toastr.s(this.$trans('email.send.succ'))
112+
}).catch(err => {
113+
this.$toastr.e(this.$trans('email.send.fail'))
114+
}).then(() => {
115+
this.sending = false
116+
})
117+
}
76118
}
77119
}
78120
}
+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
<template>
2+
<select
3+
v-bind="$attrs"
4+
v-bind:value="value"
5+
@change.prevent="onchange"
6+
>
7+
<option v-for="item in optionList" :key="item.value" :value="item.value" v-text="item.label"></option>
8+
</select>
9+
</template>
10+
11+
<script>
12+
export default {
13+
inheritAttrs: false,
14+
15+
model: {
16+
prop: 'value',
17+
event: 'change'
18+
},
19+
20+
props: ['value'],
21+
22+
data () {
23+
return {
24+
// Default storage lifetime array
25+
values: [
26+
'1 day',
27+
'2 days',
28+
'3 days',
29+
'1 week',
30+
'2 weeks',
31+
'1 month',
32+
'2 months',
33+
'3 months',
34+
'1 year',
35+
]
36+
}
37+
},
38+
39+
mounted () {
40+
// Retrieve config array
41+
if (window.config && Array.isArray(window.config.storageTimes) && window.config.storageTimes.length)
42+
this.values = window.config.storageTimes
43+
},
44+
45+
computed: {
46+
optionList () {
47+
let options = []
48+
this.values.forEach(item => {
49+
let parts = item.split(' ')
50+
let msgcode = (parts.length > 1 ? 'time.' + parts[1] : 'timespan')
51+
msgcode.endsWith('s') || (msgcode = msgcode+'s')
52+
options.push({
53+
value: item,
54+
label: `${parts[0]} ${this.$choice(msgcode, parts[0])}`
55+
})
56+
})
57+
return options
58+
}
59+
},
60+
61+
methods: {
62+
// Enable v-model binding
63+
onchange(event) {
64+
// When emitted inline the parent model seems to stay unchanged,
65+
// though moved event emitter to method
66+
this.$emit('change', event.target.value);
67+
}
68+
}
69+
}
70+
</script>

0 commit comments

Comments
 (0)