Greasy Fork is available in English.

豆瓣读书书架

无需登录,快速选择书籍加入书架,复制导出书籍信息

// ==UserScript==
// @name               豆瓣读书书架
// @name:zh            豆瓣读书书架
// @namespace          https://github.com/ilyydy/tampermonkey-script
// @version            0.0.4
// @author             ilyydy
// @description        无需登录,快速选择书籍加入书架,复制导出书籍信息
// @description:zh     无需登录,快速选择书籍加入书架,复制导出书籍信息
// @description:zh-CN  无需登录,快速选择书籍加入书架,复制导出书籍信息
// @license            MIT
// @icon               
// @supportURL         https://github.com/ilyydy/tampermonkey-script/issues
// @match              http*://*book.douban.com/*
// @match              http*://search.douban.com/book/*
// @require            https://cdn.jsdelivr.net/npm/vue@3.2.47/dist/vue.global.prod.js
// @require            data:application/javascript,window.Vue%3DVue%3B
// @require            https://cdn.jsdelivr.net/npm/element-plus@2.2.19/dist/index.full.min.js
// @require            https://cdn.jsdelivr.net/npm/xlsx@0.18.5/dist/xlsx.full.min.js
// @require            https://cdn.jsdelivr.net/npm/csv-stringify@6.2.3/dist/iife/sync.js
// @resource           element-plus/dist/index.css  https://cdn.jsdelivr.net/npm/element-plus@2.2.19/dist/index.css
// @connect            book.douban.com
// @grant              GM_getResourceText
// @grant              GM_getValue
// @grant              GM_setClipboard
// @grant              GM_setValue
// @grant              GM_xmlhttpRequest
// ==/UserScript==

(t=>{const e=document.createElement("style");e.dataset.source="vite-plugin-monkey",e.innerText=t,document.head.appendChild(e)})(".book-field[data-v-5f2bdcc9]{margin-bottom:5px}.item[data-v-bdb4b0de]{display:grid;grid-template-columns:80% 20%}.item-button[data-v-bdb4b0de]{display:grid}.affix[data-v-ed4eeb8f]{position:fixed;bottom:50px;right:50px}.book-list-container[data-v-ed4eeb8f]{margin:auto 20px}.book-list-container-header[data-v-ed4eeb8f]{margin-bottom:20px}.book-list-container-header-select[data-v-ed4eeb8f]{margin-left:10px;width:140px}.book-list-container-header-item[data-v-ed4eeb8f]{margin-left:10px}.book-list-container-empty p[data-v-ed4eeb8f]{font-size:18px;text-align:center}.book-list-container-book-item[data-v-ed4eeb8f]{margin-bottom:20px}");

(function(vue, elementPlus, sync, XLSX2) {
  "use strict";
  function _interopNamespaceDefault(e) {
    const n = Object.create(null, { [Symbol.toStringTag]: { value: "Module" } });
    if (e) {
      for (const k in e) {
        if (k !== "default") {
          const d = Object.getOwnPropertyDescriptor(e, k);
          Object.defineProperty(n, k, d.get ? d : {
            enumerable: true,
            get: () => e[k]
          });
        }
      }
    }
    n.default = e;
    return Object.freeze(n);
  }
  const XLSX__namespace = /* @__PURE__ */ _interopNamespaceDefault(XLSX2);
  const cssLoader = (e) => {
    const t = GM_getResourceText(e), o = document.createElement("style");
    return o.innerText = t, document.head.append(o), t;
  };
  cssLoader("element-plus/dist/index.css");
  /*! Element Plus Icons Vue v2.0.10 */
  var export_helper_default = (sfc, props) => {
    let target = sfc.__vccOpts || sfc;
    for (let [key, val] of props)
      target[key] = val;
    return target;
  };
  var arrow_down_vue_vue_type_script_lang_default = {
    name: "ArrowDown"
  };
  var _hoisted_16 = {
    viewBox: "0 0 1024 1024",
    xmlns: "http://www.w3.org/2000/svg"
  }, _hoisted_26 = /* @__PURE__ */ vue.createElementVNode("path", {
    fill: "currentColor",
    d: "M831.872 340.864 512 652.672 192.128 340.864a30.592 30.592 0 0 0-42.752 0 29.12 29.12 0 0 0 0 41.6L489.664 714.24a32 32 0 0 0 44.672 0l340.288-331.712a29.12 29.12 0 0 0 0-41.728 30.592 30.592 0 0 0-42.752 0z"
  }, null, -1), _hoisted_36 = [
    _hoisted_26
  ];
  function _sfc_render6(_ctx, _cache, $props, $setup, $data, $options) {
    return vue.openBlock(), vue.createElementBlock("svg", _hoisted_16, _hoisted_36);
  }
  var arrow_down_default = /* @__PURE__ */ export_helper_default(arrow_down_vue_vue_type_script_lang_default, [["render", _sfc_render6], ["__file", "arrow-down.vue"]]);
  var arrow_left_vue_vue_type_script_lang_default = {
    name: "ArrowLeft"
  };
  var _hoisted_18 = {
    viewBox: "0 0 1024 1024",
    xmlns: "http://www.w3.org/2000/svg"
  }, _hoisted_28 = /* @__PURE__ */ vue.createElementVNode("path", {
    fill: "currentColor",
    d: "M609.408 149.376 277.76 489.6a32 32 0 0 0 0 44.672l331.648 340.352a29.12 29.12 0 0 0 41.728 0 30.592 30.592 0 0 0 0-42.752L339.264 511.936l311.872-319.872a30.592 30.592 0 0 0 0-42.688 29.12 29.12 0 0 0-41.728 0z"
  }, null, -1), _hoisted_38 = [
    _hoisted_28
  ];
  function _sfc_render8(_ctx, _cache, $props, $setup, $data, $options) {
    return vue.openBlock(), vue.createElementBlock("svg", _hoisted_18, _hoisted_38);
  }
  var arrow_left_default = /* @__PURE__ */ export_helper_default(arrow_left_vue_vue_type_script_lang_default, [["render", _sfc_render8], ["__file", "arrow-left.vue"]]);
  var monkeyWindow = window;
  var GM_setValue = /* @__PURE__ */ (() => monkeyWindow.GM_setValue)();
  var GM_xmlhttpRequest = /* @__PURE__ */ (() => monkeyWindow.GM_xmlhttpRequest)();
  var GM_setClipboard = /* @__PURE__ */ (() => monkeyWindow.GM_setClipboard)();
  var GM_getValue = /* @__PURE__ */ (() => monkeyWindow.GM_getValue)();
  const BOOK_SHELF_KEY = "tampermonkey_douban_books";
  let books;
  function useStore$1(force = false) {
    if (!books || force) {
      {
        books = vue.reactive(JSON.parse(GM_getValue(BOOK_SHELF_KEY, "[]")));
        vue.watch(books, (newValue) => {
          GM_setValue(BOOK_SHELF_KEY, JSON.stringify(newValue));
        });
      }
    }
    return books;
  }
  function getBookIdx(id) {
    return books.findIndex((i) => i.id === id);
  }
  function getBook$1(id) {
    return books.find((i) => i.id === id);
  }
  function hasBook(id) {
    return getBookIdx(id) !== -1;
  }
  function addBook(book) {
    if (!hasBook(book.id)) {
      books.push({ ...book, addTime: Date.now() });
      return { success: true, msg: "" };
    } else {
      return { success: false, msg: `《${book.title}》 已在书架中` };
    }
  }
  function removeBook(id) {
    const idx = getBookIdx(id);
    if (idx === -1)
      return false;
    books.splice(idx, 1);
    return true;
  }
  function clearBook() {
    books.splice(0, books.length);
  }
  const booksStore = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
    __proto__: null,
    BOOK_SHELF_KEY,
    addBook,
    get books() {
      return books;
    },
    clearBook,
    getBook: getBook$1,
    getBookIdx,
    hasBook,
    removeBook,
    useStore: useStore$1
  }, Symbol.toStringTag, { value: "Module" }));
  function useStore(force = false) {
    useStore$1(force);
    return { booksStore };
  }
  function success(msg) {
    return elementPlus.ElMessage.success({ message: msg, showClose: true, grouping: true });
  }
  function warning(msg) {
    console.warn(msg);
    return elementPlus.ElMessage.warning({ message: msg, showClose: true, grouping: true });
  }
  function error(msg) {
    console.error(msg);
    return elementPlus.ElMessage.error({ message: msg, showClose: true, grouping: true });
  }
  function exportCsv(rows, filename, options) {
    const v = sync.stringify(rows, options);
    {
      const link = document.createElement("a");
      link.download = filename;
      const blob = new Blob([v], { type: "text/csv;charset=utf-8;" });
      link.href = URL.createObjectURL(blob);
      link.click();
      URL.revokeObjectURL(link.href);
    }
  }
  function exportXlsx(rows, filename, aoa2SheetOpts, writingOptions) {
    const worksheet = XLSX__namespace.utils.aoa_to_sheet(rows, aoa2SheetOpts);
    const workbook = XLSX__namespace.utils.book_new();
    XLSX__namespace.utils.book_append_sheet(workbook, worksheet);
    XLSX__namespace.writeFile(workbook, filename, writingOptions);
  }
  const BOOK_FIELD_MAP = {
    id: "id",
    title: "书名",
    subTitle: "副标题",
    originName: "原作名",
    authors: "作者",
    translators: "译者",
    publishingHouse: "出版社",
    publishingTime: "出版时间",
    seriesName: "丛书",
    page: "页数",
    ISBN: "ISBN",
    score: "评分",
    scorePeopleCount: "评分人数",
    coverUrl: "封面图链接",
    doubanUrl: "豆瓣链接",
    contentBrief: "内容简介"
  };
  const defaultBookFields = Object.keys(BOOK_FIELD_MAP);
  const excelFormatterMap = defaultBookFields.reduce((pre, field) => {
    switch (field) {
      case "authors":
      case "translators":
        pre[field] = (book) => book[field].join(", ");
        break;
      default:
        pre[field] = (book) => book[field];
        break;
    }
    return pre;
  }, {});
  const specialBookItemFormatters = [
    {
      key: "id",
      label: BOOK_FIELD_MAP.id,
      type: "link",
      getLink: (book) => book.doubanUrl
    },
    {
      key: "doubanUrl",
      label: BOOK_FIELD_MAP.doubanUrl,
      type: "link",
      getLink: (book) => book.doubanUrl
    },
    {
      key: "coverUrl",
      label: BOOK_FIELD_MAP.coverUrl,
      type: "link",
      getLink: (book) => book.coverUrl
    },
    {
      key: "authors",
      label: BOOK_FIELD_MAP.authors,
      type: "string",
      getText: (book) => book.authors.join(", ")
    },
    {
      key: "translators",
      label: BOOK_FIELD_MAP.translators,
      type: "string",
      getText: (book) => book.translators.join(", ")
    },
    {
      key: "contentBrief",
      label: BOOK_FIELD_MAP.contentBrief,
      type: "string",
      getText: (book) => book.contentBrief,
      styleObj: {
        "line-height": "30px",
        "white-space": "pre-line",
        "max-height": "300px",
        "padding-right": "20px",
        width: "98%",
        display: "inline-block",
        overflow: "auto"
      }
    }
  ];
  const bookItemFormatters = defaultBookFields.map((field) => {
    const v = specialBookItemFormatters.find((i) => i.key === field);
    if (v)
      return v;
    return {
      key: field,
      label: BOOK_FIELD_MAP[field],
      type: "string",
      getText: (book) => book[field]
    };
  });
  function getBookViewText(book, fields = defaultBookFields) {
    const textList = [];
    fields.forEach((i) => {
      const v = book[i];
      if (Array.isArray(v)) {
        if (v.length > 0)
          textList.push(`${BOOK_FIELD_MAP[i]}: ${v.join(", ")}`);
      } else if (v) {
        textList.push(`${BOOK_FIELD_MAP[i]}: ${v}`);
      }
    });
    return textList.join("\n");
  }
  async function copyBook(book, fields = defaultBookFields) {
    const text = getBookViewText(book, fields);
    GM_setClipboard(text, "text");
  }
  async function copyBookWithTip(book, fields = defaultBookFields) {
    try {
      await copyBook(book);
      success("复制成功");
    } catch (err) {
      error(err.msg);
    }
  }
  const exportCsvName = "豆瓣书籍导出.csv";
  function exportBookCsv(books2, filename = exportCsvName, options = { bom: true }) {
    const header = defaultBookFields.map((field) => BOOK_FIELD_MAP[field]);
    const data = [header];
    books2.forEach((book) => {
      const row = defaultBookFields.map((field) => {
        return excelFormatterMap[field](book);
      });
      data.push(row);
    });
    exportCsv(data, filename, options);
  }
  const exportXlsxName = "豆瓣书籍导出.xlsx";
  function exportBookXlsx(books2, filename = exportXlsxName, aoa2SheetOpts, writingOptions) {
    const header = defaultBookFields.map((field) => BOOK_FIELD_MAP[field]);
    const data = [header];
    books2.forEach((book) => {
      const row = defaultBookFields.map((field) => {
        return excelFormatterMap[field](book);
      });
      data.push(row);
    });
    exportXlsx(data, filename, aoa2SheetOpts, writingOptions);
  }
  const _hoisted_1$3 = { key: 0 };
  const _hoisted_2$2 = { key: 1 };
  const _hoisted_3$1 = ["href"];
  const _sfc_main$4 = /* @__PURE__ */ vue.defineComponent({
    __name: "BookFormatter",
    props: {
      book: null,
      fields: { default: () => defaultBookFields },
      hideEmptyField: { type: Boolean, default: true }
    },
    setup(__props) {
      const props = __props;
      const bookFormatter = vue.computed(
        () => bookItemFormatters.filter((i) => {
          if (!props.fields.includes(i.key))
            return false;
          if (!props.hideEmptyField)
            return true;
          const v = props.book[i.key];
          if (Array.isArray(v))
            return v.length > 0;
          return !!props.book[i.key];
        })
      );
      return (_ctx, _cache) => {
        return vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(bookFormatter), (item) => {
          return vue.openBlock(), vue.createElementBlock("div", {
            class: "book-field",
            key: item.key
          }, [
            item.key === "contentBrief" ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_1$3, vue.toDisplayString(item.label) + ":", 1)) : (vue.openBlock(), vue.createElementBlock("span", _hoisted_2$2, vue.toDisplayString(item.label) + ": ", 1)),
            item.type === "string" ? (vue.openBlock(), vue.createElementBlock("span", {
              key: 2,
              style: vue.normalizeStyle(item.styleObj)
            }, vue.toDisplayString(item.getText ? item.getText(__props.book) : __props.book[item.key]), 5)) : item.type === "link" ? (vue.openBlock(), vue.createElementBlock("a", {
              key: 3,
              href: item.getLink ? item.getLink(__props.book) : `${__props.book[item.key]}`,
              target: "_blank",
              style: vue.normalizeStyle(item.styleObj)
            }, vue.toDisplayString(__props.book[item.key]), 13, _hoisted_3$1)) : vue.createCommentVNode("", true)
          ]);
        }), 128);
      };
    }
  });
  const BookFormatter_vue_vue_type_style_index_0_scoped_5f2bdcc9_lang = "";
  const _export_sfc = (sfc, props) => {
    const target = sfc.__vccOpts || sfc;
    for (const [key, val] of props) {
      target[key] = val;
    }
    return target;
  };
  const BookFormatter = /* @__PURE__ */ _export_sfc(_sfc_main$4, [["__scopeId", "data-v-5f2bdcc9"]]);
  const _hoisted_1$2 = { class: "item" };
  const _hoisted_2$1 = { class: "item-button" };
  const _sfc_main$3 = /* @__PURE__ */ vue.defineComponent({
    __name: "BookItem",
    props: {
      book: null
    },
    emits: ["select"],
    setup(__props, { emit }) {
      const props = __props;
      const { booksStore: booksStore2 } = useStore();
      const briefFields = ["id", "title", "authors", "score", "ISBN"];
      function select() {
        emit("select", props.book);
      }
      return (_ctx, _cache) => {
        return vue.openBlock(), vue.createElementBlock("div", _hoisted_1$2, [
          vue.createElementVNode("div", null, [
            vue.createVNode(BookFormatter, {
              book: __props.book,
              fields: briefFields,
              "hide-empty-field": false
            }, null, 8, ["book"])
          ]),
          vue.createElementVNode("div", _hoisted_2$1, [
            vue.createElementVNode("div", null, [
              vue.createVNode(vue.unref(elementPlus.ElButton), {
                round: "",
                size: "small",
                type: "primary",
                onClick: select
              }, {
                default: vue.withCtx(() => [
                  vue.createTextVNode("详情")
                ]),
                _: 1
              })
            ]),
            vue.createElementVNode("div", null, [
              vue.createVNode(vue.unref(elementPlus.ElButton), {
                round: "",
                size: "small",
                type: "primary",
                onClick: _cache[0] || (_cache[0] = () => vue.unref(copyBookWithTip)(__props.book))
              }, {
                default: vue.withCtx(() => [
                  vue.createTextVNode("复制")
                ]),
                _: 1
              })
            ]),
            vue.createElementVNode("div", null, [
              vue.createVNode(vue.unref(elementPlus.ElButton), {
                round: "",
                size: "small",
                type: "danger",
                onClick: _cache[1] || (_cache[1] = ($event) => vue.unref(booksStore2).removeBook(__props.book.id))
              }, {
                default: vue.withCtx(() => [
                  vue.createTextVNode("移除")
                ]),
                _: 1
              })
            ])
          ])
        ]);
      };
    }
  });
  const BookItem_vue_vue_type_style_index_0_scoped_bdb4b0de_lang = "";
  const BookItem = /* @__PURE__ */ _export_sfc(_sfc_main$3, [["__scopeId", "data-v-bdb4b0de"]]);
  const _hoisted_1$1 = { key: 0 };
  const _sfc_main$2 = /* @__PURE__ */ vue.defineComponent({
    __name: "BookDetail",
    props: {
      book: null,
      show: { type: Boolean }
    },
    emits: ["close"],
    setup(__props, { emit }) {
      function handleClose() {
        emit("close");
      }
      return (_ctx, _cache) => {
        return vue.openBlock(), vue.createBlock(vue.unref(elementPlus.ElDialog), {
          "model-value": __props.show && !!__props.book,
          title: "书籍详情",
          width: "70%",
          style: { "max-height": "80vh" },
          "before-close": handleClose
        }, {
          default: vue.withCtx(() => [
            !!__props.book ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_1$1, [
              vue.createVNode(BookFormatter, { book: __props.book }, null, 8, ["book"])
            ])) : vue.createCommentVNode("", true)
          ]),
          _: 1
        }, 8, ["model-value"]);
      };
    }
  });
  const _withScopeId = (n) => (vue.pushScopeId("data-v-ed4eeb8f"), n = n(), vue.popScopeId(), n);
  const _hoisted_1 = { class: "affix" };
  const _hoisted_2 = { class: "book-list-container" };
  const _hoisted_3 = { class: "book-list-container-header" };
  const _hoisted_4 = { class: "book-list-container-body" };
  const _hoisted_5 = { key: 0 };
  const _hoisted_6 = {
    key: 1,
    class: "book-list-container-empty"
  };
  const _hoisted_7 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("p", null, "书架为空", -1));
  const _hoisted_8 = [
    _hoisted_7
  ];
  const _sfc_main$1 = /* @__PURE__ */ vue.defineComponent({
    __name: "BookShelf",
    setup(__props) {
      const { booksStore: booksStore2 } = useStore();
      const showDrawer = vue.ref(false);
      const sortKey = vue.ref("addTime");
      const sortAsc = vue.ref(true);
      const books2 = vue.computed(() => {
        return [...booksStore2.books].sort((a, b) => {
          const aValue = a[sortKey.value];
          const bValue = b[sortKey.value];
          if (Array.isArray(aValue) || Array.isArray(bValue))
            return 0;
          return aValue > bValue ? sortAsc.value ? 1 : -1 : sortAsc.value ? -1 : 1;
        });
      });
      const selectedBook = vue.ref(void 0);
      const showDetail = vue.ref(false);
      function selectBook(book) {
        selectedBook.value = book;
        showDetail.value = true;
      }
      return (_ctx, _cache) => {
        return vue.openBlock(), vue.createElementBlock(vue.Fragment, null, [
          vue.withDirectives(vue.createElementVNode("div", _hoisted_1, [
            vue.createVNode(vue.unref(elementPlus.ElButton), {
              icon: vue.unref(arrow_left_default),
              circle: "",
              type: "primary",
              onClick: _cache[0] || (_cache[0] = ($event) => showDrawer.value = !showDrawer.value)
            }, null, 8, ["icon"])
          ], 512), [
            [vue.vShow, !showDrawer.value]
          ]),
          vue.createVNode(vue.unref(elementPlus.ElDrawer), {
            modelValue: showDrawer.value,
            "onUpdate:modelValue": _cache[3] || (_cache[3] = ($event) => showDrawer.value = $event),
            title: "书架"
          }, {
            default: vue.withCtx(() => [
              vue.createElementVNode("div", _hoisted_2, [
                vue.createElementVNode("div", _hoisted_3, [
                  vue.createVNode(vue.unref(elementPlus.ElDropdown), {
                    disabled: vue.unref(books2).length === 0
                  }, {
                    dropdown: vue.withCtx(() => [
                      vue.createVNode(vue.unref(elementPlus.ElDropdownMenu), null, {
                        default: vue.withCtx(() => [
                          vue.createVNode(vue.unref(elementPlus.ElDropdownItem), {
                            onClick: _cache[1] || (_cache[1] = () => vue.unref(exportBookXlsx)(vue.unref(books2)))
                          }, {
                            default: vue.withCtx(() => [
                              vue.createTextVNode("excel ")
                            ]),
                            _: 1
                          }),
                          vue.createVNode(vue.unref(elementPlus.ElDropdownItem), {
                            onClick: _cache[2] || (_cache[2] = () => vue.unref(exportBookCsv)(vue.unref(books2)))
                          }, {
                            default: vue.withCtx(() => [
                              vue.createTextVNode("csv ")
                            ]),
                            _: 1
                          })
                        ]),
                        _: 1
                      })
                    ]),
                    default: vue.withCtx(() => [
                      vue.createVNode(vue.unref(elementPlus.ElButton), {
                        type: "primary",
                        round: "",
                        disabled: vue.unref(books2).length === 0
                      }, {
                        default: vue.withCtx(() => [
                          vue.createTextVNode(" 导出"),
                          vue.createVNode(vue.unref(elementPlus.ElIcon), { class: "el-icon--right" }, {
                            default: vue.withCtx(() => [
                              vue.createVNode(vue.unref(arrow_down_default))
                            ]),
                            _: 1
                          })
                        ]),
                        _: 1
                      }, 8, ["disabled"])
                    ]),
                    _: 1
                  }, 8, ["disabled"]),
                  vue.createVNode(vue.unref(elementPlus.ElButton), {
                    class: "book-list-container-header-item",
                    round: "",
                    type: "danger",
                    disabled: vue.unref(books2).length === 0,
                    onClick: vue.unref(booksStore2).clearBook
                  }, {
                    default: vue.withCtx(() => [
                      vue.createTextVNode("清空 ")
                    ]),
                    _: 1
                  }, 8, ["disabled", "onClick"])
                ]),
                vue.createElementVNode("div", _hoisted_4, [
                  vue.unref(books2).length > 0 ? (vue.openBlock(), vue.createElementBlock("div", _hoisted_5, [
                    (vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(vue.unref(books2), (book) => {
                      return vue.openBlock(), vue.createElementBlock("div", {
                        class: "book-list-container-book-item",
                        key: book.id
                      }, [
                        vue.createVNode(BookItem, {
                          book,
                          onSelect: selectBook
                        }, null, 8, ["book"])
                      ]);
                    }), 128))
                  ])) : (vue.openBlock(), vue.createElementBlock("div", _hoisted_6, _hoisted_8))
                ])
              ])
            ]),
            _: 1
          }, 8, ["modelValue"]),
          vue.createVNode(_sfc_main$2, {
            book: selectedBook.value,
            show: showDetail.value,
            onClose: _cache[4] || (_cache[4] = ($event) => showDetail.value = false)
          }, null, 8, ["book", "show"])
        ], 64);
      };
    }
  });
  const BookShelf_vue_vue_type_style_index_0_scoped_ed4eeb8f_lang = "";
  const BookShelf = /* @__PURE__ */ _export_sfc(_sfc_main$1, [["__scopeId", "data-v-ed4eeb8f"]]);
  function getMetaInfo(subjectDoc) {
    const data = {
      id: "",
      doubanUrl: "",
      title: "",
      ISBN: "",
      authors: []
    };
    const jsonElement = subjectDoc.querySelector('[type="application/ld+json"]');
    if (!jsonElement || !jsonElement.textContent) {
      return data;
    }
    const obj = JSON.parse(jsonElement.textContent);
    data.doubanUrl = obj.url;
    const id = obj.url.split("/")[4];
    data.id = id;
    data.title = obj.name;
    data.ISBN = obj.isbn;
    data.authors = obj.author.map((i) => i.name);
    return data;
  }
  function getFromA(element) {
    const aElements = [];
    let current = element;
    while (current) {
      const next = current.nextElementSibling;
      if (!next || next.tagName !== "A")
        break;
      current = next;
      aElements.push(current);
    }
    return aElements.map((i) => {
      var _a;
      return (_a = i.textContent) == null ? void 0 : _a.trim();
    });
  }
  function getFromText(element) {
    var _a, _b;
    const text = (_b = (_a = element.nextSibling) == null ? void 0 : _a.textContent) == null ? void 0 : _b.trim();
    if (!text || text === ":")
      return "";
    return text;
  }
  function getFromAOrText(element) {
    const resultFromA = getFromA(element);
    if (resultFromA.length !== 0) {
      return resultFromA;
    }
    const resultFromText = getFromText(element);
    return resultFromText ? [resultFromText] : [];
  }
  function getExtraInfo(subjectDoc) {
    const data = {
      publishingHouse: "",
      subTitle: "",
      originName: "",
      publishingTime: "",
      seriesName: "",
      page: 0,
      translators: []
    };
    const plElements = subjectDoc.querySelectorAll("#info .pl");
    Array.from(plElements).forEach((element) => {
      var _a;
      switch ((_a = element.textContent) == null ? void 0 : _a.trim()) {
        case "出版社:":
          data.publishingHouse = getFromAOrText(element)[0] ?? "";
          break;
        case "副标题:":
          data.subTitle = getFromText(element);
          break;
        case "原作名:":
          data.originName = getFromText(element);
          break;
        case "译者":
          data.translators = getFromAOrText(element);
          break;
        case "出版年:":
          data.publishingTime = getFromText(element);
          break;
        case "页数:": {
          const pageStr = getFromText(element);
          data.page = pageStr ? Number.parseInt(pageStr, 10) : 0;
          break;
        }
        case "丛书:":
          data.seriesName = getFromAOrText(element)[0] ?? "";
          break;
      }
    });
    return data;
  }
  function getScoreInfo(subjectDoc) {
    var _a, _b, _c, _d;
    const element = subjectDoc.getElementById("interest_sectl");
    const scoreStr = (_b = (_a = element == null ? void 0 : element.querySelector("strong")) == null ? void 0 : _a.textContent) == null ? void 0 : _b.trim();
    const scorePeopleCountStr = (_d = (_c = element == null ? void 0 : element.querySelector(".rating_people > span")) == null ? void 0 : _c.textContent) == null ? void 0 : _d.trim();
    let scorePeopleCount = 0;
    if (scorePeopleCountStr && !scorePeopleCountStr.includes("目前无人评价")) {
      scorePeopleCount = Number.parseInt(scorePeopleCountStr, 10);
    }
    return {
      score: scoreStr ? Number.parseFloat(scoreStr) : 0,
      scorePeopleCount
    };
  }
  function getContentBrief(subjectDoc) {
    var _a;
    const introElements = Array.from(
      subjectDoc == null ? void 0 : subjectDoc.querySelectorAll("#link-report .intro")
    );
    if (introElements.length === 0)
      return "";
    const pElements = ((_a = introElements[introElements.length - 1]) == null ? void 0 : _a.getElementsByTagName("p")) ?? [];
    return Array.from(pElements).reduce((str, current, idx) => {
      var _a2;
      const text = (_a2 = current.textContent) == null ? void 0 : _a2.trim();
      if (text)
        str += idx === 0 ? text : `
${text}`;
      return str;
    }, "");
  }
  function getCoverUrl(subjectDoc) {
    var _a;
    const imgElement = (_a = subjectDoc.getElementById("mainpic")) == null ? void 0 : _a.querySelector("img");
    return (imgElement == null ? void 0 : imgElement.src) ?? "";
  }
  function getBook(subjectDoc) {
    const book = {
      ...getMetaInfo(subjectDoc),
      ...getScoreInfo(subjectDoc),
      ...getExtraInfo(subjectDoc),
      contentBrief: getContentBrief(subjectDoc),
      coverUrl: getCoverUrl(subjectDoc)
    };
    return book;
  }
  function getAlreadyReadButton(subjectDoc) {
    return subjectDoc.querySelector("#interest_sect_level > a:nth-child(3)");
  }
  function getLasButton(subjectDoc) {
    return subjectDoc.querySelector(
      "#interest_sect_level > a:nth-last-of-type(1)"
    );
  }
  const parseHeaders = (rawHeaders = "") => {
    const headers = new Headers();
    const preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, " ");
    preProcessedHeaders.split("\r").map(function(header) {
      return header.startsWith(`
`) ? header.substring(1) : header;
    }).forEach(function(line) {
      var _a;
      const parts = line.split(":");
      const key = (_a = parts.shift()) == null ? void 0 : _a.trim();
      if (key) {
        const value = parts.join(":").trim();
        headers.append(key, value);
      }
    });
    return headers;
  };
  async function GM_Fetch(xhrRequest) {
    return new Promise((resolve, reject) => {
      GM_xmlhttpRequest({
        method: "GET",
        async onload(e) {
          const resp = new Response(e.response ?? e.responseText, {
            status: e.status,
            statusText: e.statusText,
            headers: parseHeaders(e.responseHeaders)
          });
          Object.defineProperty(resp, "url", { value: e.finalUrl });
          resolve(resp);
        },
        async onerror() {
          reject(new TypeError("Network request failed"));
        },
        async ontimeout() {
          reject(new TypeError("Network request failed"));
        },
        async onabort() {
          reject(new DOMException("Aborted", "AbortError"));
        },
        ...xhrRequest
      });
    });
  }
  const IDENTIFY_CLASS = "douban_shelf_button";
  function createBtn(doc, id, text, clickHandler, style, classList) {
    const btn = doc.createElement("a");
    const clsList = classList || ["j", "a_show_login", "colbutt", "ll"];
    btn.classList.add(...clsList);
    btn.id = id;
    btn.href = "#";
    btn.rel = "nofollow";
    if (style) {
      Object.entries(style).map(([k, v]) => {
        btn.style[k] = v;
      });
    }
    const span = doc.createElement("span");
    span.classList.add(IDENTIFY_CLASS);
    span.textContent = text;
    span.style.fontSize = "13px";
    btn.appendChild(span);
    btn.addEventListener("click", (e) => {
      e.preventDefault();
      clickHandler();
    });
    return btn;
  }
  const COPY_ID = "douban-book-copy";
  const ADD_SHELF_ID = "douban-book-add-shelf";
  const COPY = "复制";
  const ADD_SHELF = "加入书架";
  function createCopyBtn(doc, getBook2, id = COPY_ID, style, classList) {
    const btn = createBtn(
      doc,
      id,
      COPY,
      async () => {
        const book = await getBook2(doc);
        if (!book)
          return;
        try {
          await copyBookWithTip(book);
          btn.dispatchEvent(new CustomEvent(COPY, { detail: true }));
        } catch (error2) {
          btn.dispatchEvent(new CustomEvent(COPY, { detail: false }));
          throw error2;
        }
      },
      style,
      classList
    );
    return btn;
  }
  function createAddBtn(doc, getBook2, id = ADD_SHELF_ID, style, classList) {
    const btn = createBtn(
      doc,
      id,
      ADD_SHELF,
      async () => {
        const book = await getBook2(doc);
        const { booksStore: booksStore2 } = useStore();
        const r = booksStore2.addBook(book);
        if (!r.success) {
          btn.dispatchEvent(new CustomEvent(ADD_SHELF, { detail: false }));
          warning(r.msg);
        } else {
          btn.dispatchEvent(new CustomEvent(ADD_SHELF, { detail: true }));
          success("加入成功");
        }
      },
      style,
      classList
    );
    return btn;
  }
  async function getBookPageHtmlByUrl(url) {
    const response = await GM_Fetch({ url });
    if (response.status !== 200) {
      return null;
    }
    const text = await response.text();
    return text;
  }
  function useInitBtns$5(doc) {
    if (!/\/subject\/\d+\/?$/.test(new URL(doc.URL).pathname)) {
      return;
    }
    const lastButton = getLasButton(doc);
    if (!lastButton) {
      warning(`定位'读过'按钮失败`);
      return;
    }
    const span = lastButton.querySelector("span");
    if ((span == null ? void 0 : span.textContent) === ADD_SHELF) {
      return;
    }
    const input = span == null ? void 0 : span.querySelector("input");
    const alreadyReadButton = (input == null ? void 0 : input.value) === "读过" ? lastButton : getAlreadyReadButton(doc);
    if (!alreadyReadButton) {
      warning(`定位'读过'按钮失败`);
      return;
    }
    const copyBtn = createCopyBtn(doc, getBook);
    const addBtn = createAddBtn(doc, getBook);
    alreadyReadButton.after(copyBtn);
    copyBtn.after(addBtn);
    return { copyBtn, addBtn };
  }
  function init$5(doc) {
    useInitBtns$5(doc);
  }
  function getBookItemList$4(searchDoc) {
    const rootEle = searchDoc.querySelector("#wrapper #root");
    if (!rootEle)
      return [];
    return Array.from(rootEle.querySelectorAll(".item-root")).reduce(
      (list, cur) => {
        var _a, _b, _c, _d, _e;
        const aEle = cur.querySelector("a");
        if ((_a = aEle == null ? void 0 : aEle.href) == null ? void 0 : _a.startsWith("https://book.douban.com/subject/")) {
          const name = ((_c = (_b = cur.querySelector(".detail .title")) == null ? void 0 : _b.querySelector("a")) == null ? void 0 : _c.textContent) ?? "";
          const id = ((_e = (_d = aEle == null ? void 0 : aEle.href) == null ? void 0 : _d.split("https://book.douban.com/subject/")[1]) == null ? void 0 : _e.split("/")[0]) ?? "";
          list.push({
            element: cur,
            url: aEle == null ? void 0 : aEle.href,
            id,
            name
          });
        }
        return list;
      },
      []
    );
  }
  function useInitBtns$4(doc) {
    const btn = doc.getElementById(`${COPY_ID}-0`);
    if (btn) {
      return false;
    }
    const { booksStore: booksStore2 } = useStore();
    const list = getBookItemList$4(doc);
    list.forEach(({ element, url, name, id }, idx) => {
      var _a;
      let bookCache = void 0;
      const getBook$12 = async () => {
        if (bookCache)
          return bookCache;
        const bookInStore = booksStore2.getBook(id);
        if (bookInStore) {
          bookCache = {
            ...bookInStore
          };
          delete bookCache.addTime;
          return bookCache;
        }
        const html = await getBookPageHtmlByUrl(url);
        if (!html) {
          throw new Error(`获取《${name}》页面失败`);
        }
        const parser = new DOMParser();
        const htmlDoc = parser.parseFromString(html, "text/html");
        const book = getBook(htmlDoc);
        bookCache = book;
        return book;
      };
      const style = {
        marginTop: "7px"
      };
      const copyBtn = createCopyBtn(doc, getBook$12, `${COPY_ID}-${idx}`, style);
      const addBtn = createAddBtn(doc, getBook$12, `${ADD_SHELF_ID}-${idx}`, style);
      (_a = element.querySelector(".detail")) == null ? void 0 : _a.appendChild(copyBtn);
      copyBtn.after(addBtn);
    });
    return true;
  }
  function init$4(doc) {
    useInitBtns$4(doc);
  }
  function getBookItemList$3(seriesDoc) {
    return Array.from(
      seriesDoc.querySelectorAll("#content .subject-item")
    ).reduce((list, cur) => {
      var _a, _b, _c, _d;
      const aEle = (_a = cur.querySelector(".info")) == null ? void 0 : _a.querySelector("a");
      if ((_b = aEle == null ? void 0 : aEle.href) == null ? void 0 : _b.startsWith("https://book.douban.com/subject/")) {
        const id = ((_d = (_c = aEle == null ? void 0 : aEle.href) == null ? void 0 : _c.split("https://book.douban.com/subject/")[1]) == null ? void 0 : _d.split("/")[0]) ?? "";
        list.push({
          element: cur,
          url: aEle == null ? void 0 : aEle.href,
          id,
          name: (aEle == null ? void 0 : aEle.textContent) ?? ""
        });
      }
      return list;
    }, []);
  }
  function useInitBtns$3(doc) {
    if (!/\/series\/\d+\/?$/.test(new URL(doc.URL).pathname)) {
      return false;
    }
    const btn = doc.getElementById(`${COPY_ID}-0`);
    if (btn) {
      return false;
    }
    const { booksStore: booksStore2 } = useStore();
    const list = getBookItemList$3(doc);
    list.forEach(({ element, url, name, id }, idx) => {
      var _a;
      let bookCache = void 0;
      const getBook$12 = async () => {
        if (bookCache)
          return bookCache;
        const bookInStore = booksStore2.getBook(id);
        if (bookInStore) {
          bookCache = {
            ...bookInStore
          };
          delete bookCache.addTime;
          return bookCache;
        }
        const html = await getBookPageHtmlByUrl(url);
        if (!html) {
          throw new Error(`获取《${name}》页面失败`);
        }
        const parser = new DOMParser();
        const htmlDoc = parser.parseFromString(html, "text/html");
        const book = getBook(htmlDoc);
        bookCache = book;
        return book;
      };
      const style = {
        marginTop: "7px"
      };
      const copyBtn = createCopyBtn(doc, getBook$12, `${COPY_ID}-${idx}`, style);
      const addBtn = createAddBtn(doc, getBook$12, `${ADD_SHELF_ID}-${idx}`, style);
      (_a = element.querySelector(".info")) == null ? void 0 : _a.appendChild(copyBtn);
      copyBtn.after(addBtn);
    });
    return true;
  }
  function init$3(doc) {
    useInitBtns$3(doc);
  }
  function getBookItemList$2(seriesDoc) {
    return Array.from(seriesDoc.querySelectorAll("#content .bkdesc")).reduce(
      (list, cur) => {
        var _a, _b, _c;
        const aEle = cur == null ? void 0 : cur.querySelector("a");
        if ((_a = aEle == null ? void 0 : aEle.href) == null ? void 0 : _a.startsWith("https://book.douban.com/subject/")) {
          const id = ((_c = (_b = aEle == null ? void 0 : aEle.href) == null ? void 0 : _b.split("https://book.douban.com/subject/")[1]) == null ? void 0 : _c.split("/")[0]) ?? "";
          list.push({
            element: cur,
            url: aEle == null ? void 0 : aEle.href,
            id,
            name: (aEle == null ? void 0 : aEle.textContent) ?? ""
          });
        }
        return list;
      },
      []
    );
  }
  function useInitBtns$2(doc) {
    if (!/\/works\/\d+\/?$/.test(new URL(doc.URL).pathname)) {
      return false;
    }
    const btn = doc.getElementById(`${COPY_ID}-0`);
    if (btn) {
      return false;
    }
    const { booksStore: booksStore2 } = useStore();
    const list = getBookItemList$2(doc);
    list.forEach(({ element, url, name, id }, idx) => {
      var _a;
      let bookCache = void 0;
      const getBook$12 = async () => {
        if (bookCache)
          return bookCache;
        const bookInStore = booksStore2.getBook(id);
        if (bookInStore) {
          bookCache = {
            ...bookInStore
          };
          delete bookCache.addTime;
          return bookCache;
        }
        const html = await getBookPageHtmlByUrl(url);
        if (!html) {
          throw new Error(`获取《${name}》页面失败`);
        }
        const parser = new DOMParser();
        const htmlDoc = parser.parseFromString(html, "text/html");
        const book = getBook(htmlDoc);
        bookCache = book;
        return book;
      };
      const classList = ["j", "ll", "colbutt", "a_add2cart", "add2cart"];
      const copyBtn = createCopyBtn(
        doc,
        getBook$12,
        `${COPY_ID}-${idx}`,
        void 0,
        classList
      );
      const addBtn = createAddBtn(
        doc,
        getBook$12,
        `${ADD_SHELF_ID}-${idx}`,
        void 0,
        classList
      );
      (_a = element.querySelector(".rr")) == null ? void 0 : _a.appendChild(copyBtn);
      copyBtn.after(addBtn);
    });
    return true;
  }
  function init$2(doc) {
    useInitBtns$2(doc);
  }
  function getBookItemList$1(seriesDoc) {
    return Array.from(
      seriesDoc.querySelectorAll("#content .subject-item")
    ).reduce((list, cur) => {
      var _a, _b, _c, _d;
      const aEle = (_a = cur.querySelector(".info")) == null ? void 0 : _a.querySelector("a");
      if ((_b = aEle == null ? void 0 : aEle.href) == null ? void 0 : _b.startsWith("https://book.douban.com/subject/")) {
        const id = ((_d = (_c = aEle == null ? void 0 : aEle.href) == null ? void 0 : _c.split("https://book.douban.com/subject/")[1]) == null ? void 0 : _d.split("/")[0]) ?? "";
        list.push({
          element: cur,
          url: aEle == null ? void 0 : aEle.href,
          id,
          name: (aEle == null ? void 0 : aEle.textContent) ?? ""
        });
      }
      return list;
    }, []);
  }
  function useInitBtns$1(doc) {
    if (!/\/press\/\d+\/?$/.test(new URL(doc.URL).pathname)) {
      return false;
    }
    const btn = doc.getElementById(`${COPY_ID}-0`);
    if (btn) {
      return false;
    }
    const { booksStore: booksStore2 } = useStore();
    const list = getBookItemList$1(doc);
    list.forEach(({ element, url, name, id }, idx) => {
      var _a;
      let bookCache = void 0;
      const getBook$12 = async () => {
        if (bookCache)
          return bookCache;
        const bookInStore = booksStore2.getBook(id);
        if (bookInStore) {
          bookCache = {
            ...bookInStore
          };
          delete bookCache.addTime;
          return bookCache;
        }
        const html = await getBookPageHtmlByUrl(url);
        if (!html) {
          throw new Error(`获取《${name}》页面失败`);
        }
        const parser = new DOMParser();
        const htmlDoc = parser.parseFromString(html, "text/html");
        const book = getBook(htmlDoc);
        bookCache = book;
        return book;
      };
      const style = {
        marginTop: "7px"
      };
      const classList = ["j", "a_collect_btn"];
      const copyBtn = createCopyBtn(
        doc,
        getBook$12,
        `${COPY_ID}-${idx}`,
        style,
        classList
      );
      const addBtn = createAddBtn(
        doc,
        getBook$12,
        `${ADD_SHELF_ID}-${idx}`,
        style,
        classList
      );
      (_a = element.querySelector(".ft .collect-info")) == null ? void 0 : _a.appendChild(copyBtn);
      copyBtn.after(addBtn);
    });
    return true;
  }
  function init$1(doc) {
    useInitBtns$1(doc);
  }
  function getBookItemList(seriesDoc) {
    return Array.from(seriesDoc.querySelectorAll("#content li")).reduce(
      (list, cur) => {
        var _a, _b, _c;
        const aEle = cur == null ? void 0 : cur.querySelector("a");
        if ((_a = aEle == null ? void 0 : aEle.href) == null ? void 0 : _a.startsWith("https://book.douban.com/subject/")) {
          const id = ((_c = (_b = aEle == null ? void 0 : aEle.href) == null ? void 0 : _b.split("https://book.douban.com/subject/")[1]) == null ? void 0 : _c.split("/")[0]) ?? "";
          list.push({
            element: cur,
            url: aEle == null ? void 0 : aEle.href,
            id,
            name: (aEle == null ? void 0 : aEle.textContent) ?? ""
          });
        }
        return list;
      },
      []
    );
  }
  function useInitBtns(doc) {
    if (!/\/author\/\d+\/books\/?$/.test(new URL(doc.URL).pathname)) {
      return false;
    }
    const btn = doc.getElementById(`${COPY_ID}-0`);
    if (btn) {
      return false;
    }
    const { booksStore: booksStore2 } = useStore();
    const list = getBookItemList(doc);
    list.forEach(({ element, url, name, id }, idx) => {
      var _a;
      let bookCache = void 0;
      const getBook$12 = async () => {
        if (bookCache)
          return bookCache;
        const bookInStore = booksStore2.getBook(id);
        if (bookInStore) {
          bookCache = {
            ...bookInStore
          };
          delete bookCache.addTime;
          return bookCache;
        }
        const html = await getBookPageHtmlByUrl(url);
        if (!html) {
          throw new Error(`获取《${name}》页面失败`);
        }
        const parser = new DOMParser();
        const htmlDoc = parser.parseFromString(html, "text/html");
        const book = getBook(htmlDoc);
        bookCache = book;
        return book;
      };
      const classList = ["j", "a_collect_btn"];
      const copyBtn = doc.createElement("span");
      copyBtn.classList.add("gact");
      copyBtn.appendChild(
        createCopyBtn(doc, getBook$12, `${COPY_ID}-${idx}`, void 0, classList)
      );
      const addBtn = doc.createElement("span");
      addBtn.classList.add("gact");
      addBtn.appendChild(
        createAddBtn(doc, getBook$12, `${ADD_SHELF_ID}-${idx}`, void 0, classList)
      );
      (_a = element.querySelector(".author-collect")) == null ? void 0 : _a.appendChild(copyBtn);
      copyBtn.after(addBtn);
    });
    return true;
  }
  function init(doc) {
    useInitBtns(doc);
  }
  const _sfc_main = /* @__PURE__ */ vue.defineComponent({
    __name: "App",
    setup(__props) {
      vue.onMounted(async () => {
        const map = {
          ["book.douban.com/subject"]: init$5,
          ["search.douban.com/book"]: init$4,
          ["book.douban.com/series"]: init$3,
          ["book.douban.com/works"]: init$2,
          ["book.douban.com/press"]: init$1,
          ["book.douban.com/author"]: init
        };
        const url = new URL(document.URL);
        const initFunc = map[url.host + url.pathname.split("/").slice(0, 2).join("/")];
        if (initFunc) {
          initFunc(document);
        } else {
          console.warn(`${url} 没有对应的处理方法`);
        }
      });
      return (_ctx, _cache) => {
        return vue.openBlock(), vue.createBlock(BookShelf);
      };
    }
  });
  async function run(doc) {
    const app = document.createElement("div");
    doc.body.appendChild(app);
    setTimeout(() => {
      vue.createApp(_sfc_main).mount(app);
    });
  }
  run(document);
})(Vue, ElementPlus, csv_stringify_sync, XLSX);