Skip to content

Commit cf77ea9

Browse files
f15utim-smart
authored andcommittedMar 20, 2025
feat: add Array.window (#4477)
1 parent ffaa3f3 commit cf77ea9

File tree

4 files changed

+60
-0
lines changed

4 files changed

+60
-0
lines changed
 

‎.changeset/tender-kangaroos-kiss.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"effect": minor
3+
---
4+
5+
Add `Array.window` function

‎packages/effect/dtslint/Array.tst.ts

+12
Original file line numberDiff line numberDiff line change
@@ -942,6 +942,18 @@ describe("Array", () => {
942942
.type.toBe<[[string, ...Array<string>], ...Array<[string, ...Array<string>]>]>()
943943
})
944944

945+
it("window", () => {
946+
// Array
947+
expect(Array.window(strings, 2)).type.toBe<Array<Array<string>>>()
948+
expect(pipe(strings, Array.window(2))).type.toBe<Array<Array<string>>>()
949+
expect(Array.window(2)(strings)).type.toBe<Array<Array<string>>>()
950+
951+
// NonEmptyArray
952+
expect(Array.window(nonEmptyStrings, 2)).type.toBe<Array<Array<string>>>()
953+
expect(pipe(nonEmptyStrings, Array.window(2))).type.toBe<Array<Array<string>>>()
954+
expect(Array.window(2)(nonEmptyStrings)).type.toBe<Array<Array<string>>>()
955+
})
956+
945957
it("reverse", () => {
946958
// Array
947959
expect(Array.reverse(strings)).type.toBe<Array<string>>()

‎packages/effect/src/Array.ts

+31
Original file line numberDiff line numberDiff line change
@@ -1994,6 +1994,37 @@ export const chunksOf: {
19941994
return []
19951995
})
19961996

1997+
/**
1998+
* Creates sliding windows of size `n` from an `Iterable`.
1999+
* If the number of elements is less than `n` or if `n` is not greater than zero,
2000+
* an empty array is returned.
2001+
*
2002+
* @example
2003+
* ```ts
2004+
* import { Array } from "effect"
2005+
*
2006+
* const numbers = [1, 2, 3, 4, 5]
2007+
* assert.deepStrictEqual(Array.window(numbers, 3), [[1, 2, 3], [2, 3, 4], [3, 4, 5]])
2008+
* assert.deepStrictEqual(Array.window(numbers, 6), [])
2009+
* ```
2010+
*
2011+
* @category splitting
2012+
* @since 3.13.2
2013+
*/
2014+
export const window: {
2015+
(n: number): <A>(self: Iterable<A>) => Array<Array<A>>
2016+
<A>(self: Iterable<A>, n: number): Array<Array<A>>
2017+
} = dual(2, <A>(self: Iterable<A>, n: number): Array<Array<A>> => {
2018+
const input = fromIterable(self)
2019+
if (n > 0 && isNonEmptyReadonlyArray(input)) {
2020+
return Array.from(
2021+
{ length: input.length - (n - 1) },
2022+
(_, index) => input.slice(index, index + n)
2023+
)
2024+
}
2025+
return []
2026+
})
2027+
19972028
/**
19982029
* Group equal, consecutive elements of a `NonEmptyReadonlyArray` into `NonEmptyArray`s using the provided `isEquivalent` function.
19992030
*

‎packages/effect/test/Array.test.ts

+12
Original file line numberDiff line numberDiff line change
@@ -956,6 +956,18 @@ describe("Array", () => {
956956
assertSingleChunk(Arr.make(1, 2), 3)
957957
})
958958

959+
it("window", () => {
960+
deepStrictEqual(Arr.window(2)([]), [])
961+
962+
deepStrictEqual(Arr.window(2)([1, 2, 3, 4, 5]), [[1, 2], [2, 3], [3, 4], [4, 5]])
963+
deepStrictEqual(Arr.window(3)([1, 2, 3, 4, 5]), [[1, 2, 3], [2, 3, 4], [3, 4, 5]])
964+
965+
// n out of bounds
966+
deepStrictEqual(Arr.window([1, 2, 3, 4, 5], 6), [])
967+
deepStrictEqual(Arr.window([1, 2, 3, 4, 5], 0), [])
968+
deepStrictEqual(Arr.window([1, 2, 3, 4, 5], -1), [])
969+
})
970+
959971
it("min", () => {
960972
deepStrictEqual(Arr.min(Num.Order)([2, 1, 3]), 1)
961973
deepStrictEqual(Arr.min(Num.Order)([3]), 3)

0 commit comments

Comments
 (0)
Please sign in to comment.