Skip to content

Commit

Permalink
✨ feat(papago): use native api
Browse files Browse the repository at this point in the history
  • Loading branch information
summerscar committed Dec 19, 2024
1 parent 9688435 commit bb4ab7d
Show file tree
Hide file tree
Showing 12 changed files with 411 additions and 63 deletions.
16 changes: 4 additions & 12 deletions app/(home)/learn/[level]/_components/markdown-wrapper.tsx
Original file line number Diff line number Diff line change
@@ -1,31 +1,23 @@
"use client";
import { useSelectToSearch } from "@/hooks/use-select-to-search";
import { SelectToSearch } from "@/hooks/use-select-to-search";
import { usePathname } from "next/navigation";
import { type ReactNode, type RefObject, useEffect, useState } from "react";
import { type ReactNode, useEffect, useState } from "react";
import type { TocItem } from "remark-flexible-toc";

const MDContentWrapper = ({
children,
lastModified,
bottomNav,
}: { children: ReactNode; lastModified?: string; bottomNav?: ReactNode }) => {
const [containerRef, panel] = useSelectToSearch({
showAI: false,
});

return (
<article
data-last-modified={lastModified}
className="p-8 border-r-2 border-slate-900/10 flex-auto"
>
<div
className="markdown-body"
ref={containerRef as RefObject<HTMLDivElement>}
>
<SelectToSearch showAI={false} className="markdown-body">
{children}
</div>
</SelectToSearch>
{bottomNav}
{panel}
</article>
);
};
Expand Down
190 changes: 190 additions & 0 deletions app/actions/papago-translate-action.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,190 @@
"use server";
import Base64 from "crypto-js/enc-base64";
import HmacMD5 from "crypto-js/hmac-md5";

const url = "https://papago.naver.com/apis/n2mt/translate";

export const papagoTranslateAction = async (
text: string,
locale = "zh-CN",
): Promise<TranslateResult> => {
const deviceId = uuid();
const body = {
deviceId,
locale,
dict: true,
dictDisplay: 30,
honorific: false,
instant: false,
paging: false,
source: "ko",
target: locale,
text,
usageAgreed: false,
};
const headers = {
Accept: "application/json",
"Accept-Language": "zh-CN",
"device-type": "pc",
"x-apigw-partnerid": "papago",
"Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
"User-Agent":
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.114 Safari/537.36",
Referer: "https://papago.naver.com/",
Origin: "https://papago.naver.com",
...authorization(url, deviceId),
};

const response = await fetch(url, {
method: "POST",
headers: headers,
// @ts-ignore
body: new URLSearchParams(body),
mode: "no-cors",
});
if (response.status !== 200) {
throw new Error((await response.text()) || response.statusText);
}
return await response.json();
};

function uuid() {
let e = new Date().getTime();
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (a) => {
const t = ((e + 16 * Math.random()) % 16) | 0;
// biome-ignore lint/style/noCommaOperator: <explanation>
return (e = Math.floor(e / 16)), ("x" === a ? t : (3 & t) | 8).toString(16);
});
}

function authorization(url: string, uuid: string) {
const t = uuid;
const n = new Date().getTime();
return {
Authorization: `PPG ${t}:${HmacMD5(
`${t}\n${url.split("?")[0]}\n${n}`,
"v1.8.7_0c4b33bdb0",
).toString(Base64)}`,
Timestamp: n.toString(),
};
}

interface Meaning {
meaning: string;
examples: string[];
originalMeaning: string;
}

interface Po {
type: string;
meanings: Meaning[];
}

interface Item {
entry: string;
subEntry?: string | null;
matchType: string;
hanjaEntry: string | null;
phoneticSigns: PhoneticSign[];
pos: Po[];
source: string;
url: string;
mUrl: string;
expDicTypeForm: string;
locale: string;
gdid: string;
expEntrySuperscript: string;
}

interface Dict {
items?: Item[];
examples: string[];
isWordType: boolean;
}

interface PhoneticSign {
type: null;
sign: string;
}

interface Meaning {
meaning: string;
examples: string[];
originalMeaning: string;
}

interface Po {
type: string;
meanings: Meaning[];
}

interface Item {
entry: string;
subEntry?: string | null;
matchType: string;
hanjaEntry: string | null;
phoneticSigns: PhoneticSign[];
pos: Po[];
source: string;
url: string;
mUrl: string;
expDicTypeForm: string;
locale: string;
gdid: string;
expEntrySuperscript: string;
}

interface TarDict {
items: Item[];
examples: string[];
isWordType: boolean;
}

interface TlitResult {
token: string;
phoneme: string;
}

interface Message {
tlitResult: TlitResult[];
}

interface TlitSrc {
message: Message;
}

interface TlitResult {
token: string;
phoneme: string;
}

interface Message {
tlitResult: TlitResult[];
}

interface Tlit {
message: Message;
}

interface Nbest {
lang: string;
prob: number;
}

interface LangDetection {
nbests: Nbest[];
}

export interface TranslateResult {
dict: Dict;
tarDict: TarDict;
delay: number;
delaySmt: number;
srcLangType: string;
tarLangType: string;
translatedText: string;
engineType: string;
tlitSrc: TlitSrc;
tlit: Tlit;
langDetection: LangDetection;
}
3 changes: 3 additions & 0 deletions app/assets/svg/translate.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
Expand Up @@ -3,22 +3,24 @@ import CopyIcon from "@/assets/svg/copy.svg";
import AddIcon from "@/assets/svg/folder-plus.svg";
import SearchIcon from "@/assets/svg/search.svg";
import SparklesIcon from "@/assets/svg/sparkles.svg";
import TranslateIcon from "@/assets/svg/translate.svg";
import type { CSSProperties } from "react";

interface SearchButtonProps {
style?: CSSProperties;
onClick: () => void;
icon: "search" | "copy" | "sparkles" | "add";
icon: "search" | "copy" | "sparkles" | "add" | "translate";
}

const IconMap = {
search: SearchIcon,
translate: TranslateIcon,
copy: CopyIcon,
sparkles: SparklesIcon,
add: AddIcon,
};

export const SearchButton = ({
export const FloatButton = ({
style,
onClick,
icon = "search",
Expand Down
Loading

0 comments on commit bb4ab7d

Please sign in to comment.