forked from webcompat/css-fixme
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcssfixme.htm
471 lines (446 loc) · 27.4 KB
/
cssfixme.htm
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
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
<!DOCTYPE html><html><head>
<title>CSS:fixme</title>
<script src="css-browserside.js">/* This is the CSS rework parser, converted from Node.js - code to something that runs in browsers with browserify */</script>
<style type="text/css">pre strong{
background: yellow;
color:black;
}
textarea{width: 100%; min-height: 200px;}</style>
</head>
<body>
<p>This tool attempts to fix non-standard CSS by adding standardised equivalents to all declarations that use <em>-webkit-</em>prefixed properties and values that have standardised equivalents. The standardised syntax is not always same as the -webkit- syntax, so getting everything right is a challenge. Please test this tool with various <em>-webkit-</em> prefixed CSS constructs and let me <a href="mailto:hsteen@mozilla.com">know</a> if you find problems.</p>
<p>Note: some <em>-webkit-</em> prefixed properties do not (yet?) have standardised equivalents. This tool will deliberately not add pseudo-standard code on the assumption that the standards will one day match the current -webkit- implementation.</p>
<p>This uses the <a href="https://github.com/reworkcss/rework">ReworkCSS</a> library for parsing and stringifying CSS.</p>
<p>Paste CSS in this box to see what needs fixing:</p>
<textarea></textarea><br><button type="button" onclick="doTheBigStyleFixing(document.getElementsByTagName('textarea')[0].value)">Add standard equivalents for prefixed CSS</button> <label><input type="checkbox" name="compact">Compact output (whitespace only after closing brace)</label>
<pre></pre>
<!--
TODO: -webkit-border-radius: 50% 2px;
border-radius: 50% / 2px; ??
-->
<script>
// CSS property list, thanks to Jens Meiert. Extracted from http://meiert.com/en/indices/css-properties/ with this JS:
// var props = []; for(var elms = document.querySelectorAll('a code'),el,i=0; el=elms[i]; i++){props.push(el.textContent)}; JSON.stringify(props)
var allW3CSSProperties = ["align-content","align-items","align-self","alignment-adjust","alignment-baseline","all","anchor-point","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","appearance","azimuth","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","baseline-shift","binding","bleed","bookmark-label","bookmark-level","bookmark-state","bookmark-target","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","chains","clear","clip","clip-path","clip-rule","color","color-interpolation-filters","color-profile","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","contain","content","counter-increment","counter-reset","crop","cue","cue-after","cue-before","cursor","direction","display","dominant-baseline","drop-initial-after-adjust","drop-initial-after-align","drop-initial-before-adjust","drop-initial-before-align","drop-initial-size","drop-initial-value","elevation","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","float-offset","flood-color","flood-opacity","flow-from","flow-into","font","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-stretch","font-style","font-synthesis","font-variant","font-variant-alternates","font-variant-caps","font-variant-east-asian","font-variant-ligatures","font-variant-numeric","font-variant-position","font-weight","grid","grid-area","grid-auto-columns","grid-auto-flow","grid-auto-position","grid-auto-rows","grid-column","grid-column-end","grid-column-start","grid-row","grid-row-end","grid-row-start","grid-template","grid-template-areas","grid-template-columns","grid-template-rows","hanging-punctuation","height","hyphens","icon","image-orientation","image-resolution","ime-mode","inline-box-align","justify-content","justify-items","justify-self","left","letter-spacing","lighting-color","line-break","line-height","line-stacking","line-stacking-ruby","line-stacking-shift","line-stacking-strategy","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marker-offset","marks","mask","mask-box","mask-box-outset","mask-box-repeat","mask-box-slice","mask-box-source","mask-box-width","mask-clip","mask-image","mask-origin","mask-position","mask-repeat","mask-size","mask-source-type","mask-type","max-height","max-lines","max-width","min-height","min-width","move-to","nav-down","nav-index","nav-left","nav-right","nav-up","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page","page-break-after","page-break-before","page-break-inside","page-policy","pause","pause-after","pause-before","perspective","perspective-origin","pitch","pitch-range","play-during","position","presentation-level","quotes","region-fragment","rendering-intent","resize","rest","rest-after","rest-before","richness","right","rotation","rotation-point","ruby-align","ruby-overhang","ruby-position","ruby-span","shape-image-threshold","shape-outside","shape-margin","size","speak","speak-as","speak-header","speak-numeral","speak-punctuation","speech-rate","stress","string-set","tab-size","table-layout","target","target-name","target-new","target-position","text-align","text-align-last","text-combine-horizontal","text-decoration","text-decoration-color","text-decoration-line","text-decoration-skip","text-decoration-style","text-emphasis","text-emphasis-color","text-emphasis-position","text-emphasis-style","text-height","text-indent","text-justify","text-orientation","text-outline","text-overflow","text-shadow","text-space-collapse","text-transform","text-underline-position","text-wrap","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","voice-balance","voice-duration","voice-family","voice-pitch","voice-range","voice-rate","voice-stress","voice-volume","volume","white-space","widows","width","word-break","word-spacing","word-wrap","wrap-flow","wrap-through","writing-mode","z-index"];
var warnAgainstPotentiallyOverwrittenValues = true; // outputs console error if we override an existing property except the non-standard one
/* Derived from GreaseMonkey script. Adapted to give a prettier output.. */
function doTheBigStyleFixing(str){
/* This method builds a "pretty" view of the CSS with amended rules highlighted in the PRE element */
var fixupStyle = {"type": "stylesheet", stylesheet: {rules: []}};
var pre = document.getElementsByTagName('pre')[0], parent;
pre.innerHTML = '';
try{
var oldRules = css.parse(stripHTMLComments(str));
}catch(e){
console.log('failed to parse stylesheet (' + str.length + ' characters) - error message from parser: ' + e.message);
return;
}
createFixupRulesFromCSS(oldRules);
var compact = document.getElementsByName('compact')[0].checked;
for(var rule, i = 0; rule = oldRules.stylesheet.rules[i]; i++){
stringifyRuleToDOM(rule, 1, compact);
}
function stringifyRuleToDOM (rule, indent, compact) {
// rule.type, rule.selectors, rule.declarations
var strindent1 = compact ? '' : (new Array(indent)).join(' '); // indent by four spaces per indentation level
var strindent2 = compact ? '' : (new Array(indent + 1)).join(' '); // indent by four spaces per indentation level
var propValSep = compact ? ':' : ': ';
var prePropWS = compact ? '' : '\n';
if(rule.type === 'rule'){
var theSelectors = rule.selectors.join(compact? ',\n' : ', ')
pre.appendChild(document.createTextNode( '\n' + strindent1 + theSelectors+'{'))
for(var decl, j = 0; decl = rule.declarations[j]; j++){
// decl.type, decl.property, decl.value
// custom: decl._fxjsdefined
parent = decl._fxjsdefined ? pre.appendChild(document.createElement('strong')) : pre;
if(decl.type === 'declaration'){
parent.appendChild(document.createTextNode(prePropWS + strindent2 + decl.property + propValSep + decl.value + ';'))
}else if(decl.type === 'comment'){
parent.appendChild(document.createTextNode( prePropWS + '/*' + decl.comment + '*/'))
}
}
pre.appendChild(document.createTextNode(prePropWS + strindent1 + '}\n'))
}else if(rule.type === 'comment'){
pre.appendChild(document.createTextNode( prePropWS + '/*' + rule.comment + '*/'))
}else if(rule.type === 'media'){
pre.appendChild(document.createTextNode(prePropWS + strindent1 + '@media ' + rule.media + '{' ));
for(var subrule, k = 0; subrule = rule.rules[k]; k++){
stringifyRuleToDOM(subrule, indent+1);
}
pre.appendChild(document.createTextNode(prePropWS + strindent1 + '}'))
}else if(rule.type === 'charset'){
pre.appendChild(document.createTextNode(prePropWS + strindent1 + '@charset ' + rule.charset + ';'))
}
}
}
/*document.getElementsByTagName('textarea')[0].addEventListener('change', function(e){
doTheBigStyleFixing(e.target.value);
}, false);*/
function createFixupRulesFromCSS(obj){
var fixupRules = [], fixupDeclarations, prop, value;
for(var rule, i = 0; rule = obj.stylesheet.rules[i]; i++){
insertFixupDeclarations(rule);
}
function insertFixupDeclarations(rule){
// rule.type, rule.selectors, rule.declarations (rule.comment, rule.media)
if(rule.declarations){
fixupDeclarations = [];
for(var decl, j = 0; decl = rule.declarations[j]; j++){
// decl.type, decl.property, decl.value
prop = '', value = '';
// sometimes sites include a background fallback colour but in the *same* declaration as -webkit-gradient.
// It is better practise to have a separate 'background' declaration for the fallback colour. Let's add it.
if(decl.property === 'background' && /(#\w{3,6})\s*-webkit-/.test(decl.value)){ // TODO: in reality, the color and -webkit-foo() could be in any order.. is this a problem?
fixupDeclarations.push({type:'declaration', property:'background', value:RegExp.$1, _fxjsdefined:true});
decl.value = decl.value.replace(/(#\w{3,6})\s*-webkit-/, '-webkit-');
}
if(decl.property === 'display' && /box$/.test(decl.value) || /(box-|flex-)/.test(decl.property)){
var tmp = createFixupFlexboxDeclaration(decl, rule.declarations);
prop = tmp.property, value = tmp.value;
}else if(/-webkit-(linear-|radial-)?gradient/i.test(decl.value)){
var tmp = createFixupGradientDeclaration(decl, rule.declarations);
prop = tmp.property, value = tmp.value;
}else{
if(/-webkit-/.test(decl.property)){
prop = decl.property.substr(8);
}
if(/-webkit-/.test(decl.value)){
value = decl.value.replace(/-webkit-/g, '');
}
}
if(prop || value){
prop = prop || decl.property;
value = value || decl.value
if(hasDeclaration(fixupDeclarations.concat(rule.declarations), prop, value, false, true))continue;
if(warnAgainstPotentiallyOverwrittenValues){
var existingValue = getValueForProperty(fixupDeclarations.concat(rule.declarations), prop, false);
if(existingValue !== undefined && !/(-webkit-|box)/.test(existingValue)){ // now it gets complicated. There is a declaration for the property we want to add, but the value is different..
console.log('Whoa, Sir! We want to add "' + prop + ':' + value +'", but found "' + prop + ':' + existingValue +'"');
}
}
// We follow the standards - better not to pseudo-standardise -webkit-something
if(allW3CSSProperties.indexOf(prop) === -1)continue;
fixupDeclarations.push({type:'declaration', property:prop, value:value, _fxjsdefined:true});
// extra gotcha: per Gecko's reading of the spec, border-image will only appear if border-style or border-width is set..
if(prop === 'border-image' && getValueForProperty(fixupDeclarations.concat(rule.declarations), 'border-style', false) === undefined){
//fixupDeclarations[fixupDeclarations.length - 1].value += ' fill';
fixupDeclarations.push({type:'declaration', property:'border-style', value:'solid', _fxjsdefined:true});
}
}
}
if(fixupDeclarations.length > 0){
//fixupRules.push({type:'rule', selectors:rule.selectors.concat([]), declarations:fixupDeclarations.concat([])});
rule.declarations = rule.declarations.concat(fixupDeclarations); // add our custom declarations to the end of the current rules
}
}else if(rule.rules){
for (var k = 0; k < rule.rules.length; k++) {
insertFixupDeclarations(rule.rules[k]);
};
}
}
}
function createFixupFlexboxDeclaration(decl, parent){
var propname = decl.property, value = decl.value;
// remove -webkit- prefixing from names, values
if(/^-webkit-/.test(propname)){
propname = propname.substr(8);
}
if(/^-webkit-/.test(value)){
value = value.substr(8);
}
var mappings = {
'display':{
valueMap: {
'box':'flex',
'flexbox':'flex',
'inline-box':'inline-flex',
'inline-flexbox':'inline-flex'
}
},
'box-align':{
newName: 'align-items',
valueMap:{
'start': 'flex-start',
'end': 'flex-end'
}
},
'flex-direction':{
valueMap:{
'lr': 'row',
'rl': 'row-reverse',
'tb': 'column',
'bt': 'column-reverse'
}
},
'box-pack':{
newName:'justify-content',
valueMap:{
'start': 'flex-start',
'end': 'flex-end',
'justify': 'space-between'
}
},
'box-ordinal-group':{
newName:'order',
valueMap:{}
},
'box-flex':{
newName:'flex',
valueMap:{}
}
};
mappings['flex-align'] = mappings['box-align']; // 2009 => 2011
mappings['flex-order'] = mappings['box-ordinal-group']; // 2009 => 2011
if(propname in mappings){
if(value in mappings[propname].valueMap){
value = mappings[propname].valueMap[value];
}
propname = mappings[propname].newName || propname;
}
// some stuff is more complicated than a simple substitution..
// box-flex:0 maps to 'none', other values need 'auto' appended - thanks to Daniel Holbert
if(propname === 'flex'){
if(decl.value == 0){
value = 'none';
}else{
value = decl.value+' auto';
}
}
// box-direction, box-orient is a bit of a mess - these two 2009 draft values turn into 2011's flex-direction, which again has different values for final spec
if(propname === 'box-direction' || propname === 'box-orient'){
var dir, orient;
if(propname === 'box-direction'){
dir = value;
orient = getValueForProperty(parent, 'box-orient', true);
}else {
orient = value;
dir = getValueForProperty(parent, 'box-direction', true);
}
// horizontal,normal => row, vertical,normal => column. horizontal,reverse => row-reverse etc..
// lr, rl etc are handled by the simpler mapping above, so we don't need to worry about those
value = orient === 'vertical' ? 'column' : 'row';
if(dir === 'reverse'){
value += '-reverse';
}
propname = 'flex-direction';
}
return {type:'declaration', property:propname, value:value, _fxjsdefined:true};
}
function createFixupGradientDeclaration(decl, parent){
var value = decl.value.trim(), newValue='', prop = decl.property.replace(/-webkit-/, '');
// -webkit-gradient(<type>, <point> [, <radius>]?, <point> [, <radius>]? [, <stop>]*)
// fff -webkit-gradient(linear,0 0,0 100%,from(#fff),to(#f6f6f6));
// Sometimes there is code before the -webkit-gradient, for example when it's part of a more complex background: shorthand declaration
// we'll extract and keep any stuff before -webkit-gradient before we try parsing the gradient part
var head = value.substr(0, value.indexOf('-webkit-gradient'));
if(head){
value = value.substr(head.length);
}
var m = value.match(/-webkit-gradient\s*\(\s*(linear|radial)\s*(.*)/);
if(m){ // yay, really old syntax...
// extracting the values..
var parts = oldGradientParser(value), type; //GM_log(JSON.stringify(parts, null, 2))
for(var i = 0; i < parts.length; i++){
if(!parts[i].args)continue;
if(parts[i].name === '-webkit-gradient'){
type = parts[i].args[0].name;
newValue += type + '-gradient('; // radial or linear
}
newValue += standardizeOldGradientArgs(type, parts[i].args.slice(1));
newValue += ')' // end of gradient method
if (i < parts.length - 1) {
newValue += ', '
}
}
}else{ // we're dealing with more modern syntax - should be somewhat easier, at least for linear gradients.
// Fix three things: remove -webkit-, add 'to ' before reversed top/bottom keywords (linear) or 'at ' before position keywords (radial), recalculate deg-values
// -webkit-linear-gradient( [ [ <angle> | [top | bottom] || [left | right] ],]? <color-stop>[, <color-stop>]+);
newValue = value.replace(/-webkit-/, '');
// Keywords top, bottom, left, right: can be stand-alone or combined pairwise but in any order ('top left' or 'left top')
// These give the starting edge or corner in the -webkit syntax. The standardised equivalent is 'to ' plus opposite values for linear gradients, 'at ' plus same values for radial gradients
if(newValue.indexOf('linear') > -1){
newValue = newValue.replace(/(top|bottom|left|right)+\s*(top|bottom|left|right)*/, function(str){
var words = str.split(/\s+/);
for(var i=0; i<words.length; i++){
switch(words[i].toLowerCase()){
case 'top':
words[i] = 'bottom';
break;
case 'bottom':
words[i] = 'top';
break;
case 'left':
words[i] = 'right';
break;
case 'right':
words[i] = 'left';
}
}
str = words.join(' ');
return ( 'to ' + str);
});
}else{
newValue = newValue.replace(/(top|bottom|left|right)+\s/, 'at $1 ');
}
newValue = newValue.replace(/\d+deg/, function (val) {
return (360 - (parseInt(val)-90))+'deg';
});
}
// Similar to the code for "head" above..
var tail = value.substr(value.lastIndexOf(')')+1);
if( tail && tail.trim() !== ','){ // sometimes the gradient is a part of a more complex declaration (for example an image shorthand with stuff like no-repeat added), and what's after the gradient needs to be included
newValue += tail;
}
if(head){
newValue = head + newValue;
}
// GM_log(newValue)
return {type:'declaration', property:prop, value:newValue, _fxjsdefined:true};
}
function getValueForProperty(declarations, property, prefixAgnostic){
for(var i=0;i<declarations.length;i++){
if(declarations[i].property == property || (prefixAgnostic && declarations[i].property.substr(8) == property)) return declarations[i].value;
}
}
function hasDeclaration(declarations, property, value, prefixAgnostic, checkFunctionNameOnly){
if(checkFunctionNameOnly && value.indexOf('(') > -1){
value = value.match(/\b(\w+)\(/)[1];
var rx = new RegExp("(^|[\\s:,]+)" + value + "\\s*\\(", "i");
}else{
checkFunctionNameOnly = false;
}
for(var i=0;i<declarations.length;i++){
if(declarations[i].property == property || (prefixAgnostic && declarations[i].property.substr(8) == property)){
if(declarations[i].value === value || checkFunctionNameOnly && rx.test(declarations[i].value)){
return true;
}
}
}
return false;
}
function oldGradientParser(str){
/** This method takes a legacy -webkit-gradient() method call and parses it
to pull out the values, function names and their arguments.
It returns something like [{name:'-webkit-gradient',args:[{name:'linear'}, {name:'top left'} ... ]}]
*/
var objs = [{}], path=[], current, word='', separator_chars = [',', '(', ')'];
current = objs[0], path[0] = objs;
//str = str.replace(/\s*\(/g, '('); // sorry, ws in front of ( would make parsing a lot harder
for(var i = 0; i < str.length; i++){
if(separator_chars.indexOf(str[i]) === -1){
word += str[i];
}else{ // now we have a "separator" - presumably we've also got a "word" or value
current.name = word.trim();
//GM_log(word+' '+path.length+' '+str[i])
word = '';
if(str[i] === '('){ // we assume the 'word' is a function, for example -webkit-gradient() or rgb(), so we create a place to record the arguments
if(!('args' in current)){
current.args = [];
}
current.args.push({});
path.push(current.args);
current = current.args[current.args.length - 1];
path.push(current);
}else if(str[i] === ')'){ // function is ended, no more arguments - go back to appending details to the previous branch of the tree
current = path.pop(); // drop 'current'
current = path.pop(); // drop 'args' reference
}else{
path.pop(); // remove 'current' object from path, we have no arguments to add
var current_parent = path[path.length - 1] || objs; // last object on current path refers to array that contained the previous "current"
current_parent.push({}); // we need a new object to hold this "word" or value
current = current_parent[current_parent.length - 1]; // that object is now the 'current'
path.push(current);
//GM_log(path.length)
}
}
}
return objs;
}
/* Given an array of args for "-webkit-gradient(...)" returned by
* oldGradientParser(), this function constructs a string representing the
* equivalent arguments for a standard "linear-gradient(...)" or
* "radial-gradient(...)" expression.
*
* @param type Either 'linear' or 'radial'.
* @param args An array of args for a "-webkit-gradient(...)" expression,
* provided by oldGradientParser() (not including gradient type).
*/
function standardizeOldGradientArgs(type, args){
var stdArgStr = "";
var stops = [];
if(/^linear/.test(type)){
// linear gradient, args 1 and 2 tend to be start/end keywords
var points = [].concat(args[0].name.split(/\s+/), args[1].name.split(/\s+/)); // example: [left, top, right, top]
// Old webkit syntax "uses a two-point syntax that lets you explicitly state where a linear gradient starts and ends"
// if start/end keywords are percentages, let's massage the values a little more..
var rxPercTest = /\d+\%/;
if(rxPercTest.test(points[0]) || points[0] == 0){
var startX = parseInt(points[0]), startY = parseInt(points[1]), endX = parseInt(points[2]), endY = parseInt(points[3]);
stdArgStr += ((Math.atan2(endY- startY, endX - startX)) * (180 / Math.PI)+90) + 'deg';
}else{
if(points[1] === points[3]){ // both 'top' or 'bottom, this linear gradient goes left-right
stdArgStr += 'to ' + points[2];
}else if(points[0] === points[2]){ // both 'left' or 'right', this linear gradient goes top-bottom
stdArgStr += 'to ' + points[3];
}else if(points[1] === 'top'){ // diagonal gradient - from top left to opposite corner is 135deg
stdArgStr += '135deg';
}else{
stdArgStr += '45deg';
}
}
}else if(/^radial/i.test(type)){ // oooh, radial gradients..
stdArgStr += 'circle ' + args[3].name.replace(/(\d+)$/, '$1px') + ' at ' + args[0].name.replace(/(\d+) /, '$1px ').replace(/(\d+)$/, '$1px');
}
var toColor;
for(var j = type === 'linear' ? 2 : 4; j < args.length; j++){
var position, color, colorIndex;
if(args[j].name === 'color-stop'){
position = args[j].args[0].name;
colorIndex = 1;
}else if (args[j].name === 'to') {
position = '100%';
colorIndex = 0;
}else if (args[j].name === 'from') {
position = '0%';
colorIndex = 0;
};
if (position.indexOf('%') === -1) { // original Safari syntax had 0.5 equivalent to 50%
position = (parseFloat(position) * 100) +'%';
};
color = args[j].args[colorIndex].name;
if (args[j].args[colorIndex].args) { // the color is itself a function call, like rgb()
color += '(' + colorValue(args[j].args[colorIndex].args) + ')';
};
if (args[j].name === 'from'){
stops.unshift(color + ' ' + position);
}else if(args[j].name === 'to'){
toColor = color;
}else{
stops.push(color + ' ' + position);
}
}
// translating values to right syntax
for(var j = 0; j < stops.length; j++){
stdArgStr += ', ' + stops[j];
}
if(toColor){
stdArgStr += ', ' + toColor + ' 100%';
}
return stdArgStr;
}
function colorValue(obj){
var ar = [];
for (var i = 0; i < obj.length; i++) {
ar.push(obj[i].name);
};
return ar.join(', ');
}
function stripHTMLComments (str) {
str = str.replace(/^\s*<!--/, '');
str = str.replace(/-->\s*$/, '');
return str;
}
function GM_log(){}
</script>
</body>
</html>