Skip to content

Commit b378a21

Browse files
committed
Added stream insertion operator.
1 parent 844e4bf commit b378a21

File tree

3 files changed

+85
-1
lines changed

3 files changed

+85
-1
lines changed

api/Stream.h

+32-1
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,37 @@ class Stream : public Print
128128

129129
#undef NO_IGNORE_CHAR
130130

131+
// Structure to store a value and a modifier.
132+
template <class T>
133+
struct Format {
134+
T data;
135+
int modifier;
136+
};
137+
138+
// Helper function that creates a `Format` object.
139+
template <class T>
140+
Format<T> format(T const data, int const modifier) {
141+
Format<T> fmt {data, modifier};
142+
return fmt;
143+
}
144+
145+
// Stream insertion operator for plain data types.
146+
template <class T>
147+
Stream& operator <<(Stream& stream, T const data) {
148+
stream.print(data);
149+
return stream;
150+
}
151+
152+
// Stream insertion operator with modifiers (e.g., BIN, HEX, number of digits, etc.).
153+
template <class T>
154+
Stream& operator <<(Stream& stream, Format<T> const& parameters) {
155+
stream.print(parameters.data, parameters.modifier);
156+
return stream;
157+
}
158+
131159
}
132160

133-
using arduino::Stream;
161+
using arduino::Stream;
162+
using arduino::Format;
163+
using arduino::format;
164+
using arduino::operator <<;

test/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@ set(TEST_SRCS
5555
src/Stream/test_find.cpp
5656
src/Stream/test_findUntil.cpp
5757
src/Stream/test_getTimeout.cpp
58+
src/Stream/test_insertion_operator.cpp
5859
src/Stream/test_parseFloat.cpp
5960
src/Stream/test_parseInt.cpp
6061
src/Stream/test_readBytes.cpp
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
/*
2+
* Copyright (c) 2020 Arduino. All rights reserved.
3+
*/
4+
5+
/**************************************************************************************
6+
* INCLUDE
7+
**************************************************************************************/
8+
9+
#include <catch.hpp>
10+
11+
#include <StreamMock.h>
12+
13+
/**************************************************************************************
14+
* TEST CODE
15+
**************************************************************************************/
16+
17+
TEST_CASE ("Testing 'Format' initialisation", "[Stream-insertion-operator-01]") {
18+
Format<char> fmt {'a', 2};
19+
REQUIRE(fmt.data == 'a');
20+
REQUIRE(fmt.modifier == 2);
21+
}
22+
23+
TEST_CASE ("Testing 'format' helper function", "[Stream-insertion-operator-02]") {
24+
Format<char> fmt {format('a', 2)};
25+
REQUIRE(fmt.data == 'a');
26+
REQUIRE(fmt.modifier == 2);
27+
}
28+
29+
TEST_CASE ("Testing basic insertion operator", "[Stream-insertion-operator-03]") {
30+
StreamMock mock;
31+
mock << 'a' << 12 << 'b' << 34; // Note we cannot test C strings because `StreamMock` has its own << operator.
32+
REQUIRE(mock.available() == 6);
33+
34+
char buf[10];
35+
mock.readBytes(buf, 6);
36+
REQUIRE(not strncmp(buf, "a12b34", 6));
37+
}
38+
39+
TEST_CASE ("Testing insertion operator with modifiers", "[Stream-insertion-operator-04]") {
40+
StreamMock mock;
41+
mock << format(1.2, 4);
42+
REQUIRE(mock.available() == 6);
43+
44+
char buf[10];
45+
mock.readBytes(buf, 6);
46+
REQUIRE(not strncmp(buf, "1.2000", 6));
47+
48+
mock << format(12, BIN);
49+
REQUIRE(mock.available() == 4);
50+
mock.readBytes(buf, 4);
51+
REQUIRE(not strncmp(buf, "1100", 4));
52+
}

0 commit comments

Comments
 (0)