From 32c63691d645585288453f718384a39b78cf54ec Mon Sep 17 00:00:00 2001 From: Tenth-crew <68376741+Tenth-crew@users.noreply.github.com> Date: Mon, 22 Jul 2024 11:58:29 +0800 Subject: [PATCH] feat: display openrank information in developer information hover card (#832) * Modified the developer name acquisition method and tried other monitoring methods * modify evenlistener method and add openrank info * feat: add the developer-avator-openrank * Modified the presentation of OpenRank * Improved the situation when openrank does not exist * Introduce openrank information to the bottom of the hovercard * Modify the names of some functions and fields * Add view.tsx and Adjust some details * Adjust openrank icon position * Solved the problem of mismatch between OpenRank information and people * Fixed the bug that OpenRank information was added multiple times * Adjust the display position of OpenRank * Introducing the renderTo function --- .../developer-hovercard-info/base64.ts | 5 ++ .../developer-hovercard-info/index.tsx | 82 +++++++++++++++++++ .../developer-hovercard-info/view.tsx | 35 ++++++++ src/pages/ContentScripts/index.ts | 1 + 4 files changed, 123 insertions(+) create mode 100644 src/pages/ContentScripts/features/developer-hovercard-info/base64.ts create mode 100644 src/pages/ContentScripts/features/developer-hovercard-info/index.tsx create mode 100644 src/pages/ContentScripts/features/developer-hovercard-info/view.tsx diff --git a/src/pages/ContentScripts/features/developer-hovercard-info/base64.ts b/src/pages/ContentScripts/features/developer-hovercard-info/base64.ts new file mode 100644 index 00000000..7d2a7452 --- /dev/null +++ b/src/pages/ContentScripts/features/developer-hovercard-info/base64.ts @@ -0,0 +1,5 @@ +export const rocketLight = + ''; + +export const rocketDark = + ''; diff --git a/src/pages/ContentScripts/features/developer-hovercard-info/index.tsx b/src/pages/ContentScripts/features/developer-hovercard-info/index.tsx new file mode 100644 index 00000000..b6f5c6cf --- /dev/null +++ b/src/pages/ContentScripts/features/developer-hovercard-info/index.tsx @@ -0,0 +1,82 @@ +import features from '../../../../feature-manager'; +import { getOpenrank } from '../../../../api/developer'; +import elementReady from 'element-ready'; +import React from 'react'; +import ReactDOM from 'react-dom'; +import View from './view'; + +const featureId = features.getFeatureID(import.meta.url); + +const getDeveloperLatestOpenrank = async (developerName: string): Promise => { + const data = await getOpenrank(developerName); + if (data) { + const values = Object.values(data) as string[]; + const latestValue = values[values.length - 1]; + return latestValue; + } + return null; +}; + +const getDeveloperName = (target: HTMLElement): string | null => { + const hovercardUrlAttribute = target.getAttribute('data-hovercard-url'); + if (!hovercardUrlAttribute) return null; + + const matches = hovercardUrlAttribute.match(/\/users\/([^\/]+)\/hovercard/); + return matches ? matches[1] : null; +}; + +const renderTo = (container: HTMLElement, developerName: string, openrank: string) => { + const openRankContainer = document.createElement('div'); + container.appendChild(openRankContainer); + ReactDOM.render(, openRankContainer); +}; + +const init = async (): Promise => { + let abortController = new AbortController(); + const hovercardSelector = '[data-hovercard-type="user"]'; + document.querySelectorAll(hovercardSelector).forEach((element) => { + // isProcessing is used to Prevent OpenRank from adding duplicates + element.addEventListener('mouseover', async () => { + abortController.abort(); + abortController = new AbortController(); + const signal = abortController.signal; + + const developerName = getDeveloperName(element as HTMLElement) as string; + + // Create a unique identifier for the popover + const popoverId = `popover-${developerName}`; + + // Get the floating card container + const $popoverContainer = + 'body > div.logged-in.env-production.page-responsive > div.Popover.js-hovercard-content.position-absolute > div > div > div'; + const popover = await elementReady($popoverContainer, { stopOnDomReady: false }); + + const openRankDiv = popover?.querySelector('.hypercrx-openrank-info'); + const existingDeveloperName = openRankDiv?.getAttribute('data-developer-name'); + if (existingDeveloperName === developerName) { + return; + } + openRankDiv?.remove(); + + // Set the popover's unique identifier + // make the current OpenRank information and person match + popover?.setAttribute('data-popover-id', popoverId); + + const openrank = await getDeveloperLatestOpenrank(developerName); + + if (!openrank) { + return; + } + + if (!signal.aborted && popover && popover.getAttribute('data-popover-id') === popoverId) { + // Check if the popover is still associated with the correct developer + renderTo(popover, developerName, openrank); + } + }); + }); +}; + +features.add(featureId, { + awaitDomReady: false, + init, +}); diff --git a/src/pages/ContentScripts/features/developer-hovercard-info/view.tsx b/src/pages/ContentScripts/features/developer-hovercard-info/view.tsx new file mode 100644 index 00000000..7c317137 --- /dev/null +++ b/src/pages/ContentScripts/features/developer-hovercard-info/view.tsx @@ -0,0 +1,35 @@ +import React from 'react'; +import getGithubTheme from '../../../../helpers/get-github-theme'; +import '../../../../helpers/i18n'; +import { rocketLight, rocketDark } from './base64'; + +interface OpenRankProps { + developerName: string; + openrank: string; +} + +const View: React.FC = ({ developerName, openrank }) => { + const theme = getGithubTheme() as 'light' | 'dark'; + + const textColor = theme === 'light' ? '#717981' : '#878f98'; + const fontSize = '13px'; + + return ( +
+
+ + + OpenRank {openrank} + +
+
+ ); +}; + +export default View; diff --git a/src/pages/ContentScripts/index.ts b/src/pages/ContentScripts/index.ts index bab274a3..11a5dc6f 100644 --- a/src/pages/ContentScripts/index.ts +++ b/src/pages/ContentScripts/index.ts @@ -13,3 +13,4 @@ import './features/repo-networks'; import './features/developer-networks'; import './features/oss-gpt'; import './features/repo-activity-racing-bar'; +import './features/developer-hovercard-info';