@pisell/pisellos
Version:
一个可扩展的前端模块化SDK框架,支持插件系统
224 lines (207 loc) • 9.62 kB
JavaScript
function _createForOfIteratorHelper(o, allowArrayLike) { var it = typeof Symbol !== "undefined" && o[Symbol.iterator] || o["@@iterator"]; if (!it) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = it.call(o); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it.return != null) it.return(); } finally { if (didErr) throw err; } } }; }
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; return arr2; }
import dayjs from "dayjs";
/**
* 计算资源在指定时间段内的总可用时间(以分钟为单位)
* @param resource 需要计算可用时间的资源
* @param timeSlots 要检查可用性的时间段
* @param currentCapacity 当前预约所需的容量
* @returns 总可用时间(分钟)
*/
export function calculateResourceAvailableTime(_ref) {
var resource = _ref.resource,
timeSlots = _ref.timeSlots,
_ref$currentCapacity = _ref.currentCapacity,
currentCapacity = _ref$currentCapacity === void 0 ? 1 : _ref$currentCapacity;
// 过滤出与给定日期相同的资源时间
var matchingTimes = resource.times.filter(function (time) {
return dayjs(time.start_at).isSame(dayjs(timeSlots.start_at), 'day');
});
if (matchingTimes.length === 0) return 0;
// 计算所有时间段与目标时间槽的重叠部分
var overlaps = [];
var _iterator = _createForOfIteratorHelper(matchingTimes),
_step;
try {
for (_iterator.s(); !(_step = _iterator.n()).done;) {
var time = _step.value;
// 计算实际重叠的时间段
var overlapStart = dayjs(time.start_at).isAfter(dayjs(timeSlots.start_at)) ? dayjs(time.start_at) : dayjs(timeSlots.start_at);
var overlapEnd = dayjs(time.end_at).isBefore(dayjs(timeSlots.end_at)) ? dayjs(time.end_at) : dayjs(timeSlots.end_at);
// 只有当重叠时间段有效时才添加
if (overlapStart.isBefore(overlapEnd)) {
overlaps.push({
start: overlapStart,
end: overlapEnd
});
}
}
// 合并重叠的时间段,计算并集
} catch (err) {
_iterator.e(err);
} finally {
_iterator.f();
}
if (overlaps.length === 0) return 0;
// 按开始时间排序
overlaps.sort(function (a, b) {
return a.start.diff(b.start);
});
// 合并重叠的时间段
var merged = [];
var current = overlaps[0];
for (var i = 1; i < overlaps.length; i++) {
var next = overlaps[i];
// 如果当前时间段与下一个时间段重叠或相邻,则合并
if (current.end.isSameOrAfter(next.start)) {
current.end = current.end.isAfter(next.end) ? current.end : next.end;
} else {
// 不重叠,添加当前时间段到结果中,开始处理下一个
merged.push(current);
current = next;
}
}
merged.push(current);
// 计算所有合并后时间段的总时长
var totalAvailableMinutes = 0;
for (var _i = 0, _merged = merged; _i < _merged.length; _i++) {
var segment = _merged[_i];
totalAvailableMinutes += segment.end.diff(segment.start, 'minute');
}
return totalAvailableMinutes;
}
/**
* 查找最快可用的资源,如果有多个资源在相同时间点可用,则选择空闲时间最长的资源
* @param resources 资源列表
* @param currentCapacity 当前预约所需的容量
* @param countMap 已预约数量映射
* @returns 最快可用的资源
*/
export function findFastestAvailableResource(_ref2) {
var resources = _ref2.resources,
_ref2$currentCapacity = _ref2.currentCapacity,
currentCapacity = _ref2$currentCapacity === void 0 ? 1 : _ref2$currentCapacity,
_ref2$countMap = _ref2.countMap,
countMap = _ref2$countMap === void 0 ? {} : _ref2$countMap;
var currentTime = dayjs();
var fastestTime = null;
var fastestResources = [];
// 遍历所有资源,找到最快可用的时间点
var _iterator2 = _createForOfIteratorHelper(resources),
_step2;
try {
for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
var _resource = _step2.value;
// 获取资源当天的时间段
var todayTimes = _resource.times.filter(function (time) {
return dayjs(time.start_at).isSame(currentTime, 'day');
});
if (todayTimes.length === 0) continue;
// 按开始时间排序
todayTimes.sort(function (a, b) {
return dayjs(a.start_at).diff(dayjs(b.start_at));
});
// 找到第一个可用的时间段
var _iterator3 = _createForOfIteratorHelper(todayTimes),
_step3;
try {
var _loop = function _loop() {
var time = _step3.value;
var startTime = dayjs(time.start_at);
// 如果开始时间在当前时间之前,跳过
if (startTime.isBefore(currentTime)) return 0; // continue
// 检查这个时间段是否可用
if (_resource.resourceType === 'single') {
var _time$event_list;
// 单个预约类型:检查是否有预约
var hasBooking = (_time$event_list = time.event_list) === null || _time$event_list === void 0 ? void 0 : _time$event_list.some(function (event) {
return dayjs(event.start_at).isSame(startTime);
});
if (!hasBooking) {
if (!fastestTime || startTime.isSame(fastestTime)) {
fastestTime = startTime;
fastestResources.push(_resource);
} else if (startTime.isBefore(fastestTime)) {
fastestTime = startTime;
fastestResources = [_resource];
}
return 1; // break
}
} else {
var _time$event_list2;
// 多个预约类型:检查容量
var totalCapacity = _resource.capacity || 0;
var usedCapacity = ((_time$event_list2 = time.event_list) === null || _time$event_list2 === void 0 ? void 0 : _time$event_list2.reduce(function (sum, event) {
return dayjs(event.start_at).isSame(startTime) ? sum + (event.pax || 0) : sum;
}, 0)) || 0;
var remainingCapacity = totalCapacity - usedCapacity;
if (remainingCapacity >= (countMap[_resource.id] || 0) + currentCapacity) {
if (!fastestTime || startTime.isSame(fastestTime)) {
fastestTime = startTime;
fastestResources.push(_resource);
} else if (startTime.isBefore(fastestTime)) {
fastestTime = startTime;
fastestResources = [_resource];
}
return 1; // break
}
}
},
_ret;
for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
_ret = _loop();
if (_ret === 0) continue;
if (_ret === 1) break;
}
} catch (err) {
_iterator3.e(err);
} finally {
_iterator3.f();
}
}
// 如果没有找到可用资源,返回null
} catch (err) {
_iterator2.e(err);
} finally {
_iterator2.f();
}
if (!fastestTime || fastestResources.length === 0) return null;
// 如果只有一个最快可用的资源,直接返回
if (fastestResources.length === 1) return fastestResources[0];
// 如果有多个最快可用的资源,比较它们的空闲时间
var maxIdleTime = 0;
var selectedResource = null;
for (var _i2 = 0, _fastestResources = fastestResources; _i2 < _fastestResources.length; _i2++) {
var resource = _fastestResources[_i2];
var idleTime = calculateResourceAvailableTime({
resource: resource,
timeSlots: {
start_time: fastestTime.format('HH:mm'),
end_time: fastestTime.format('HH:mm'),
start_at: fastestTime,
end_at: fastestTime
},
currentCapacity: (countMap[resource.id] || 0) + currentCapacity
});
if (idleTime > maxIdleTime) {
maxIdleTime = idleTime;
selectedResource = resource;
}
}
return selectedResource;
}
/**
* 给定一个时间列表,通过开始和结束时间过滤出符合条件的时间段
*
* @export
* @param {TimeSliceItem[]} times
* @param {Dayjs} startTime
* @param {Dayjs} endTime
* @return {*}
*/
export function filterConditionTimeSlots(times, startTime, endTime) {
return times.filter(function (n) {
return !dayjs(n.start_at).isAfter(dayjs(startTime)) && !dayjs(n.end_at).isBefore(dayjs(endTime));
});
}