gulpfile.js 6.6 KB
Newer Older
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
var gulp = require('gulp');
var replace = require('gulp-replace');
var rename = require('gulp-rename');

const languages = [
	{ name: 'css', identifiers: ['css', 'css.erb'], source: 'source.css' },
	{ name: 'basic', identifiers: ['html', 'htm', 'shtml', 'xhtml', 'inc', 'tmpl', 'tpl'], source: 'text.html.basic' },
	{ name: 'ini', identifiers: ['ini', 'conf'], source: 'source.ini' },
	{ name: 'java', identifiers: ['java', 'bsh'], source: 'source.java' },
	{ name: 'lua', identifiers: ['lua'], source: 'source.lua' },
	{ name: 'makefile', identifiers: ['Makefile', 'makefile', 'GNUmakefile', 'OCamlMakefile'], source: 'source.makefile' },
	{ name: 'perl', identifiers: ['perl', 'pl', 'pm', 'pod', 't', 'PL', 'psgi', 'vcl'], source: 'source.perl' },
	{ name: 'r', identifiers: ['R', 'r', 's', 'S', 'Rprofile'], source: 'source.r' },
	{ name: 'ruby', identifiers: ['ruby', 'rb', 'rbx', 'rjs', 'Rakefile', 'rake', 'cgi', 'fcgi', 'gemspec', 'irbrc', 'Capfile', 'ru', 'prawn', 'Cheffile', 'Gemfile', 'Guardfile', 'Hobofile', 'Vagrantfile', 'Appraisals', 'Rantfile', 'Berksfile', 'Berksfile.lock', 'Thorfile', 'Puppetfile'], source: 'source.ruby' },
	// 	Left to its own devices, the PHP grammar will match HTML as a combination of operators
	// and constants. Therefore, HTML must take precedence over PHP in order to get proper
	// syntax highlighting.
	{ name: 'php', identifiers: ['php', 'php3', 'php4', 'php5', 'phpt', 'phtml', 'aw', 'ctp'], source: ['text.html.basic', 'text.html.php#language'] },
	{ name: 'sql', identifiers: ['sql', 'ddl', 'dml'], source: 'source.sql' },
	{ name: 'vs_net', identifiers: ['vb'], source: 'source.asp.vb.net' },
	{ name: 'xml', identifiers: ['xml', 'xsd', 'tld', 'jsp', 'pt', 'cpt', 'dtml', 'rss', 'opml'], source: 'text.xml' },
	{ name: 'xsl', identifiers: ['xsl', 'xslt'], source: 'text.xml.xsl' },
	{ name: 'yaml', identifiers: ['yaml', 'yml'], source: 'source.yaml' },
	{ name: 'dosbatch', identifiers: ['bat', 'batch'], source: 'source.dosbatch' },
	{ name: 'clojure', identifiers: ['clj', 'cljs', 'clojure'], source: 'source.clojure' },
	{ name: 'coffee', identifiers: ['coffee', 'Cakefile', 'coffee.erb'], source: 'source.coffee' },
	{ name: 'c', identifiers: ['c', 'h'], source: 'source.c' },
	{ name: 'cpp', identifiers: ['cpp', 'c\\+\\+', 'cxx'], source: 'source.cpp' },
	{ name: 'diff', identifiers: ['patch', 'diff', 'rej'], source: 'source.diff' },
	{ name: 'dockerfile', identifiers: ['dockerfile', 'Dockerfile'], source: 'source.dockerfile' },
	{ name: 'git_commit', identifiers: ['COMMIT_EDITMSG', 'MERGE_MSG'], source: 'text.git-commit' },
	{ name: 'git_rebase', identifiers: ['git-rebase-todo'], source: 'text.git-rebase' },
	{ name: 'go', identifiers: ['go', 'golang'], source: 'source.go' },
	{ name: 'groovy', identifiers: ['groovy', 'gvy'], source: 'source.groovy' },
	{ name: 'jade', identifiers: ['jade'], source: 'text.jade' },

	{ name: 'js', identifiers: ['js', 'jsx', 'javascript'], source: 'source.js' },
	{ name: 'js_regexp', identifiers: ['regexp'], source: 'source.js.regexp' },
	{ name: 'json', identifiers: ['json', 'sublime-settings', 'sublime-menu', 'sublime-keymap', 'sublime-mousemap', 'sublime-theme', 'sublime-build', 'sublime-project', 'sublime-completions'], source: 'source.json' },
	{ name: 'less', identifiers: ['less'], source: 'source.css.less' },
	{ name: 'objc', identifiers: ['objectivec', 'objective-c', 'mm', 'objc', 'obj-c', 'm', 'h'], source: 'source.objc' },
	{ name: 'perl6', identifiers: ['perl6', 'p6', 'pl6', 'pm6', 'nqp'], source: 'source.perl.6' },
	{ name: 'powershell', identifiers: ['powershell', 'ps1', 'psm1', 'psd1'], source: 'source.powershell' },
	{ name: 'python', identifiers: ['python', 'py', 'py3', 'rpy', 'pyw', 'cpy', 'SConstruct', 'Sconstruct', 'sconstruct', 'SConscript', 'gyp', 'gypi'], source: 'source.python' },
	{ name: 'regexp_python', identifiers: ['re'], source: 'source.regexp.python' },
	{ name: 'rust', identifiers: ['rust', 'rs'], source: 'source.rust' },
	{ name: 'scala', identifiers: ['scala', 'sbt'], source: 'source.scala' },
	{ name: 'shell', identifiers: ['shell', 'sh', 'bash', 'zsh', 'bashrc', 'bash_profile', 'bash_login', 'profile', 'bash_logout', '.textmate_init'], source: 'source.shell' },
	{ name: 'ts', identifiers: ['typescript', 'ts'], source: 'source.ts' },
	{ name: 'tsx', identifiers: ['tsx'], source: 'source.tsx' },
	{ name: 'csharp', identifiers: ['cs', 'csharp', 'c#'], source: 'source.cs' },
	{ name: 'fsharp', identifiers: ['fs', 'fsharp', 'f#'], source: 'source.fsharp' },
];

const fencedCodeBlockDefinition = (name, identifiers, sourceScope) => {
	if (!Array.isArray(sourceScope)) {
		sourceScope = [sourceScope];
	}

	const scopes = sourceScope.map(scope =>
		`<dict>
	<key>include</key>
	<string>${scope}</string>
</dict>`).join('\n');

	return `<key>fenced_code_block_${name}</key>
<dict>
	<key>begin</key>
	<string>(^|\\G)(\\s*)([\`~]{3,})\\s*((${identifiers.join('|')})(\\s+[^\`~]*)?$)</string>
	<key>name</key>
	<string>markup.fenced_code.block.markdown</string>
	<key>end</key>
	<string>(^|\\G)(\\2|\\s{0,3})(\\3)\\s*$</string>
	<key>beginCaptures</key>
	<dict>
		<key>3</key>
		<dict>
			<key>name</key>
			<string>punctuation.definition.markdown</string>
		</dict>
		<key>5</key>
		<dict>
			<key>name</key>
			<string>fenced_code.block.language</string>
		</dict>
		<key>6</key>
		<dict>
			<key>name</key>
			<string>fenced_code.block.language.attributes</string>
		</dict>
	</dict>
	<key>endCaptures</key>
	<dict>
		<key>3</key>
		<dict>
			<key>name</key>
			<string>punctuation.definition.markdown</string>
		</dict>
	</dict>
	<key>patterns</key>
	<array>
		<dict>
			<key>begin</key>
			<string>(^|\\G)(\\s*)(.*)</string>
			<key>while</key>
			<string>(^|\\G)(?!\\s*([\`~]{3,})\\s*$)</string>
			<key>patterns</key>
			<array>
${indent(4, scopes)}
			</array>
		</dict>
	</array>
</dict>`;
};

const indent = (count, text) => {
	const indent = new Array(count + 1).join('\t');
	return text.replace(/^/gm, indent);
};

const fencedCodeBlockInclude = (name) =>
	`<dict>
	<key>include</key>
	<string>#fenced_code_block_${name}</string>
</dict>`;


const fencedCodeBlockDefinitions = () =>
	languages
		.map(language => fencedCodeBlockDefinition(language.name, language.identifiers, language.source))
		.join('\n');



const fencedCodeBlockIncludes = () =>
	languages
		.map(language => fencedCodeBlockInclude(language.name))
		.join('\n');


gulp.task('default', function () {
	gulp.src(['markdown.tmLanguage.base'])
		.pipe(replace('{{languageIncludes}}', indent(4, fencedCodeBlockIncludes())))
		.pipe(replace('{{languageDefinitions}}', indent(4, fencedCodeBlockDefinitions())))
		.pipe(rename('markdown.tmLanguage'))
		.pipe(gulp.dest('.'));
});