diff --git a/VERSION b/VERSION index 34a83616bb5aa9a70c5713bc45cd45498a50ba24..54d1a4f2a4a7f6afc19897c88a7b73c17ccc54fb 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -0.12.1 +0.13.0 diff --git a/lib/SeMeet/Controller/Invites.pm b/lib/SeMeet/Controller/Invites.pm index 09fd480b546599ab7d562a131557e0d47a5e198e..a2335d13c78820c98665ae24a04476b678d21a69 100644 --- a/lib/SeMeet/Controller/Invites.pm +++ b/lib/SeMeet/Controller/Invites.pm @@ -54,7 +54,7 @@ sub meet($c) { my $invite = $c->schema->resultset('Invite')->search({ token => uc($c->stash->{token}), - expire => {'>' => \'now()'}, +# expire => {'>' => \'now()'}, # zatim nebudeme limitovat casove })->first; if ( ! $invite ) { diff --git a/lib/SeMeet/Controller/Meets.pm b/lib/SeMeet/Controller/Meets.pm index 4f4670e81909c6a886870439aa76f48bed1cf2b4..4fd168985752dff6c7757b811d8fed0f9d674cce 100644 --- a/lib/SeMeet/Controller/Meets.pm +++ b/lib/SeMeet/Controller/Meets.pm @@ -48,7 +48,8 @@ sub get($c) { my $meet = $c->_get($c->stash->{id}, 0) // return; - my $users = { 0 => [], 1 => [] }; + my $users = { 0 => [], 1 => [] }; + my @invites = (); USER: foreach my $user ( $meet->users ) { @@ -58,11 +59,21 @@ sub get($c) { } } + INVITE: + foreach my $invite ( $meet->invites ) { + push @invites, { + id => $invite->id, + token => $invite->token, + displayname => $invite->displayname, + } + } + $c->render(openapi => $c->spec_filter({ $meet->get_columns, groups => $meet->cached_groups, #TODO: direct moderators => $users->{1}, participants => $users->{0}, + invites => \@invites, }, 'Meet')); } @@ -94,6 +105,7 @@ sub list($c) { groups => $meet->cached_groups, moderators => $meet->cached_moderators, participants => [], + invites => [], }, 'Meet') } @@ -171,6 +183,18 @@ sub delete_user($c) { $c->render( status => 204, text => '' ); } +sub delete_invite($c) { + $c->openapi->valid_input or return; + + my $meet = $c->_get($c->stash->{id}, 2) // return; + + $meet->delete_related('invites', + { id => $c->stash->{invite_id} } + ); + + $c->render( status => 204, text => '' ); +} + sub delete($c) { $c->openapi->valid_input or return; diff --git a/lib/SeMeet/Schema/Result/Meet.pm b/lib/SeMeet/Schema/Result/Meet.pm index aaa221c446202bb556966c2d547e28f274b7b7b4..3a0c1516c6c4fe37e3df82484444434b77b143c7 100644 --- a/lib/SeMeet/Schema/Result/Meet.pm +++ b/lib/SeMeet/Schema/Result/Meet.pm @@ -65,6 +65,11 @@ __PACKAGE__->has_many( { 'foreign.meet_id' => 'self.id', }, ); +__PACKAGE__->has_many( + invites => 'SeMeet::Schema::Result::Invite', + { 'foreign.meet_id' => 'self.id', }, +); + __PACKAGE__->inflate_column('properties', { inflate => sub { from_json(shift); }, deflate => sub { to_json(shift); }, diff --git a/openapi.yaml b/openapi.yaml index 69009f10612650474d4b3f1a115e4e8f86b72ef7..04533ad772dc3f8ad4bbf9e19d9428638c9521e2 100644 --- a/openapi.yaml +++ b/openapi.yaml @@ -52,6 +52,10 @@ components: type: array items: $ref: '#/components/schemas/UserInList' + invites: + type: array + items: + $ref: '#/components/schemas/InviteInList' GroupInList: type: object properties: @@ -101,6 +105,19 @@ components: email: type: string description: E-mail + InviteInList: + type: object + properties: + id: + type: integer + readOnly: true + token: + type: string + description: Access token + readOnly: true + displayname: + type: string + description: Jmeno, Prijmeni paths: /meets: @@ -423,6 +440,33 @@ paths: responses: 204: description: User deleted + /meets/{id}/invites/{invite_id}: + delete: + x-mojo-to: meets#delete_invite + security: + - Bearer: [] + tags: + - meets + summary: "Smazat pozvanku" + operationId: deleteInviteFromMeet + parameters: + - name: id + in: path + required: true + example: 100345 + description: "Identifikator mistnosti" + schema: + type: integer + - name: invite_id + in: path + required: true + example: 100345 + description: "Identifikator pozvanky" + schema: + type: integer + responses: + 204: + description: Invite deleted /invites: post: x-mojo-to: invites#create diff --git a/templates/includes/meet_invites.html.ep b/templates/includes/meet_invites.html.ep index 7daf8cb3d6644eeb8385c51111071d2c0b6482f7..f95dbfc0e941cc00cab4d76bc0fe32c25831e8fd 100644 --- a/templates/includes/meet_invites.html.ep +++ b/templates/includes/meet_invites.html.ep @@ -1,4 +1,18 @@ -% if ( $roles->{owner} || $roles->{moderator} ) { +% if ( $roles->{owner} || $roles->{moderator} ) { +<div class="mb-4"> +<span v-for="invite in meet.invites" class="chip chip--violet-400 mr-1 mb-1 rounded"> +{{ invite.displayname }} +% if ( $roles->{owner} || $roles->{moderator} ) { + <span class="icon" title="Zobrazit pozvánku"> + <i class="ico--link ml-3 cursor-pointer" @click="showInvite(invite)"></i> + </span> + <span class="icon" title="Smazat pozvánku"> + <i class="ico--cross ml-3 cursor-pointer" @click="removeInvite(invite.id)"></i> + </span> +% } +</span> +</div> + <div class="grid grid-cols-5"> <div class="mr-2 col-span-2"> @@ -8,16 +22,29 @@ <div class="mr-2 col-span-3"> <label class="form-field__label"><%=l 'E-mail' %></label> + <div class="flex flex-row"> + <input type="text" class="w-full text-input mr-2" value="" v-model.lazy="newInvite.email" placeholder="<%=l 'INPUT_INVITE_EMAIL_PLACEHOLDER' %>"/> + <button class="btn btn--violet-400 btn--hoveractive text-sm ml-2" @click="addInvite()"> + <div class="btn__body" style="white-space: nowrap;"><%=l 'Create invite' %></div> + </button> + </div> + </div> + +% if (0) { + <div class="mr-2 col-span-2"> + <label class="form-field__label"><%=l 'Platnost do' %></label> <div class="flex flex-row"> - <input type="text" class="w-full text-input mr-2" value="" v-model.lazy="newInvite.email" placeholder="<%=l 'INPUT_INVITE_EMAIL_PLACEHOLDER' %>"/> - <button class="btn btn--violet-400 btn--hoveractive text-sm" @click="addInvite()"> + <date-picker format="DD.MM.YYYY" v-model.lazy="newInvite.expire"></date-picker> + <button class="btn btn--violet-400 btn--hoveractive text-sm ml-2" @click="addInvite()"> <div class="btn__body" style="white-space: nowrap;"><%=l 'Create invite' %></div> </button> </div> </div> +% } </div> + % } diff --git a/templates/layouts/default.html.ep b/templates/layouts/default.html.ep index 5f5e8b712309db4722b3a592bda2621830a4a363..d0b1ca9a443cbc3484471356948b424da67ec8b0 100644 --- a/templates/layouts/default.html.ep +++ b/templates/layouts/default.html.ep @@ -28,6 +28,13 @@ % } <script src="https://cdn-unpkg.pirati.cz/vue@2.7.8/dist/vue.min.js"></script> <script src="<%= config->{styleguide} %>js/main.bundle.js"></script> + + <script src="https://cdn-unpkg.pirati.cz/vue2-datepicker@3.11.1/index.min.js"></script> + <script src="https://cdn-unpkg.pirati.cz/vue2-datepicker@3.11.1/locale/cs.js"></script> + <link href="https://cdn-unpkg.pirati.cz/vue2-datepicker@3.11.1/index.css" rel="stylesheet"> + <style> + .mx-input { height: 48px; border-color: #adadad; border-radius: 0; } + </style> </head> <body> diff --git a/templates/meet.html.ep b/templates/meet.html.ep index cc89bcbed7ee0d66e970cc0ee0d47cc5bb46e55f..3363c27cb153ee26cc821b65e8d7bd832e4c2269 100644 --- a/templates/meet.html.ep +++ b/templates/meet.html.ep @@ -4,11 +4,11 @@ %= include 'includes/meet' -<div class="grid grid-cols-4 border border-b-0 divide-x text-center"> +<div class="grid grid-cols-5 border border-b-0 divide-x text-center"> <div @click="active_tab='groups'" class="p-4 bg-grey-125" :class="tabClass('groups')"><%=l 'Authorized groups' %></div> <div @click="active_tab='participants'" class="p-4 bg-grey-125" :class="tabClass('participants')"><%=l 'Authorized persons' %></div> <div @click="active_tab='moderators'" class="p-4 bg-grey-125" :class="tabClass('moderators')"><%=l 'Moderators' %></div> -%#<div @click="active_tab='invites'" class="p-4 bg-grey-125" :class="tabClass('invites')"><%=l 'Invites' %></div> +<div @click="active_tab='invites'" class="p-4 bg-grey-125" :class="tabClass('invites')"><%=l 'Invites' %></div> <div @click="active_tab='form'" class="p-4 bg-grey-125" :class="tabClass('form')"><%=l $roles->{owner} ? 'Configuration':'About meet' %></div> </div> @@ -22,9 +22,9 @@ <div v-if="active_tab == 'moderators'"> %= include 'includes/meet_moderators' </div> -%# <div v-if="active_tab == 'invites'"> -%#= include 'includes/meet_invites' -%# </div> + <div v-if="active_tab == 'invites'"> +%= include 'includes/meet_invites' + </div> <div v-if="active_tab == 'form'"> % if ( $roles->{owner} ) { %= include 'includes/meet_form' @@ -34,6 +34,22 @@ </div> </div> +<div class="modal__overlay" v-if="inviteInfoVisible" v-cloak> + <div class="modal__content" role="dialog"> + <div class="modal__container" role="dialog"> + <div class="modal__container-body elevation-10"> + <button class="modal__close" title="ZavĹ™Ăt" @click="inviteInfoVisible = false"><i class="ico--cross"></i></button> + <div class="card"> + <div class="card__body w-120 text-center"> + <p class="card-body-text">Adresa pozvánky byla zkopĂrovana do schránky</p> + <b><a v-bind:href="selectedInvite.full_url">{{ selectedInvite.full_url }}</a></b> + </div> + </div> + </div> + </div> + </div> +</div> + </div><%# APP %> @@ -57,6 +73,8 @@ meet: {}, newInvite: {}, + selectedInvite: {}, + inviteInfoVisible: false, active_tab: 'groups', delete_confirm_visible: false, @@ -225,8 +243,8 @@ }, addInvite: function() { + console.log(this.newInvite); if ( this.newInvite.displayname == '' || this.newInvite.email == '' ) { - console.log('XXX'); return true; } @@ -245,8 +263,14 @@ }) }, + showInvite: function(invite) { + this.selectedInvite.full_url = 'https://<%= config->{domain} %>/guest/'+ invite.token ; + this.inviteInfoVisible = true; + navigator.clipboard.writeText(this.selectedInvite.full_url); + }, + removeInvite: function(id) { - fetch(INVITES_URL + id, { + fetch(MEET_URL + '/invites/' + id, { method: "DELETE", headers: API_HEADERS, })