diff --git a/src/twig/twig.test.ts b/src/twig/twig.test.ts
index 7288dad4..c42318d6 100644
--- a/src/twig/twig.test.ts
+++ b/src/twig/twig.test.ts
@@ -7,6 +7,636 @@
import { testTokenization } from '../test/testRunner';
+/**
+ * HTML Tests, without the html substate
+ */
+testTokenization(['twig', 'css', 'javascript'], [
+
+ // Open Start Tag #1'
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: '' }
+ ]
+ }],
+
+ // Open Start Tag #4
+ [{
+ line: 'i ',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: 'delimiter' }
+ ]
+ }],
+
+ // Complete Start Tag with Whitespace
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'delimiter' }
+ ]
+ }],
+
+ // bug 9809 - Complete Start Tag with Namespaceprefix
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 8, type: 'delimiter' }
+ ]
+ }],
+
+ // Complete End Tag
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 2, type: 'tag' },
+ { startIndex: 5, type: 'delimiter' }
+ ]
+ }],
+
+ // Complete End Tag with Whitespace
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 2, type: 'tag' },
+ { startIndex: 5, type: '' },
+ { startIndex: 7, type: 'delimiter' }
+ ]
+ }],
+
+ // Empty Tag
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #1
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 7, type: '' },
+ { startIndex: 8, type: 'attribute.name' },
+ { startIndex: 12, type: 'delimiter' },
+ { startIndex: 13, type: 'attribute.value' },
+ { startIndex: 30, type: 'delimiter' },
+ { startIndex: 31, type: 'keyword.js' },
+ { startIndex: 34, type: '' },
+ { startIndex: 35, type: 'identifier.js' },
+ { startIndex: 36, type: 'delimiter.js' },
+ { startIndex: 37, type: '' },
+ { startIndex: 38, type: 'number.js' },
+ { startIndex: 40, type: 'delimiter.js' },
+ { startIndex: 41, type: 'delimiter' },
+ { startIndex: 43, type: 'tag' },
+ { startIndex: 49, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #2
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 2, type: 'tag' },
+ { startIndex: 8, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #3
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 2, type: 'tag' },
+ { startIndex: 8, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #4
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'keyword.js' },
+ { startIndex: 3, type: '' },
+ { startIndex: 4, type: 'identifier.js' },
+ { startIndex: 5, type: 'delimiter.js' },
+ { startIndex: 6, type: '' },
+ { startIndex: 7, type: 'number.js' },
+ { startIndex: 9, type: 'delimiter.js' },
+ { startIndex: 10, type: 'delimiter' },
+ { startIndex: 12, type: 'tag' },
+ { startIndex: 18, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #5
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: '' },
+ { startIndex: 2, type: 'delimiter' },
+ { startIndex: 4, type: 'tag' },
+ { startIndex: 10, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #6
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 7, type: 'delimiter' },
+ { startIndex: 8, type: 'identifier.js' },
+ { startIndex: 9, type: 'delimiter' },
+ { startIndex: 11, type: 'tag' },
+ { startIndex: 17, type: 'delimiter' },
+ // { startIndex:18, type: 'delimiter' },
+ { startIndex: 19, type: 'tag' },
+ { startIndex: 25, type: 'delimiter' },
+ { startIndex: 26, type: 'identifier.js' },
+ { startIndex: 27, type: 'delimiter' },
+ { startIndex: 29, type: 'tag' },
+ { startIndex: 35, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #7
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 7, type: '' },
+ { startIndex: 8, type: 'attribute.name' },
+ { startIndex: 12, type: 'delimiter' },
+ { startIndex: 13, type: 'attribute.value' },
+ { startIndex: 30, type: 'delimiter' },
+ // { startIndex:31, type: 'delimiter' },
+ { startIndex: 33, type: 'tag' },
+ { startIndex: 39, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #8
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 7, type: 'delimiter' },
+ { startIndex: 8, type: 'keyword.js' },
+ { startIndex: 11, type: '' },
+ { startIndex: 12, type: 'identifier.js' },
+ { startIndex: 13, type: 'delimiter.js' },
+ { startIndex: 14, type: '' },
+ { startIndex: 15, type: 'number.js' },
+ { startIndex: 17, type: 'delimiter.js' },
+ { startIndex: 18, type: 'delimiter' },
+ { startIndex: 20, type: 'tag' },
+ { startIndex: 26, type: 'delimiter' }
+ ]
+ }],
+
+ // Embedded Content #9
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 7, type: '' },
+ { startIndex: 8, type: 'attribute.name' },
+ { startIndex: 12, type: 'delimiter' },
+ { startIndex: 13, type: 'attribute.value' },
+ { startIndex: 30, type: '' },
+ { startIndex: 31, type: 'attribute.name' },
+ { startIndex: 34, type: 'delimiter' },
+ { startIndex: 35, type: 'attribute.value' },
+ { startIndex: 44, type: 'delimiter' },
+ // { startIndex:45, type: 'delimiter' },
+ { startIndex: 47, type: 'tag' },
+ { startIndex: 53, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Attribute
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: 'attribute.value' },
+ { startIndex: 14, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Empty Attribute Value
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: 'attribute.value' },
+ { startIndex: 14, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with empty attributes
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: 'attribute.value' },
+ { startIndex: 11, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Attributes
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: 'attribute.value' },
+ { startIndex: 14, type: '' },
+ { startIndex: 15, type: 'attribute.name' },
+ { startIndex: 18, type: 'delimiter' },
+ { startIndex: 19, type: 'attribute.value' },
+ { startIndex: 24, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Attributes, no quotes
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: 'attribute.name' }, // slightly incorrect
+ { startIndex: 12, type: '' },
+ { startIndex: 13, type: 'attribute.name' },
+ { startIndex: 16, type: 'delimiter' },
+ { startIndex: 17, type: 'attribute.name' }, // slightly incorrect
+ { startIndex: 24, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Attribute And Whitespace
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' },
+ { startIndex: 9, type: '' },
+ { startIndex: 11, type: 'attribute.value' },
+ { startIndex: 16, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Attribute And Whitespace #2
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: '' },
+ { startIndex: 9, type: 'delimiter' },
+ { startIndex: 10, type: '' },
+ { startIndex: 11, type: 'attribute.value' },
+ { startIndex: 16, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Name-Only-Attribute #1
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Name-Only-Attribute #2
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: '' },
+ { startIndex: 9, type: 'attribute.name' },
+ { startIndex: 12, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Interesting Attribute Name
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 5, type: 'attribute.name' },
+ { startIndex: 8, type: '' },
+ { startIndex: 11, type: 'delimiter' },
+ { startIndex: 12, type: 'attribute.value' },
+ { startIndex: 17, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Angular Attribute Name
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 4, type: '' },
+ { startIndex: 6, type: 'attribute.name' },
+ { startIndex: 13, type: '' },
+ { startIndex: 15, type: 'attribute.name' },
+ { startIndex: 20, type: '' },
+ { startIndex: 21, type: 'delimiter' },
+ { startIndex: 22, type: 'attribute.value' },
+ { startIndex: 27, type: '' },
+ { startIndex: 29, type: 'attribute.name' },
+ { startIndex: 34, type: '' },
+ { startIndex: 35, type: 'delimiter' },
+ { startIndex: 36, type: 'attribute.value' },
+ { startIndex: 50, type: '' },
+ { startIndex: 52, type: 'attribute.name' },
+ { startIndex: 56, type: 'delimiter' },
+ { startIndex: 57, type: 'attribute.value' },
+ { startIndex: 72, type: 'delimiter' }
+ ]
+ }],
+
+ // Tag with Invalid Attribute Value
+ [{
+ line: '',
+ tokens: [
+ { startIndex: 0, type: 'metatag.content' },
+ { startIndex: 11, type: 'metatag' }
+ ]
+ }],
+
+ // PR #14
+ [{
+ line: 'asd',
+ tokens: [
+ { startIndex: 0, type: 'delimiter' },
+ { startIndex: 1, type: 'tag' },
+ { startIndex: 9, type: 'delimiter' },
+ { startIndex: 10, type: '' },
+ { startIndex: 13, type: 'delimiter' },
+ { startIndex: 15, type: 'tag' },
+ { startIndex: 23, type: 'delimiter' }
+ ]
+ }]
+]);
+
+/**
+ * Twig Tests
+ */
testTokenization(['twig'], [
/**
* Comments