Box 邀请协作者

解除Box网盘的协作限制, 通过API向指定邮箱发送邀请

// ==UserScript==
// @name               Box Invite Collaborator
// @name:zh-CN         Box 邀请协作者
// @namespace          http://tampermonkey.net/
// @version            0.1
// @description        Remove Box's collaboration limit, send invitation to specified email via API
// @description:zh-CN  解除Box网盘的协作限制, 通过API向指定邮箱发送邀请
// @author             0xYYP
// @match              https://app.box.com/*
// @icon               
// @grant              GM_xmlhttpRequest
// @license            GPL
// ==/UserScript==

let requestToken = '';
let xRequestToken = '';

// 监视特定的HTTP请求以获取令牌
function monitorRequestAndGetTokens() {
  const originalOpen = XMLHttpRequest.prototype.open;

  XMLHttpRequest.prototype.open = function() {
    this.addEventListener('readystatechange', function() {
      if (this.readyState === 4 && this.responseURL.endsWith('/onboarding/tracking')) {
        requestToken = this._headers['Request-Token'] || '';
        xRequestToken = this._headers['X-Request-Token'] || '';
      }
    });
    originalOpen.apply(this, arguments);
  };

  const originalSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;
  XMLHttpRequest.prototype.setRequestHeader = function(header, value) {
    this._headers = this._headers || {};
    this._headers[header] = value;
    originalSetRequestHeader.apply(this, arguments);
  };
}

// 在页面上创建UI元素
function createUI() {
  const config = { childList: true, subtree: true };
  let hasInserted = false;
  
  const observer = new MutationObserver((mutations) => {
    if (hasInserted) return;
    mutations.forEach((mutation) => {
      if (mutation.addedNodes) {
        mutation.addedNodes.forEach((node) => {
          if (node.nodeType === Node.ELEMENT_NODE) {
            const element = node.getElementsByClassName('sharing-sidebar-section')[0];
            if (element) {
              const html = `
              <button id="inviteButton" class="btn-plain CollabInfo" type="submit">
                <div class="sharing-item">
                  <span class="sidebar-icon-container">
                    <svg width="16" height="16" viewBox="0 0 16 16" class="icon" focusable="false" aria-hidden="true"
                      role="presentation">
                      <path fill="#222" fill-rule="evenodd"
                        d="M8 9.5a6.497 6.497 0 015.63 3.251.5.5 0 01-.865.5A5.497 5.497 0 008 10.5a5.497 5.497 0 00-4.767 2.754.5.5 0 11-.866-.5A6.497 6.497 0 018 9.5zM8 2a3 3 0 110 6 3 3 0 010-6zm0 1a2 2 0 100 4 2 2 0 000-4z">
                      </path>
                    </svg>
                  </span>
                  <div class="description-wrapper">
                    <div class="txt-ellipsis"><span>协作者</span></div>
                    <p class="item-subtext txt-ellipsis">
                      <span id="inviteSpan">邀请人员</span>
                      <input id="inviteInput" type="text" style="
                          display: none;
                          width: 190px;
                          height: 20px;
                          border-radius: 2px;
                          text-align: center;
                          font-size: 14px;
                          line-height: 20px;
                          margin: 2px 0 0 0;
                        " placeholder="输入被邀请人邮箱, 回车发送" />
                    </p>
                  </div>
                </div>
              </button>
              `;
              element.insertAdjacentHTML('afterbegin', html);
              hasInserted = true;
              observer.disconnect();
              attachUIEvents();
            }
          }
        });
      }
    });
  });
  
  observer.observe(document.body, config);
}

// 附加事件到UI元素
function attachUIEvents() {
  const inviteSpan = document.getElementById("inviteSpan");
  const inviteInput = document.getElementById("inviteInput");
  const inviteButton = document.getElementById("inviteButton");

  inviteButton.addEventListener("click", function() {
    inviteSpan.style.display = "none";
    inviteInput.style.display = "block";
    inviteInput.focus();
  });

  inviteInput.addEventListener('keydown', function(event) {
    if (event.key === 'Enter') {
      const email = inviteInput.value;
      inviteSpan.style.display = "block";
      inviteInput.style.display = "none";
      const dir = window.location.pathname.split("/").pop();
      inviteUser(email, dir);
    }
  });
}

// 向指定邮箱和目录发送邀请
function inviteUser(email, dir) {
  GM_xmlhttpRequest({
    method: "POST",
    url: `https://app.box.com/app-api/enduserapp/item/d_${dir}/invite-collaborators`,
    headers: {
      "authority": "app.box.com",
      "accept": "application/json",
      "content-type": "application/json",
      "cookie": document.cookie,
      "origin": "https://app.box.com",
      "referer": `https://app.box.com/folder/${dir}`,
      "request-token": requestToken,
      "x-request-token": xRequestToken
    },
    data: JSON.stringify({
      "emails": email,
      "groupIDs": "",
      "emailMessage": "",
      "permission": "Editor",
      "numsOfInvitees": 1,
      "numOfInviteeGroups": 0
    }),
    onload: function (response) {
      alert(`向 ${email} 的协作请求已发送!`);
    }
  });
}

(function() {
  monitorRequestAndGetTokens();
  createUI();
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址