You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							621 lines
						
					
					
						
							21 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							621 lines
						
					
					
						
							21 KiB
						
					
					
				
								<template>
							 | 
						|
								  <li>
							 | 
						|
								    <!-- Last viewed separator -->
							 | 
						|
								    <div
							 | 
						|
								      v-if="record.jwtId == lastViewedClaimId"
							 | 
						|
								      class="border-b border-dashed border-slate-300 text-orange-400 mt-4 mb-6 font-bold text-sm"
							 | 
						|
								    >
							 | 
						|
								      <span class="block w-fit mx-auto -mb-2.5 bg-white px-2">
							 | 
						|
								        You've already seen all the following
							 | 
						|
								      </span>
							 | 
						|
								    </div>
							 | 
						|
								
							 | 
						|
								    <div
							 | 
						|
								      class="flex items-center justify-between gap-2 text-lg bg-slate-200 border border-slate-300 border-b-0 rounded-t-md px-3 sm:px-4 py-1 sm:py-2"
							 | 
						|
								    >
							 | 
						|
								      <div class="flex items-center gap-2">
							 | 
						|
								        <router-link
							 | 
						|
								          v-if="record.issuerDid && !isHiddenDid(record.issuerDid)"
							 | 
						|
								          :to="{
							 | 
						|
								            path: '/did/' + encodeURIComponent(record.issuerDid),
							 | 
						|
								          }"
							 | 
						|
								          title="More details about this person"
							 | 
						|
								        >
							 | 
						|
								          <EntityIcon
							 | 
						|
								            :entity-id="record.issuerDid"
							 | 
						|
								            class="rounded-full bg-white overflow-hidden !size-[2rem] object-cover"
							 | 
						|
								          />
							 | 
						|
								        </router-link>
							 | 
						|
								        <font-awesome
							 | 
						|
								          v-else-if="isHiddenDid(record.issuerDid)"
							 | 
						|
								          icon="eye-slash"
							 | 
						|
								          class="text-slate-400 !size-[2rem] cursor-pointer"
							 | 
						|
								          @click="notifyHiddenPerson"
							 | 
						|
								        />
							 | 
						|
								        <font-awesome
							 | 
						|
								          v-else
							 | 
						|
								          icon="person-circle-question"
							 | 
						|
								          class="text-slate-400 !size-[2rem] cursor-pointer"
							 | 
						|
								          @click="notifyUnknownPerson"
							 | 
						|
								        />
							 | 
						|
								
							 | 
						|
								        <div>
							 | 
						|
								          <h3 v-if="record.issuer.known" class="font-semibold leading-tight">
							 | 
						|
								            {{ record.issuer.displayName }}
							 | 
						|
								          </h3>
							 | 
						|
								          <p class="ms-auto text-xs text-slate-500 italic">
							 | 
						|
								            {{ friendlyDate }}
							 | 
						|
								          </p>
							 | 
						|
								        </div>
							 | 
						|
								      </div>
							 | 
						|
								
							 | 
						|
								      <a
							 | 
						|
								        class="cursor-pointer"
							 | 
						|
								        data-testid="circle-info-link"
							 | 
						|
								        @click="emitLoadClaim(record.jwtId)"
							 | 
						|
								      >
							 | 
						|
								        <font-awesome icon="circle-info" class="fa-fw text-slate-500" />
							 | 
						|
								      </a>
							 | 
						|
								    </div>
							 | 
						|
								
							 | 
						|
								    <div class="bg-slate-100 rounded-b-md border border-slate-300 p-3 sm:p-4">
							 | 
						|
								      <!-- Record Image -->
							 | 
						|
								      <div
							 | 
						|
								        v-if="record.image"
							 | 
						|
								        class="bg-cover mb-2 -mt-3 sm:-mt-4 -mx-3 sm:-mx-4"
							 | 
						|
								        :style="`background-image: url(${record.image});`"
							 | 
						|
								      >
							 | 
						|
								        <a
							 | 
						|
								          class="block bg-slate-100/50 backdrop-blur-md px-6 py-4 cursor-pointer"
							 | 
						|
								          @click="emitViewImage(record.image)"
							 | 
						|
								        >
							 | 
						|
								          <img
							 | 
						|
								            class="w-full h-auto max-w-lg max-h-96 object-contain mx-auto drop-shadow-md"
							 | 
						|
								            :src="record.image"
							 | 
						|
								            alt="Activity image"
							 | 
						|
								          />
							 | 
						|
								        </a>
							 | 
						|
								      </div>
							 | 
						|
								
							 | 
						|
								      <!-- Emoji Section -->
							 | 
						|
								      <div
							 | 
						|
								        v-if="hasEmojis || isRegistered"
							 | 
						|
								        class="float-right ml-3 mb-1 bg-white rounded border border-slate-300 px-1.5 py-0.5 max-w-[240px]"
							 | 
						|
								      >
							 | 
						|
								        <div class="flex items-center justify-between gap-1">
							 | 
						|
								          <!-- Existing Emojis Display -->
							 | 
						|
								          <div v-if="hasEmojis" class="flex flex-wrap gap-1">
							 | 
						|
								            <button
							 | 
						|
								              v-for="(count, emoji) in record.emojiCount"
							 | 
						|
								              :key="emoji"
							 | 
						|
								              class="inline-flex items-center gap-0.5 px-1 py-0.5 text-xs bg-slate-50 hover:bg-slate-100 rounded border border-slate-200 transition-colors cursor-pointer"
							 | 
						|
								              :class="{
							 | 
						|
								                'bg-blue-50 border-blue-200': isUserEmojiWithoutLoading(emoji),
							 | 
						|
								                'opacity-75 cursor-wait': loadingEmojis,
							 | 
						|
								              }"
							 | 
						|
								              :title="
							 | 
						|
								                loadingEmojis
							 | 
						|
								                  ? 'Loading...'
							 | 
						|
								                  : !emojisOnActivity?.isResolved
							 | 
						|
								                    ? 'Click to load your emojis'
							 | 
						|
								                    : isUserEmojiWithoutLoading(emoji)
							 | 
						|
								                      ? 'Click to remove your emoji'
							 | 
						|
								                      : 'Click to add this emoji'
							 | 
						|
								              "
							 | 
						|
								              :disabled="!isRegistered"
							 | 
						|
								              @click="toggleThisEmoji(emoji)"
							 | 
						|
								            >
							 | 
						|
								              <!-- Show spinner when loading -->
							 | 
						|
								              <div v-if="loadingEmojis" class="animate-spin text-xs">
							 | 
						|
								                <font-awesome icon="spinner" class="fa-spin" />
							 | 
						|
								              </div>
							 | 
						|
								              <span v-else class="text-sm leading-none">{{ emoji }}</span>
							 | 
						|
								              <span class="text-xs text-slate-600 font-medium leading-none">{{
							 | 
						|
								                count
							 | 
						|
								              }}</span>
							 | 
						|
								            </button>
							 | 
						|
								          </div>
							 | 
						|
								
							 | 
						|
								          <!-- Add Emoji Button -->
							 | 
						|
								          <button
							 | 
						|
								            v-if="isRegistered"
							 | 
						|
								            class="inline-flex px-1 py-0.5 text-xs bg-slate-100 hover:bg-slate-200 rounded border border-slate-300 transition-colors items-center justify-center ml-2 ml-auto"
							 | 
						|
								            :title="showEmojiPicker ? 'Close emoji picker' : 'Add emoji'"
							 | 
						|
								            @click="toggleEmojiPicker"
							 | 
						|
								          >
							 | 
						|
								            <span class="px-2 text-sm leading-none">{{
							 | 
						|
								              showEmojiPicker ? "x" : "😊"
							 | 
						|
								            }}</span>
							 | 
						|
								          </button>
							 | 
						|
								        </div>
							 | 
						|
								
							 | 
						|
								        <!-- Emoji Picker (placeholder for now) -->
							 | 
						|
								        <div
							 | 
						|
								          v-if="showEmojiPicker"
							 | 
						|
								          class="mt-1 p-1.5 bg-slate-50 rounded border border-slate-300"
							 | 
						|
								        >
							 | 
						|
								          <!-- Temporary emoji buttons for testing -->
							 | 
						|
								          <div class="flex flex-wrap gap-3 mt-1">
							 | 
						|
								            <button
							 | 
						|
								              v-for="emoji in QUICK_EMOJIS"
							 | 
						|
								              :key="emoji"
							 | 
						|
								              class="p-0.5 hover:bg-slate-200 rounded text-base transition-opacity"
							 | 
						|
								              :class="{
							 | 
						|
								                'opacity-75 cursor-wait': loadingEmojis,
							 | 
						|
								              }"
							 | 
						|
								              :disabled="loadingEmojis"
							 | 
						|
								              @click="toggleThisEmoji(emoji)"
							 | 
						|
								            >
							 | 
						|
								              <!-- Show spinner when loading -->
							 | 
						|
								              <div v-if="loadingEmojis" class="animate-spin text-sm">⟳</div>
							 | 
						|
								              <span v-else>{{ emoji }}</span>
							 | 
						|
								            </button>
							 | 
						|
								          </div>
							 | 
						|
								        </div>
							 | 
						|
								      </div>
							 | 
						|
								
							 | 
						|
								      <!-- Description -->
							 | 
						|
								      <p class="font-medium">
							 | 
						|
								        <a class="block cursor-pointer" @click="emitLoadClaim(record.jwtId)">
							 | 
						|
								          <vue-markdown
							 | 
						|
								            :source="truncatedDescription"
							 | 
						|
								            class="markdown-content"
							 | 
						|
								          />
							 | 
						|
								        </a>
							 | 
						|
								      </p>
							 | 
						|
								
							 | 
						|
								      <div
							 | 
						|
								        class="clear-right relative flex justify-between gap-4 max-w-[40rem] mx-auto mt-4"
							 | 
						|
								      >
							 | 
						|
								        <!-- Source -->
							 | 
						|
								        <div
							 | 
						|
								          class="w-[7rem] sm:w-[12rem] text-center bg-white border border-slate-200 rounded p-2 sm:p-3"
							 | 
						|
								        >
							 | 
						|
								          <div class="relative w-fit mx-auto">
							 | 
						|
								            <div>
							 | 
						|
								              <!-- Project Icon -->
							 | 
						|
								              <div v-if="record.providerPlanName">
							 | 
						|
								                <router-link
							 | 
						|
								                  :to="{
							 | 
						|
								                    path:
							 | 
						|
								                      '/project/' +
							 | 
						|
								                      encodeURIComponent(record.providerPlanHandleId || ''),
							 | 
						|
								                  }"
							 | 
						|
								                  title="View project details"
							 | 
						|
								                >
							 | 
						|
								                  <ProjectIcon
							 | 
						|
								                    :entity-id="record.providerPlanHandleId || ''"
							 | 
						|
								                    :icon-size="48"
							 | 
						|
								                    class="rounded size-[3rem] sm:size-[4rem] *:w-full *:h-full"
							 | 
						|
								                  />
							 | 
						|
								                </router-link>
							 | 
						|
								              </div>
							 | 
						|
								              <!-- Identicon for DIDs -->
							 | 
						|
								              <div v-else-if="record.agentDid">
							 | 
						|
								                <router-link
							 | 
						|
								                  v-if="!isHiddenDid(record.agentDid)"
							 | 
						|
								                  :to="{
							 | 
						|
								                    path: '/did/' + encodeURIComponent(record.agentDid),
							 | 
						|
								                  }"
							 | 
						|
								                  title="More details about this person"
							 | 
						|
								                >
							 | 
						|
								                  <EntityIcon
							 | 
						|
								                    :entity-id="record.agentDid"
							 | 
						|
								                    :profile-image-url="record.issuer.profileImageUrl"
							 | 
						|
								                    class="rounded-full bg-slate-100 overflow-hidden !size-[3rem] sm:!size-[4rem]"
							 | 
						|
								                  />
							 | 
						|
								                </router-link>
							 | 
						|
								                <font-awesome
							 | 
						|
								                  v-else
							 | 
						|
								                  icon="eye-slash"
							 | 
						|
								                  class="text-slate-300 !size-[3rem] sm:!size-[4rem]"
							 | 
						|
								                  @click="notifyHiddenPerson"
							 | 
						|
								                />
							 | 
						|
								              </div>
							 | 
						|
								              <!-- Unknown Person -->
							 | 
						|
								              <div v-else>
							 | 
						|
								                <font-awesome
							 | 
						|
								                  icon="person-circle-question"
							 | 
						|
								                  class="text-slate-300 text-[3rem] sm:text-[4rem]"
							 | 
						|
								                  @click="notifyUnknownPerson"
							 | 
						|
								                />
							 | 
						|
								              </div>
							 | 
						|
								            </div>
							 | 
						|
								          </div>
							 | 
						|
								          <div
							 | 
						|
								            v-if="record.providerPlanName || record.giver.known"
							 | 
						|
								            class="text-xs mt-2 truncate"
							 | 
						|
								          >
							 | 
						|
								            <font-awesome
							 | 
						|
								              :icon="record.providerPlanName ? 'users' : 'user'"
							 | 
						|
								              class="fa-fw text-slate-400"
							 | 
						|
								            />
							 | 
						|
								            {{ record.providerPlanName || record.giver.displayName }}
							 | 
						|
								          </div>
							 | 
						|
								        </div>
							 | 
						|
								
							 | 
						|
								        <!-- Arrow -->
							 | 
						|
								        <div
							 | 
						|
								          class="absolute inset-x-[7rem] sm:inset-x-[12rem] mx-2 top-1/2 -translate-y-1/2"
							 | 
						|
								        >
							 | 
						|
								          <div
							 | 
						|
								            class="text-sm text-center leading-none font-semibold pe-2 sm:pe-4"
							 | 
						|
								          >
							 | 
						|
								            {{ fetchAmount }}
							 | 
						|
								          </div>
							 | 
						|
								
							 | 
						|
								          <div class="flex items-center">
							 | 
						|
								            <hr
							 | 
						|
								              class="grow border-t-[18px] sm:border-t-[24px] border-slate-300"
							 | 
						|
								            />
							 | 
						|
								
							 | 
						|
								            <div
							 | 
						|
								              class="shrink-0 w-0 h-0 border border-slate-300 border-t-[20px] sm:border-t-[25px] border-t-transparent border-b-[20px] sm:border-b-[25px] border-b-transparent border-s-[27px] sm:border-s-[34px] border-e-0"
							 | 
						|
								            ></div>
							 | 
						|
								          </div>
							 | 
						|
								        </div>
							 | 
						|
								
							 | 
						|
								        <!-- Destination -->
							 | 
						|
								        <div
							 | 
						|
								          class="w-[7rem] sm:w-[12rem] text-center bg-white border border-slate-200 rounded p-2 sm:p-3"
							 | 
						|
								        >
							 | 
						|
								          <div class="relative w-fit mx-auto">
							 | 
						|
								            <div>
							 | 
						|
								              <!-- Project Icon -->
							 | 
						|
								              <div v-if="record.recipientProjectName">
							 | 
						|
								                <router-link
							 | 
						|
								                  :to="{
							 | 
						|
								                    path:
							 | 
						|
								                      '/project/' +
							 | 
						|
								                      encodeURIComponent(record.fulfillsPlanHandleId || ''),
							 | 
						|
								                  }"
							 | 
						|
								                  title="View project details"
							 | 
						|
								                >
							 | 
						|
								                  <ProjectIcon
							 | 
						|
								                    :entity-id="record.fulfillsPlanHandleId || ''"
							 | 
						|
								                    :icon-size="48"
							 | 
						|
								                    class="rounded size-[3rem] sm:size-[4rem] *:w-full *:h-full"
							 | 
						|
								                  />
							 | 
						|
								                </router-link>
							 | 
						|
								              </div>
							 | 
						|
								              <!-- Identicon for DIDs -->
							 | 
						|
								              <div v-else-if="record.recipientDid">
							 | 
						|
								                <router-link
							 | 
						|
								                  v-if="!isHiddenDid(record.recipientDid)"
							 | 
						|
								                  :to="{
							 | 
						|
								                    path: '/did/' + encodeURIComponent(record.recipientDid),
							 | 
						|
								                  }"
							 | 
						|
								                  title="More details about this person"
							 | 
						|
								                >
							 | 
						|
								                  <EntityIcon
							 | 
						|
								                    :entity-id="record.recipientDid"
							 | 
						|
								                    :profile-image-url="record.receiver.profileImageUrl"
							 | 
						|
								                    class="rounded-full bg-slate-100 overflow-hidden !size-[3rem] sm:!size-[4rem]"
							 | 
						|
								                  />
							 | 
						|
								                </router-link>
							 | 
						|
								                <font-awesome
							 | 
						|
								                  v-else
							 | 
						|
								                  icon="eye-slash"
							 | 
						|
								                  class="text-slate-300 !size-[3rem] sm:!size-[4rem]"
							 | 
						|
								                  @click="notifyHiddenPerson"
							 | 
						|
								                />
							 | 
						|
								              </div>
							 | 
						|
								              <!-- Unknown Person -->
							 | 
						|
								              <div v-else>
							 | 
						|
								                <font-awesome
							 | 
						|
								                  icon="person-circle-question"
							 | 
						|
								                  class="text-slate-300 text-[3rem] sm:text-[4rem]"
							 | 
						|
								                  @click="notifyUnknownPerson"
							 | 
						|
								                />
							 | 
						|
								              </div>
							 | 
						|
								            </div>
							 | 
						|
								          </div>
							 | 
						|
								          <div
							 | 
						|
								            v-if="record.recipientProjectName || record.receiver.known"
							 | 
						|
								            class="text-xs mt-2 truncate"
							 | 
						|
								          >
							 | 
						|
								            <font-awesome
							 | 
						|
								              :icon="record.recipientProjectName ? 'users' : 'user'"
							 | 
						|
								              class="fa-fw text-slate-400"
							 | 
						|
								            />
							 | 
						|
								            {{ record.recipientProjectName || record.receiver.displayName }}
							 | 
						|
								          </div>
							 | 
						|
								        </div>
							 | 
						|
								      </div>
							 | 
						|
								    </div>
							 | 
						|
								  </li>
							 | 
						|
								</template>
							 | 
						|
								
							 | 
						|
								<script lang="ts">
							 | 
						|
								import { Component, Prop, Vue, Emit } from "vue-facing-decorator";
							 | 
						|
								import VueMarkdown from "vue-markdown-render";
							 | 
						|
								
							 | 
						|
								import { logger } from "../utils/logger";
							 | 
						|
								import {
							 | 
						|
								  createAndSubmitClaim,
							 | 
						|
								  getHeaders,
							 | 
						|
								  isHiddenDid,
							 | 
						|
								} from "../libs/endorserServer";
							 | 
						|
								import EntityIcon from "./EntityIcon.vue";
							 | 
						|
								import ProjectIcon from "./ProjectIcon.vue";
							 | 
						|
								import { createNotifyHelpers, NotifyFunction, TIMEOUTS } from "@/utils/notify";
							 | 
						|
								import {
							 | 
						|
								  NOTIFY_PERSON_HIDDEN,
							 | 
						|
								  NOTIFY_UNKNOWN_PERSON,
							 | 
						|
								} from "@/constants/notifications";
							 | 
						|
								import { EmojiSummaryRecord, GenericVerifiableCredential } from "@/interfaces";
							 | 
						|
								import { GiveRecordWithContactInfo } from "@/interfaces/give";
							 | 
						|
								import { PromiseTracker } from "@/libs/util";
							 | 
						|
								
							 | 
						|
								@Component({
							 | 
						|
								  components: {
							 | 
						|
								    EntityIcon,
							 | 
						|
								    ProjectIcon,
							 | 
						|
								    VueMarkdown,
							 | 
						|
								  },
							 | 
						|
								})
							 | 
						|
								export default class ActivityListItem extends Vue {
							 | 
						|
								  readonly QUICK_EMOJIS = ["👍", "👏", "❤️", "🎉", "😊", "😆", "🔥"];
							 | 
						|
								
							 | 
						|
								  @Prop() record!: GiveRecordWithContactInfo;
							 | 
						|
								  @Prop() lastViewedClaimId?: string;
							 | 
						|
								  @Prop() isRegistered!: boolean;
							 | 
						|
								  @Prop() activeDid!: string;
							 | 
						|
								  @Prop() apiServer!: string;
							 | 
						|
								
							 | 
						|
								  isHiddenDid = isHiddenDid;
							 | 
						|
								  notify!: ReturnType<typeof createNotifyHelpers>;
							 | 
						|
								  $notify!: NotifyFunction;
							 | 
						|
								
							 | 
						|
								  // Emoji-related data
							 | 
						|
								  showEmojiPicker = false;
							 | 
						|
								  loadingEmojis = false; // Track if emojis are currently loading
							 | 
						|
								
							 | 
						|
								  emojisOnActivity: PromiseTracker<EmojiSummaryRecord[]> | null = null; // load this only when needed
							 | 
						|
								
							 | 
						|
								  created() {
							 | 
						|
								    this.notify = createNotifyHelpers(this.$notify);
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  notifyHiddenPerson() {
							 | 
						|
								    this.notify.warning(NOTIFY_PERSON_HIDDEN.message, TIMEOUTS.STANDARD);
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  notifyUnknownPerson() {
							 | 
						|
								    this.notify.warning(NOTIFY_UNKNOWN_PERSON.message, TIMEOUTS.STANDARD);
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  get fetchAmount(): string {
							 | 
						|
								    const claim = this.record.fullClaim;
							 | 
						|
								
							 | 
						|
								    const amount = claim.object?.amountOfThisGood
							 | 
						|
								      ? this.displayAmount(claim.object.unitCode, claim.object.amountOfThisGood)
							 | 
						|
								      : "";
							 | 
						|
								
							 | 
						|
								    return amount;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  get description(): string {
							 | 
						|
								    const claim = this.record.fullClaim;
							 | 
						|
								
							 | 
						|
								    return `${claim?.description || ""}`;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  get truncatedDescription(): string {
							 | 
						|
								    const desc = this.description;
							 | 
						|
								    if (desc.length <= 300) {
							 | 
						|
								      return desc;
							 | 
						|
								    }
							 | 
						|
								    return desc.substring(0, 300) + "...";
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  private displayAmount(code: string, amt: number) {
							 | 
						|
								    return `${amt} ${this.currencyShortWordForCode(code, amt === 1)}`;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  private currencyShortWordForCode(unitCode: string, single: boolean) {
							 | 
						|
								    return unitCode === "HUR" ? (single ? "hour" : "hours") : unitCode;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  // Emit methods using @Emit decorator
							 | 
						|
								  @Emit("viewImage")
							 | 
						|
								  emitViewImage(imageUrl: string) {
							 | 
						|
								    return imageUrl;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  @Emit("loadClaim")
							 | 
						|
								  emitLoadClaim(jwtId: string) {
							 | 
						|
								    return jwtId;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  get friendlyDate(): string {
							 | 
						|
								    const date = new Date(this.record.issuedAt);
							 | 
						|
								    return date.toLocaleDateString(undefined, {
							 | 
						|
								      year: "numeric",
							 | 
						|
								      month: "short",
							 | 
						|
								      day: "numeric",
							 | 
						|
								    });
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  // Emoji-related computed properties and methods
							 | 
						|
								  get hasEmojis(): boolean {
							 | 
						|
								    return Object.keys(this.record.emojiCount).length > 0;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  triggerUserEmojiLoad(): PromiseTracker<EmojiSummaryRecord[]> {
							 | 
						|
								    if (!this.emojisOnActivity) {
							 | 
						|
								      const promise = new Promise<EmojiSummaryRecord[]>((resolve) => {
							 | 
						|
								        (async () => {
							 | 
						|
								          this.axios
							 | 
						|
								            .get(
							 | 
						|
								              `${this.apiServer}/api/v2/report/emoji?parentHandleId=${encodeURIComponent(this.record.handleId)}`,
							 | 
						|
								              { headers: await getHeaders(this.activeDid) },
							 | 
						|
								            )
							 | 
						|
								            .then((response) => {
							 | 
						|
								              const userEmojiRecords = response.data.data.filter(
							 | 
						|
								                (e: EmojiSummaryRecord) => e.issuerDid === this.activeDid,
							 | 
						|
								              );
							 | 
						|
								              resolve(userEmojiRecords);
							 | 
						|
								            })
							 | 
						|
								            .catch((error) => {
							 | 
						|
								              logger.error("Error loading user emojis:", error);
							 | 
						|
								              resolve([]);
							 | 
						|
								            });
							 | 
						|
								        })();
							 | 
						|
								      });
							 | 
						|
								
							 | 
						|
								      this.emojisOnActivity = new PromiseTracker(promise);
							 | 
						|
								    }
							 | 
						|
								    return this.emojisOnActivity;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  /**
							 | 
						|
								   *
							 | 
						|
								   * @param emoji - The emoji to check.
							 | 
						|
								   * @returns True if the emoji is in the user's emojis, false otherwise.
							 | 
						|
								   *
							 | 
						|
								   * @note This method is quick and synchronous, and can check resolved emojis
							 | 
						|
								   * without triggering a server request. Returns false if emojis haven't been loaded yet.
							 | 
						|
								   */
							 | 
						|
								  isUserEmojiWithoutLoading(emoji: string): boolean {
							 | 
						|
								    if (this.emojisOnActivity?.isResolved && this.emojisOnActivity.value) {
							 | 
						|
								      return this.emojisOnActivity.value.some(
							 | 
						|
								        (record) => record.text === emoji,
							 | 
						|
								      );
							 | 
						|
								    }
							 | 
						|
								    return false;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  async toggleEmojiPicker() {
							 | 
						|
								    this.triggerUserEmojiLoad(); // trigger it, but don't wait for it to complete
							 | 
						|
								    this.showEmojiPicker = !this.showEmojiPicker;
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  async toggleThisEmoji(emoji: string) {
							 | 
						|
								    // Start loading indicator
							 | 
						|
								    this.loadingEmojis = true;
							 | 
						|
								    this.showEmojiPicker = false; // always close the picker when an emoji is clicked
							 | 
						|
								
							 | 
						|
								    try {
							 | 
						|
								      this.triggerUserEmojiLoad(); // trigger just in case
							 | 
						|
								
							 | 
						|
								      const userEmojiList = await this.emojisOnActivity!.promise; // must wait now that they've chosen
							 | 
						|
								
							 | 
						|
								      const userHasEmoji: boolean = userEmojiList.some(
							 | 
						|
								        (record) => record.text === emoji,
							 | 
						|
								      );
							 | 
						|
								
							 | 
						|
								      if (userHasEmoji) {
							 | 
						|
								        this.$notify(
							 | 
						|
								          {
							 | 
						|
								            group: "modal",
							 | 
						|
								            type: "confirm",
							 | 
						|
								            title: "Remove Emoji",
							 | 
						|
								            text: `Do you want to remove your ${emoji} ?`,
							 | 
						|
								            yesText: "Remove",
							 | 
						|
								            onYes: async () => {
							 | 
						|
								              await this.removeEmoji(emoji);
							 | 
						|
								            },
							 | 
						|
								          },
							 | 
						|
								          TIMEOUTS.MODAL,
							 | 
						|
								        );
							 | 
						|
								      } else {
							 | 
						|
								        // User doesn't have this emoji, add it
							 | 
						|
								        await this.submitEmoji(emoji);
							 | 
						|
								      }
							 | 
						|
								    } finally {
							 | 
						|
								      // Remove loading indicator
							 | 
						|
								      this.loadingEmojis = false;
							 | 
						|
								    }
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  async submitEmoji(emoji: string) {
							 | 
						|
								    try {
							 | 
						|
								      // Create an Emoji claim and send to the server
							 | 
						|
								      const emojiClaim: GenericVerifiableCredential = {
							 | 
						|
								        "@context": "https://endorser.ch",
							 | 
						|
								        "@type": "Emoji",
							 | 
						|
								        text: emoji,
							 | 
						|
								        parentItem: { lastClaimId: this.record.jwtId },
							 | 
						|
								      };
							 | 
						|
								      const claim = await createAndSubmitClaim(
							 | 
						|
								        emojiClaim,
							 | 
						|
								        this.activeDid,
							 | 
						|
								        this.apiServer,
							 | 
						|
								        this.axios,
							 | 
						|
								      );
							 | 
						|
								      if (claim.success && !claim.embeddedRecordError) {
							 | 
						|
								        // Update emoji count
							 | 
						|
								        this.record.emojiCount[emoji] =
							 | 
						|
								          (this.record.emojiCount[emoji] || 0) + 1;
							 | 
						|
								
							 | 
						|
								        // Create a new emoji record (we'll get the actual jwtId from the server response later)
							 | 
						|
								        const newEmojiRecord: EmojiSummaryRecord = {
							 | 
						|
								          issuerDid: this.activeDid,
							 | 
						|
								          jwtId: claim.claimId || "",
							 | 
						|
								          text: emoji,
							 | 
						|
								          parentHandleId: this.record.jwtId,
							 | 
						|
								        };
							 | 
						|
								
							 | 
						|
								        // Update user emojis list by creating a new promise with the updated data
							 | 
						|
								        // (Trigger shouldn't be necessary since all calls should come through a toggle, but just in case someone calls this directly)
							 | 
						|
								        this.triggerUserEmojiLoad();
							 | 
						|
								        const currentEmojis = await this.emojisOnActivity!.promise; // must wait now that they've clicked one
							 | 
						|
								        this.emojisOnActivity = new PromiseTracker(
							 | 
						|
								          Promise.resolve([...currentEmojis, newEmojiRecord]),
							 | 
						|
								        );
							 | 
						|
								      } else {
							 | 
						|
								        this.notify.error("Failed to add emoji.", TIMEOUTS.STANDARD);
							 | 
						|
								      }
							 | 
						|
								    } catch (error) {
							 | 
						|
								      logger.error("Error submitting emoji:", error);
							 | 
						|
								      this.notify.error("Got error adding emoji.", TIMEOUTS.STANDARD);
							 | 
						|
								    }
							 | 
						|
								  }
							 | 
						|
								
							 | 
						|
								  async removeEmoji(emoji: string) {
							 | 
						|
								    try {
							 | 
						|
								      // Create an Emoji claim and send to the server
							 | 
						|
								      const emojiClaim: GenericVerifiableCredential = {
							 | 
						|
								        "@context": "https://endorser.ch",
							 | 
						|
								        "@type": "Emoji",
							 | 
						|
								        text: emoji,
							 | 
						|
								        parentItem: { lastClaimId: this.record.jwtId },
							 | 
						|
								      };
							 | 
						|
								      const claim = await createAndSubmitClaim(
							 | 
						|
								        emojiClaim,
							 | 
						|
								        this.activeDid,
							 | 
						|
								        this.apiServer,
							 | 
						|
								        this.axios,
							 | 
						|
								      );
							 | 
						|
								      if (claim.success && !claim.embeddedRecordError) {
							 | 
						|
								        // Update emoji count
							 | 
						|
								        const newCount = Math.max(0, (this.record.emojiCount[emoji] || 0) - 1);
							 | 
						|
								        if (newCount === 0) {
							 | 
						|
								          delete this.record.emojiCount[emoji];
							 | 
						|
								        } else {
							 | 
						|
								          this.record.emojiCount[emoji] = newCount;
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								        // Update user emojis list by creating a new promise with the updated data
							 | 
						|
								        // (Trigger shouldn't be necessary since all calls should come through a toggle, but just in case someone calls this directly)
							 | 
						|
								        this.triggerUserEmojiLoad();
							 | 
						|
								        const currentEmojis = await this.emojisOnActivity!.promise; // must wait now that they've clicked one
							 | 
						|
								        this.emojisOnActivity = new PromiseTracker(
							 | 
						|
								          Promise.resolve(
							 | 
						|
								            currentEmojis.filter(
							 | 
						|
								              (record) =>
							 | 
						|
								                record.issuerDid === this.activeDid && record.text !== emoji,
							 | 
						|
								            ),
							 | 
						|
								          ),
							 | 
						|
								        );
							 | 
						|
								      } else {
							 | 
						|
								        this.notify.error("Failed to remove emoji.", TIMEOUTS.STANDARD);
							 | 
						|
								      }
							 | 
						|
								    } catch (error) {
							 | 
						|
								      logger.error("Error removing emoji:", error);
							 | 
						|
								      this.notify.error("Got error removing emoji.", TIMEOUTS.STANDARD);
							 | 
						|
								    }
							 | 
						|
								  }
							 | 
						|
								}
							 | 
						|
								</script>
							 | 
						|
								
							 |