Teambition

Beautify the Teambition.

// ==UserScript==
// @name               Teambition
// @name:zh-CN         Teambition
// @description        Beautify the Teambition.
// @description:zh-CN  美化Teambition。
// @namespace          https://github.com/HaleShaw
// @version            1.0.3
// @author             HaleShaw
// @copyright          2022+, HaleShaw (https://github.com/HaleShaw)
// @license            AGPL-3.0-or-later
// @homepage           https://github.com/HaleShaw/TM-Teambition
// @supportURL         https://github.com/HaleShaw/TM-Teambition/issues
// @contributionURL    https://www.jianwudao.com/
// @icon               
// @match              https://www.teambition.com/project/*
// @match              https://www.teambition.com/organization/*
// @match              https://apps.teambition.com/work-time-client/*
// @compatible	       Chrome
// @grant              GM_addStyle
// @grant              GM_getValue
// @grant              GM_setValue
// @grant              GM_registerMenuCommand
// ==/UserScript==

// ==OpenUserJS==
// @author             HaleShaw
// @collaborator       HaleShaw
// ==/OpenUserJS==

(function () {
  'use strict';

  let data;

  const mainStyle = `
  /* 任务面板 */

  /* 看板模式下,将任务卡片的任务ID左移,鼠标悬浮时使“…”按钮不覆盖任务ID */
  .task-card .task-content-set .task-card-footer>* {
      margin-right: 25px !important;
  }

  /* 隐藏工作流顶部右侧的“更多”按钮 */
  div.kanban-single-lane-body-wrapper>div.kanban-single-list>div.kanban-single-list-contents>div>div>div:last-child {
    display: none !important;
  }

  /* -------------------------------------------------------------------------- */
  /* 工作流配置页面 */

  /* 将整个页面周围边距清除 */
  .taskflow-config-contents {
      bottom: 0px !important;
      left: 0px !important;
      right: 0px !important;
      top: 60px !important;
  }

  /* 减小行高 */
  .taskflow-config-table-block-item {
      height: 50px !important;
  }

  /* 减小列宽 */
  .taskflow-config-table-status-column {
      width: 110px !important;
  }

  .status-label-with-menu {
      padding: 0 !important;
      width: 110px !important;
  }
  /* -------------------------------------------------------------------------- */
  /* 迭代面板 */
  .sprint-panel-component-title-wrapper {
      padding: 4px 16px !important;
  }
  .sprint-panel-component-sprint-content>.sprint-name {
      padding: 3px 0 !important;
  }
  .sprint-panel-item .sprint-control-option {
      height: 28px !important;
  }

  .sprint-panel-item .sprint-subtitle {
      margin-bottom: 0px !important;
  }
  `;

  const loadAllStyle = `
  .loadAll {
    background-color: #f2fbff!important;
    border-color: transparent!important;
    color: #1b9aee!important;
    font-size: 14px;
    height: 28px;
    line-height: 26px;
    min-width: 52px;
    padding: 0 7px;
    margin-right: 15px;
    border-radius: 4px;
    border-width: 1px;
    box-shadow: none;
    transition: background-color .3s ease,color .3s ease,border-color .3s ease;
  }

  .loadAll:hover{
    color: #ffffff !important;
    background-color: #1b9aee !important;
    border-color: #1b9aee !important;
  }
  `;

  const worktimeStyle = `
    /* 头部标题 */
    #root header h4 {
        display: none !important;
    }
    /* 头部日期选择器 */
    ._18ivZlWaPJH2YVLnpZrb0N>header .LZNYW1Yb6acagP8pDYBcB .vkkNP00hyMhbgKHQS-CCT {
        margin-right: 8px !important;
    }
    ._18ivZlWaPJH2YVLnpZrb0N>header .LZNYW1Yb6acagP8pDYBcB ._2gIxRD0Aw3Wz4V58lFor-k {
        margin-left: 8px !important;
    }

    #root > div > div > header > div:first-child > h4 {
      padding: 0 5px !important;
    }

    #root > div > div > header > div:last-child {
      flex-grow: 0 !important;
      flex-basis: 240px !important;
    }

    /* 表头第一列 */
    ._1a3bcSp1CtO4ejX9ugVzzO._2Q3J8umLBh0SslZCOuVtuw {
      width: 180px !important;
    }

    /* 表体第一列 */
    ._1a3bcSp1CtO4ejX9ugVzzO {
      padding: 0 0px 0 28px !important;
    }

    .rt-table > .rt-tbody > .rt-tr-group > .rt-tr > .rt-td:first-child,
    .rt-table > .rt-thead > .rt-tr > .rt-th:first-child,
    .rt-table > .rt-tfoot > .rt-tr > .rt-td:first-child {
      flex: 180 0 auto !important;
      width: 180px !important;
      max-width: 180px !important;
    }

    .rt-table > .rt-tbody > .rt-tr-group > .rt-tr > .rt-td:not(:first-child),
    .rt-table > .rt-thead > .rt-tr > .rt-th:not(:first-child),
    .rt-table > .rt-tfoot > .rt-tr > .rt-td:not(:first-child) {
      flex: 124 0 auto !important;
      width: 124px !important;
      max-width: 124px !important;
    }

    /* 隐藏悬浮边框 */
    ._1SmlEXYUElETMoqOAF1ym0:hover {
      border: none !important;
    }
    ._3AzO2C7wToS7pqSbOxyEXm {
      border: none !important;
    }

    /* 直接展示实际和计划工时 */
    ._3sRT72VQQvRBhWYMmH0hD2 {
      display: none !important;
    }
    .JqBbgp5bIj-_fWcuSdn8X {
      display: block !important;
    }

    /* footer总计工时 */
    .Z8fZCPB9x3aQBvMMYEzWI {
      padding: 0px 0px !important;
    }

    button.cusBtn {
      outline: none;
      padding: 2px 10px;
      border: none;
      border-radius: 4px;
      color: #595959;
      line-height: 26px;
      background-color: #e5e5e5;
      font-size: 14px;
      cursor: pointer;
      margin-left: 5px;
    }
    button.cusBtn:hover {
      color: #1b9aee;
    }

    button.cusBtn.selected {
      background-color: #888888;
      color: #ffffff;
    }

    select.cusSelect {
      font-size: 14px;
      border: 1px solid #e5e5e5;
      border-radius: 4px;
      color: #262626;
      transition: border 218ms;
      padding: 0px 8px;
      margin-left: 10px;
      height: 30px;
    }

    /* 计划时间过大 */
    span.time.plan.greater {
      color: #ff2222;
    }

    /* 计划时间过小 */
    span.time.plan.less {
      color: #f29900;
    }
    `;

  const settingHTML = `
<div id="settingContent">
  <div id="settingTitle">
    <span class="icon"></span><span class="title">Teambition - 工时应用设置</span>
  </div>
  <ol id="settingBody">
    <li>
      <span class="setting title">工时应用URL</span>
      <div class="setting comment">
        <a>a.在左侧边栏上,点击“更多”按钮 -> “全部应用” -> “工时”</a><span class="errMsg url"></span>
        <a>b.页面加载完成后,按F12进入开发者模式,找到第一个iframe的地址,即是工时应用的地址</a>
        <a class="code">document.querySelectorAll('iframe')[0].src</a>
        <button type="button" id="getUrl">一键获取</button>
      </div>
      <input type="text" class="setting text url" placeholder="https://appshell.teambition.com/api/v1/organization/..."></input>
    </li>
    <li>
      <span class="setting title">分组过滤按钮</span>
      <div class="setting comment">
        <a>配置每个按钮过滤部分人员。请使用标准JSON格式,使用英文双引号。</a><span class="errMsg filter"></span>
      </div>
      <textarea
        type="text"
        class="setting text filter"
        placeholder="{
    'button name':['username','username'],
    'button name':['username','username']
}"></textarea>
    </li>
    <li>
      <span class="setting title">移除人员</span>
      <div class="setting comment">
        <a>可将不必要的人员从表格中移除。请使用标准JSON格式,使用英文双引号。</a><span class="errMsg remove"></span>
      </div>
      <input type="text" class="setting text remove" placeholder="['username','username']"></input>
    </li>
  </ol>
  <div id="settingFooter">
    <button type="button" id="settingCancel" class="setting button">取消</button>
    <button type="button" id="settingSave" class="setting button">保存</button>
  </div>
</div>
`;

  const settingStyle = `
#settingPanel {
  display: none;
  justify-content: center;
  align-items: center;
  position: fixed;
  top: 0;
  right: 0;
  bottom: 0;
  left: 0;
  background-color: rgba(0, 0, 0, 0.5);
  z-index: 200000000;
  overflow: auto;
  font-family: arial, sans-serif;
  min-height: 100%;
  font-size: 16px;
  transition: 0.5s;
  opacity: 1;
  user-select: none;
  -moz-user-select: none;
  padding-bottom: 80px;
  box-sizing: border-box;
}

#settingContent {
  display: flex;
  flex-wrap: wrap;
  padding: 20px;
  background-color: #f7f7f7;
  border-radius: 4px;
  position: absolute;
  width: 50%;
  transition: 0.5s;
}

#settingTitle {
  width: 100%;
  margin: 15px 0;
}

#settingTitle > span.icon {
  width: 20px;
  height: 20px;
  display: inline-block;
  background-image: url("https://tcs-ga.teambition.net/thumbnail/312ca08e98b657c56f74c8f5eae0b66dd8d2/w/20/h/20");
  background-position: center center;
  background-repeat: no-repeat;
  background-size: 100%;
  border-radius: 4px;
}

#settingTitle > span.title {
  font-size: 18px;
  vertical-align: top;
  font-weight: 500;
}

#settingBody {
  padding: 20px 20px 20px 40px;
  background-color: #fff;
  border: 1px solid #fff;
  border-radius: 5px;
  width: 100%;
  list-style: decimal;
}

span.setting.title {
  font-size: 1.125rem;
  font-weight: 500;
  line-height: 2rem;
  display: block;
}

span.setting.title:not(:first-child) {
  margin-top: 8px;
}

.setting.comment > a {
  color: #666;
  display: inline-block;
  line-height: 24px;
  text-decoration: none;
  user-select: text;
}

.setting.comment > a.code {
  background: #fafafa;
  line-height: 150%;
  padding-right: 10px;
  padding-left: 10px;
  border-left: 2px solid #6ce26c;
  display: block !important;
}

.setting.text {
  width: 100%;
  margin-top: 5px;
  line-height: 24px;
}

.setting.text.url {
  width: calc(100% - 90px) !important;
}

textarea.setting.text {
  min-height: 126px;
  max-height: 200px;
}

#getUrl {
  position: absolute;
  right: 40px;
  padding: 4px 8px;
  margin-top: 5px;
  cursor: pointer;
  outline-style: none;
  border-radius: 3px;
  color: #fff;
  border: 1px solid #1b9aee;
  background-color: #1b9aee;
}

#settingFooter {
  text-align: center;
  margin: auto;
  margin-top: 15px;
  border-radius: 4px;
}

.setting.button {
  padding: 5px 10px;
  cursor: pointer;
  outline-style: none;
  border-radius: 3px;
}

#settingCancel {
  border: 1px solid #e7e7e7;
  background-color: #e7e7e7;
}

#settingCancel:hover {
  border: 1px solid #979797;
  background-color: #979797;
}

#settingSave {
  margin-left: 20px;
  color: #fff;
  border: 1px solid #1b9aee;
  background-color: #1b9aee;
}

#settingSave:hover,
#getUrl:hover {
  border: 1px solid #0171c2;
  background-color: #0171c2;
}

span.errMsg {
  color: red;
  margin-left: 10px;
}
  `;

  let members = [];

  const planTimeMaximum = 10;
  const planTimeMinimum = 7;

  // The number of times the worktime page was loaded.
  const loadCount = 5;

  main();

  function main() {
    GM_addStyle(mainStyle);
    logInfo(GM_info.script.name, GM_info.script.version);
    data = GM_getValue('worktime');
    GM_registerMenuCommand("设置", () => {
      new Setting();
    });

    if (window.location.href.startsWith('https://apps.teambition.com/work-time-client')) {
      beautifyWorktime();
    } else {
      setTimeout(() => {
        hideAddingStatus();
        listenDom();
        loadAll();
        addSideButton();
      }, 4000);
    }
  }

  function beautifyWorktime() {
    GM_addStyle(worktimeStyle);

    let timeout = 0;
    let table;
    let interval = setInterval(() => {
      timeout += 1;
      table = document.querySelector('.ReactTable > .rt-table');

      // Stop the circulator when the element is found or after 10 seconds(100*100 ms).
      if (timeout == 100 || table) {
        clearInterval(interval);
        loadAllWorktime();
        choosePlanTime();
        addFilterButton();
      }
    }, 100);
  }

  /**
   * Listen the parent DOM. While the children is changing, and the top board is TaskBoard then hide the panel.
   */
  function listenDom() {
    let parentObj = document.querySelector('div#teambition-web-content>div.project-app-view>div.project-app-inner');
    if (!parentObj) {
      return;
    }
    let innerListener = new MutationObserver(() => {
      setTimeout(() => { hideAddingStatus() }, 4000);
    });

    let listener = new MutationObserver((mutationRecords) => {
      top:
      for (let i = 0; i < mutationRecords.length; i++) {
        let addedNodes = mutationRecords[i].addedNodes;
        let removedNodes = mutationRecords[i].removedNodes;
        for (let j = 0; j < addedNodes.length; j++) {
          if (addedNodes[j].className == 'board-view') {
            // When you enter the task panel, start the task listener.
            setTimeout(() => {
              let innerTaskBoard = document.querySelector('div.board-view > div.board-flex-view > div.board-right-view');
              if (!innerTaskBoard) {
                return;
              }
              hideAddingStatus();
              innerListener.observe(innerTaskBoard, {
                childList: true
              });
            }, 4000);
            break top;
          }
        }
        for (let j = 0; j < removedNodes.length; j++) {
          if (removedNodes[j].className == 'board-view') {
            // When you leave the task panel, stop the task listener.
            innerListener.disconnect();
            break top;
          }
        }
      }
    });
    listener.observe(parentObj, {
      childList: true
    });

    listenTaskBoard();
  }

  /**
   * While the content of the task board changing then hide the adding button.
   */
  function listenTaskBoard() {
    let taskListener = new MutationObserver(() => {
      setTimeout(() => { hideAddingStatus() }, 3000);
    });

    let taskBoard = document.querySelector('div.board-view > div.board-flex-view > div.board-right-view');
    if (taskBoard) {
      hideAddingStatus();
      taskListener.observe(taskBoard, {
        childList: true
      });
    }
  }

  function hideAddingStatus() {
    let status = document.querySelector('div.kanban-single-lane-body-wrapper>div.kanban-single-list:last-child');
    if (!status) {
      return;
    }
    let statusText = status.querySelector('div>div>button>span:last-child');
    if (!statusText) {
      return;
    }
    if ('添加状态' == statusText.textContent) {
      status.style.display = 'none';
    }
  }

  /**
   * Add the button of loading all the content.
   */
  function loadAll() {
    GM_addStyle(loadAllStyle);
    let parent = $('nav.project-navigation.sub-nav.sub-navigator > div.row-flex.nav-container > div.nav-footer');
    if (!parent || parent.length == 0) {
      return;
    }
    let loadAllButton = $('<button type="button" class="loadAll" title="表格视图下,加载所有数据">LoadAll</button>');
    loadAllButton.click(function (e) {
      let test = getLoadButton();
      load(test);
    });
    loadAllButton.insertBefore(parent.children().eq(0));
  }

  /**
   * Load content.
   * @param {Object} element The button of loading.
   */
  function load(element) {
    if (element) {
      element.click();
      setTimeout(() => {
        let test = getLoadButton();
        load(test);
        // load(getLoadButton());
      }, 1000);
    }
  }

  /**
   * Get the button of loading.
   * @returns The button of loading.
   */
  function getLoadButton() {
    let button = null;
    $('div').each((index, element) => {
      if (element.textContent.startsWith('加载更多')) {
        button = element;
        return;
      }
    })
    return button;
  }

  /**
   * Add the side button which used for opening the worktime page.
   */
  function addSideButton() {
    let data = GM_getValue('worktime');
    if (!data || !data?.url || data.url.trim() == '') {
      return;
    }

    let parent = $('#nav_bar-apps > div.nav_bar_detail-apps > ul').last();
    let button = $(`
      <li class="nav_bar_detail-app">
        <span class="next-badge nav_bar_detail-badge">
          <span class="nav_bar_detail-app-icon" style="display: inline-block">
            <span
              style="
                width: 100%;
                height: 100%;
                display: inline-block;
                background-image: url('https://tcs-ga.teambition.net/thumbnail/312ca08e98b657c56f74c8f5eae0b66dd8d2/w/20/h/20');
                background-position: center center;
                background-repeat: no-repeat;
                background-size: 100%;
                border-radius: 4px;
              "
            ></span>
          </span>
          <span></span>
        </span>
        <span class="nav_bar_detail-app-text">工时</span>
      </li>
    `);
    button.click(() => {
      window.open(data?.url, "_blank");
    });
    if (parent.length < 1) {
      return;
    }
    parent.append(button);
  }

  /**
   * Load all of the worktime list.
   */
  function loadAllWorktime() {
    let table = document.querySelector('.ReactTable > .rt-table');
    for (var i = 0; i < loadCount; i++) {
      (function (t) {
        setTimeout(function () {
          table.scroll(0, 10000);
        }, 1000 * t);
      })(i)
    }
    setTimeout(() => {
      table.scroll(0, 0);
    }, loadCount * 1000);

    setTimeout(() => {
      simplifyHeader();
      removeWorktime();
      cleanWord();
      AppendMemberNumber();
      addMemberSelect();
      markAbnormalTime();
    }, (loadCount + 1) * 1000);
  }

  /**
   * Simplify header.
   */
  function simplifyHeader() {
    let spanList = document.querySelectorAll('._38MgU8RZnSct2M6Qs3ZR2i');
    for (let i = 0; i < spanList.length; i++) {
      const text = spanList[i].textContent;
      spanList[i].textContent = text.replace('工时', "");
    }
  }

  /**
   * Add a button for clean the unuseful word.
   */
  function cleanWord() {
    let spans = document.querySelectorAll('._3RNDj8L2GanC2QT53PysSS > span');
    for (let i = 0; i < spans.length; i++) {
      spans[i].innerHTML = spans[i].innerHTML.replace('合计 :&nbsp;&nbsp;', '');
    }
    let total = document.querySelector('.rt-tfoot > .rt-tr > .rt-td > div > div');
    if (total) {
      total.innerHTML = total.innerHTML.replace('总计 :&nbsp;&nbsp;', '');
    }

    let values = document.querySelectorAll('.rt-td span.JqBbgp5bIj-_fWcuSdn8X');
    for (let i = 0; i < values.length; i++) {
      const element = values[i];
      let text = values[i].innerText;
      if (text.endsWith(' 小时')) {
        values[i].innerText = text.replace(' 小时', '');
      }
    }

    document.querySelectorAll('span._3sRT72VQQvRBhWYMmH0hD2').forEach((item) => {
      item.remove();
    });
  }

  /**
   * Append the number of members at the 'All' button.
   */
  function AppendMemberNumber() {
    let button = document.querySelector('.team.cusBtn.all');
    let trList = document.querySelectorAll('.rt-table > .rt-tbody > .rt-tr-group');
    if (!button || !trList) {
      return;
    }
    button.textContent += `(${trList.length})`;

    // Collect member list.
    for (let i = 0; i < trList.length; i++) {
      const nameElement = trList[i].querySelector('.rt-tr > .rt-td:first-child > div > div > div');
      const username = nameElement.textContent;
      members.push(username);
    }
    members.sort((a, b) => { return a.localeCompare(b) });
  }

  /**
   * The Plan Time is selected by default.
   */
  function choosePlanTime() {
    let times = document.querySelectorAll('#root header h4 + div > div');
    if (times.length == 2) {
      times[1].click();
    }
  }

  /**
   * Add the buttons for filtering worktime list.
   */
  function addFilterButton() {
    let parent = document.querySelector('#root header > div');
    if (!parent) {
      return;
    }
    let teamMember = data?.filter;
    if (!teamMember) {
      return;
    }
    const teams = Object.keys(teamMember);
    for (let i = 0; i < teams.length; i++) {
      let button = document.createElement('button');
      button.className = 'team cusBtn';
      button.textContent = `${teams[i]}(${teamMember[teams[i]].length})`;
      button.title = teamMember[teams[i]].join(',');
      button.onclick = function () {
        changeButtonStyle(button.textContent);
        let trList = document.querySelectorAll('.rt-table > .rt-tbody > .rt-tr-group');
        for (let j = 0; j < trList.length; j++) {
          const nameElement = trList[j].querySelector('.rt-tr > .rt-td:first-child > div > div > div');
          const username = nameElement.textContent;
          if (teamMember[teams[i]].indexOf(username) != -1) {
            trList[j].style.display = 'flex';
          } else {
            trList[j].style.display = 'none';
          }
        }
        changeSelectStatus();
      };
      parent.append(button);
    }
    parent.append(createAllButton());
  }

  /**
   * Create the button for displaying all the worktime list.
   */
  function createAllButton() {
    let button = document.createElement('button');
    button.className = 'team cusBtn all';
    button.textContent = 'All';
    button.onclick = function () {
      changeButtonStyle(button.textContent);
      let trList = document.querySelectorAll('.rt-table > .rt-tbody > .rt-tr-group');
      for (let j = 0; j < trList.length; j++) {
        trList[j].style.display = 'flex';
      }
      changeSelectStatus();
    };
    return button;
  }

  /**
   * Change the style of the button.
   * @param {String} button The name of the button.
   */
  function changeButtonStyle(button) {
    let buttons = document.querySelectorAll('button.team.cusBtn');
    for (let i = 0; i < buttons.length; i++) {
      if (buttons[i].textContent == button) {
        addClass(buttons[i], 'selected');
      } else {
        removeClass(buttons[i], 'selected');
      }
    }
  }

  /**
   * Set the first element be selected.
   */
  function changeSelectStatus() {
    let select = document.querySelector('select.cusSelect');
    if (select) {
      select.selectedIndex = 0;
    }
  }

  /**
   * Remove the worktime item in the table.
   */
  function removeWorktime() {
    let removeMembers = data?.remove;
    if (!removeMembers) {
      return;
    }
    let trList = document.querySelectorAll('.rt-table > .rt-tbody > .rt-tr-group');
    for (let i = 0; i < trList.length; i++) {
      const nameElement = trList[i].querySelector('.rt-tr > .rt-td:first-child > div > div > div');
      const username = nameElement.textContent;
      if (removeMembers.indexOf(username) != -1) {
        trList[i].remove();
      }
    }
  }

  /**
   * Add the select element for searching member.
   */
  function addMemberSelect() {
    let parent = document.querySelector('#root header > div');
    if (!parent || members.length == 0) {
      return;
    }
    let select = document.createElement('select');
    select.className = 'cusSelect';

    let hiddenOption = document.createElement('option');
    hiddenOption.style.display = 'none';
    select.append(hiddenOption);

    for (let i = 0; i < members.length; i++) {
      let option = document.createElement('option');
      option.value = members[i];
      option.textContent = members[i];
      select.append(option);
    }
    select.onchange = function () {
      let selectedMember = select.options[select.selectedIndex].value;
      let trList = document.querySelectorAll('.rt-table > .rt-tbody > .rt-tr-group');
      for (let j = 0; j < trList.length; j++) {
        const nameElement = trList[j].querySelector('.rt-tr > .rt-td:first-child > div > div > div');
        const username = nameElement.textContent;
        if (username == selectedMember) {
          trList[j].style.display = 'flex';
        } else {
          trList[j].style.display = 'none';
        }
      }

      // Remove the selected style for all the fileter button.
      let buttons = document.querySelectorAll('button.team.cusBtn');
      for (let i = 0; i < buttons.length; i++) {
        removeClass(buttons[i], 'selected');
      }
    }
    parent.append(select);
  }

  /**
   * Color the abnormal times.
   */
  function markAbnormalTime() {
    let spanList = document.querySelectorAll('.rt-tbody .rt-tr .rt-td:not(:first-child) span.JqBbgp5bIj-_fWcuSdn8X');
    for (let i = 0; i < spanList.length; i++) {
      let timeArr = spanList[i].textContent.split('/');
      const actualTime = new Number(timeArr[0].trim()).toFixed(0);
      const planTime = new Number(timeArr[1].trim()).toFixed(0);
      let innerHTML;
      if (planTime >= planTimeMaximum) {
        innerHTML = `<span>${actualTime}</span><span>&nbsp;/&nbsp;</span><span class="time plan greater">${planTime}</span>`;
      }
      else if (planTime <= planTimeMinimum) {
        innerHTML = `<span>${actualTime}</span><span>&nbsp;/&nbsp;</span><span class="time plan less">${planTime}</span>`;
      }
      else {
        innerHTML = `<span>${actualTime}</span><span>&nbsp;/&nbsp;</span><span class="time plan">${planTime}</span>`;
      }
      spanList[i].innerHTML = innerHTML;
    }
  }
  class Setting {
    constructor() {
      this.init()
    }
    init() {
      let self = this;
      setTimeout(() => {
        if (!document.getElementById('settingPanel')) {
          self.addElement();
          self.addListener();
          GM_addStyle(settingStyle);
        }
        self.loadConfiguration();
        self.show();
      }, 300);
    }

    addElement() {
      let settingPanel = document.createElement('div');
      settingPanel.setAttribute('id', 'settingPanel');
      settingPanel.innerHTML = settingHTML;
      document.body.append(settingPanel);
    }

    addListener() {
      let self = this;
      let panel = document.getElementById('settingPanel');

      document.getElementById('getUrl').onclick = function () {
        let iframes = document.querySelectorAll('iframe');
        if (iframes.length > 0) {
          const url = iframes[0].src;
          if (url.startsWith('https://appshell.teambition.com/api/v1/organization/')) {
            document.querySelector('input.setting.text.url').value = url;
            document.querySelector('span.errMsg.url').textContent = '';
          } else {
            document.querySelector('span.errMsg.url').textContent = '未获取到,请先打开工时应用。';
          }
        } else {
          document.querySelector('span.errMsg.url').textContent = '未检测到iframe,请先打开工时应用。';
        }
      }

      document.getElementById('settingCancel').onclick = function () {
        panel.style.display = 'none';
      }

      document.getElementById('settingSave').onclick = function () {
        if (self.saveConfiguration()) {
          panel.style.display = 'none';
          location.reload();
        }
      }

      document.onkeyup = function (e) {
        var theEvent = e || window.event;
        var code = theEvent.keyCode || theEvent.which || theEvent.charCode;
        // Escape
        if (code == 27 && panel) {
          panel.style.display = 'none';
        }
      };
    }

    loadConfiguration() {
      let data = GM_getValue('worktime');
      if (!data) {
        return;
      }
      const urlValue = data?.url;
      if (urlValue && urlValue.trim() != '') {
        document.querySelector('.setting.text.url').value = urlValue;
      }
      const filterValue = data?.filter;
      if (filterValue) {
        document.querySelector('.setting.text.filter').value = JSON.stringify(filterValue);
      }
      const removeValue = data?.remove;
      if (removeValue) {
        document.querySelector('.setting.text.remove').value = JSON.stringify(removeValue);
      }
      document.querySelector('span.errMsg.url').textContent = '';
      document.querySelector('span.errMsg.filter').textContent = '';
      document.querySelector('span.errMsg.remove').textContent = '';
    }

    saveConfiguration() {
      let urlEle = document.querySelector('.setting.text.url');
      let filterEle = document.querySelector('.setting.text.filter');
      let filterValue;
      try {
        filterValue = JSON.parse(filterEle.value);
      } catch (error) {
        console.error(error);
        document.querySelector('span.errMsg.filter').textContent = 'JSON格式不正确,请修改后重试!';
        return false;
      }
      document.querySelector('span.errMsg.filter').textContent = '';

      let removeEle = document.querySelector('.setting.text.remove');
      let removeValue;
      try {
        removeValue = JSON.parse(removeEle.value);
      } catch (error) {
        console.error(error);
        document.querySelector('span.errMsg.remove').textContent = '数组格式不正确,请修改后重试!';
        return false;
      }
      document.querySelector('span.errMsg.remove').textContent = '';
      const data = {
        'url': urlEle.value,
        'filter': filterValue,
        'remove': removeValue
      }
      GM_setValue('worktime', data);
      return true;
    }

    show() {
      document.getElementById('settingPanel').style.display = 'flex';
    }
  }

  /**
   * Add class name for the element.
   * @param {Object} element The DOM elment object.
   * @param {String} value the class name of the element.
   */
  function addClass(element, value) {
    if (!element.className) {
      element.className = value;
    } else if (element.className.indexOf(value) == -1) {
      let newClassName = element.className;
      newClassName += " ";
      newClassName += value;
      element.className = newClassName;
    }
  }

  /**
   * Remove class name for the element.
   * @param {Object} element The DOM elment object.
   * @param {String} value the class name of the element.
   */
  function removeClass(element, value) {
    if (element.className) {
      let newClassName = element.className;
      newClassName = newClassName.replace(value, '');
      newClassName = newClassName.trim();
      element.className = newClassName;
    }
  }

  /**
   * Log the title and version at the front of the console.
   * @param {String} title title.
   * @param {String} version script version.
   */
  function logInfo(title, version) {
    console.clear();
    const titleStyle = 'color:white;background-color:#606060';
    const versionStyle = 'color:white;background-color:#1475b2';
    const logTitle = ' ' + title + ' ';
    const logVersion = ' ' + version + ' ';
    console.log('%c' + logTitle + '%c' + logVersion, titleStyle, versionStyle);
  }
})();