sassc4 2.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (216) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +18 -0
  3. data/.gitmodules +3 -0
  4. data/.travis.yml +16 -0
  5. data/CHANGELOG.md +97 -0
  6. data/CODE_OF_CONDUCT.md +10 -0
  7. data/Gemfile +2 -0
  8. data/LICENSE.txt +22 -0
  9. data/README.md +80 -0
  10. data/Rakefile +51 -0
  11. data/ext/depend +4 -0
  12. data/ext/extconf.rb +92 -0
  13. data/ext/libsass/VERSION +1 -0
  14. data/ext/libsass/contrib/plugin.cpp +60 -0
  15. data/ext/libsass/include/sass/base.h +97 -0
  16. data/ext/libsass/include/sass/context.h +174 -0
  17. data/ext/libsass/include/sass/functions.h +139 -0
  18. data/ext/libsass/include/sass/values.h +145 -0
  19. data/ext/libsass/include/sass/version.h +12 -0
  20. data/ext/libsass/include/sass.h +15 -0
  21. data/ext/libsass/include/sass2scss.h +120 -0
  22. data/ext/libsass/src/MurmurHash2.hpp +91 -0
  23. data/ext/libsass/src/ast.cpp +953 -0
  24. data/ext/libsass/src/ast.hpp +1064 -0
  25. data/ext/libsass/src/ast2c.cpp +80 -0
  26. data/ext/libsass/src/ast2c.hpp +39 -0
  27. data/ext/libsass/src/ast_def_macros.hpp +140 -0
  28. data/ext/libsass/src/ast_fwd_decl.cpp +31 -0
  29. data/ext/libsass/src/ast_fwd_decl.hpp +274 -0
  30. data/ext/libsass/src/ast_helpers.hpp +316 -0
  31. data/ext/libsass/src/ast_sel_cmp.cpp +396 -0
  32. data/ext/libsass/src/ast_sel_super.cpp +539 -0
  33. data/ext/libsass/src/ast_sel_unify.cpp +275 -0
  34. data/ext/libsass/src/ast_sel_weave.cpp +616 -0
  35. data/ext/libsass/src/ast_selectors.cpp +1070 -0
  36. data/ext/libsass/src/ast_selectors.hpp +523 -0
  37. data/ext/libsass/src/ast_supports.cpp +114 -0
  38. data/ext/libsass/src/ast_supports.hpp +121 -0
  39. data/ext/libsass/src/ast_values.cpp +1154 -0
  40. data/ext/libsass/src/ast_values.hpp +498 -0
  41. data/ext/libsass/src/b64/cencode.h +32 -0
  42. data/ext/libsass/src/b64/encode.h +79 -0
  43. data/ext/libsass/src/backtrace.cpp +50 -0
  44. data/ext/libsass/src/backtrace.hpp +29 -0
  45. data/ext/libsass/src/base64vlq.cpp +47 -0
  46. data/ext/libsass/src/base64vlq.hpp +30 -0
  47. data/ext/libsass/src/bind.cpp +312 -0
  48. data/ext/libsass/src/bind.hpp +15 -0
  49. data/ext/libsass/src/c2ast.cpp +64 -0
  50. data/ext/libsass/src/c2ast.hpp +14 -0
  51. data/ext/libsass/src/c99func.c +54 -0
  52. data/ext/libsass/src/cencode.c +106 -0
  53. data/ext/libsass/src/check_nesting.cpp +393 -0
  54. data/ext/libsass/src/check_nesting.hpp +70 -0
  55. data/ext/libsass/src/color_maps.cpp +652 -0
  56. data/ext/libsass/src/color_maps.hpp +323 -0
  57. data/ext/libsass/src/color_spaces.cpp +241 -0
  58. data/ext/libsass/src/color_spaces.hpp +227 -0
  59. data/ext/libsass/src/constants.cpp +199 -0
  60. data/ext/libsass/src/constants.hpp +200 -0
  61. data/ext/libsass/src/context.cpp +870 -0
  62. data/ext/libsass/src/context.hpp +140 -0
  63. data/ext/libsass/src/cssize.cpp +521 -0
  64. data/ext/libsass/src/cssize.hpp +71 -0
  65. data/ext/libsass/src/dart_helpers.hpp +199 -0
  66. data/ext/libsass/src/debug.hpp +43 -0
  67. data/ext/libsass/src/debugger.hpp +964 -0
  68. data/ext/libsass/src/emitter.cpp +297 -0
  69. data/ext/libsass/src/emitter.hpp +101 -0
  70. data/ext/libsass/src/environment.cpp +260 -0
  71. data/ext/libsass/src/environment.hpp +124 -0
  72. data/ext/libsass/src/error_handling.cpp +239 -0
  73. data/ext/libsass/src/error_handling.hpp +248 -0
  74. data/ext/libsass/src/eval.cpp +1543 -0
  75. data/ext/libsass/src/eval.hpp +110 -0
  76. data/ext/libsass/src/eval_selectors.cpp +75 -0
  77. data/ext/libsass/src/expand.cpp +875 -0
  78. data/ext/libsass/src/expand.hpp +98 -0
  79. data/ext/libsass/src/extender.cpp +1226 -0
  80. data/ext/libsass/src/extender.hpp +399 -0
  81. data/ext/libsass/src/extension.cpp +43 -0
  82. data/ext/libsass/src/extension.hpp +89 -0
  83. data/ext/libsass/src/file.cpp +531 -0
  84. data/ext/libsass/src/file.hpp +124 -0
  85. data/ext/libsass/src/fn_colors.cpp +836 -0
  86. data/ext/libsass/src/fn_colors.hpp +99 -0
  87. data/ext/libsass/src/fn_lists.cpp +285 -0
  88. data/ext/libsass/src/fn_lists.hpp +34 -0
  89. data/ext/libsass/src/fn_maps.cpp +94 -0
  90. data/ext/libsass/src/fn_maps.hpp +30 -0
  91. data/ext/libsass/src/fn_miscs.cpp +248 -0
  92. data/ext/libsass/src/fn_miscs.hpp +40 -0
  93. data/ext/libsass/src/fn_numbers.cpp +246 -0
  94. data/ext/libsass/src/fn_numbers.hpp +45 -0
  95. data/ext/libsass/src/fn_selectors.cpp +205 -0
  96. data/ext/libsass/src/fn_selectors.hpp +35 -0
  97. data/ext/libsass/src/fn_strings.cpp +268 -0
  98. data/ext/libsass/src/fn_strings.hpp +34 -0
  99. data/ext/libsass/src/fn_utils.cpp +159 -0
  100. data/ext/libsass/src/fn_utils.hpp +62 -0
  101. data/ext/libsass/src/inspect.cpp +1126 -0
  102. data/ext/libsass/src/inspect.hpp +101 -0
  103. data/ext/libsass/src/json.cpp +1436 -0
  104. data/ext/libsass/src/json.hpp +117 -0
  105. data/ext/libsass/src/kwd_arg_macros.hpp +28 -0
  106. data/ext/libsass/src/lexer.cpp +122 -0
  107. data/ext/libsass/src/lexer.hpp +304 -0
  108. data/ext/libsass/src/listize.cpp +70 -0
  109. data/ext/libsass/src/listize.hpp +37 -0
  110. data/ext/libsass/src/mapping.hpp +19 -0
  111. data/ext/libsass/src/memory/allocator.cpp +48 -0
  112. data/ext/libsass/src/memory/allocator.hpp +138 -0
  113. data/ext/libsass/src/memory/config.hpp +20 -0
  114. data/ext/libsass/src/memory/memory_pool.hpp +186 -0
  115. data/ext/libsass/src/memory/shared_ptr.cpp +33 -0
  116. data/ext/libsass/src/memory/shared_ptr.hpp +332 -0
  117. data/ext/libsass/src/memory.hpp +12 -0
  118. data/ext/libsass/src/operation.hpp +223 -0
  119. data/ext/libsass/src/operators.cpp +267 -0
  120. data/ext/libsass/src/operators.hpp +30 -0
  121. data/ext/libsass/src/ordered_map.hpp +112 -0
  122. data/ext/libsass/src/output.cpp +320 -0
  123. data/ext/libsass/src/output.hpp +47 -0
  124. data/ext/libsass/src/parser.cpp +3059 -0
  125. data/ext/libsass/src/parser.hpp +395 -0
  126. data/ext/libsass/src/parser_selectors.cpp +189 -0
  127. data/ext/libsass/src/permutate.hpp +164 -0
  128. data/ext/libsass/src/plugins.cpp +188 -0
  129. data/ext/libsass/src/plugins.hpp +57 -0
  130. data/ext/libsass/src/position.cpp +163 -0
  131. data/ext/libsass/src/position.hpp +147 -0
  132. data/ext/libsass/src/prelexer.cpp +1780 -0
  133. data/ext/libsass/src/prelexer.hpp +484 -0
  134. data/ext/libsass/src/remove_placeholders.cpp +86 -0
  135. data/ext/libsass/src/remove_placeholders.hpp +37 -0
  136. data/ext/libsass/src/sass.cpp +156 -0
  137. data/ext/libsass/src/sass.hpp +147 -0
  138. data/ext/libsass/src/sass2scss.cpp +895 -0
  139. data/ext/libsass/src/sass_context.cpp +742 -0
  140. data/ext/libsass/src/sass_context.hpp +129 -0
  141. data/ext/libsass/src/sass_functions.cpp +210 -0
  142. data/ext/libsass/src/sass_functions.hpp +50 -0
  143. data/ext/libsass/src/sass_values.cpp +362 -0
  144. data/ext/libsass/src/sass_values.hpp +82 -0
  145. data/ext/libsass/src/settings.hpp +19 -0
  146. data/ext/libsass/src/source.cpp +69 -0
  147. data/ext/libsass/src/source.hpp +95 -0
  148. data/ext/libsass/src/source_data.hpp +32 -0
  149. data/ext/libsass/src/source_map.cpp +202 -0
  150. data/ext/libsass/src/source_map.hpp +65 -0
  151. data/ext/libsass/src/stylesheet.cpp +22 -0
  152. data/ext/libsass/src/stylesheet.hpp +57 -0
  153. data/ext/libsass/src/to_value.cpp +114 -0
  154. data/ext/libsass/src/to_value.hpp +46 -0
  155. data/ext/libsass/src/units.cpp +507 -0
  156. data/ext/libsass/src/units.hpp +110 -0
  157. data/ext/libsass/src/utf8/checked.h +336 -0
  158. data/ext/libsass/src/utf8/core.h +332 -0
  159. data/ext/libsass/src/utf8/unchecked.h +235 -0
  160. data/ext/libsass/src/utf8.h +34 -0
  161. data/ext/libsass/src/utf8_string.cpp +104 -0
  162. data/ext/libsass/src/utf8_string.hpp +38 -0
  163. data/ext/libsass/src/util.cpp +723 -0
  164. data/ext/libsass/src/util.hpp +105 -0
  165. data/ext/libsass/src/util_string.cpp +125 -0
  166. data/ext/libsass/src/util_string.hpp +73 -0
  167. data/ext/libsass/src/values.cpp +140 -0
  168. data/ext/libsass/src/values.hpp +12 -0
  169. data/lib/sassc/dependency.rb +17 -0
  170. data/lib/sassc/engine.rb +141 -0
  171. data/lib/sassc/error.rb +37 -0
  172. data/lib/sassc/functions_handler.rb +73 -0
  173. data/lib/sassc/import_handler.rb +50 -0
  174. data/lib/sassc/importer.rb +31 -0
  175. data/lib/sassc/native/native_context_api.rb +147 -0
  176. data/lib/sassc/native/native_functions_api.rb +159 -0
  177. data/lib/sassc/native/sass2scss_api.rb +10 -0
  178. data/lib/sassc/native/sass_input_style.rb +13 -0
  179. data/lib/sassc/native/sass_output_style.rb +12 -0
  180. data/lib/sassc/native/sass_value.rb +97 -0
  181. data/lib/sassc/native/string_list.rb +10 -0
  182. data/lib/sassc/native.rb +64 -0
  183. data/lib/sassc/sass_2_scss.rb +9 -0
  184. data/lib/sassc/script/functions.rb +8 -0
  185. data/lib/sassc/script/value/bool.rb +32 -0
  186. data/lib/sassc/script/value/color.rb +95 -0
  187. data/lib/sassc/script/value/list.rb +136 -0
  188. data/lib/sassc/script/value/map.rb +69 -0
  189. data/lib/sassc/script/value/number.rb +389 -0
  190. data/lib/sassc/script/value/string.rb +96 -0
  191. data/lib/sassc/script/value.rb +137 -0
  192. data/lib/sassc/script/value_conversion/base.rb +13 -0
  193. data/lib/sassc/script/value_conversion/bool.rb +13 -0
  194. data/lib/sassc/script/value_conversion/color.rb +18 -0
  195. data/lib/sassc/script/value_conversion/list.rb +25 -0
  196. data/lib/sassc/script/value_conversion/map.rb +21 -0
  197. data/lib/sassc/script/value_conversion/number.rb +13 -0
  198. data/lib/sassc/script/value_conversion/string.rb +17 -0
  199. data/lib/sassc/script/value_conversion.rb +69 -0
  200. data/lib/sassc/script.rb +17 -0
  201. data/lib/sassc/util/normalized_map.rb +117 -0
  202. data/lib/sassc/util.rb +231 -0
  203. data/lib/sassc/version.rb +5 -0
  204. data/lib/sassc.rb +57 -0
  205. data/sassc.gemspec +69 -0
  206. data/test/css_color_level4_test.rb +168 -0
  207. data/test/custom_importer_test.rb +127 -0
  208. data/test/engine_test.rb +314 -0
  209. data/test/error_test.rb +29 -0
  210. data/test/fixtures/paths.scss +10 -0
  211. data/test/functions_test.rb +340 -0
  212. data/test/native_test.rb +213 -0
  213. data/test/output_style_test.rb +107 -0
  214. data/test/sass_2_scss_test.rb +14 -0
  215. data/test/test_helper.rb +45 -0
  216. metadata +396 -0
@@ -0,0 +1,248 @@
1
+ #include "ast.hpp"
2
+ #include "expand.hpp"
3
+ #include "fn_utils.hpp"
4
+ #include "fn_miscs.hpp"
5
+ #include "util_string.hpp"
6
+
7
+ namespace Sass {
8
+
9
+ namespace Functions {
10
+
11
+ //////////////////////////
12
+ // INTROSPECTION FUNCTIONS
13
+ //////////////////////////
14
+
15
+ Signature type_of_sig = "type-of($value)";
16
+ BUILT_IN(type_of)
17
+ {
18
+ Expression* v = ARG("$value", Expression);
19
+ return SASS_MEMORY_NEW(String_Quoted, pstate, v->type());
20
+ }
21
+
22
+ Signature variable_exists_sig = "variable-exists($name)";
23
+ BUILT_IN(variable_exists)
24
+ {
25
+ sass::string s = Util::normalize_underscores(unquote(ARG("$name", String_Constant)->value()));
26
+
27
+ if(d_env.has("$"+s)) {
28
+ return SASS_MEMORY_NEW(Boolean, pstate, true);
29
+ }
30
+ else {
31
+ return SASS_MEMORY_NEW(Boolean, pstate, false);
32
+ }
33
+ }
34
+
35
+ Signature global_variable_exists_sig = "global-variable-exists($name)";
36
+ BUILT_IN(global_variable_exists)
37
+ {
38
+ sass::string s = Util::normalize_underscores(unquote(ARG("$name", String_Constant)->value()));
39
+
40
+ if(d_env.has_global("$"+s)) {
41
+ return SASS_MEMORY_NEW(Boolean, pstate, true);
42
+ }
43
+ else {
44
+ return SASS_MEMORY_NEW(Boolean, pstate, false);
45
+ }
46
+ }
47
+
48
+ Signature function_exists_sig = "function-exists($name)";
49
+ BUILT_IN(function_exists)
50
+ {
51
+ String_Constant* ss = Cast<String_Constant>(env["$name"]);
52
+ if (!ss) {
53
+ error("$name: " + (env["$name"]->to_string()) + " is not a string for `function-exists'", pstate, traces);
54
+ }
55
+
56
+ sass::string name = Util::normalize_underscores(unquote(ss->value()));
57
+
58
+ if(d_env.has(name+"[f]")) {
59
+ return SASS_MEMORY_NEW(Boolean, pstate, true);
60
+ }
61
+ else {
62
+ return SASS_MEMORY_NEW(Boolean, pstate, false);
63
+ }
64
+ }
65
+
66
+ Signature mixin_exists_sig = "mixin-exists($name)";
67
+ BUILT_IN(mixin_exists)
68
+ {
69
+ sass::string s = Util::normalize_underscores(unquote(ARG("$name", String_Constant)->value()));
70
+
71
+ if(d_env.has(s+"[m]")) {
72
+ return SASS_MEMORY_NEW(Boolean, pstate, true);
73
+ }
74
+ else {
75
+ return SASS_MEMORY_NEW(Boolean, pstate, false);
76
+ }
77
+ }
78
+
79
+ Signature feature_exists_sig = "feature-exists($feature)";
80
+ BUILT_IN(feature_exists)
81
+ {
82
+ sass::string s = unquote(ARG("$feature", String_Constant)->value());
83
+
84
+ static const auto *const features = new std::unordered_set<sass::string> {
85
+ "global-variable-shadowing",
86
+ "extend-selector-pseudoclass",
87
+ "at-error",
88
+ "units-level-3",
89
+ "custom-property"
90
+ };
91
+ return SASS_MEMORY_NEW(Boolean, pstate, features->find(s) != features->end());
92
+ }
93
+
94
+ Signature call_sig = "call($function, $args...)";
95
+ BUILT_IN(call)
96
+ {
97
+ sass::string function;
98
+ Function* ff = Cast<Function>(env["$function"]);
99
+ String_Constant* ss = Cast<String_Constant>(env["$function"]);
100
+
101
+ if (ss) {
102
+ function = Util::normalize_underscores(unquote(ss->value()));
103
+ std::cerr << "DEPRECATION WARNING: ";
104
+ std::cerr << "Passing a string to call() is deprecated and will be illegal" << std::endl;
105
+ std::cerr << "in Sass 4.0. Use call(get-function(" + quote(function) + ")) instead." << std::endl;
106
+ std::cerr << std::endl;
107
+ } else if (ff) {
108
+ function = ff->name();
109
+ }
110
+
111
+ List_Obj arglist = SASS_MEMORY_COPY(ARG("$args", List));
112
+
113
+ Arguments_Obj args = SASS_MEMORY_NEW(Arguments, pstate);
114
+ // sass::string full_name(name + "[f]");
115
+ // Definition* def = d_env.has(full_name) ? Cast<Definition>((d_env)[full_name]) : 0;
116
+ // Parameters* params = def ? def->parameters() : 0;
117
+ // size_t param_size = params ? params->length() : 0;
118
+ for (size_t i = 0, L = arglist->length(); i < L; ++i) {
119
+ ExpressionObj expr = arglist->value_at_index(i);
120
+ // if (params && params->has_rest_parameter()) {
121
+ // Parameter_Obj p = param_size > i ? (*params)[i] : 0;
122
+ // List* list = Cast<List>(expr);
123
+ // if (list && p && !p->is_rest_parameter()) expr = (*list)[0];
124
+ // }
125
+ if (arglist->is_arglist()) {
126
+ ExpressionObj obj = arglist->at(i);
127
+ Argument_Obj arg = (Argument*) obj.ptr(); // XXX
128
+ args->append(SASS_MEMORY_NEW(Argument,
129
+ pstate,
130
+ expr,
131
+ arg ? arg->name() : "",
132
+ arg ? arg->is_rest_argument() : false,
133
+ arg ? arg->is_keyword_argument() : false));
134
+ } else {
135
+ args->append(SASS_MEMORY_NEW(Argument, pstate, expr));
136
+ }
137
+ }
138
+ Function_Call_Obj func = SASS_MEMORY_NEW(Function_Call, pstate, function, args);
139
+
140
+ Expand expand(ctx, &d_env, &selector_stack, &original_stack);
141
+ func->via_call(true); // calc invoke is allowed
142
+ if (ff) func->func(ff);
143
+ return Cast<PreValue>(func->perform(&expand.eval));
144
+ }
145
+
146
+ ////////////////////
147
+ // BOOLEAN FUNCTIONS
148
+ ////////////////////
149
+
150
+ Signature not_sig = "not($value)";
151
+ BUILT_IN(sass_not)
152
+ {
153
+ return SASS_MEMORY_NEW(Boolean, pstate, ARG("$value", Expression)->is_false());
154
+ }
155
+
156
+ Signature if_sig = "if($condition, $if-true, $if-false)";
157
+ BUILT_IN(sass_if)
158
+ {
159
+ Expand expand(ctx, &d_env, &selector_stack, &original_stack);
160
+ ExpressionObj cond = ARG("$condition", Expression)->perform(&expand.eval);
161
+ bool is_true = !cond->is_false();
162
+ ExpressionObj res = ARG(is_true ? "$if-true" : "$if-false", Expression);
163
+ ExpressionObj rv = res->perform(&expand.eval);
164
+ ValueObj value = Cast<Value>(rv);
165
+ if (value != nullptr) {
166
+ value->set_delayed(false);
167
+ return value.detach();
168
+ }
169
+ rv->set_delayed(false);
170
+ return nullptr;
171
+ }
172
+
173
+ //////////////////////////
174
+ // MISCELLANEOUS FUNCTIONS
175
+ //////////////////////////
176
+
177
+ Signature inspect_sig = "inspect($value)";
178
+ BUILT_IN(inspect)
179
+ {
180
+ Expression* v = ARG("$value", Expression);
181
+ if (v->concrete_type() == Expression::NULL_VAL) {
182
+ return SASS_MEMORY_NEW(String_Constant, pstate, "null");
183
+ } else if (v->concrete_type() == Expression::BOOLEAN && v->is_false()) {
184
+ return SASS_MEMORY_NEW(String_Constant, pstate, "false");
185
+ } else if (v->concrete_type() == Expression::STRING) {
186
+ String_Constant *s = Cast<String_Constant>(v);
187
+ if (s->quote_mark()) {
188
+ return SASS_MEMORY_NEW(String_Constant, pstate, quote(s->value(), s->quote_mark()));
189
+ } else {
190
+ return s;
191
+ }
192
+ } else {
193
+ // ToDo: fix to_sass for nested parentheses
194
+ Sass_Output_Style old_style;
195
+ old_style = ctx.c_options.output_style;
196
+ ctx.c_options.output_style = TO_SASS;
197
+ Emitter emitter(ctx.c_options);
198
+ Inspect i(emitter);
199
+ i.in_declaration = false;
200
+ v->perform(&i);
201
+ ctx.c_options.output_style = old_style;
202
+ return SASS_MEMORY_NEW(String_Quoted, pstate, i.get_buffer());
203
+ }
204
+ }
205
+
206
+ Signature content_exists_sig = "content-exists()";
207
+ BUILT_IN(content_exists)
208
+ {
209
+ if (!d_env.has_global("is_in_mixin")) {
210
+ error("Cannot call content-exists() except within a mixin.", pstate, traces);
211
+ }
212
+ return SASS_MEMORY_NEW(Boolean, pstate, d_env.has_lexical("@content[m]"));
213
+ }
214
+
215
+ Signature get_function_sig = "get-function($name, $css: false)";
216
+ BUILT_IN(get_function)
217
+ {
218
+ String_Constant* ss = Cast<String_Constant>(env["$name"]);
219
+ if (!ss) {
220
+ error("$name: " + (env["$name"]->to_string()) + " is not a string for `get-function'", pstate, traces);
221
+ }
222
+
223
+ sass::string name = Util::normalize_underscores(unquote(ss->value()));
224
+ sass::string full_name = name + "[f]";
225
+
226
+ Boolean_Obj css = ARG("$css", Boolean);
227
+ if (!css->is_false()) {
228
+ Definition* def = SASS_MEMORY_NEW(Definition,
229
+ pstate,
230
+ name,
231
+ SASS_MEMORY_NEW(Parameters, pstate),
232
+ SASS_MEMORY_NEW(Block, pstate, 0, false),
233
+ Definition::FUNCTION);
234
+ return SASS_MEMORY_NEW(Function, pstate, def, true);
235
+ }
236
+
237
+
238
+ if (!d_env.has_global(full_name)) {
239
+ error("Function not found: " + name, pstate, traces);
240
+ }
241
+
242
+ Definition* def = Cast<Definition>(d_env[full_name]);
243
+ return SASS_MEMORY_NEW(Function, pstate, def, false);
244
+ }
245
+
246
+ }
247
+
248
+ }
@@ -0,0 +1,40 @@
1
+ #ifndef SASS_FN_MISCS_H
2
+ #define SASS_FN_MISCS_H
3
+
4
+ #include "fn_utils.hpp"
5
+
6
+ namespace Sass {
7
+
8
+ namespace Functions {
9
+
10
+ extern Signature type_of_sig;
11
+ extern Signature variable_exists_sig;
12
+ extern Signature global_variable_exists_sig;
13
+ extern Signature function_exists_sig;
14
+ extern Signature mixin_exists_sig;
15
+ extern Signature feature_exists_sig;
16
+ extern Signature call_sig;
17
+ extern Signature not_sig;
18
+ extern Signature if_sig;
19
+ extern Signature set_nth_sig;
20
+ extern Signature content_exists_sig;
21
+ extern Signature get_function_sig;
22
+
23
+ BUILT_IN(type_of);
24
+ BUILT_IN(variable_exists);
25
+ BUILT_IN(global_variable_exists);
26
+ BUILT_IN(function_exists);
27
+ BUILT_IN(mixin_exists);
28
+ BUILT_IN(feature_exists);
29
+ BUILT_IN(call);
30
+ BUILT_IN(sass_not);
31
+ BUILT_IN(sass_if);
32
+ BUILT_IN(set_nth);
33
+ BUILT_IN(content_exists);
34
+ BUILT_IN(get_function);
35
+
36
+ }
37
+
38
+ }
39
+
40
+ #endif
@@ -0,0 +1,246 @@
1
+ // sass.hpp must go before all system headers to get the
2
+ // __EXTENSIONS__ fix on Solaris.
3
+ #include "sass.hpp"
4
+
5
+ #include <cstdint>
6
+ #include <cstdlib>
7
+ #include <cmath>
8
+ #include <ctime>
9
+ #include <random>
10
+ #include <sstream>
11
+ #include <iomanip>
12
+ #include <algorithm>
13
+ #include <thread>
14
+
15
+ #include "ast.hpp"
16
+ #include "units.hpp"
17
+ #include "fn_utils.hpp"
18
+ #include "fn_numbers.hpp"
19
+
20
+ #ifdef __MINGW32__
21
+ #include "windows.h"
22
+ #include "wincrypt.h"
23
+ #endif
24
+
25
+ namespace Sass {
26
+
27
+ namespace Functions {
28
+
29
+ #ifdef __MINGW32__
30
+ uint64_t GetSeed()
31
+ {
32
+ HCRYPTPROV hp = 0;
33
+ BYTE rb[8];
34
+ CryptAcquireContext(&hp, 0, 0, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT);
35
+ CryptGenRandom(hp, sizeof(rb), rb);
36
+ CryptReleaseContext(hp, 0);
37
+
38
+ uint64_t seed;
39
+ memcpy(&seed, &rb[0], sizeof(seed));
40
+
41
+ return seed;
42
+ }
43
+ #else
44
+ uint64_t GetSeed()
45
+ {
46
+ // Init universe entropy
47
+ uint64_t rnd = 42;
48
+ // Try to get random number from system
49
+ try {
50
+ std::random_device rd;
51
+ rnd = rd();
52
+ }
53
+ // On certain system this can throw since either
54
+ // underlying hardware or software can be buggy.
55
+ // https://github.com/sass/libsass/issues/3151
56
+ catch (std::exception&) {
57
+ }
58
+ // Don't trust anyone to be random, so we
59
+ // add a little entropy of our own.
60
+ rnd ^= std::time(NULL) ^ std::clock() ^
61
+ std::hash<std::thread::id>()
62
+ (std::this_thread::get_id());
63
+ // Return entropy
64
+ return rnd;
65
+ }
66
+ #endif
67
+
68
+ // note: the performance of many implementations of
69
+ // random_device degrades sharply once the entropy pool
70
+ // is exhausted. For practical use, random_device is
71
+ // generally only used to seed a PRNG such as mt19937.
72
+ static std::mt19937 rand(static_cast<unsigned int>(GetSeed()));
73
+
74
+ ///////////////////
75
+ // NUMBER FUNCTIONS
76
+ ///////////////////
77
+
78
+ Signature percentage_sig = "percentage($number)";
79
+ BUILT_IN(percentage)
80
+ {
81
+ Number_Obj n = ARGN("$number");
82
+ if (!n->is_unitless()) error("argument $number of `" + sass::string(sig) + "` must be unitless", pstate, traces);
83
+ return SASS_MEMORY_NEW(Number, pstate, n->value() * 100, "%");
84
+ }
85
+
86
+ Signature round_sig = "round($number)";
87
+ BUILT_IN(round)
88
+ {
89
+ Number_Obj r = ARGN("$number");
90
+ r->value(Sass::round(r->value(), ctx.c_options.precision));
91
+ r->pstate(pstate);
92
+ return r.detach();
93
+ }
94
+
95
+ Signature ceil_sig = "ceil($number)";
96
+ BUILT_IN(ceil)
97
+ {
98
+ Number_Obj r = ARGN("$number");
99
+ r->value(std::ceil(r->value()));
100
+ r->pstate(pstate);
101
+ return r.detach();
102
+ }
103
+
104
+ Signature floor_sig = "floor($number)";
105
+ BUILT_IN(floor)
106
+ {
107
+ Number_Obj r = ARGN("$number");
108
+ r->value(std::floor(r->value()));
109
+ r->pstate(pstate);
110
+ return r.detach();
111
+ }
112
+
113
+ Signature abs_sig = "abs($number)";
114
+ BUILT_IN(abs)
115
+ {
116
+ Number_Obj r = ARGN("$number");
117
+ r->value(std::abs(r->value()));
118
+ r->pstate(pstate);
119
+ return r.detach();
120
+ }
121
+
122
+ Signature min_sig = "min($numbers...)";
123
+ BUILT_IN(min)
124
+ {
125
+ List* arglist = ARG("$numbers", List);
126
+ Number_Obj least;
127
+ size_t L = arglist->length();
128
+ if (L == 0) {
129
+ error("At least one argument must be passed.", pstate, traces);
130
+ }
131
+ for (size_t i = 0; i < L; ++i) {
132
+ ExpressionObj val = arglist->value_at_index(i);
133
+ Number_Obj xi = Cast<Number>(val);
134
+ if (!xi) {
135
+ error("\"" + val->to_string(ctx.c_options) + "\" is not a number for `min'", pstate, traces);
136
+ }
137
+ if (least) {
138
+ if (*xi < *least) least = xi;
139
+ } else least = xi;
140
+ }
141
+ return least.detach();
142
+ }
143
+
144
+ Signature max_sig = "max($numbers...)";
145
+ BUILT_IN(max)
146
+ {
147
+ List* arglist = ARG("$numbers", List);
148
+ Number_Obj greatest;
149
+ size_t L = arglist->length();
150
+ if (L == 0) {
151
+ error("At least one argument must be passed.", pstate, traces);
152
+ }
153
+ for (size_t i = 0; i < L; ++i) {
154
+ ExpressionObj val = arglist->value_at_index(i);
155
+ Number_Obj xi = Cast<Number>(val);
156
+ if (!xi) {
157
+ error("\"" + val->to_string(ctx.c_options) + "\" is not a number for `max'", pstate, traces);
158
+ }
159
+ if (greatest) {
160
+ if (*greatest < *xi) greatest = xi;
161
+ } else greatest = xi;
162
+ }
163
+ return greatest.detach();
164
+ }
165
+
166
+ Signature random_sig = "random($limit:false)";
167
+ BUILT_IN(random)
168
+ {
169
+ AST_Node_Obj arg = env["$limit"];
170
+ Value* v = Cast<Value>(arg);
171
+ Number* l = Cast<Number>(arg);
172
+ Boolean* b = Cast<Boolean>(arg);
173
+ if (l) {
174
+ double lv = l->value();
175
+ if (lv < 1) {
176
+ sass::ostream err;
177
+ err << "$limit " << lv << " must be greater than or equal to 1 for `random'";
178
+ error(err.str(), pstate, traces);
179
+ }
180
+ bool eq_int = std::fabs(trunc(lv) - lv) < NUMBER_EPSILON;
181
+ if (!eq_int) {
182
+ sass::ostream err;
183
+ err << "Expected $limit to be an integer but got " << lv << " for `random'";
184
+ error(err.str(), pstate, traces);
185
+ }
186
+ std::uniform_real_distribution<> distributor(1, lv + 1);
187
+ uint_fast32_t distributed = static_cast<uint_fast32_t>(distributor(rand));
188
+ return SASS_MEMORY_NEW(Number, pstate, (double)distributed);
189
+ }
190
+ else if (b) {
191
+ std::uniform_real_distribution<> distributor(0, 1);
192
+ double distributed = static_cast<double>(distributor(rand));
193
+ return SASS_MEMORY_NEW(Number, pstate, distributed);
194
+ } else if (v) {
195
+ traces.push_back(Backtrace(pstate));
196
+ throw Exception::InvalidArgumentType(pstate, traces, "random", "$limit", "number", v);
197
+ } else {
198
+ traces.push_back(Backtrace(pstate));
199
+ throw Exception::InvalidArgumentType(pstate, traces, "random", "$limit", "number");
200
+ }
201
+ }
202
+
203
+ Signature unique_id_sig = "unique-id()";
204
+ BUILT_IN(unique_id)
205
+ {
206
+ sass::ostream ss;
207
+ std::uniform_real_distribution<> distributor(0, 4294967296); // 16^8
208
+ uint_fast32_t distributed = static_cast<uint_fast32_t>(distributor(rand));
209
+ ss << "u" << std::setfill('0') << std::setw(8) << std::hex << distributed;
210
+ return SASS_MEMORY_NEW(String_Quoted, pstate, ss.str());
211
+ }
212
+
213
+ Signature unit_sig = "unit($number)";
214
+ BUILT_IN(unit)
215
+ {
216
+ Number_Obj arg = ARGN("$number");
217
+ sass::string str(quote(arg->unit(), '"'));
218
+ return SASS_MEMORY_NEW(String_Quoted, pstate, str);
219
+ }
220
+
221
+ Signature unitless_sig = "unitless($number)";
222
+ BUILT_IN(unitless)
223
+ {
224
+ Number_Obj arg = ARGN("$number");
225
+ bool unitless = arg->is_unitless();
226
+ return SASS_MEMORY_NEW(Boolean, pstate, unitless);
227
+ }
228
+
229
+ Signature comparable_sig = "comparable($number1, $number2)";
230
+ BUILT_IN(comparable)
231
+ {
232
+ Number_Obj n1 = ARGN("$number1");
233
+ Number_Obj n2 = ARGN("$number2");
234
+ if (n1->is_unitless() || n2->is_unitless()) {
235
+ return SASS_MEMORY_NEW(Boolean, pstate, true);
236
+ }
237
+ // normalize into main units
238
+ n1->normalize(); n2->normalize();
239
+ Units &lhs_unit = *n1, &rhs_unit = *n2;
240
+ bool is_comparable = (lhs_unit == rhs_unit);
241
+ return SASS_MEMORY_NEW(Boolean, pstate, is_comparable);
242
+ }
243
+
244
+ }
245
+
246
+ }
@@ -0,0 +1,45 @@
1
+ #ifndef SASS_FN_NUMBERS_H
2
+ #define SASS_FN_NUMBERS_H
3
+
4
+ #include "fn_utils.hpp"
5
+
6
+ namespace Sass {
7
+
8
+ namespace Functions {
9
+
10
+ // return a number object (copied since we want to have reduced units)
11
+ #define ARGN(argname) get_arg_n(argname, env, sig, pstate, traces) // Number copy
12
+
13
+ extern Signature percentage_sig;
14
+ extern Signature round_sig;
15
+ extern Signature ceil_sig;
16
+ extern Signature floor_sig;
17
+ extern Signature abs_sig;
18
+ extern Signature min_sig;
19
+ extern Signature max_sig;
20
+ extern Signature inspect_sig;
21
+ extern Signature random_sig;
22
+ extern Signature unique_id_sig;
23
+ extern Signature unit_sig;
24
+ extern Signature unitless_sig;
25
+ extern Signature comparable_sig;
26
+
27
+ BUILT_IN(percentage);
28
+ BUILT_IN(round);
29
+ BUILT_IN(ceil);
30
+ BUILT_IN(floor);
31
+ BUILT_IN(abs);
32
+ BUILT_IN(min);
33
+ BUILT_IN(max);
34
+ BUILT_IN(inspect);
35
+ BUILT_IN(random);
36
+ BUILT_IN(unique_id);
37
+ BUILT_IN(unit);
38
+ BUILT_IN(unitless);
39
+ BUILT_IN(comparable);
40
+
41
+ }
42
+
43
+ }
44
+
45
+ #endif