溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在JS中實(shí)現(xiàn)一個(gè)用戶禁止保存圖片功能

發(fā)布時(shí)間:2020-11-19 14:38:46 來源:億速云 閱讀:827 作者:Leah 欄目:開發(fā)技術(shù)

今天就跟大家聊聊有關(guān)如何在JS中實(shí)現(xiàn)一個(gè)用戶禁止保存圖片功能,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

場(chǎng)景

在業(yè)務(wù)需求中不希望用戶保存圖片,因?yàn)槭且恍┕﹥?nèi)部使用的圖片。

思路

  • 添加事件禁止選擇、拖拽、右鍵(簡(jiǎn)單的禁止用戶保存圖片,但無法阻止用戶打開控制臺(tái)查看,或是直接抓包)
  • 將之轉(zhuǎn)換為 canvas(讓瀏覽器認(rèn)為不是圖片以此禁止用戶對(duì)之進(jìn)行圖片的操作,但無法阻止抓包)
  • 禁止用戶使用控制臺(tái)查看源碼(阻止瀏覽器打開控制臺(tái),但無法阻止抓包)
  • 傳輸圖片使用自定義格式(可以阻止抓包,但需要后臺(tái)配合)

注:以下內(nèi)容使用 react+ts 實(shí)現(xiàn)

添加事件禁止選擇、拖拽、右鍵

簡(jiǎn)而言之,這是一種簡(jiǎn)單有效的方式,能夠在用戶不打開控制臺(tái)的情況下阻止用戶保存圖片。

export function preventDefaultListener(e: any) {
 e.preventDefault()
}

;<img
 src={props.url}
 alt=""
 style={{
  //禁止用戶選擇
  userSelect: 'none',
  //禁止所有鼠標(biāo)事件,過于強(qiáng)大,圖片僅用于展示可用
  // pointerEvents: 'none',
 }}
 onTouchStart={preventDefaultListener}
 onContextMenu={preventDefaultListener}
 onDragStart={preventDefaultListener}
/>

將之轉(zhuǎn)換為 canvas

另一種思路是將圖片轉(zhuǎn)換為 canvas 避免用戶使用 img 相關(guān)的操作。

將圖片轉(zhuǎn)成 canvas

export async function imageToCanvas(url: string, canvas: HTMLCanvasElement) {
 return new Promise((resolve, reject) => {
  //新建Image對(duì)象,引入當(dāng)前目錄下的圖片
  const img = new Image()
  img.src = url
  const c = canvas.getContext('2d')!

  //圖片初始化完成后調(diào)用
  img.onload = function () {
   //將canvas的寬高設(shè)置為圖像的寬高
   canvas.width = img.width
   canvas.height = img.height

   //canvas畫圖片
   c.drawImage(img, 0, 0, img.width, img.height)
   resolve()
  }
  img.addEventListener('error', (e) => {
   reject(e)
  })
 })
}

禁用 canvas 事件

const throwFn = () => {
 throw new Error(
  "Uncaught DOMException: Failed to execute 'toDataURL' on 'HTMLCanvasElement': Tainted canvases may not be exported.",
 )
}

const $canvasRef = useRef<HTMLCanvasElement>(null)
 useEffect(() => {
   ;(async () => {
     await imageToCanvas(props.url, $canvasRef.current!)
     $canvasRef.current!.toBlob = throwFn
     $canvasRef.current!.toDataURL = throwFn
   })()
 }, [])
 return (
   <canvas
     ref={$canvasRef}
     onTouchStart={preventDefaultListener}
     onContextMenu={preventDefaultListener}
   />
 )

禁止用戶使用控制臺(tái)查看源碼

如果能禁止用戶操作控制臺(tái),那么自然能夠避免用戶查看源碼了,下面是一個(gè)簡(jiǎn)單的實(shí)現(xiàn)。

/**
 * 兼容異步函數(shù)的返回值
 * @param res 返回值
 * @param callback 同步/異步結(jié)果的回調(diào)函數(shù)
 * @typeparam T 處理參數(shù)的類型,如果是 Promise 類型,則取出其泛型類型
 * @typeparam Param 處理參數(shù)具體的類型,如果是 Promise 類型,則指定為原類型
 * @typeparam R 返回值具體的類型,如果是 Promise 類型,則指定為 Promise 類型,否則為原類型
 * @returns 處理后的結(jié)果,如果是同步的,則返回結(jié)果是同步的,否則為異步的
 */
export function compatibleAsync<T = any, Param = T | Promise<T>, R = T>(
 res: Param,
 callback: (r: T) => R,
): Param extends Promise<T> &#63; Promise<R> : R {
 return (res instanceof Promise
  &#63; res.then(callback)
  : callback(res as any)) as any
}

/**
 * 測(cè)試函數(shù)的執(zhí)行時(shí)間
 * 注:如果函數(shù)返回 Promise,則該函數(shù)也會(huì)返回 Promise,否則直接返回執(zhí)行時(shí)間
 * @param fn 需要測(cè)試的函數(shù)
 * @returns 執(zhí)行的毫秒數(shù)
 */
export function timing<R>(
 fn: (...args: any[]) => R,
 // 函數(shù)返回類型是 Promise 的話,則返回 Promise<number>,否則返回 number
): R extends Promise<any> &#63; Promise<number> : number {
 const begin = performance.now()
 const res = fn()
 return compatibleAsync(res, () => performance.now() - begin)
}
/**
 * 禁止他人調(diào)試網(wǎng)站相關(guān)方法的集合對(duì)象
 */
export class AntiDebug {
 /**
  * 不停循環(huán) debugger 防止有人調(diào)試代碼
  * @returns 取消函數(shù)
  */
 public static cyclingDebugger(): Function {
  const res = setInterval(() => {
   debugger
  }, 100)
  return () => clearInterval(res)
 }
 /**
  * 檢查是否正在 debugger 并調(diào)用回調(diào)函數(shù)
  * @param fn 回調(diào)函數(shù),默認(rèn)為重載頁面
  * @returns 取消函數(shù)
  */
 public static checkDebug(
  fn: Function = () => window.location.reload(),
 ): Function {
  const res = setInterval(() => {
   const diff = timing(() => {
    debugger
   })
   if (diff > 500) {
    console.log(diff)
    fn()
   }
  }, 1000)
  return () => clearInterval(res)
 }
}

傳輸圖片使用自定義格式

該功能需要服務(wù)端配合,故而此處贊不實(shí)現(xiàn),可以參考 微信讀書,就是將文本轉(zhuǎn)為 canvas,數(shù)據(jù)傳輸也進(jìn)行了加密,可以在很大程度上防止普通用戶想要復(fù)制/下載的行為了。

看完上述內(nèi)容,你們對(duì)如何在JS中實(shí)現(xiàn)一個(gè)用戶禁止保存圖片功能有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI