通义听语文件排序

实现对通义听语的列表和导入阿里云文件时进行排序

// ==UserScript==
// @name         通义听语文件排序
// @namespace    https://ymjin.blog.csdn.net/
// @version      0.2
// @license      AGPL License
// @description  实现对通义听语的列表和导入阿里云文件时进行排序
// @author       明金同学
// @match        *://tingwu.aliyun.com/*
// @icon         
// @grant        none
// ==/UserScript==

(function () {
  'use strict';
  // 声明排序方向变量,默认为升序
  let sortDirection = 'asc';
  // 定义一个标志变量,用于标识当前的排序状态,初始值为 true,代表升序
  var ascending = true;
  // 定义一个标志变量,用于标识监听器是否已经添加,初始值为 false
  var listenerAdded = false;

  // 定义我们的排序和监听器添加函数
  function sortAndListen (element) {
    // 点击事件
    element.addEventListener('click', function () {
      // 1. 选取元素
      var elements = document.querySelectorAll('.ant-list-item');

      // 2. 放入数组
      var elementsArray = Array.from(elements);

      // 3. 排序
      elementsArray.sort(function (a, b) {
        var aTitle = a.querySelector('.ant-list-item-meta-title div').textContent;
        var bTitle = b.querySelector('.ant-list-item-meta-title div').textContent;
        // 根据 ascending 变量的值决定是升序还是降序
        return ascending ? aTitle.localeCompare(bTitle) : bTitle.localeCompare(aTitle);
      });

      // 4. 获取元素的父容器并插入排序后的元素
      var container = elements[0].parentNode;
      elementsArray.forEach(function (element) {
        container.appendChild(element);
      });

      // 每次点击后,反转排序状态
      ascending = !ascending;
    });

    // 设置监听器已经添加的标志
    listenerAdded = true;
  }
  // MutationObserver 回调
  function callback (mutationsList, observera) {
    // 检查是否已经添加了监听器,如果已经添加了,那么就断开 observera
    if (listenerAdded) {
      observera.disconnect();
      return;
    }
    // 检查每一个 mutation
    for (let mutation of mutationsList) {
      if (mutation.type === 'childList') {
        var element = document.querySelector('.sc-keuYuY.ioFVbe');
        if (element) {
          sortAndListen(element);
          break;
        }
      }
    }
  }

  function sortElements (selector, attribute) {
    // 选取元素
    var elements = document.querySelectorAll('.sc-kBRoID.bnHVRk');
    if (!elements.length) {
      console.error('No elements found for selector ".sc-kBRoID.bnHVRk"');
      return;
    }

    // 放入数组
    var elementsArray = Array.from(elements);

    // 从数组中移除.sc-eZYNyq fLbdNx
    var excludeElement = document.querySelector('.sc-eZYNyq.fLbdNx');
    if (excludeElement) {
      var index = elementsArray.indexOf(excludeElement);
      if (index > -1) {
        elementsArray.splice(index, 1);
      }
    }

    // 排序
    elementsArray.sort(function (a, b) {
      var aAttribute = a.querySelector(selector).textContent;
      var bAttribute = b.querySelector(selector).textContent;

      // 如果选择的是时长,将其转换为秒
      if (selector === '.sc-iLLODe.dyILtT') {
        aAttribute = aAttribute.split(':').reduce((acc, time) => (60 * acc) + +time);
        bAttribute = bAttribute.split(':').reduce((acc, time) => (60 * acc) + +time);
      }
      // 如果选择的是文件大小,将其转换为数字
      else if (selector === '.sc-iLLODe.dyILwY') {
        aAttribute = parseFloat(aAttribute);
        bAttribute = parseFloat(bAttribute);
      }
      // 如果选择的是创建时间,将其转换为Date对象
      else if (selector === '.sc-iLLODe.dyIKAj') {
        const today = new Date().toISOString().split('T')[0];
        aAttribute = new Date(aAttribute.replace('今天', today));
        bAttribute = new Date(bAttribute.replace('今天', today));
      }

      // 根据sortDirection变量选择排序方向
      if (sortDirection === 'asc') {
        return aAttribute > bAttribute ? 1 : -1;
      } else {
        return bAttribute > aAttribute ? 1 : -1;
      }
    });

    // 获取元素的父容器并插入排序后的元素
    var container = document.querySelector('.sc-kBRoID.bnHVRk').parentNode;
    if (!container) {
      console.error('No parent container found for selector ".sc-kBRoID.bnHVRk"');
      return;
    }

    elementsArray.forEach(function (element) {
      container.appendChild(element);
    });

    // 最后再将被排除的元素添加回来
    if (excludeElement) {
      container.appendChild(excludeElement);
    }

    // 每次排序后更改排序方向
    sortDirection = sortDirection === 'asc' ? 'desc' : 'asc';
  }

  // 选取排序元素并添加点击事件监听器
  function addClickListener () {
    ['.sc-bhqpjJ.LwBoU', '.sc-iLLODe.dyILtT', '.sc-iLLODe.dyILwY', '.sc-iLLODe.dyIKAj'].forEach(selector => {
      var sortElement = document.querySelector(selector);
      if (!sortElement) {
        console.error(`No element found for selector "${selector}"`);
      } else {
        // 先移除之前可能添加的监听器,避免循环添加
        sortElement.removeEventListener('click', () => sortElements(selector));
        // 添加监听器
        sortElement.addEventListener('click', () => sortElements(selector));
      }
    });
  }

  // 创建一个 observera 实例
  const observera = new MutationObserver(callback);

  // 配置 observera:
  const config = { attributes: true, childList: true, subtree: true };

  // 开始观察:
  observera.observe(document.body, config);

  // 创建观察器实例
  var observer = new MutationObserver(function (mutations) {
    mutations.forEach(function (mutation) {
      // 如果没有新增节点,则返回
      if (!mutation.addedNodes) return

      for (var i = 0; i < mutation.addedNodes.length; i++) {
        // 如果新增的节点是我们需要的.sc-EgOXT ijozuf元素
        if (mutation.addedNodes[i].classList.contains('sc-EgOXT') && mutation.addedNodes[i].classList.contains('ijozuf')) {
          // 添加点击事件监听器
          addClickListener();
          break;
        }
      }
    })
  });

  // 传入观察目标对象及观察选项
  observer.observe(document.body, {
    childList: true, // 子节点的变动(指新增,删除或者更改)
    subtree: true, // 所有后代节点的变动
    attributes: false, // 属性的变动
    characterData: false // 数据的变动
  });

})();