Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(tally): integrate tally poll #2715

Merged
merged 6 commits into from
Apr 2, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 3 additions & 2 deletions .env.example
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,6 @@ ACTIVE_RECORD_ENCRYPTION_PRIMARY_KEY=ACTIVE_RECORD_ENCRYPTION_PRIMARY_KEY
ACTIVE_RECORD_ENCRYPTION_DETERMINISTIC_KEY=ACTIVE_RECORD_ENCRYPTION_DETERMINISTIC_KEY
ACTIVE_RECORD_KEY_DERIVATION_SALT=ACTIVE_RECORD_KEY_DERIVATION_SALT

MAZE_API_KEY=change_me

# Third-party services
## SMS
BREVO_API_V3_KEY=change_me
Expand Down Expand Up @@ -55,3 +53,6 @@ RDV_SOLIDARITES_OAUTH_APP_ID=zC24y16rYftyrBgTj8h08g1NZKkwStXWe3E_lLMGoHc
RDV_SOLIDARITES_OAUTH_APP_SECRET=development-EdtuETfEK5Lr_--kx6S_QlItIJov-iThILw6M8IyTus
CRISP_WEBSITE_ID=change_me
ENABLE_CRISP=false


TALLY_NEW_UPLOAD_FORM_ID=change_me
49 changes: 49 additions & 0 deletions app/javascript/controllers/tally_controller.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { Controller } from "@hotwired/stimulus";

export default class extends Controller {
async showPopup() {
if (this.#hasAlreadyAnswered || !this.element.dataset.tallyFormId) return

await this.#loadScript()
this.#displayForm()
}

#loadScript() {
return new Promise((resolve) => {
if (!document.querySelector("script#tally-script")) {
const script = document.createElement("script");
script.src = "https://tally.so/widgets/embed.js";
script.id = "tally-script"
script.onload = resolve
document.head.appendChild(script);
} else {
Comment on lines +13 to +19
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

je sais pas si ça va passer avec les nouveaux csp, mais on peut pas le loader sur la page avant au pire ?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Je trouve ça plutôt bien de justement le loader au besoin depuis là. Mais je pense qu'on doit pouvoir juste faire une exception non ?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Si y a pas de souci niveau csp ça me va de le laisser là

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

J'ai fait le check après avoir mergé c'est ok visiblement 👍

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

En dev ça passe ?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oui oui, chez moi ça fonctionne bien, même après redémarrage du serveur :
image

resolve();
}
})
}

#displayForm() {
setTimeout(() => {
window.Tally.openPopup(this.element.dataset.tallyFormId, {
onSubmit: () => {
this.#hasAlreadyAnswered = true

// Give time for the user to read success message in Tally's popup
setTimeout(() => window.Tally.closePopup(this.element.dataset.tallyFormId), 1000);
}
})
}, this.element.dataset.tallyDelayInMs || 0)
}

set #hasAlreadyAnswered(value) {
localStorage.setItem(this.#hasAlreadyAnsweredKey, value)
}

get #hasAlreadyAnswered() {
return !!localStorage.getItem(this.#hasAlreadyAnsweredKey)
}

get #hasAlreadyAnsweredKey() {
return `tally-answered-${this.element.dataset.tallyFormId}`
}
}
32 changes: 0 additions & 32 deletions app/javascript/react/components/MazePoll.jsx

This file was deleted.

2 changes: 0 additions & 2 deletions app/javascript/react/pages/users/Uploads.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@ import EnrichWithContactFile from "../../components/users/EnrichWithContactFile"
import BatchActionsButtons from "../../components/users/BatchActionsButtons";
import DisplayReferentsColumnButton from "../../components/users/DisplayReferentsColumnButton";
import UsersList from "../../components/users/UsersList";
import MazePoll from "../../components/MazePoll";

import uploadFile from "../../lib/uploadFile";
import retrieveUpToDateUsers from "../../lib/retrieveUpToDateUsers";
Expand Down Expand Up @@ -232,7 +231,6 @@ const UsersUploads = observer(
</div>
</div>
<UsersList users={users} />
<MazePoll />
</>
)}
</>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@
</div>
<div>
<% if @all_invitations_attempted %>
<%= link_to "Terminer", @user_list_upload.structure_users_path, class: "btn btn-primary d-block mx-auto", data: { turbo_frame: "_top" } %>
<%= link_to "Terminer", @user_list_upload.structure_users_path, class: "btn btn-primary d-block mx-auto", data: { turbo_frame: "_top", controller: "tally", "tally-form-id": ENV["TALLY_NEW_UPLOAD_FORM_ID"], "tally-delay-in-ms": 1000, action: "click->tally#showPopup" } %>
<% else %>
<%= link_to "Terminer", "#", class: "btn btn-primary d-block mx-auto disabled" %>
<% end %>
Expand Down
15 changes: 14 additions & 1 deletion app/views/user_list_uploads/user_save_attempts/index.html.erb
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,20 @@
<div class="container sticky-footer">
<div class="py-3 px-4 bg-white d-flex justify-content-end">
<div>
<%= link_to "Terminer et revenir à l'accueil", @user_list_upload.structure_users_path, class: "btn btn-blue-out #{'link-disabled' unless @all_saves_attempted}", data: { turbo_frame: "_top" } %>
<%=
link_to(
"Terminer et revenir à l'accueil",
@user_list_upload.structure_users_path,
class: "btn btn-blue-out #{'link-disabled' unless @all_saves_attempted}",
data: {
turbo_frame: "_top",
controller: "tally",
"tally-form-id": ENV["TALLY_NEW_UPLOAD_FORM_ID"],
"tally-delay-in-ms": 1000,
action: "click->tally#showPopup"
}
)
%>
</div>
<div class="ms-3">
<% if @user_list_upload.invitations_enabled? %>
Expand Down
8 changes: 4 additions & 4 deletions config/initializers/content_security_policy.rb
Original file line number Diff line number Diff line change
Expand Up @@ -9,19 +9,19 @@
matomo = "matomo.inclusion.beta.gouv.fr"
crisp = ["*.crisp.chat", "wss://client.relay.crisp.chat"]
sentry = "sentry.incubateur.net"
maze = "*.maze.co"
tally = "tally.so"
flourish = ["flo.uri.sh", "https://public.flourish.studio/resources/embed.js"] # for deployment map

Rails.application.config.content_security_policy do |policy|
policy.default_src :self
policy.font_src :self, :data, *crisp
policy.img_src :self, :data, s3_bucket, *crisp
policy.media_src :self, s3_bucket
policy.frame_src :self, *flourish, maze
policy.frame_src :self, *flourish, tally
policy.object_src :none
policy.script_src :self, matomo, *crisp, *flourish, maze, sentry
policy.script_src :self, matomo, *crisp, *flourish, tally, sentry
policy.style_src :self, :unsafe_inline, *crisp
policy.connect_src :self, rdv_solidarites, sentry, matomo, maze, *crisp
policy.connect_src :self, rdv_solidarites, sentry, matomo, tally, *crisp
policy.form_action :self, rdv_solidarites
policy.frame_ancestors :self, rdv_solidarites, matomo
policy.worker_src :self, :blob
Expand Down
1 change: 0 additions & 1 deletion config/webpack/webpack.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,6 @@ module.exports = {
"process.env.RAILS_ENV": JSON.stringify(process.env.RAILS_ENV),
"process.env.MATOMO_CONTAINER_ID": JSON.stringify(process.env.MATOMO_CONTAINER_ID),
"process.env.CRISP_WEBSITE_ID": JSON.stringify(process.env.CRISP_WEBSITE_ID),
"process.env.MAZE_API_KEY": JSON.stringify(process.env.MAZE_API_KEY),
}),
// Include plugins
new RemoveEmptyScriptsPlugin(),
Expand Down