string_literals/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
//! A very tiny crate with Rust macros to more easily create String types.
//!
//! When creating string literals in Rust, the given type is `&str`. To create an owned `String` type,
//! you either need to:
//!
//! - pass in the literal to `String::from()`,
//! - call `.to_owned()`, or `.to_string()` on the `&str` literal
//!
//! ### Strings
//!
//! ```rust
//! use string_literals::s;
//!
//! let old = "Hello, world!".to_owned();
//! let new = s!("Hello, world!");
//! ```
//!
//! ### Arrays, vectors
//!
//! ```rust
//! use string_literals::{string_arr, string_vec};
//!
//! let old_arr: [String; 3] = ["Banana".to_owned(), "Apple".to_owned(), "Orange".to_owned()];
//! let new_arr: [String; 3] = string_arr!["Banana", "Apple", "Orange"];
//!
//! let old_vec = vec!["Banana".to_owned(), "Apple".to_owned(), "Orange".to_owned()];
//! let new_vec = string_vec!["Banana", "Apple", "Orange"];
//! ```
//!
//! ### Hash maps
//!
//! ```rust
//! use std::collections::HashMap;
//! use string_literals::string_hashmap;
//!
//! let mut old1: HashMap<String, String> = HashMap::new();
//! old1.insert("Banana".to_owned(), "Good".to_owned());
//! old1.insert("Apple".to_owned(), "Okay".to_owned());
//!
//! let old2: HashMap<String, String> = HashMap::from([
//!     ("Banana".to_owned(), "Good".to_owned()),
//!     ("Apple".to_owned(), "Okay".to_owned()),
//! ]);
//!
//! let new: HashMap<String, String> = string_hashmap! {
//!     "Banana" => "Good",
//!     "Apple" => "Okay",
//! };
//! ```

/// Create a [`String`] literal
///
/// This provides a slightly shorter alternative to creating a `String`
/// from a literal.
///
/// # Examples
/// Empty strings:
/// ```
/// use string_literals::s;
///
/// let old = String::new();
/// let s1 = s!();
/// let s2 = s!("");
/// assert!(s1.is_empty());
/// assert!(s2.is_empty());
/// ```
///
/// Non-empty strings:
/// ```
/// use string_literals::s;
///
/// let old1 = String::from("Alice");
/// let old2 = "Alice".to_owned();
/// let old3 = "Alice".to_string();
/// let new  = s!("Alice");
/// assert_eq!(new, String::from("Alice"));
/// ```
#[macro_export]
macro_rules! s {
	() => {
		String::new()
	};
	("") => {
		String::new()
	};
	($s: expr) => {
		$s.to_owned()
	};
}

/// Create an [`array`] of `[String; N]` with string literals
///
/// # Examples
/// Empty arrays:
/// ```
/// use string_literals::string_arr;
///
/// let arr: [String; 0] = string_arr![];
/// assert!(arr.is_empty());
/// ```
///
/// Non-empty arrays:
/// ```
/// use string_literals::string_arr;
///
/// let old: [String; 2] = [String::from("Alice"), String::from("Bob")];
/// let new: [String; 2] = string_arr!["Alice", "Bob"];
/// assert_eq!(new.len(), 2);
/// assert_eq!(new[0], String::from("Alice"));
/// assert_eq!(new[1], String::from("Bob"));
/// ```
#[macro_export]
macro_rules! string_arr {
	() => {
		[]
	};
	($($x:expr),+ $(,)?) => {
		[$($x.to_owned()),*]
	}
}

/// Create a [`Vec`] of `Vec<String>` with string literals.
///
/// This macro also allows zero arguments. In this case however, it would be
/// shorter to call `vec![]` or `Vec::new()`.
///
/// # Examples
/// ```
/// use string_literals::string_vec;
///
/// let old: Vec<String> = vec![String::from("Alice"), String::from("Bob")];
/// let new: Vec<String> = string_vec!["Alice", "Bob"];
/// assert_eq!(new.len(), 2);
/// assert_eq!(new[0], "Alice".to_owned());
/// assert_eq!(new[1], "Bob".to_owned());
/// ```
#[macro_export]
macro_rules! string_vec {
	() => {
		Vec::new()
	};
	($($x:expr),+ $(,)?) => {
		vec![$($x.to_owned()),+]
	};
}

/// Create a [`HashMap`](std::collections::HashMap) of `HashMap<String, String>` with string literals
///
/// This macro also allows zero arguments (an empty hash map). In this case
/// however, it would be shorter in length to call `HashMap::new()`.
///
/// # Examples
/// Empty hash maps:
/// ```
/// use std::collections::HashMap;
/// use string_literals::string_hashmap;
///
/// let map: HashMap<String, String> = string_hashmap!{};
/// assert!(map.is_empty());
/// ```
///
/// Non-empty hash maps:
/// ```
/// use std::collections::HashMap;
/// use string_literals::string_hashmap;
///
/// let map: HashMap<String, String> = string_hashmap!{
///     "Banana" => "Good",
///     "Apple" => "Okay",
/// };
/// assert_eq!(map[&"Banana".to_owned()], "Good".to_owned());
/// assert_eq!(map[&"Apple".to_owned()], "Okay".to_owned());
/// ```
#[macro_export]
macro_rules! string_hashmap {
	() => {
		::std::collections::HashMap::new()
	};
	($($k:expr => $v:expr),+ $(,)?) => {
		::std::collections::HashMap::from([
			$(($k.to_owned(), $v.to_owned())),+
		])
	};
}