Это можно решить с помощью findIndex
.В следующем примере сортировка будет помещать Mail
первым согласно массиву prevRates
:
const rates = [
{name:'UPS', isChoosen:true, cost:63},
{name:'Mail', isChoosen:false, cost:23},
{name:'FedEx', isChoosen:false, cost:33}
];
const prevRates = [
{name: 'Mail'},
{name: 'UPS'},
{name: 'FedEx'},
];
let keepRateOrder = rates.sort((a, b) => {
return prevRates.findIndex(p => p.name === a.name) - prevRates.findIndex(p => p.name === b.name);
});
console.log(keepRateOrder);
Вы можете сделать то же самое с indexOf
, если сначала вы map
.Это приводит к несколько более чистому коду:
const rates = [
{name:'UPS', isChoosen:true, cost:63},
{name:'Mail', isChoosen:false, cost:23},
{name:'FedEx', isChoosen:false, cost:33}
];
const prevRates = [
{name: 'Mail'},
{name: 'UPS'},
{name: 'FedEx'},
].map(x => x.name);
let keepRateOrder = rates.sort((a, b) => {
return prevRates.indexOf(a.name) - prevRates.indexOf(b.name);
});
console.log(keepRateOrder);
И вот еще одно решение, использующее хэш исходных индексов, созданный с использованием reduceRight
:
const rates = [
{name:'UPS', isChoosen:true, cost:63},
{name:'Mail', isChoosen:false, cost:23},
{name:'FedEx', isChoosen:false, cost:33}
]
const prevRates = [
{name: 'Mail'},
{name: 'UPS'},
{name: 'FedEx'},
].reduceRight((a, x, i) => (a[x.name] = i, a), {});
let keepRateOrder = rates.sort((a, b) => {
return prevRates[a.name] - prevRates[b.name];
});
console.log(keepRateOrder);
Поскольку вы сказали, что элементы могут быть добавлены или удалены из исходного массива, обратите внимание, что все вышеперечисленные решения будут размещать новые элементы первыми (так как их индекс в prevRates
массив будет возвращен как -1
).Если вы хотите, чтобы новые элементы появлялись в конце, вам нужно сделать что-то вроде этого:
const rates = [
{name:'UPS', isChoosen:true, cost:63},
{name:'Mail', isChoosen:false, cost:23},
{name:'Foo'},
{name:'FedEx', isChoosen:false, cost:33},
];
const prevRates = [
{name: 'Mail'},
{name: 'UPS'},
{name: 'FedEx'},
].map(x => x.name);
let keepRateOrder = rates.sort((a, b) => {
const aIndex = prevRates.indexOf(a.name);
const bIndex = prevRates.indexOf(b.name);
return (aIndex === -1 ? Number.MAX_VALUE : aIndex) - (bIndex === -1 ? Number.MAX_VALUE : bIndex);
});
console.log(keepRateOrder);