Initial code import. 

This plug-in is pretty functional already:

* Working editor with content-assist based on grammar, context and data type
* Imports of .proto files works
* Content-assist for common options
* Outline View
* Preferences for using protoc on save
* Error markers based on protoc's output
diff --git a/com.google.eclipse.protobuf.ui/.classpath b/com.google.eclipse.protobuf.ui/.classpath
new file mode 100644
index 0000000..7e8449d
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/.classpath
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<classpath>
+	<classpathentry kind="src" path="src"/>
+	<classpathentry kind="src" path="src-gen"/>
+	<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/J2SE-1.5"/>
+	<classpathentry kind="con" path="org.eclipse.pde.core.requiredPlugins"/>
+	<classpathentry kind="output" path="bin"/>
+</classpath>
diff --git a/com.google.eclipse.protobuf.ui/.project b/com.google.eclipse.protobuf.ui/.project
new file mode 100644
index 0000000..aa5cde0
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/.project
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<projectDescription>
+	<name>com.google.eclipse.protobuf.ui</name>
+	<comment></comment>
+	<projects>
+	</projects>
+	<buildSpec>
+		<buildCommand>
+			<name>org.eclipse.jdt.core.javabuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+		<buildCommand>
+			<name>org.eclipse.pde.ManifestBuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+		<buildCommand>
+			<name>org.eclipse.pde.SchemaBuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+	</buildSpec>
+	<natures>
+		<nature>org.eclipse.jdt.core.javanature</nature>
+		<nature>org.eclipse.pde.PluginNature</nature>
+	</natures>
+</projectDescription>
diff --git a/com.google.eclipse.protobuf.ui/.settings/org.eclipse.jdt.core.prefs b/com.google.eclipse.protobuf.ui/.settings/org.eclipse.jdt.core.prefs
new file mode 100644
index 0000000..47c08ff
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/.settings/org.eclipse.jdt.core.prefs
@@ -0,0 +1,81 @@
+#Wed Apr 06 10:56:25 PDT 2011
+eclipse.preferences.version=1
+org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled
+org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.5
+org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve
+org.eclipse.jdt.core.compiler.compliance=1.5
+org.eclipse.jdt.core.compiler.debug.lineNumber=generate
+org.eclipse.jdt.core.compiler.debug.localVariable=generate
+org.eclipse.jdt.core.compiler.debug.sourceFile=generate
+org.eclipse.jdt.core.compiler.problem.annotationSuperInterface=warning
+org.eclipse.jdt.core.compiler.problem.assertIdentifier=error
+org.eclipse.jdt.core.compiler.problem.autoboxing=ignore
+org.eclipse.jdt.core.compiler.problem.comparingIdentical=warning
+org.eclipse.jdt.core.compiler.problem.deadCode=warning
+org.eclipse.jdt.core.compiler.problem.deprecation=warning
+org.eclipse.jdt.core.compiler.problem.deprecationInDeprecatedCode=disabled
+org.eclipse.jdt.core.compiler.problem.deprecationWhenOverridingDeprecatedMethod=disabled
+org.eclipse.jdt.core.compiler.problem.discouragedReference=warning
+org.eclipse.jdt.core.compiler.problem.emptyStatement=ignore
+org.eclipse.jdt.core.compiler.problem.enumIdentifier=error
+org.eclipse.jdt.core.compiler.problem.fallthroughCase=ignore
+org.eclipse.jdt.core.compiler.problem.fatalOptionalError=disabled
+org.eclipse.jdt.core.compiler.problem.fieldHiding=warning
+org.eclipse.jdt.core.compiler.problem.finalParameterBound=warning
+org.eclipse.jdt.core.compiler.problem.finallyBlockNotCompletingNormally=warning
+org.eclipse.jdt.core.compiler.problem.forbiddenReference=error
+org.eclipse.jdt.core.compiler.problem.hiddenCatchBlock=warning
+org.eclipse.jdt.core.compiler.problem.includeNullInfoFromAsserts=disabled
+org.eclipse.jdt.core.compiler.problem.incompatibleNonInheritedInterfaceMethod=warning
+org.eclipse.jdt.core.compiler.problem.incompleteEnumSwitch=ignore
+org.eclipse.jdt.core.compiler.problem.indirectStaticAccess=warning
+org.eclipse.jdt.core.compiler.problem.localVariableHiding=warning
+org.eclipse.jdt.core.compiler.problem.methodWithConstructorName=warning
+org.eclipse.jdt.core.compiler.problem.missingDeprecatedAnnotation=ignore
+org.eclipse.jdt.core.compiler.problem.missingHashCodeMethod=warning
+org.eclipse.jdt.core.compiler.problem.missingOverrideAnnotation=ignore
+org.eclipse.jdt.core.compiler.problem.missingOverrideAnnotationForInterfaceMethodImplementation=enabled
+org.eclipse.jdt.core.compiler.problem.missingSerialVersion=warning
+org.eclipse.jdt.core.compiler.problem.missingSynchronizedOnInheritedMethod=warning
+org.eclipse.jdt.core.compiler.problem.noEffectAssignment=warning
+org.eclipse.jdt.core.compiler.problem.noImplicitStringConversion=warning
+org.eclipse.jdt.core.compiler.problem.nonExternalizedStringLiteral=ignore
+org.eclipse.jdt.core.compiler.problem.nullReference=warning
+org.eclipse.jdt.core.compiler.problem.overridingPackageDefaultMethod=warning
+org.eclipse.jdt.core.compiler.problem.parameterAssignment=warning
+org.eclipse.jdt.core.compiler.problem.possibleAccidentalBooleanAssignment=warning
+org.eclipse.jdt.core.compiler.problem.potentialNullReference=warning
+org.eclipse.jdt.core.compiler.problem.rawTypeReference=warning
+org.eclipse.jdt.core.compiler.problem.redundantNullCheck=warning
+org.eclipse.jdt.core.compiler.problem.redundantSuperinterface=warning
+org.eclipse.jdt.core.compiler.problem.reportMethodCanBePotentiallyStatic=ignore
+org.eclipse.jdt.core.compiler.problem.reportMethodCanBeStatic=ignore
+org.eclipse.jdt.core.compiler.problem.specialParameterHidingField=disabled
+org.eclipse.jdt.core.compiler.problem.staticAccessReceiver=warning
+org.eclipse.jdt.core.compiler.problem.suppressOptionalErrors=disabled
+org.eclipse.jdt.core.compiler.problem.suppressWarnings=enabled
+org.eclipse.jdt.core.compiler.problem.syntheticAccessEmulation=ignore
+org.eclipse.jdt.core.compiler.problem.typeParameterHiding=warning
+org.eclipse.jdt.core.compiler.problem.unavoidableGenericTypeProblems=enabled
+org.eclipse.jdt.core.compiler.problem.uncheckedTypeOperation=warning
+org.eclipse.jdt.core.compiler.problem.undocumentedEmptyBlock=ignore
+org.eclipse.jdt.core.compiler.problem.unhandledWarningToken=warning
+org.eclipse.jdt.core.compiler.problem.unnecessaryElse=warning
+org.eclipse.jdt.core.compiler.problem.unnecessaryTypeCheck=warning
+org.eclipse.jdt.core.compiler.problem.unqualifiedFieldAccess=ignore
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownException=warning
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionExemptExceptionAndThrowable=enabled
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionIncludeDocCommentReference=enabled
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionWhenOverriding=disabled
+org.eclipse.jdt.core.compiler.problem.unusedImport=warning
+org.eclipse.jdt.core.compiler.problem.unusedLabel=warning
+org.eclipse.jdt.core.compiler.problem.unusedLocal=warning
+org.eclipse.jdt.core.compiler.problem.unusedObjectAllocation=ignore
+org.eclipse.jdt.core.compiler.problem.unusedParameter=ignore
+org.eclipse.jdt.core.compiler.problem.unusedParameterIncludeDocCommentReference=enabled
+org.eclipse.jdt.core.compiler.problem.unusedParameterWhenImplementingAbstract=disabled
+org.eclipse.jdt.core.compiler.problem.unusedParameterWhenOverridingConcrete=disabled
+org.eclipse.jdt.core.compiler.problem.unusedPrivateMember=warning
+org.eclipse.jdt.core.compiler.problem.unusedWarningToken=warning
+org.eclipse.jdt.core.compiler.problem.varargsArgumentNeedCast=warning
+org.eclipse.jdt.core.compiler.source=1.5
diff --git a/com.google.eclipse.protobuf.ui/META-INF/MANIFEST.MF b/com.google.eclipse.protobuf.ui/META-INF/MANIFEST.MF
new file mode 100644
index 0000000..4098738
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/META-INF/MANIFEST.MF
@@ -0,0 +1,29 @@
+Manifest-Version: 1.0

+Bundle-ManifestVersion: 2

+Bundle-Name: %Bundle-Name

+Bundle-Vendor: %Bundle-Vendor

+Bundle-Version: 1.0.0

+Bundle-SymbolicName: com.google.eclipse.protobuf.ui; singleton:=true

+Bundle-ActivationPolicy: lazy

+Require-Bundle: com.google.eclipse.protobuf;visibility:=reexport,

+ org.eclipse.xtext.ui,

+ org.eclipse.ui.editors;bundle-version="3.5.0",

+ org.eclipse.ui.ide;bundle-version="3.5.0",

+ org.eclipse.xtext.ui.shared,

+ org.eclipse.ui,

+ org.eclipse.xtext.builder,

+ org.antlr.runtime,

+ org.eclipse.core.runtime,

+ org.eclipse.core.databinding,

+ org.eclipse.core.databinding.beans,

+ org.eclipse.core.databinding.observable,

+ org.eclipse.core.databinding.property,

+ org.eclipse.jface.databinding,

+ com.ibm.icu,

+ org.eclipse.emf.databinding,

+ org.eclipse.core.resources

+Import-Package: org.apache.log4j,

+ org.apache.commons.logging

+Bundle-RequiredExecutionEnvironment: J2SE-1.5

+Export-Package: com.google.eclipse.protobuf.ui.contentassist.antlr

+Bundle-Activator: com.google.eclipse.protobuf.ui.internal.ProtobufActivator

diff --git a/com.google.eclipse.protobuf.ui/OSGI-INF/l10n/bundle.properties b/com.google.eclipse.protobuf.ui/OSGI-INF/l10n/bundle.properties
new file mode 100644
index 0000000..b699865
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/OSGI-INF/l10n/bundle.properties
@@ -0,0 +1,19 @@
+#Properties file for com.google.eclipse.protobuf.ui
+Bundle-Vendor = Google Inc.
+Bundle-Name = com.google.eclipse.protobuf.ui
+editor.name = Protocol Buffer Editor
+page.name = Protocol Buffer
+page.name.0 = Syntax Coloring
+page.name.1 = Templates
+page.name.2 = Compiler
+page.name.3 = Protocol Buffer Compiler
+keyword.label = Protocol Buffer
+command.description = Trigger expensive validation
+command.name = Validate
+command.tooltip = Trigger expensive validation
+command.description.0 = Open the quick outline.
+command.name.0 = Quick Outline
+command.tooltip.0 = Open Quick Outline
+command.description.1 = Insert semicolon.
+command.name.1 = Insert semicolon
+command.tooltip.1 = Insert semicolon
\ No newline at end of file
diff --git a/com.google.eclipse.protobuf.ui/build.properties b/com.google.eclipse.protobuf.ui/build.properties
new file mode 100644
index 0000000..300fb1a
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/build.properties
@@ -0,0 +1,6 @@
+source.. = src/,\
+          src-gen/
+bin.includes = META-INF/,\
+               .,\
+               plugin.xml,\
+               OSGI-INF/
diff --git a/com.google.eclipse.protobuf.ui/icons/empty.gif b/com.google.eclipse.protobuf.ui/icons/empty.gif
new file mode 100644
index 0000000..9cd97c2
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/empty.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/enum.gif b/com.google.eclipse.protobuf.ui/icons/enum.gif
new file mode 100644
index 0000000..15535f5
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/enum.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/extend.gif b/com.google.eclipse.protobuf.ui/icons/extend.gif
new file mode 100644
index 0000000..e8abf43
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/extend.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/extensions.gif b/com.google.eclipse.protobuf.ui/icons/extensions.gif
new file mode 100644
index 0000000..13500b7
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/extensions.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/import.gif b/com.google.eclipse.protobuf.ui/icons/import.gif
new file mode 100644
index 0000000..9e44ce5
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/import.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/literal.gif b/com.google.eclipse.protobuf.ui/icons/literal.gif
new file mode 100644
index 0000000..5b881d8
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/literal.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/message.gif b/com.google.eclipse.protobuf.ui/icons/message.gif
new file mode 100644
index 0000000..e24e9e3
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/message.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/option.gif b/com.google.eclipse.protobuf.ui/icons/option.gif
new file mode 100644
index 0000000..9d8c615
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/option.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/package.gif b/com.google.eclipse.protobuf.ui/icons/package.gif
new file mode 100644
index 0000000..131c28d
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/package.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/pb.gif b/com.google.eclipse.protobuf.ui/icons/pb.gif
new file mode 100644
index 0000000..4fcebf9
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/pb.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/property-opt.gif b/com.google.eclipse.protobuf.ui/icons/property-opt.gif
new file mode 100644
index 0000000..6e3302d
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/property-opt.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/property-rep.gif b/com.google.eclipse.protobuf.ui/icons/property-rep.gif
new file mode 100644
index 0000000..ae61259
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/property-rep.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/property-req.gif b/com.google.eclipse.protobuf.ui/icons/property-req.gif
new file mode 100644
index 0000000..5611cce
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/property-req.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/property.gif b/com.google.eclipse.protobuf.ui/icons/property.gif
new file mode 100644
index 0000000..cd83b96
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/property.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/icons/protobuf.gif b/com.google.eclipse.protobuf.ui/icons/protobuf.gif
new file mode 100644
index 0000000..5f161d9
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/icons/protobuf.gif
Binary files differ
diff --git a/com.google.eclipse.protobuf.ui/plugin.xml b/com.google.eclipse.protobuf.ui/plugin.xml
new file mode 100644
index 0000000..6a3942f
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/plugin.xml
@@ -0,0 +1,243 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<?eclipse version="3.0"?>
+
+<plugin>
+
+    <extension
+            point="org.eclipse.ui.editors">
+        <editor
+              class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.XtextEditor"
+              contributorClass="org.eclipse.ui.editors.text.TextEditorActionContributor"
+              default="true"
+              extensions="proto"
+              icon="icons/pb.gif"
+              id="com.google.eclipse.protobuf.Protobuf"
+              name="%editor.name">
+        </editor>
+    </extension>
+    <extension
+        point="org.eclipse.ui.handlers">
+        <handler
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.hyperlinking.OpenDeclarationHandler"
+            commandId="org.eclipse.xtext.ui.editor.hyperlinking.OpenDeclaration">
+            <activeWhen>
+                <reference
+                    definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+                </reference>
+            </activeWhen>
+        </handler>
+        <handler
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.handler.ValidateActionHandler"
+            commandId="com.google.eclipse.protobuf.Protobuf.validate">
+         <activeWhen>
+            <reference
+                    definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+            </reference>
+         </activeWhen>
+      </handler>
+      <handler
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:com.google.eclipse.protobuf.ui.commands.SmartSemicolonHandler"
+            commandId="com.google.eclipse.protobuf.ui.smartSemicolon">
+         <activeWhen>
+            <reference
+                    definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+            </reference>
+         </activeWhen>
+      </handler>
+    </extension>
+    <extension point="org.eclipse.core.expressions.definitions">
+        <definition id="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+            <and>
+                <reference definitionId="isActiveEditorAnInstanceOfXtextEditor"/>
+                <with variable="activeEditor">
+                    <test property="org.eclipse.xtext.ui.editor.XtextEditor.languageName" 
+                        value="com.google.eclipse.protobuf.Protobuf" 
+                        forcePluginActivation="true"/>
+                </with>        
+            </and>
+        </definition>
+    </extension>
+    <extension
+            point="org.eclipse.ui.preferencePages">
+        <page
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.preferences.LanguageRootPreferencePage"
+            id="com.google.eclipse.protobuf.Protobuf"
+            name="%page.name">
+            <keywordReference id="com.google.eclipse.protobuf.ui.keyword_Protobuf"/>
+        </page>
+        <page
+            category="com.google.eclipse.protobuf.Protobuf"
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.syntaxcoloring.SyntaxColoringPreferencePage"
+            id="com.google.eclipse.protobuf.Protobuf.coloring"
+            name="%page.name.0">
+            <keywordReference id="com.google.eclipse.protobuf.ui.keyword_Protobuf"/>
+        </page>
+        <page
+            category="com.google.eclipse.protobuf.Protobuf"
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.templates.XtextTemplatePreferencePage"
+            id="com.google.eclipse.protobuf.Protobuf.templates"
+            name="%page.name.1">
+            <keywordReference id="com.google.eclipse.protobuf.ui.keyword_Protobuf"/>
+        </page>
+        <page
+            category="com.google.eclipse.protobuf.Protobuf"
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:com.google.eclipse.protobuf.ui.preferences.CompilerPreferencePage"
+            id="com.google.eclipse.protobuf.ui.preferences.CompilerPreferencePage"
+            name="%page.name.2">
+            <keywordReference id="com.google.eclipse.protobuf.ui.keyword_Protobuf"/>
+        </page>
+    </extension>
+    <extension
+        point="org.eclipse.ui.keywords">
+        <keyword
+            id="com.google.eclipse.protobuf.ui.keyword_Protobuf"
+            label="%keyword.label"/>
+    </extension>
+    <extension
+         point="org.eclipse.ui.commands">
+      <command
+            description="%command.description"
+            id="com.google.eclipse.protobuf.Protobuf.validate"
+            name="%command.name">
+      </command>
+    </extension>
+    <extension point="org.eclipse.ui.menus">
+        <menuContribution
+            locationURI="popup:#TextEditorContext?after=group.edit">
+             <command
+                 commandId="com.google.eclipse.protobuf.Protobuf.validate"
+                 style="push"
+                 tooltip="%command.tooltip">
+            <visibleWhen checkEnabled="false">
+                <reference
+                    definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+                </reference>
+            </visibleWhen>
+         </command>  
+         </menuContribution>
+    </extension>
+    <extension point="org.eclipse.ui.menus">
+		<menuContribution locationURI="popup:#TextEditorContext?endof=group.find">
+			<command commandId="org.eclipse.xtext.ui.editor.FindReferences">
+				<visibleWhen checkEnabled="false">
+                	<reference definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+                	</reference>
+            	</visibleWhen>
+			</command>
+		</menuContribution>
+	</extension>
+	<extension point="org.eclipse.ui.handlers">
+	    <handler
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.findrefs.FindReferencesHandler"
+            commandId="org.eclipse.xtext.ui.editor.FindReferences">
+            <activeWhen>
+                <reference
+                    definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+                </reference>
+            </activeWhen>
+        </handler>
+    </extension>   
+
+<!-- adding resource factories -->
+
+	<extension
+		point="org.eclipse.emf.ecore.extension_parser">
+		<parser
+			class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.resource.IResourceFactory"
+			type="proto">
+		</parser>
+	</extension>
+	<extension point="org.eclipse.xtext.extension_resourceServiceProvider">
+        <resourceServiceProvider
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.resource.IResourceUIServiceProvider"
+            uriExtension="proto">
+        </resourceServiceProvider>
+    </extension>
+
+
+
+	<!-- Quick Outline -->
+	<extension
+		point="org.eclipse.ui.handlers">
+		<handler 
+			class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.outline.quickoutline.ShowQuickOutlineActionHandler"
+			commandId="org.eclipse.xtext.ui.editor.outline.QuickOutline">
+			<activeWhen>
+				<reference
+					definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened">
+				</reference>
+			</activeWhen>
+		</handler>
+	</extension>
+	<extension
+		point="org.eclipse.ui.commands">
+		<command
+			description="%command.description.0"
+			id="org.eclipse.xtext.ui.editor.outline.QuickOutline"
+			name="%command.name.0">
+		</command>
+  <command
+        categoryId="org.eclipse.ui.category.textEditor"
+        description="%command.description.1"
+        id="com.google.eclipse.protobuf.ui.smartSemicolon"
+        name="%command.name.1">
+  </command>
+	</extension>
+	<extension point="org.eclipse.ui.menus">
+		<menuContribution
+			locationURI="popup:#TextEditorContext?after=group.open">
+			<command commandId="org.eclipse.xtext.ui.editor.outline.QuickOutline"
+				style="push"
+				tooltip="%command.tooltip.0">
+				<visibleWhen checkEnabled="false">
+					<reference definitionId="com.google.eclipse.protobuf.Protobuf.Editor.opened"/>
+				</visibleWhen>
+			</command>
+		</menuContribution>
+	</extension>
+   <!-- quickfix marker resolution generator -->
+   <extension
+         point="org.eclipse.ui.ide.markerResolution">
+      <markerResolutionGenerator
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:org.eclipse.xtext.ui.editor.quickfix.MarkerResolutionGenerator">
+      </markerResolutionGenerator>
+   </extension>
+   <extension
+         point="org.eclipse.ui.bindings">
+      <key
+            commandId="com.google.eclipse.protobuf.ui.smartSemicolon"
+            contextId="org.eclipse.xtext.ui.XtextEditorScope"
+            schemeId="org.eclipse.ui.defaultAcceleratorConfiguration"
+            sequence=";">
+      </key>
+   </extension>
+   <extension
+         point="org.eclipse.ui.propertyPages">
+      <page
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:com.google.eclipse.protobuf.ui.preferences.CompilerPreferencePage"
+            id="com.google.eclipse.protobuf.ui.properties.CompilerPropertyPage"
+            name="%page.name.3"
+            selectionFilter="single">
+      </page>
+   </extension>
+   <extension
+         point="org.eclipse.xtext.builder.participant">
+      <participant
+            class="com.google.eclipse.protobuf.ui.ProtobufExecutableExtensionFactory:com.google.eclipse.protobuf.ui.builder.ProtobufBuildParticipant">
+      </participant>
+   </extension>
+   <extension
+         id="pbmarker"
+         name="Protocol Buffer Marker"
+         point="org.eclipse.core.resources.markers">
+      <super
+            type="org.eclipse.core.resources.problemmarker">
+      </super>
+      <super
+            type="org.eclipse.core.resources.textmarker">
+      </super>
+      <persistent
+            value="true">
+      </persistent>
+   </extension>
+</plugin>
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/ProtobufUiModule.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/ProtobufUiModule.java
new file mode 100644
index 0000000..99e78a5
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/ProtobufUiModule.java
@@ -0,0 +1,56 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui;
+
+import org.eclipse.ui.plugin.AbstractUIPlugin;
+import org.eclipse.ui.views.contentoutline.IContentOutlinePage;
+import org.eclipse.xtext.ui.editor.IXtextEditorCallback;
+import org.eclipse.xtext.ui.editor.outline.actions.IOutlineContribution;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreInitializer;
+
+import com.google.eclipse.protobuf.ui.builder.AutoAddNatureEditorCallback;
+import com.google.eclipse.protobuf.ui.outline.LinkWithEditor;
+import com.google.eclipse.protobuf.ui.outline.ProtobufOutlinePage;
+import com.google.eclipse.protobuf.ui.preferences.CompilerPreferencesInitializer;
+import com.google.inject.Binder;
+import com.google.inject.name.Names;
+
+/**
+ * Use this class to register components to be used within the IDE.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufUiModule extends AbstractProtobufUiModule {
+
+  public ProtobufUiModule(AbstractUIPlugin plugin) {
+    super(plugin);
+  }
+
+  @Override public Class<? extends IContentOutlinePage> bindIContentOutlinePage() {
+    return ProtobufOutlinePage.class;
+  }
+
+  @Override public Class<? extends IXtextEditorCallback> bindIXtextEditorCallback() {
+    return AutoAddNatureEditorCallback.class;
+  }
+
+  /** {@inheritDoc} */
+  @Override public void configureToggleLinkWithEditorOutlineContribution(Binder binder) {
+    binder.bind(IOutlineContribution.class)
+          .annotatedWith(IOutlineContribution.LinkWithEditor.class)
+          .to(LinkWithEditor.class);
+  }
+
+  public void configureCompilerPreferencesInitializer(Binder binder) {
+    binder.bind(IPreferenceStoreInitializer.class)
+          .annotatedWith(Names.named("compilerPreferences"))
+          .to(CompilerPreferencesInitializer.class);
+  }
+
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/AutoAddNatureEditorCallback.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/AutoAddNatureEditorCallback.java
new file mode 100644
index 0000000..5c46840
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/AutoAddNatureEditorCallback.java
@@ -0,0 +1,38 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.builder;
+
+import org.eclipse.core.resources.IProject;
+import org.eclipse.core.resources.IResource;
+import org.eclipse.xtext.builder.nature.ToggleXtextNatureAction;
+import org.eclipse.xtext.builder.nature.XtextNature;
+import org.eclipse.xtext.ui.editor.AbstractDirtyStateAwareEditorCallback;
+import org.eclipse.xtext.ui.editor.XtextEditor;
+
+import com.google.inject.Inject;
+
+/**
+ * Automatically adds <code>{@link XtextNature}</code> and <code>{@link ProtobufNature}</code> to a project if needed
+ * (e.g. when opening a 'Protocol Buffer' editor for the first time.)
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class AutoAddNatureEditorCallback extends AbstractDirtyStateAwareEditorCallback {
+
+  @Inject private ToggleXtextNatureAction xtext;
+
+  @Override public void afterCreatePartControl(XtextEditor editor) {
+    super.afterCreatePartControl(editor);
+    IResource resource = editor.getResource();
+    if (resource == null) return;
+    IProject project = resource.getProject();
+    if (!project.isAccessible() || project.isHidden()) return;
+    if (!xtext.hasNature(project)) xtext.toggleNature(project);
+  }
+}
\ No newline at end of file
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/MarkerFactory.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/MarkerFactory.java
new file mode 100644
index 0000000..7a850b5
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/MarkerFactory.java
@@ -0,0 +1,56 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.builder;
+
+import static java.lang.Integer.parseInt;
+import static org.eclipse.core.resources.IMarker.*;
+
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+import org.eclipse.core.resources.IFile;
+import org.eclipse.core.resources.IMarker;
+import org.eclipse.core.runtime.CoreException;
+
+/**
+ * Parses the output of protoc and create error markers if necessary.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+class MarkerFactory {
+
+  static final String MARKER_ID = "com.google.eclipse.protobuf.ui.pbmarker";
+
+  /*
+   * (.*):(\\d+):(\\d+):\\s*(.*)
+   * --1- ---2-- ---3-- -*- --4-
+   *
+   * 1: file name
+   * 2: line number
+   * 3: column
+   * *: whitespace
+   * 4: description
+   */
+  private static final Pattern ERROR_PATTERN = Pattern.compile("(.*):(\\d+):(\\d+):\\s*(.*)");
+
+  void parseAndCreateMarkerIfNecessary(String line, IFile file) throws CoreException {
+    parseError(line, file);
+  }
+
+  private void parseError(String line, IFile file) throws CoreException {
+    Matcher errorMatcher = ERROR_PATTERN.matcher(line);
+    if (!errorMatcher.matches()) return;
+    int lineNumber = parseInt(errorMatcher.group(2));
+    String description = errorMatcher.group(4);
+    IMarker marker = file.createMarker(MARKER_ID);
+    marker.setAttribute(SEVERITY, SEVERITY_ERROR);
+    marker.setAttribute(MESSAGE, description);
+    marker.setAttribute(LINE_NUMBER, lineNumber);
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtobufBuildParticipant.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtobufBuildParticipant.java
new file mode 100644
index 0000000..96f1022
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtobufBuildParticipant.java
@@ -0,0 +1,130 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.builder;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferences.loadPreferences;
+import static com.google.eclipse.protobuf.ui.preferences.RefreshTarget.PROJECT;
+import static org.eclipse.core.resources.IResource.DEPTH_INFINITE;
+
+import java.io.*;
+import java.util.List;
+
+import org.apache.log4j.Logger;
+import org.eclipse.core.resources.*;
+import org.eclipse.core.runtime.*;
+import org.eclipse.emf.common.util.URI;
+import org.eclipse.xtext.builder.IXtextBuilderParticipant;
+import org.eclipse.xtext.resource.IResourceDescription;
+import org.eclipse.xtext.resource.IResourceDescription.Delta;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreAccess;
+
+import com.google.eclipse.protobuf.ui.preferences.*;
+import com.google.inject.Inject;
+
+/**
+ * Calls protoc to generate Java, C++ or Python code from .proto files.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufBuildParticipant implements IXtextBuilderParticipant {
+
+  private static final NullProgressMonitor NO_MONITOR = new NullProgressMonitor();
+
+  private static Logger logger = Logger.getLogger(ProtobufBuildParticipant.class);
+
+  @Inject private IPreferenceStoreAccess preferenceStoreAccess;
+  @Inject private MarkerFactory markerFactory;
+  @Inject private ProtocCommandFactory commandFactory;
+
+  public void build(IBuildContext context, IProgressMonitor monitor) throws CoreException {
+    IProject project = context.getBuiltProject();
+    CompilerPreferences preferences = loadPreferences(preferenceStoreAccess, project);
+    if (!preferences.compileProtoFiles) return;
+    List<Delta> deltas = context.getDeltas();
+    if (deltas.isEmpty()) return;
+    IFolder outputFolder = findOrCreateOutputFolder(project, preferences.outputFolderName);
+    for (Delta d : deltas) {
+      IResourceDescription newResource = d.getNew();
+      String path = filePath(newResource);
+      if (path == null) continue;
+      IFile source = project.getWorkspace().getRoot().getFile(new Path(path));
+      generateSingleProto(source, preferences.protocPath, preferences.language, pathOf(outputFolder));
+    }
+    if (preferences.refreshResources) refresh(outputFolder, preferences.refreshTarget, monitor);
+  }
+
+  private static IFolder findOrCreateOutputFolder(IProject project, String outputFolderName) throws CoreException {
+    IFolder outputFolder = project.getFolder(outputFolderName);
+    if (!outputFolder.exists()) outputFolder.create(true, true, NO_MONITOR);
+    return outputFolder;
+  }
+
+  private static String filePath(IResourceDescription r) {
+    if (r == null) return null;
+    URI uri = r.getURI();
+    if (uri.scheme() == null) return uri.toFileString();
+    StringBuilder b = new StringBuilder();
+    int segmentCount = uri.segmentCount();
+    for (int i = 1; i < segmentCount; i++)
+      b.append("/").append(uri.segment(i));
+    return b.length() == 0 ? null : b.toString();
+  }
+
+  private void generateSingleProto(IFile source, String protocPath, TargetLanguage language, String outputFolderPath) {
+    String command = commandFactory.protocCommand(source, protocPath, language, outputFolderPath);
+    try {
+      source.deleteMarkers(MarkerFactory.MARKER_ID, true, DEPTH_INFINITE);
+      Process process = Runtime.getRuntime().exec(command);
+      processStream(process.getErrorStream(), source);
+      process.destroy();
+    } catch (Exception ex) {
+      // TODO show error message
+      ex.printStackTrace();
+    }
+  }
+
+  private void processStream(InputStream stream, IFile source) {
+    InputStreamReader reader = null;
+    try {
+      reader = new InputStreamReader(stream);
+      BufferedReader bufferedReader = new BufferedReader(reader);
+      String line = null;
+      while ((line = bufferedReader.readLine()) != null) {
+        markerFactory.parseAndCreateMarkerIfNecessary(line, source);
+        System.out.println("[protoc] " + line);
+      }
+    } catch (Exception e) {
+      logger.fatal("Execution of protoc on [" + source.getName() + "] failed", e);
+    } finally {
+      close(reader);
+    }
+  }
+
+  private static void close(Reader reader) {
+    if (reader == null) return;
+    try {
+      reader.close();
+    } catch (IOException ignored) {}
+  }
+
+  private static String pathOf(IResource r) {
+    return r.getLocation().toOSString();
+  }
+
+  private static void refresh(IFolder outputFolder, RefreshTarget refreshTarget, IProgressMonitor monitor)
+      throws CoreException {
+    IResource target = refreshTarget(outputFolder, refreshTarget);
+    target.refreshLocal(DEPTH_INFINITE, monitor);
+  }
+
+  private static IResource refreshTarget(IFolder outputFolder, RefreshTarget refreshTarget) {
+    if (refreshTarget.equals(PROJECT)) return outputFolder.getProject();
+    return outputFolder;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtocCommandFactory.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtocCommandFactory.java
new file mode 100644
index 0000000..67d79b5
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/builder/ProtocCommandFactory.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.builder;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.core.resources.IFile;
+import org.eclipse.core.runtime.IPath;
+
+import com.google.eclipse.protobuf.ui.preferences.TargetLanguage;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+class ProtocCommandFactory {
+  private static final Map<TargetLanguage, String> LANG_OUT_FLAG = new HashMap<TargetLanguage, String>();
+
+  static {
+    for (TargetLanguage lang : TargetLanguage.values())
+      LANG_OUT_FLAG.put(lang, "--" + lang.name().toLowerCase() + "_out=");
+  }
+
+  String protocCommand(IFile protoFile, String protocPath, TargetLanguage language, String outputFolderPath) {
+    IPath protoFilePath = protoFile.getLocation();
+    StringBuilder command = new StringBuilder();
+    command.append(protocPath).append(" ");
+    String protoFileFolder = protoFilePath.toFile().getParentFile().toString();
+    command.append("-I=").append(protoFileFolder).append(" ");
+    command.append(langOutFlag(language)).append(outputFolderPath).append(" ");
+    command.append(protoFilePath.toOSString());
+    return command.toString();
+  }
+
+  private String langOutFlag(TargetLanguage targetLanguage) {
+    return LANG_OUT_FLAG.get(targetLanguage);
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartInsertHandler.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartInsertHandler.java
new file mode 100644
index 0000000..3dcecea
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartInsertHandler.java
@@ -0,0 +1,40 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.commands;
+
+import static org.eclipse.xtext.ui.editor.utils.EditorUtils.getActiveXtextEditor;
+
+import org.eclipse.core.commands.AbstractHandler;
+import org.eclipse.core.commands.ExecutionEvent;
+import org.eclipse.swt.custom.StyledText;
+import org.eclipse.swt.widgets.Control;
+import org.eclipse.xtext.ui.editor.XtextEditor;
+
+/**
+ * Base class for command handlers that insert content in an editor.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public abstract class SmartInsertHandler extends AbstractHandler {
+
+  /** {@inheritDoc} */
+  public final Object execute(ExecutionEvent event) {
+    XtextEditor activeEditor = getActiveXtextEditor();
+    if (activeEditor != null) insertContent(activeEditor);
+    return null;
+  }
+
+  protected abstract void insertContent(XtextEditor editor);
+
+  protected static StyledText styledTextFrom(XtextEditor editor) {
+    Object adapter = editor.getAdapter(Control.class);
+    if (adapter instanceof StyledText) return (StyledText) adapter;
+    return null;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartSemicolonHandler.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartSemicolonHandler.java
new file mode 100644
index 0000000..2ccb99f
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/commands/SmartSemicolonHandler.java
@@ -0,0 +1,121 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.commands;
+
+import static com.google.eclipse.protobuf.protobuf.Modifier.REPEATED;
+
+import java.util.regex.Pattern;
+
+import org.eclipse.emf.ecore.EObject;
+import org.eclipse.swt.custom.StyledText;
+import org.eclipse.xtext.resource.XtextResource;
+import org.eclipse.xtext.ui.editor.XtextEditor;
+import org.eclipse.xtext.ui.editor.contentassist.ContentAssistContext;
+import org.eclipse.xtext.ui.editor.contentassist.antlr.ParserBasedContentAssistContextFactory;
+import org.eclipse.xtext.util.concurrent.IUnitOfWork;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.ui.grammar.*;
+import com.google.eclipse.protobuf.ui.util.*;
+import com.google.inject.Inject;
+
+/**
+ * Inserts a semicolon at the end of a line, regardless of the current position of the caret in the editor. If the
+ * line of code being edited is a property or enum literal and if it does not have an index yet, this handler will
+ * insert an index with a proper value as well.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class SmartSemicolonHandler extends SmartInsertHandler {
+
+  private static final Pattern LITERAL_WITH_INDEX = Pattern.compile("[\\s]+(.*)[\\s]+=[\\s]+[\\d]+(.*)");
+
+  private static final Pattern PROPERTY_WITH_INDEX =
+      Pattern.compile("[\\s]+(.*)[\\s]+(.*)[\\s]+(.*)[\\s]+=[\\s]+[\\d]+(.*)");
+
+  private final CompoundElements compoundElements;
+
+  @Inject private ParserBasedContentAssistContextFactory contextFactory;
+  @Inject private Literals literals;
+  @Inject private Properties properties;
+
+  private final String semicolon;
+
+  @Inject public SmartSemicolonHandler(CompoundElements compoundElements, Keywords keywords) {
+    this.compoundElements = compoundElements;
+    semicolon = keywords.semicolon().getValue();
+  }
+
+  /** {@inheritDoc} */
+  @Override protected void insertContent(XtextEditor editor) {
+    StyledText styledText = styledTextFrom(editor);
+    int originalCaretOffset = styledText.getCaretOffset();
+    int lineAtOffset = styledText.getLineAtOffset(originalCaretOffset);
+    int offsetAtLine = styledText.getOffsetAtLine(lineAtOffset);
+    String line = styledText.getLine(lineAtOffset);
+    if (line.endsWith(semicolon)) {
+      behaveLikeRegularEditing(originalCaretOffset, styledText);
+      return;
+    }
+    int endOfLineOffset = offsetAtLine + line.length();
+    styledText.setCaretOffset(endOfLineOffset);
+    String contentToInsert = contentToInsert(line, editor, originalCaretOffset);
+    styledText.insert(contentToInsert);
+    styledText.setCaretOffset(endOfLineOffset + contentToInsert.length());
+  }
+
+  private void behaveLikeRegularEditing(int caretOffset, StyledText styledText) {
+    styledText.insert(semicolon);
+    styledText.setCaretOffset(caretOffset + semicolon.length());
+  }
+
+  private String contentToInsert(final String line, final XtextEditor editor, final int offset) {
+    return editor.getDocument().readOnly(new IUnitOfWork<String, XtextResource>() {
+      public String exec(XtextResource state) {
+        ContentAssistContext[] context = contextFactory.create(editor.getInternalSourceViewer(), offset, state);
+        if (context == null || context.length == 0) return semicolon;
+        for (ContentAssistContext c : context) {
+          EObject model = c.getCurrentModel();
+          if (model instanceof Literal)
+            return contentToInsert(line, (Literal) model);
+          if (model instanceof Property)
+            return contentToInsert(line, (Property) model);
+        }
+        return semicolon;
+      }
+    });
+  }
+
+  private String contentToInsert(String line, Literal literal) {
+    boolean hasIndexAlready = LITERAL_WITH_INDEX.matcher(line).matches();
+    if (hasIndexAlready) return semicolon;
+    int index = literals.calculateIndexOf(literal);
+    return defaultIndexAndSemicolonToInsert(line, index);
+  }
+
+  private String contentToInsert(String line, Property property) {
+    boolean hasIndexAlready = PROPERTY_WITH_INDEX.matcher(line).matches();
+    if (hasIndexAlready) return semicolon;
+    int index = properties.calculateIndexOf(property);
+    if (REPEATED.equals(property.getModifier())) {
+      String format = "= %d " + compoundElements.packedInBrackets() + "%s";
+      return indexAndSemicolonToInsert(format, line, index);
+    }
+    return defaultIndexAndSemicolonToInsert(line, index);
+  }
+
+  private String defaultIndexAndSemicolonToInsert(String line, int index) {
+    return indexAndSemicolonToInsert("= %d%s", line, index);
+  }
+
+  private String indexAndSemicolonToInsert(String format, String line, int index) {
+    String content = String.format(format, index, semicolon);
+    return (line.endsWith(" ")) ? content : " " + content;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/contentassist/ProtobufProposalProvider.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/contentassist/ProtobufProposalProvider.java
new file mode 100644
index 0000000..253046f
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/contentassist/ProtobufProposalProvider.java
@@ -0,0 +1,307 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.contentassist;
+
+import static com.google.eclipse.protobuf.protobuf.Modifier.*;
+import static com.google.eclipse.protobuf.protobuf.ScalarType.STRING;
+import static java.lang.String.valueOf;
+
+import org.eclipse.emf.ecore.EObject;
+import org.eclipse.jface.text.contentassist.ICompletionProposal;
+import org.eclipse.swt.custom.StyledText;
+import org.eclipse.swt.graphics.Image;
+import org.eclipse.xtext.*;
+import org.eclipse.xtext.ui.PluginImageHelper;
+import org.eclipse.xtext.ui.editor.contentassist.*;
+
+import com.google.common.collect.ImmutableList;
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.protobuf.Enum;
+import com.google.eclipse.protobuf.scoping.GlobalScope;
+import com.google.eclipse.protobuf.ui.grammar.*;
+import com.google.eclipse.protobuf.ui.labeling.Images;
+import com.google.eclipse.protobuf.ui.util.*;
+import com.google.eclipse.protobuf.util.EObjectFinder;
+import com.google.inject.Inject;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ *
+ * @see http://www.eclipse.org/Xtext/documentation/latest/xtext.html#contentAssist on how to customize content assistant
+ */
+public class ProtobufProposalProvider extends AbstractProtobufProposalProvider {
+
+  @Inject private CompoundElements compoundElements;
+  @Inject private EObjectFinder finder;
+  @Inject private GlobalScope globalScope;
+  @Inject private PluginImageHelper imageHelper;
+  @Inject private Images imageRegistry;
+  @Inject private Keywords keywords;
+  @Inject private Literals literals;
+  @Inject private Properties properties;
+  @Inject private Strings strings;
+
+  @Override public void completeOption_Name(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    EObject container = model.eContainer();
+    if (container instanceof Protobuf) {
+      proposeCommonFileOptions(context, acceptor);
+      return;
+    }
+  }
+
+  private void proposeCommonFileOptions(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    for (Property fileOption : globalScope.fileOptions()) {
+      String displayString = fileOption.getName();
+      String proposalText = displayString + " " + keywords.equalSign().getValue() + " ";
+      boolean isStringOption = properties.isStringProperty(fileOption);
+      if (isStringOption)
+        proposalText = proposalText + compoundElements.emptyString() + keywords.semicolon().getValue();
+      ICompletionProposal proposal = createCompletionProposal(proposalText, displayString, context);
+      if (isStringOption && proposal instanceof ConfigurableCompletionProposal) {
+        // set cursor between the proposal's quotes
+        ConfigurableCompletionProposal configurable = (ConfigurableCompletionProposal) proposal;
+        configurable.setCursorPosition(proposalText.length() - 2);
+      }
+      acceptor.accept(proposal);
+    }
+  }
+
+  @Override public void completeOption_Value(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    Option option = (Option) model;
+    Property fileOption = globalScope.lookupFileOption(option.getName());
+    if (fileOption == null) return;
+    if (globalScope.isOptimizeForOption(option)) {
+      proposeAndAccept(globalScope.optimizedMode(), context, acceptor);
+      return;
+    }
+    if (properties.isStringProperty(fileOption)) {
+      proposeEmptyString(context, acceptor);
+      return;
+    }
+    if (properties.isBoolProperty(fileOption)) {
+      proposeBooleanValues(context, acceptor);
+      return;
+    }
+  }
+
+  private void proposeBooleanValues(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    proposeAndAccept(keywords.boolFalse().getValue(), context, acceptor);
+    proposeAndAccept(keywords.boolTrue().getValue(), context, acceptor);
+  }
+
+  @Override public void complete_ID(EObject model, RuleCall ruleCall, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {}
+
+  @Override public void complete_STRING(EObject model, RuleCall ruleCall, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    if (model instanceof Property && isProposalForDefaultValue(context)) {
+      Property p = (Property) model;
+      if (!isStringProperty(p)) return;
+      proposeEmptyString(context, acceptor);
+      return;
+    }
+    if (model instanceof Option) return;
+    super.complete_STRING(model, ruleCall, context, acceptor);
+  }
+
+  private void proposeEmptyString(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    ICompletionProposal proposal = createCompletionProposal(compoundElements.emptyString(), context);
+    if (proposal instanceof ConfigurableCompletionProposal) {
+      ConfigurableCompletionProposal configurable = (ConfigurableCompletionProposal) proposal;
+      configurable.setCursorPosition(1);
+    }
+    acceptor.accept(proposal);
+  }
+
+  private boolean isProposalForDefaultValue(ContentAssistContext context) {
+    return isProposalForAssignment(keywords.defaultValue().getValue(), context);
+  }
+
+  private boolean isProposalForAssignment(String feature, ContentAssistContext context) {
+    ImmutableList<AbstractElement> grammarElements = context.getFirstSetGrammarElements();
+    for (AbstractElement e : grammarElements) {
+      if (!(e instanceof Assignment)) continue;
+      Assignment a = (Assignment) e;
+      String equalSign = keywords.equalSign().getValue();
+      if (feature.equals(a.getFeature()) && equalSign.equals(a.getOperator())) return true;
+    }
+    return false;
+  }
+
+  @Override public void completeKeyword(Keyword keyword, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    if (keyword == null) return;
+    if (isKeywordEqualToPreviousWordInEditor(keyword, context)) return;
+    if (keyword.equals(keywords.boolTrue()) || keyword.equals(keywords.boolFalse())) {
+      if (!isBoolProposalValid(context)) return;
+    }
+    if (keyword.equals(keywords.openingBracket())) {
+      boolean proposalWasHandledAlready = proposeOpenBracket(context, acceptor);
+      if (proposalWasHandledAlready) return;
+    }
+    if (keyword.equals(keywords.packed())) {
+      proposePackedOption(context, acceptor);
+      return;
+    }
+    if (keyword.equals(keywords.defaultValue())) {
+      proposeDefaultValue(context, acceptor);
+      return;
+    }
+    super.completeKeyword(keyword, context, acceptor);
+  }
+
+  private boolean isKeywordEqualToPreviousWordInEditor(Keyword keyword, ContentAssistContext context) {
+    StyledText styledText = context.getViewer().getTextWidget();
+    String value = keyword.getValue();
+    int valueLength = value.length();
+    int start = styledText.getCaretOffset() - valueLength;
+    if (start < 0) return false;
+    String previousWord = styledText.getTextRange(start, valueLength);
+    return value.equals(previousWord);
+  }
+
+  private boolean isBoolProposalValid(ContentAssistContext context) {
+    EObject model = context.getCurrentModel();
+    if (model instanceof Property) return properties.isBoolProperty((Property) model);
+    if (model instanceof Option) {
+      Property fileOption = globalScope.lookupFileOption(((Option) model).getName());
+      return fileOption != null && properties.isBoolProperty(fileOption);
+    }
+    return false;
+  }
+
+  private boolean proposeOpenBracket(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    EObject model = context.getCurrentModel();
+    if (!(model instanceof Property)) return false;
+    Property p = (Property) model;
+    Modifier modifier = p.getModifier();
+    if (OPTIONAL.equals(modifier)) {
+      String display = compoundElements.defaultValueInBrackets();
+      int cursorPosition = display.indexOf(keywords.closingBracket().getValue());
+      if (isStringProperty(p)) {
+        display = compoundElements.defaultStringValueInBrackets();
+        cursorPosition++;
+      }
+      ICompletionProposal proposal = createCompletionProposal(display, context);
+      if (proposal instanceof ConfigurableCompletionProposal) {
+        ConfigurableCompletionProposal configurable = (ConfigurableCompletionProposal) proposal;
+        configurable.setCursorPosition(cursorPosition);
+      }
+      acceptor.accept(proposal);
+    }
+    if (REPEATED.equals(modifier)) proposeAndAccept(compoundElements.packedInBrackets(), context, acceptor);
+    return true;
+  }
+
+  private void proposePackedOption(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    Modifier modifier = extractModifierFromModel(context);
+    if (!REPEATED.equals(modifier)) return;
+    proposeAndAccept(compoundElements.packed(), context, acceptor);
+  }
+
+  private void proposeDefaultValue(ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    Modifier modifier = extractModifierFromModel(context);
+    if (!OPTIONAL.equals(modifier)) return;
+    String display = compoundElements.defaultValue();
+    int cursorPosition = display.length();
+    if (isStringProperty((Property) context.getCurrentModel())) {
+      display = compoundElements.defaultStringValue();
+      cursorPosition++;
+    }
+    ICompletionProposal proposal = createCompletionProposal(display, context);
+    if (proposal instanceof ConfigurableCompletionProposal) {
+      ConfigurableCompletionProposal configurable = (ConfigurableCompletionProposal) proposal;
+      configurable.setCursorPosition(cursorPosition);
+    }
+    acceptor.accept(proposal);
+  }
+
+  private Modifier extractModifierFromModel(ContentAssistContext context) {
+    EObject model = context.getCurrentModel();
+    // this is most likely a bug in Xtext:
+    if (!(model instanceof Property)) model = context.getPreviousModel();
+    if (!(model instanceof Property)) return null;
+    return ((Property) model).getModifier();
+  }
+
+  private boolean isStringProperty(Property p) {
+    return STRING.equals(finder.scalarTypeOfProperty(p));
+  }
+
+  @Override public void completeLiteral_Index(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    int index = literals.calculateIndexOf((Literal) model);
+    proposeIndex(index, context, acceptor);
+  }
+
+  @Override public void completeProperty_Default(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    Enum enumType = finder.enumTypeOfProperty((Property) model);
+    if (enumType == null) return;
+    proposeAndAccept(enumType, context, acceptor);
+  }
+
+  private void proposeAndAccept(Enum enumType, ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    Image image = imageHelper.getImage(imageRegistry.imageFor(Literal.class));
+    for (Literal literal : enumType.getLiterals())
+      proposeAndAccept(literal.getName(), image, context, acceptor);
+  }
+
+  private void proposeAndAccept(String proposalText, Image image, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    ICompletionProposal proposal = createCompletionProposal(proposalText, proposalText, image, context);
+    acceptor.accept(proposal);
+  }
+
+  @Override public void completeProperty_Index(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    int index = properties.calculateIndexOf((Property) model);
+    proposeIndex(index, context, acceptor);
+  }
+
+  private void proposeIndex(int index, ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    proposeAndAccept(valueOf(index), context, acceptor);
+  }
+
+  @Override public void completeProperty_Name(EObject model, Assignment assignment, ContentAssistContext context,
+      ICompletionProposalAcceptor acceptor) {
+    String typeName = strings.firstCharToLowerCase(properties.nameOfTypeIn((Property) model));
+    int index = 1;
+    String name = typeName + index;
+    for (EObject o : model.eContainer().eContents()) {
+      if (o == model || !(o instanceof Property)) continue;
+      Property p = (Property) o;
+      if (!name.equals(p.getName())) continue;
+      name = typeName + (++index);
+    }
+    proposeAndAccept(name, context, acceptor);
+  }
+
+  private ICompletionProposal createCompletionProposal(String proposal, String displayString,
+      ContentAssistContext context) {
+    return createCompletionProposal(proposal, displayString, defaultImage(), context);
+  }
+
+  private void proposeAndAccept(String proposalText, ContentAssistContext context, ICompletionProposalAcceptor acceptor) {
+    ICompletionProposal proposal = createCompletionProposal(proposalText, context);
+    acceptor.accept(proposal);
+  }
+
+  @Override protected ICompletionProposal createCompletionProposal(String proposal,
+      ContentAssistContext contentAssistContext) {
+    return createCompletionProposal(proposal, null, defaultImage(), getPriorityHelper().getDefaultPriority(),
+        contentAssistContext.getPrefix(), contentAssistContext);
+  }
+
+  private Image defaultImage() {
+    return imageHelper.getImage(imageRegistry.defaultImage());
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/CompoundElements.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/CompoundElements.java
new file mode 100644
index 0000000..a648ecf
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/CompoundElements.java
@@ -0,0 +1,83 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.grammar;
+
+import org.eclipse.xtext.Keyword;
+
+import com.google.inject.*;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class CompoundElements {
+
+  private static final String EMPTY_STRING = "\"\"";
+
+  private final String inBracketsFormat;
+
+  private final String defaultValue;
+  private final String defaultValueInBrackets;
+  private final String defaultStringValue;
+  private final String defaultStringValueInBrackets;
+  private final String packed;
+  private final String packedInBrackets;
+
+  @Inject public CompoundElements(Keywords keywords) {
+    inBracketsFormat = keywords.openingBracket().getValue() + "%s" + keywords.closingBracket().getValue();
+    defaultValue = format("%s %s", keywords.defaultValue(), keywords.equalSign());
+    defaultValueInBrackets = inBrackets(defaultValue);
+    defaultStringValue = format("%s %s %s", keywords.defaultValue(), keywords.equalSign(), EMPTY_STRING);
+    defaultStringValueInBrackets = inBrackets(defaultStringValue);
+    packed = format("%s %s %s", keywords.packed(), keywords.equalSign(), keywords.boolTrue());
+    packedInBrackets = inBrackets(packed);
+  }
+
+  private static String format(String format, Object...values) {
+    int count = values.length;
+    Object[] cleanValues = new Object[count];
+    for (int i = 0; i < count; i++) {
+      Object value = values[i];
+      cleanValues[i] = (value instanceof Keyword) ? ((Keyword) value).getValue() : value;
+    }
+    return String.format(format, cleanValues);
+  }
+
+  private String inBrackets(String element) {
+    return String.format(inBracketsFormat, element);
+  }
+
+  public String defaultValue() {
+    return defaultValue;
+  }
+
+  public String defaultValueInBrackets() {
+    return defaultValueInBrackets;
+  }
+
+  public String defaultStringValue() {
+    return defaultStringValue;
+  }
+
+  public String defaultStringValueInBrackets() {
+    return defaultStringValueInBrackets;
+  }
+
+  public String emptyString() {
+    return EMPTY_STRING;
+  }
+
+  public String packed() {
+    return packed;
+  }
+
+  public String packedInBrackets() {
+    return packedInBrackets;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/Keywords.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/Keywords.java
new file mode 100644
index 0000000..7030ca6
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/grammar/Keywords.java
@@ -0,0 +1,155 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.grammar;
+
+import static org.eclipse.xtext.GrammarUtil.containedKeywords;
+
+import java.util.List;
+
+import org.eclipse.xtext.*;
+
+import com.google.inject.*;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Keywords {
+
+  private Keyword bool;
+  private Keyword boolTrue;
+  private Keyword boolFalse;
+  private Keyword openingBracket;
+  private Keyword closingBracket;
+  private Keyword defaultValue;
+  private Keyword equalSign;
+  private Keyword packed;
+  private Keyword semicolon;
+  private Keyword string;
+
+  @Inject public Keywords(IGrammarAccess grammarAccess) {
+    List<Keyword> allKeywords = containedKeywords(grammarAccess.getGrammar());
+    for (Keyword k : allKeywords) {
+      if (assignIfIsBool(k)) continue;
+      if (assignIfIsBoolTrue(k)) continue;
+      if (assignIfIsBoolFalse(k)) continue;
+      if (assignIfIsOpeningBracket(k)) continue;
+      if (assignIfIsClosingBracket(k)) continue;
+      if (assignIfIsDefaultValue(k)) continue;
+      if (assignIfIsEqualSign(k)) continue;
+      if (assignIfIsPacked(k)) continue;
+      if (assignIfIsSemicolon(k)) continue;
+      if (assignIfIsString(k)) continue;
+    }
+  }
+
+  private boolean assignIfIsBool(Keyword k) {
+    if (!isKeywordValue(k, "bool")) return false;
+    bool = k;
+    return true;
+  }
+
+  private boolean assignIfIsBoolTrue(Keyword k) {
+    if (!isKeywordValue(k, "true")) return false;
+    boolTrue = k;
+    return true;
+  }
+
+  private boolean assignIfIsBoolFalse(Keyword k) {
+    if (!isKeywordValue(k, "false")) return false;
+    boolFalse = k;
+    return true;
+  }
+
+  private boolean assignIfIsOpeningBracket(Keyword k) {
+    if (!isKeywordValue(k, "[")) return false;
+    openingBracket = k;
+    return true;
+  }
+
+  private boolean assignIfIsClosingBracket(Keyword k) {
+    if (!isKeywordValue(k, "]")) return false;
+    closingBracket = k;
+    return true;
+  }
+
+  private boolean assignIfIsDefaultValue(Keyword k) {
+    if (!isKeywordValue(k, "default")) return false;
+    defaultValue = k;
+    return true;
+  }
+
+  private boolean assignIfIsEqualSign(Keyword k) {
+    if (!isKeywordValue(k, "=")) return false;
+    equalSign = k;
+    return true;
+  }
+
+  private boolean assignIfIsPacked(Keyword k) {
+    if (!isKeywordValue(k, "packed")) return false;
+    packed = k;
+    return true;
+  }
+
+  private boolean assignIfIsSemicolon(Keyword k) {
+    if (!isKeywordValue(k, ";")) return false;
+    semicolon = k;
+    return true;
+  }
+
+  private boolean assignIfIsString(Keyword k) {
+    if (!isKeywordValue(k, "string")) return false;
+    string = k;
+    return true;
+  }
+
+  private static boolean isKeywordValue(Keyword k, String expectedValue) {
+    return expectedValue.equals(k.getValue());
+  }
+
+  public Keyword bool() {
+    return bool;
+  }
+
+  public Keyword boolTrue() {
+    return boolTrue;
+  }
+
+  public Keyword boolFalse() {
+    return boolFalse;
+  }
+
+  public Keyword openingBracket() {
+    return openingBracket;
+  }
+
+  public Keyword closingBracket() {
+    return closingBracket;
+  }
+
+  public Keyword defaultValue() {
+    return defaultValue;
+  }
+
+  public Keyword equalSign() {
+    return equalSign;
+  }
+
+  public Keyword packed() {
+    return packed;
+  }
+
+  public Keyword semicolon() {
+    return semicolon;
+  }
+
+  public Keyword string() {
+    return string;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Images.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Images.java
new file mode 100644
index 0000000..c53092b
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Images.java
@@ -0,0 +1,93 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.labeling;
+
+import static com.google.eclipse.protobuf.protobuf.Modifier.*;
+import static java.util.Arrays.asList;
+
+import java.util.*;
+
+import org.eclipse.xtext.Keyword;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.protobuf.Enum;
+import com.google.eclipse.protobuf.protobuf.Package;
+import com.google.inject.Singleton;
+
+/**
+ * Registry of all images used in the 'Protocol Buffer' editor.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Images {
+
+  private static final String DEFAULT_IMAGE = "empty.gif";
+
+  private static final Map<Modifier, String> IMAGES_BY_MODIFIER = new HashMap<Modifier, String>();
+  static {
+    IMAGES_BY_MODIFIER.put(OPTIONAL, "property-opt.gif");
+    IMAGES_BY_MODIFIER.put(REPEATED, "property-rep.gif");
+    IMAGES_BY_MODIFIER.put(REQUIRED, "property-req.gif");
+  }
+
+  private static final Map<Class<?>, String> IMAGES_BY_TYPE = new HashMap<Class<?>, String>();
+  static {
+    IMAGES_BY_TYPE.put(Enum.class, "enum.gif");
+    IMAGES_BY_TYPE.put(ExtendMessage.class, "extend.gif");
+    IMAGES_BY_TYPE.put(Import.class, "import.gif");
+    IMAGES_BY_TYPE.put(Literal.class, "literal.gif");
+    IMAGES_BY_TYPE.put(Message.class, "message.gif");
+    IMAGES_BY_TYPE.put(Option.class, "option.gif");
+    IMAGES_BY_TYPE.put(Package.class, "package.gif");
+    IMAGES_BY_TYPE.put(Protobuf.class, "protobuf.gif");
+  }
+
+  private static final List<String> STANDALONE_IMAGES = asList("extensions.gif");
+
+  public String imageFor(Object o) {
+    if (o instanceof Property) {
+      Property p = (Property) o;
+      return imageFor(p.getModifier());
+    }
+    if (o instanceof Keyword) {
+      Keyword k = (Keyword) o;
+      return imageFor(k);
+    }
+    return imageFor(o.getClass());
+  }
+
+  public String imageFor(Class<?> type) {
+    String image = IMAGES_BY_TYPE.get(type);
+    if (image != null) return image;
+    Class<?>[] interfaces = type.getInterfaces();
+    if (interfaces == null || interfaces.length != 1) return DEFAULT_IMAGE;
+    return imageFor(interfaces[0]);
+  }
+
+  private String imageFor(Keyword k) {
+    String value = k.getValue();
+    Modifier m = Modifier.getByName(value);
+    String image = IMAGES_BY_MODIFIER.get(m);
+    if (image != null) return image;
+    String imageName = value + ".gif";
+    if (IMAGES_BY_TYPE.containsValue(imageName) || STANDALONE_IMAGES.contains(imageName)) return imageName;
+    return DEFAULT_IMAGE;
+  }
+
+  private String imageFor(Modifier m) {
+    String image = IMAGES_BY_MODIFIER.get(m);
+    if (image != null) return image;
+    return "property.gif";
+  }
+
+  public String defaultImage() {
+    return DEFAULT_IMAGE;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Labels.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Labels.java
new file mode 100644
index 0000000..966c26d
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/Labels.java
@@ -0,0 +1,69 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.labeling;
+
+import static com.google.eclipse.protobuf.scoping.SimpleImportUriResolver.URI_PREFIX;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.ui.util.Properties;
+import com.google.inject.Inject;
+import com.google.inject.Singleton;
+
+/**
+ * Registry of commonly used text in the 'Protocol Buffer' editor.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Labels {
+
+  @Inject private Properties properties;
+
+  private static final String LITERAL_FORMAT = "%s [%d]";
+  private static final String PROPERTY_FORMAT = "%s [%d] : %s";
+
+  public String labelFor(Object o) {
+    if (o instanceof Import) {
+      Import i = (Import) o;
+      return labelFor(i);
+    }
+    if (o instanceof Literal) {
+      Literal l = (Literal) o;
+      return labelFor(l);
+    }
+    if (o instanceof Property) {
+      Property p = (Property) o;
+      return labelFor(p);
+    }
+    if (o instanceof Protobuf) {
+      Protobuf p = (Protobuf) o;
+      return labelFor(p);
+    }
+    return null;
+  }
+
+  private String labelFor(Import i) {
+    String uri = i.getImportURI();
+    if (uri == null || !uri.startsWith(URI_PREFIX)) return uri;
+    return uri.substring(URI_PREFIX.length());
+  }
+
+  private String labelFor(Literal l) {
+    return String.format(LITERAL_FORMAT, l.getName(), l.getIndex());
+  }
+
+  private String labelFor(Property p) {
+    return String.format(PROPERTY_FORMAT, p.getName(), p.getIndex(), properties.nameOfTypeIn(p));
+  }
+
+  private String labelFor(Protobuf p) {
+    // TODO show this text till I figure out how to hide 'Protobuf' node in outline view
+    return "Protocol Buffer";
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufDescriptionLabelProvider.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufDescriptionLabelProvider.java
new file mode 100644
index 0000000..57ca234
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufDescriptionLabelProvider.java
@@ -0,0 +1,20 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.labeling;
+
+import org.eclipse.xtext.ui.label.DefaultDescriptionLabelProvider;
+
+/**
+ * Provides labels for a {@code IEObjectDescription}s and {@code IResourceDescription}s.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ *
+ * @see http://www.eclipse.org/Xtext/documentation/latest/xtext.html#labelProvider
+ */
+public class ProtobufDescriptionLabelProvider extends DefaultDescriptionLabelProvider {}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufLabelProvider.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufLabelProvider.java
new file mode 100644
index 0000000..71423d3
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/labeling/ProtobufLabelProvider.java
@@ -0,0 +1,40 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.labeling;
+
+import org.eclipse.emf.edit.ui.provider.AdapterFactoryLabelProvider;
+import org.eclipse.xtext.ui.label.DefaultEObjectLabelProvider;
+
+import com.google.inject.Inject;
+
+/**
+ * Provides labels for a {@code EObject}s.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ *
+ * @see http://www.eclipse.org/Xtext/documentation/latest/xtext.html#labelProvider
+ */
+public class ProtobufLabelProvider extends DefaultEObjectLabelProvider {
+
+  @Inject private Labels labels;
+  @Inject private Images images;
+
+  @Inject public ProtobufLabelProvider(AdapterFactoryLabelProvider delegate) {
+    super(delegate);
+  }
+
+  @Override public Object text(Object o) {
+    String text = labels.labelFor(o);
+    return (text != null) ? text : super.text(o);
+  }
+
+  @Override public Object image(Object o) {
+    return images.imageFor(o);
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/LinkWithEditor.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/LinkWithEditor.java
new file mode 100644
index 0000000..d8fdfd2
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/LinkWithEditor.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.outline;
+
+import java.util.Map;
+
+import org.eclipse.xtext.ui.editor.outline.actions.*;
+import org.eclipse.xtext.ui.editor.outline.impl.OutlinePage;
+
+import com.google.common.collect.Maps;
+import com.google.inject.*;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class LinkWithEditor extends LinkWithEditorOutlineContribution {
+
+  @Inject private Provider<OutlineWithEditorLinker> outlineWithEditorLinkerProvider;
+
+  private final Map<OutlinePage, OutlineWithEditorLinker> page2linker = Maps.newHashMap();
+
+  /** {@inheritDoc} */
+  @Override public void register(OutlinePage outlinePage) {
+    addPropertyChangeListener();
+    OutlineWithEditorLinker outlineWithEditorLinker = outlineWithEditorLinkerProvider.get();
+    outlineWithEditorLinker.activate(outlinePage);
+    getPreferenceStoreAccess().getPreferenceStore().addPropertyChangeListener(outlineWithEditorLinker);
+    outlineWithEditorLinker.setLinkingEnabled(true);
+    page2linker.put(outlinePage, outlineWithEditorLinker);
+  }
+
+  @Override
+  public void deregister(OutlinePage outlinePage) {
+    removePropertyChangeListener();
+    OutlineWithEditorLinker outlineWithEditorLinker = page2linker.remove(outlinePage);
+    outlineWithEditorLinker.deactivate();
+    getPreferenceStoreAccess().getPreferenceStore().removePropertyChangeListener(outlineWithEditorLinker);
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlinePage.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlinePage.java
new file mode 100644
index 0000000..493b16b
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlinePage.java
@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.outline;
+
+import org.eclipse.xtext.ui.editor.outline.impl.OutlinePage;
+
+/**
+ * Outline Page for Protocol Buffer editors.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufOutlinePage extends OutlinePage {
+
+  /**
+   * Indicates that the root node and its immediate children of the Outline View need to be expanded.
+   * @return 3.
+   */
+  @Override protected int getDefaultExpansionLevel() {
+    return 3;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlineTreeProvider.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlineTreeProvider.java
new file mode 100644
index 0000000..0b6faa7
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/outline/ProtobufOutlineTreeProvider.java
@@ -0,0 +1,29 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.outline;
+
+import org.eclipse.xtext.ui.editor.outline.impl.DefaultOutlineTreeProvider;
+
+import com.google.eclipse.protobuf.protobuf.*;
+
+/**
+ * Customization of the default outline structure
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufOutlineTreeProvider extends DefaultOutlineTreeProvider {
+
+  boolean _isLeaf(Property p) {
+    return true;
+  }
+
+  boolean _isLeaf(Option o) {
+    return true;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferenceNames.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferenceNames.java
new file mode 100644
index 0000000..69a3441
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferenceNames.java
@@ -0,0 +1,30 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+final class CompilerPreferenceNames {
+
+  static final String ENABLE_PROJECT_SETTINGS = "compiler.enableProjectSettings";
+  static final String COMPILE_PROTO_FILES = "compiler.compileProtoFiles";
+  static final String USE_PROTOC_IN_SYSTEM_PATH = "compiler.useProtocInSystemPath";
+  static final String USE_PROTOC_IN_CUSTOM_PATH = "compiler.useProtocInCustomPath";
+  static final String PROTOC_FILE_PATH = "compiler.protocFilePath";
+  static final String GENERATE_JAVA_CODE = "compiler.generateJavaCode";
+  static final String GENERATE_CPP_CODE = "compiler.generateCppCode";
+  static final String GENERATE_PYTHON_CODE = "compiler.generatePythonCode";
+  static final String OUTPUT_FOLDER_NAME = "compiler.outputFolderName";
+  static final String REFRESH_RESOURCES = "compiler.refreshResources";
+  static final String REFRESH_PROJECT = "compiler.refreshProject";
+  static final String REFRESH_OUTPUT_FOLDER = "compiler.refreshOutputProject";
+
+  private CompilerPreferenceNames() {}
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencePage.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencePage.java
new file mode 100644
index 0000000..aeb4ffe
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencePage.java
@@ -0,0 +1,452 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferenceNames.*;
+import static com.google.eclipse.protobuf.ui.preferences.Messages.*;
+import static org.eclipse.core.resources.IResource.FOLDER;
+import static org.eclipse.core.runtime.IStatus.OK;
+
+import java.io.File;
+import java.util.Map;
+
+import org.eclipse.core.resources.*;
+import org.eclipse.core.runtime.IAdaptable;
+import org.eclipse.core.runtime.IStatus;
+import org.eclipse.jface.preference.IPreferenceStore;
+import org.eclipse.jface.preference.PreferencePage;
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.events.*;
+import org.eclipse.swt.layout.GridData;
+import org.eclipse.swt.layout.GridLayout;
+import org.eclipse.swt.widgets.*;
+import org.eclipse.ui.*;
+import org.eclipse.ui.dialogs.PreferencesUtil;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreAccess;
+
+import com.google.inject.Inject;
+
+/**
+ * Preference page for protobuf compiler.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class CompilerPreferencePage extends PreferencePage implements IWorkbenchPreferencePage, IWorkbenchPropertyPage {
+
+  private static final String PREFERENCE_PAGE_ID = "com.google.eclipse.protobuf.ui.preferences.CompilerPreferencePage";
+  private Button btnEnableProjectSettings;
+  private Link lnkEnableWorkspaceSettings;
+  private Button btnCompileProtoFiles;
+  private TabFolder tabFolder;
+  private TabItem tbtmMain;
+  private TabItem tbtmRefresh;
+  private Group grpCompilerLocation;
+  private Button btnUseProtocInSystemPath;
+  private Button btnUseProtocInCustomPath;
+  private Text txtProtocFilePath;
+  private Button btnProtocPathBrowse;
+  private Group grpTargetLanguage;
+  private Button btnJava;
+  private Button btnCpp;
+  private Button btnPython;
+  private Group grpOutput;
+  private Text txtOutputFolderName;
+  private Label lblOutputFolderName;
+
+  private IProject project;
+
+  private final IPreferenceStoreAccess preferenceStoreAccess;
+
+  private Map<String, Object> dataMap;
+  private Button btnRefreshResources;
+  private Group grpRefresh;
+  private Button btnRefreshProject;
+  private Button btnRefreshOutputFolder;
+
+  @Inject public CompilerPreferencePage(IPreferenceStoreAccess preferenceStoreAccess) {
+    this.preferenceStoreAccess = preferenceStoreAccess;
+  }
+
+  /** {@inheritDoc} */
+  @Override protected Control createContents(Composite parent) {
+    // generated by WindowBuilder
+    Composite contents = new Composite(parent, NONE);
+    contents.setLayout(new GridLayout(3, false));
+
+    if (isPropertyPage()) {
+      btnEnableProjectSettings = new Button(contents, SWT.CHECK);
+      btnEnableProjectSettings.setText(CompilerPreferencePage_enableProjectSettings);
+
+      lnkEnableWorkspaceSettings = new Link(contents, SWT.NONE);
+      lnkEnableWorkspaceSettings.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, false, false, 1, 1));
+      lnkEnableWorkspaceSettings.setText("<a>" + CompilerPreferencePage_configureWorkspaceSettings + "</a>");
+
+      Label label = new Label(contents, SWT.SEPARATOR | SWT.HORIZONTAL);
+      label.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 2, 1));
+    }
+    new Label(contents, SWT.NONE);
+
+    btnCompileProtoFiles = new Button(contents, SWT.CHECK);
+    btnCompileProtoFiles.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, false, 2, 1));
+    btnCompileProtoFiles.setText(CompilerPreferencePage_compileOnSave);
+
+    tabFolder = new TabFolder(contents, SWT.NONE);
+    tabFolder.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 2, 1));
+
+    tbtmMain = new TabItem(tabFolder, SWT.NONE);
+    tbtmMain.setText(CompilerPreferencePage_mainTab);
+
+    Composite cmpMain = new Composite(tabFolder, SWT.NONE);
+    tbtmMain.setControl(cmpMain);
+    cmpMain.setLayout(new GridLayout(1, false));
+
+    grpCompilerLocation = new Group(cmpMain, SWT.NONE);
+    grpCompilerLocation.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    grpCompilerLocation.setLayout(new GridLayout(4, false));
+    grpCompilerLocation.setText(CompilerPreferencePage_location);
+
+    btnUseProtocInSystemPath = new Button(grpCompilerLocation, SWT.RADIO);
+    btnUseProtocInSystemPath.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, false, 4, 1));
+    btnUseProtocInSystemPath.setText(CompilerPreferencePage_systemPath);
+
+    btnUseProtocInCustomPath = new Button(grpCompilerLocation, SWT.RADIO);
+    btnUseProtocInCustomPath.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, false, 4, 1));
+    btnUseProtocInCustomPath.setText(CompilerPreferencePage_customPath);
+    new Label(grpCompilerLocation, SWT.NONE);
+
+    txtProtocFilePath = new Text(grpCompilerLocation, SWT.BORDER);
+    txtProtocFilePath.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    txtProtocFilePath.setEditable(false);
+    new Label(grpCompilerLocation, SWT.NONE);
+
+    btnProtocPathBrowse = new Button(grpCompilerLocation, SWT.NONE);
+    btnProtocPathBrowse.setText(CompilerPreferencePage_browseCustomPath);
+
+    grpTargetLanguage = new Group(cmpMain, SWT.NONE);
+    grpTargetLanguage.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    grpTargetLanguage.setLayout(new GridLayout(1, false));
+    grpTargetLanguage.setText(CompilerPreferencePage_targetLanguage);
+
+    btnJava = new Button(grpTargetLanguage, SWT.RADIO);
+    btnJava.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, false, 1, 1));
+    btnJava.setText(CompilerPreferencePage_generateJava);
+
+    btnCpp = new Button(grpTargetLanguage, SWT.RADIO);
+    btnCpp.setLayoutData(new GridData(SWT.LEFT, SWT.CENTER, true, false, 1, 1));
+    btnCpp.setText(CompilerPreferencePage_generateCpp);
+
+    btnPython = new Button(grpTargetLanguage, SWT.RADIO);
+    btnPython.setText(CompilerPreferencePage_generatePython);
+
+    grpOutput = new Group(cmpMain, SWT.NONE);
+    grpOutput.setLayout(new GridLayout(3, false));
+    grpOutput.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    grpOutput.setText(CompilerPreferencePage_generatedCode);
+
+    lblOutputFolderName = new Label(grpOutput, SWT.NONE);
+    lblOutputFolderName.setText(CompilerPreferencePage_outputFolderName);
+    new Label(grpOutput, SWT.NONE);
+
+    txtOutputFolderName = new Text(grpOutput, SWT.BORDER);
+    txtOutputFolderName.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+
+    tbtmRefresh = new TabItem(tabFolder, SWT.NONE);
+    tbtmRefresh.setText(CompilerPreferencePage_refreshTab);
+
+    Composite cmpRefresh = new Composite(tabFolder, SWT.NONE);
+    tbtmRefresh.setControl(cmpRefresh);
+    cmpRefresh.setLayout(new GridLayout(1, false));
+
+    btnRefreshResources = new Button(cmpRefresh, SWT.CHECK);
+    btnRefreshResources.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    btnRefreshResources.setText(CompilerPreferencePage_refreshResources);
+
+    grpRefresh = new Group(cmpRefresh, SWT.NONE);
+    grpRefresh.setLayout(new GridLayout(1, false));
+    grpRefresh.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+
+    btnRefreshProject = new Button(grpRefresh, SWT.RADIO);
+    btnRefreshProject.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    btnRefreshProject.setText(CompilerPreferencePage_refreshProject);
+
+    btnRefreshOutputFolder = new Button(grpRefresh, SWT.RADIO);
+    btnRefreshOutputFolder.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false, 1, 1));
+    btnRefreshOutputFolder.setText(CompilerPreferencePage_refreshOutputProject);
+    new Label(contents, SWT.NONE);
+
+    updateFromPreferenceStore();
+    addEventListeners();
+
+    return contents;
+  }
+
+  private void updateFromPreferenceStore() {
+    IPreferenceStore store = doGetPreferenceStore();
+    boolean compileProtoFiles = store.getBoolean(COMPILE_PROTO_FILES);
+    btnCompileProtoFiles.setSelection(compileProtoFiles);
+    btnUseProtocInSystemPath.setSelection(store.getBoolean(USE_PROTOC_IN_SYSTEM_PATH));
+    btnUseProtocInCustomPath.setSelection(store.getBoolean(USE_PROTOC_IN_CUSTOM_PATH));
+    txtProtocFilePath.setText(store.getString(PROTOC_FILE_PATH));
+    btnJava.setSelection(store.getBoolean(GENERATE_JAVA_CODE));
+    btnCpp.setSelection(store.getBoolean(GENERATE_CPP_CODE));
+    btnPython.setSelection(store.getBoolean(GENERATE_PYTHON_CODE));
+    txtOutputFolderName.setText(store.getString(OUTPUT_FOLDER_NAME));
+    btnRefreshResources.setSelection(store.getBoolean(REFRESH_RESOURCES));
+    btnRefreshProject.setSelection(store.getBoolean(REFRESH_PROJECT));
+    btnRefreshOutputFolder.setSelection(store.getBoolean(REFRESH_OUTPUT_FOLDER));
+    boolean enableCompilerOptions = compileProtoFiles;
+    if (isPropertyPage()) {
+      boolean useProjectSettings = store.getBoolean(ENABLE_PROJECT_SETTINGS);
+      btnEnableProjectSettings.setSelection(useProjectSettings);
+      setPropertySpecificOptionsEnabled(useProjectSettings);
+      enableCompilerOptions = enableCompilerOptions && useProjectSettings;
+    }
+    setCompilerOptionsEnabled(enableCompilerOptions);
+  }
+
+  private void addEventListeners() {
+    btnCompileProtoFiles.addSelectionListener(new SelectionAdapter() {
+      @Override public void widgetSelected(SelectionEvent e) {
+        boolean selected = btnCompileProtoFiles.getSelection();
+        setCompilerOptionsEnabled(selected);
+        checkState();
+      }
+    });
+    btnUseProtocInSystemPath.addSelectionListener(new SelectionAdapter() {
+      @Override public void widgetSelected(SelectionEvent e) {
+        boolean selected = btnCompileProtoFiles.getSelection();
+        setCompilerCustomPathOptionsEnabled(!selected);
+        checkState();
+      }
+    });
+    btnUseProtocInCustomPath.addSelectionListener(new SelectionAdapter() {
+      @Override public void widgetSelected(SelectionEvent e) {
+        boolean selected = btnCompileProtoFiles.getSelection();
+        setCompilerCustomPathOptionsEnabled(selected);
+        checkState();
+      }
+    });
+    btnProtocPathBrowse.addSelectionListener(new SelectionAdapter() {
+      @Override public void widgetSelected(SelectionEvent e) {
+        FileDialog dialog = new FileDialog(getShell(), SWT.OPEN | SWT.SHEET);
+        String file = dialog.open();
+        if (file != null) txtProtocFilePath.setText(file);
+      }
+    });
+    btnRefreshResources.addSelectionListener(new SelectionAdapter() {
+      @Override public void widgetSelected(SelectionEvent e) {
+        refreshResourcesOptionsEnabled(btnRefreshResources.getSelection());
+      }
+    });
+    ModifyListener modifyListener = new ModifyListener() {
+      public void modifyText(ModifyEvent e) {
+        checkState();
+      }
+    };
+    txtProtocFilePath.addModifyListener(modifyListener);
+    txtOutputFolderName.addModifyListener(modifyListener);
+    if (isPropertyPage()) {
+      btnEnableProjectSettings.addSelectionListener(new SelectionAdapter() {
+        @Override public void widgetSelected(SelectionEvent e) {
+          boolean useProjectSettings = btnEnableProjectSettings.getSelection();
+          setPropertySpecificOptionsEnabled(useProjectSettings);
+          setCompilerOptionsEnabled(isEnabledAndSelected(btnCompileProtoFiles));
+        }
+      });
+      lnkEnableWorkspaceSettings.addSelectionListener(new SelectionAdapter() {
+        @Override public void widgetSelected(SelectionEvent e) {
+          openWorkspacePreferences(dataMap);
+        }
+      });
+    }
+  }
+
+  private void openWorkspacePreferences(Object data) {
+    String id = PREFERENCE_PAGE_ID;
+    PreferencesUtil.createPreferenceDialogOn(getShell(), id, new String[] { id }, data).open();
+  }
+
+  private void checkState() {
+    String outputFolderName = txtOutputFolderName.getText();
+    if (outputFolderName == null || outputFolderName.length() == 0) {
+      pageIsNowInvalid(CompilerPreferencePage_error_noOutputFolderName);
+      return;
+    }
+    IWorkspace workspace = ResourcesPlugin.getWorkspace();
+    IStatus validFolderName = workspace.validateName(outputFolderName, FOLDER);
+    if (validFolderName.getCode() != OK) {
+      pageIsNowInvalid(validFolderName.getMessage());
+      return;
+    }
+    if (!customPathOptionSelectedAndEnabled()) {
+      pageIsNowValid();
+      return;
+    }
+    String text = txtProtocFilePath.getText();
+    if (text == null || text.length() == 0) {
+      pageIsNowInvalid(CompilerPreferencePage_error_noSelection);
+      return;
+    }
+    File file = new File(text);
+    if (!file.isFile() || !"protoc".equals(file.getName())) {
+      pageIsNowInvalid(CompilerPreferencePage_error_invalidProtoc);
+      return;
+    }
+    pageIsNowValid();
+  }
+
+  private void pageIsNowValid() {
+    setErrorMessage(null);
+    setValid(true);
+  }
+
+  private void pageIsNowInvalid(String errorMessage) {
+    setErrorMessage(errorMessage);
+    setValid(false);
+  }
+
+  /** {@inheritDoc} */
+  @Override protected void performDefaults() {
+    IPreferenceStore store = doGetPreferenceStore();
+    boolean compileProtoFiles = store.getDefaultBoolean(COMPILE_PROTO_FILES);
+    btnCompileProtoFiles.setSelection(compileProtoFiles);
+    btnUseProtocInSystemPath.setSelection(store.getDefaultBoolean(USE_PROTOC_IN_SYSTEM_PATH));
+    btnUseProtocInCustomPath.setSelection(store.getDefaultBoolean(USE_PROTOC_IN_CUSTOM_PATH));
+    txtProtocFilePath.setText(store.getDefaultString(PROTOC_FILE_PATH));
+    btnJava.setSelection(store.getDefaultBoolean(GENERATE_JAVA_CODE));
+    btnCpp.setSelection(store.getDefaultBoolean(GENERATE_CPP_CODE));
+    btnPython.setSelection(store.getDefaultBoolean(GENERATE_PYTHON_CODE));
+    txtOutputFolderName.setText(store.getDefaultString(OUTPUT_FOLDER_NAME));
+    btnRefreshResources.setSelection(store.getDefaultBoolean(REFRESH_RESOURCES));
+    btnRefreshProject.setSelection(store.getDefaultBoolean(REFRESH_PROJECT));
+    btnRefreshOutputFolder.setSelection(store.getDefaultBoolean(REFRESH_OUTPUT_FOLDER));
+    boolean enableCompilerOptions = compileProtoFiles;
+    if (isPropertyPage()) {
+      boolean useProjectSettings = store.getDefaultBoolean(ENABLE_PROJECT_SETTINGS);
+      btnEnableProjectSettings.setSelection(useProjectSettings);
+      setPropertySpecificOptionsEnabled(useProjectSettings);
+      enableCompilerOptions = enableCompilerOptions && useProjectSettings;
+    }
+    setCompilerOptionsEnabled(enableCompilerOptions);
+    super.performDefaults();
+  }
+
+  private void setPropertySpecificOptionsEnabled(boolean enabled) {
+    btnCompileProtoFiles.setEnabled(enabled);
+    lnkEnableWorkspaceSettings.setEnabled(!enabled);
+  }
+
+  private void setCompilerOptionsEnabled(boolean enabled) {
+    tabFolder.setEnabled(enabled);
+    setCompilerPathOptionsEnabled(enabled);
+    setTargetLanguageOptionsEnabled(enabled);
+    setOutputOptionsEnabled(enabled);
+    setRefreshOptionsEnabled(enabled);
+  }
+
+  private void setCompilerPathOptionsEnabled(boolean enabled) {
+    grpCompilerLocation.setEnabled(enabled);
+    btnUseProtocInSystemPath.setEnabled(enabled);
+    btnUseProtocInCustomPath.setEnabled(enabled);
+    setCompilerCustomPathOptionsEnabled(customPathOptionSelectedAndEnabled());
+  }
+
+  private void setCompilerCustomPathOptionsEnabled(boolean enabled) {
+    txtProtocFilePath.setEnabled(enabled);
+    btnProtocPathBrowse.setEnabled(enabled);
+  }
+
+  private boolean customPathOptionSelectedAndEnabled() {
+    return isEnabledAndSelected(btnUseProtocInCustomPath);
+  }
+
+  private boolean isEnabledAndSelected(Button b) {
+    return b.isEnabled() && b.getSelection();
+  }
+
+  private void setTargetLanguageOptionsEnabled(boolean enabled) {
+    grpTargetLanguage.setEnabled(enabled);
+    btnJava.setEnabled(enabled);
+    btnCpp.setEnabled(enabled);
+    btnPython.setEnabled(enabled);
+  }
+
+  private void setOutputOptionsEnabled(boolean enabled) {
+    grpOutput.setEnabled(enabled);
+    lblOutputFolderName.setEnabled(enabled);
+    txtOutputFolderName.setEnabled(enabled);
+  }
+
+  private void setRefreshOptionsEnabled(boolean enabled) {
+    btnRefreshResources.setEnabled(enabled);
+    refreshResourcesOptionsEnabled(isEnabledAndSelected(btnRefreshResources));
+  }
+
+  private void refreshResourcesOptionsEnabled(boolean enabled) {
+    grpRefresh.setEnabled(enabled);
+    btnRefreshProject.setEnabled(enabled);
+    btnRefreshOutputFolder.setEnabled(enabled);
+  }
+
+  /** {@inheritDoc} */
+  public void init(IWorkbench workbench) {}
+
+  @Override public boolean performOk() {
+    savePreferences();
+    return true;
+  }
+
+  private void savePreferences() {
+    IPreferenceStore store = getPreferenceStore();
+    if (isPropertyPage()) {
+      store.setValue(ENABLE_PROJECT_SETTINGS, btnEnableProjectSettings.getSelection());
+    }
+    store.setValue(COMPILE_PROTO_FILES, btnCompileProtoFiles.getSelection());
+    store.setValue(USE_PROTOC_IN_SYSTEM_PATH, btnUseProtocInSystemPath.getSelection());
+    store.setValue(USE_PROTOC_IN_CUSTOM_PATH, btnUseProtocInCustomPath.getSelection());
+    store.setValue(PROTOC_FILE_PATH, txtProtocFilePath.getText());
+    store.setValue(GENERATE_JAVA_CODE, btnJava.getSelection());
+    store.setValue(GENERATE_CPP_CODE, btnCpp.getSelection());
+    store.setValue(GENERATE_PYTHON_CODE, btnPython.getSelection());
+    store.setValue(OUTPUT_FOLDER_NAME, txtOutputFolderName.getText());
+    store.setValue(REFRESH_RESOURCES, btnRefreshResources.getSelection());
+    store.setValue(REFRESH_PROJECT, btnRefreshProject.getSelection());
+    store.setValue(REFRESH_OUTPUT_FOLDER, btnRefreshOutputFolder.getSelection());
+  }
+
+  /** {@inheritDoc} */
+  public IAdaptable getElement() {
+    return project;
+  }
+
+  /** {@inheritDoc} */
+  public void setElement(IAdaptable element) {
+    this.project = (IProject) element.getAdapter(IProject.class);
+  }
+
+  @Override protected IPreferenceStore doGetPreferenceStore() {
+    if (isPropertyPage()) return preferenceStoreAccess.getWritablePreferenceStore(currentProject());
+    return preferenceStoreAccess.getWritablePreferenceStore();
+  }
+
+  private boolean isPropertyPage() {
+    return project != null;
+  }
+
+  private IProject currentProject() {
+    if (project == null)
+      throw new IllegalStateException("Not a property page case, but current project was requested.");
+    return project;
+  }
+
+  /** {@inheritDoc} */
+  @SuppressWarnings("unchecked") @Override public void applyData(Object data) {
+    if (data instanceof Map) this.dataMap = (Map<String, Object>) data;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferences.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferences.java
new file mode 100644
index 0000000..2488865
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferences.java
@@ -0,0 +1,47 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferenceNames.*;
+
+import org.eclipse.core.resources.IProject;
+import org.eclipse.jface.preference.IPreferenceStore;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreAccess;
+
+/**
+ * Compiler preferences, retrieved from an <code>{@link IPreferenceStore}</code>.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class CompilerPreferences {
+
+  public final boolean compileProtoFiles;
+  public final String protocPath;
+  public final TargetLanguage language;
+  public final String outputFolderName;
+  public final boolean refreshResources;
+  public final RefreshTarget refreshTarget;
+
+  public static CompilerPreferences loadPreferences(IPreferenceStoreAccess access, IProject project) {
+    IPreferenceStore store = access.getWritablePreferenceStore(project);
+    boolean useProjectPreferences = store.getBoolean(ENABLE_PROJECT_SETTINGS);
+    if (!useProjectPreferences) store = access.getWritablePreferenceStore();
+    return new CompilerPreferences(store);
+  }
+
+  private CompilerPreferences(IPreferenceStore store) {
+    compileProtoFiles = store.getBoolean(COMPILE_PROTO_FILES);
+    boolean useProtocInSystemPath = store.getBoolean(USE_PROTOC_IN_SYSTEM_PATH);
+    protocPath = (useProtocInSystemPath) ? "protoc" : store.getString(PROTOC_FILE_PATH);
+    language = TargetLanguage.find(store);
+    outputFolderName = store.getString(OUTPUT_FOLDER_NAME);
+    refreshResources = store.getBoolean(REFRESH_RESOURCES);
+    refreshTarget = RefreshTarget.find(store);
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencesInitializer.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencesInitializer.java
new file mode 100644
index 0000000..2895400
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/CompilerPreferencesInitializer.java
@@ -0,0 +1,35 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferenceNames.*;
+
+import org.eclipse.jface.preference.IPreferenceStore;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreAccess;
+import org.eclipse.xtext.ui.editor.preferences.IPreferenceStoreInitializer;
+
+/**
+ * Initializes default values for the "Compiler" preferences.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class CompilerPreferencesInitializer implements IPreferenceStoreInitializer {
+
+  /** {@inheritDoc} */
+  public void initialize(IPreferenceStoreAccess access) {
+    IPreferenceStore store = access.getWritablePreferenceStore();
+    store.setDefault(ENABLE_PROJECT_SETTINGS, false);
+    store.setDefault(USE_PROTOC_IN_SYSTEM_PATH, true);
+    store.setDefault(GENERATE_JAVA_CODE, true);
+    store.setDefault(OUTPUT_FOLDER_NAME, "src-gen");
+    store.setDefault(REFRESH_RESOURCES, true);
+    store.setDefault(REFRESH_OUTPUT_FOLDER, true);
+  }
+
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.java
new file mode 100644
index 0000000..89a0673
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.java
@@ -0,0 +1,46 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import org.eclipse.osgi.util.NLS;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class Messages extends NLS {
+
+  static {
+    Class<Messages> targetType = Messages.class;
+    NLS.initializeMessages(targetType.getName(), targetType);
+  }
+
+  private Messages() {}
+
+  public static String CompilerPreferencePage_enableProjectSettings;
+  public static String CompilerPreferencePage_configureWorkspaceSettings;
+  public static String CompilerPreferencePage_mainTab;
+  public static String CompilerPreferencePage_refreshTab;
+  public static String CompilerPreferencePage_browseCustomPath;
+  public static String CompilerPreferencePage_compileOnSave;
+  public static String CompilerPreferencePage_customPath;
+  public static String CompilerPreferencePage_location;
+  public static String CompilerPreferencePage_systemPath;
+  public static String CompilerPreferencePage_targetLanguage;
+  public static String CompilerPreferencePage_generateJava;
+  public static String CompilerPreferencePage_generateCpp;
+  public static String CompilerPreferencePage_generatePython;
+  public static String CompilerPreferencePage_generatedCode;
+  public static String CompilerPreferencePage_outputFolderName;
+  public static String CompilerPreferencePage_refreshResources;
+  public static String CompilerPreferencePage_refreshProject;
+  public static String CompilerPreferencePage_refreshOutputProject;
+  public static String CompilerPreferencePage_error_noSelection;
+  public static String CompilerPreferencePage_error_invalidProtoc;
+  public static String CompilerPreferencePage_error_noOutputFolderName;
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.properties b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.properties
new file mode 100644
index 0000000..b69400c
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/Messages.properties
@@ -0,0 +1,22 @@
+CompilerPreferencePage_enableProjectSettings=Enable project specific settings
+CompilerPreferencePage_configureWorkspaceSettings=Configure Workspace Settings...
+CompilerPreferencePage_mainTab=&Main
+CompilerPreferencePage_refreshTab=&Refresh
+CompilerPreferencePage_browseCustomPath=&Browse...
+CompilerPreferencePage_compileOnSave=Compile .proto files on &save
+CompilerPreferencePage_customPath=Use protoc &in:
+CompilerPreferencePage_location=Compiler location (protoc)
+CompilerPreferencePage_systemPath=Use protoc in &PATH
+CompilerPreferencePage_targetLanguage=Target Language
+CompilerPreferencePage_generateJava=&Java
+CompilerPreferencePage_generateCpp=&C++
+CompilerPreferencePage_generatePython=&Python
+CompilerPreferencePage_generatedCode=Generated Code
+CompilerPreferencePage_outputFolderName=Folder Name: *
+CompilerPreferencePage_refreshResources=Refresh resources upon completion.
+CompilerPreferencePage_refreshProject=Project
+CompilerPreferencePage_refreshOutputProject=Folder containing generated code
+CompilerPreferencePage_error_noSelection=Select the path of protoc
+CompilerPreferencePage_error_invalidProtoc=The selected file is not protoc
+CompilerPreferencePage_error_noOutputFolderName=Enter the name of the output folder
+
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/RefreshTarget.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/RefreshTarget.java
new file mode 100644
index 0000000..8b6e0e7
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/RefreshTarget.java
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferenceNames.REFRESH_PROJECT;
+
+import org.eclipse.jface.preference.IPreferenceStore;
+
+/**
+ * The type of resource to refresh after calling protoc.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public enum RefreshTarget {
+
+  PROJECT, OUTPUT_FOLDER;
+
+  static RefreshTarget find(IPreferenceStore store) {
+    if (store.getBoolean(REFRESH_PROJECT)) return PROJECT;
+    return OUTPUT_FOLDER;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/TargetLanguage.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/TargetLanguage.java
new file mode 100644
index 0000000..a1dcfef
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/preferences/TargetLanguage.java
@@ -0,0 +1,30 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.preferences;
+
+import static com.google.eclipse.protobuf.ui.preferences.CompilerPreferenceNames.*;
+
+import org.eclipse.jface.preference.IPreferenceStore;
+
+/**
+ * Languages supported by protoc.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public enum TargetLanguage {
+
+  JAVA, CPP, PYTHON;
+
+  static TargetLanguage find(IPreferenceStore store) {
+    if (store.getBoolean(GENERATE_JAVA_CODE)) return JAVA;
+    if (store.getBoolean(GENERATE_CPP_CODE)) return CPP;
+    if (store.getBoolean(GENERATE_PYTHON_CODE)) return PYTHON;
+    return JAVA;
+  }
+}
\ No newline at end of file
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/quickfix/ProtobufQuickfixProvider.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/quickfix/ProtobufQuickfixProvider.java
new file mode 100644
index 0000000..59fc306
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/quickfix/ProtobufQuickfixProvider.java
@@ -0,0 +1,29 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.quickfix;
+
+import org.eclipse.xtext.ui.editor.quickfix.DefaultQuickfixProvider;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufQuickfixProvider extends DefaultQuickfixProvider {
+
+//	@Fix(MyJavaValidator.INVALID_NAME)
+//	public void capitalizeName(final Issue issue, IssueResolutionAcceptor acceptor) {
+//		acceptor.accept(issue, "Capitalize name", "Capitalize the name.", "upcase.png", new IModification() {
+//			public void apply(IModificationContext context) throws BadLocationException {
+//				IXtextDocument xtextDocument = context.getXtextDocument();
+//				String firstLetter = xtextDocument.get(issue.getOffset(), 1);
+//				xtextDocument.replace(issue.getOffset(), 1, firstLetter.toUpperCase());
+//			}
+//		});
+//	}
+
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Literals.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Literals.java
new file mode 100644
index 0000000..9d8d8c0
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Literals.java
@@ -0,0 +1,35 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.util;
+
+import static java.lang.Math.max;
+
+import org.eclipse.emf.ecore.EObject;
+
+import com.google.eclipse.protobuf.protobuf.Literal;
+import com.google.inject.Singleton;
+
+/**
+ * Utility methods for instances of <code>{@link Literal}</code>.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Literals {
+
+  public int calculateIndexOf(Literal l) {
+    int index = 0;
+    for (EObject o : l.eContainer().eContents()) {
+      if (o == l || !(o instanceof Literal)) continue;
+      Literal c = (Literal) o;
+      index = max(index, c.getIndex());
+    }
+    return ++index;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Properties.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Properties.java
new file mode 100644
index 0000000..7e2fadc
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Properties.java
@@ -0,0 +1,62 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.util;
+
+import static java.lang.Math.max;
+
+import org.eclipse.emf.ecore.EObject;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.ui.grammar.Keywords;
+import com.google.inject.Inject;
+import com.google.inject.Singleton;
+
+/**
+ * Utility methods for instances of <code>{@link Property}</code>.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Properties {
+
+  @Inject private Keywords keywords;
+
+  public boolean isStringProperty(Property p) {
+    return keywords.string().getValue().equals(nameOfTypeIn(p));
+  }
+
+  public boolean isBoolProperty(Property p) {
+    return keywords.bool().getValue().equals(nameOfTypeIn(p));
+  }
+
+  /**
+   * Returns the name of the type of the given <code>{@link Property}</code>.
+   * @param p the given {@code Property}.
+   * @return the name of the type of the given {@code Property}.
+   */
+  public String nameOfTypeIn(Property p) {
+    AbstractTypeReference r = p.getType();
+    if (r instanceof ScalarTypeReference) return ((ScalarTypeReference) r).getScalar().getName();
+    if (r instanceof TypeReference) {
+      Type type = ((TypeReference) r).getType();
+      return type == null ? null : type.getName();
+    }
+    return r.toString();
+  }
+
+  public int calculateIndexOf(Property p) {
+    int index = 0;
+    for (EObject o : p.eContainer().eContents()) {
+      if (o == p || !(o instanceof Property)) continue;
+      Property c = (Property) o;
+      index = max(index, c.getIndex());
+    }
+    return ++index;
+  }
+}
diff --git a/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Strings.java b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Strings.java
new file mode 100644
index 0000000..6a9f9cb
--- /dev/null
+++ b/com.google.eclipse.protobuf.ui/src/com/google/eclipse/protobuf/ui/util/Strings.java
@@ -0,0 +1,31 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.ui.util;
+
+import static java.lang.Character.toLowerCase;
+
+import com.google.inject.Singleton;
+
+/**
+ * Utility classes for {@code String}s.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class Strings {
+
+  public String firstCharToLowerCase(String s) {
+    if (s == null) return null;
+    if (s.length() == 0) return s;
+    char[] chars = s.toCharArray();
+    chars[0] = toLowerCase(chars[0]);
+    return new String(chars);
+  }
+
+}
diff --git a/com.google.eclipse.protobuf/.antlr-generator-3.2.0.jar b/com.google.eclipse.protobuf/.antlr-generator-3.2.0.jar
new file mode 100644
index 0000000..4243492
--- /dev/null
+++ b/com.google.eclipse.protobuf/.antlr-generator-3.2.0.jar
Binary files differ
diff --git a/com.google.eclipse.protobuf/.classpath b/com.google.eclipse.protobuf/.classpath
new file mode 100644
index 0000000..7e8449d
--- /dev/null
+++ b/com.google.eclipse.protobuf/.classpath
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<classpath>
+	<classpathentry kind="src" path="src"/>
+	<classpathentry kind="src" path="src-gen"/>
+	<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/J2SE-1.5"/>
+	<classpathentry kind="con" path="org.eclipse.pde.core.requiredPlugins"/>
+	<classpathentry kind="output" path="bin"/>
+</classpath>
diff --git a/com.google.eclipse.protobuf/.project b/com.google.eclipse.protobuf/.project
new file mode 100644
index 0000000..b524930
--- /dev/null
+++ b/com.google.eclipse.protobuf/.project
@@ -0,0 +1,34 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<projectDescription>
+	<name>com.google.eclipse.protobuf</name>
+	<comment></comment>
+	<projects>
+	</projects>
+	<buildSpec>
+		<buildCommand>
+			<name>org.eclipse.jdt.core.javabuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+		<buildCommand>
+			<name>org.eclipse.pde.ManifestBuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+		<buildCommand>
+			<name>org.eclipse.pde.SchemaBuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+		<buildCommand>
+			<name>org.eclipse.xtext.ui.shared.xtextBuilder</name>
+			<arguments>
+			</arguments>
+		</buildCommand>
+	</buildSpec>
+	<natures>
+		<nature>org.eclipse.jdt.core.javanature</nature>
+		<nature>org.eclipse.pde.PluginNature</nature>
+		<nature>org.eclipse.xtext.ui.shared.xtextNature</nature>
+	</natures>
+</projectDescription>
diff --git a/com.google.eclipse.protobuf/.settings/org.eclipse.jdt.core.prefs b/com.google.eclipse.protobuf/.settings/org.eclipse.jdt.core.prefs
new file mode 100644
index 0000000..002551b
--- /dev/null
+++ b/com.google.eclipse.protobuf/.settings/org.eclipse.jdt.core.prefs
@@ -0,0 +1,81 @@
+#Wed Apr 06 10:56:09 PDT 2011
+eclipse.preferences.version=1
+org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled
+org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.5
+org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve
+org.eclipse.jdt.core.compiler.compliance=1.5
+org.eclipse.jdt.core.compiler.debug.lineNumber=generate
+org.eclipse.jdt.core.compiler.debug.localVariable=generate
+org.eclipse.jdt.core.compiler.debug.sourceFile=generate
+org.eclipse.jdt.core.compiler.problem.annotationSuperInterface=warning
+org.eclipse.jdt.core.compiler.problem.assertIdentifier=error
+org.eclipse.jdt.core.compiler.problem.autoboxing=ignore
+org.eclipse.jdt.core.compiler.problem.comparingIdentical=warning
+org.eclipse.jdt.core.compiler.problem.deadCode=warning
+org.eclipse.jdt.core.compiler.problem.deprecation=warning
+org.eclipse.jdt.core.compiler.problem.deprecationInDeprecatedCode=disabled
+org.eclipse.jdt.core.compiler.problem.deprecationWhenOverridingDeprecatedMethod=disabled
+org.eclipse.jdt.core.compiler.problem.discouragedReference=warning
+org.eclipse.jdt.core.compiler.problem.emptyStatement=ignore
+org.eclipse.jdt.core.compiler.problem.enumIdentifier=error
+org.eclipse.jdt.core.compiler.problem.fallthroughCase=ignore
+org.eclipse.jdt.core.compiler.problem.fatalOptionalError=disabled
+org.eclipse.jdt.core.compiler.problem.fieldHiding=warning
+org.eclipse.jdt.core.compiler.problem.finalParameterBound=warning
+org.eclipse.jdt.core.compiler.problem.finallyBlockNotCompletingNormally=warning
+org.eclipse.jdt.core.compiler.problem.forbiddenReference=error
+org.eclipse.jdt.core.compiler.problem.hiddenCatchBlock=warning
+org.eclipse.jdt.core.compiler.problem.includeNullInfoFromAsserts=disabled
+org.eclipse.jdt.core.compiler.problem.incompatibleNonInheritedInterfaceMethod=warning
+org.eclipse.jdt.core.compiler.problem.incompleteEnumSwitch=ignore
+org.eclipse.jdt.core.compiler.problem.indirectStaticAccess=warning
+org.eclipse.jdt.core.compiler.problem.localVariableHiding=warning
+org.eclipse.jdt.core.compiler.problem.methodWithConstructorName=warning
+org.eclipse.jdt.core.compiler.problem.missingDeprecatedAnnotation=ignore
+org.eclipse.jdt.core.compiler.problem.missingHashCodeMethod=warning
+org.eclipse.jdt.core.compiler.problem.missingOverrideAnnotation=ignore
+org.eclipse.jdt.core.compiler.problem.missingOverrideAnnotationForInterfaceMethodImplementation=enabled
+org.eclipse.jdt.core.compiler.problem.missingSerialVersion=warning
+org.eclipse.jdt.core.compiler.problem.missingSynchronizedOnInheritedMethod=warning
+org.eclipse.jdt.core.compiler.problem.noEffectAssignment=warning
+org.eclipse.jdt.core.compiler.problem.noImplicitStringConversion=warning
+org.eclipse.jdt.core.compiler.problem.nonExternalizedStringLiteral=ignore
+org.eclipse.jdt.core.compiler.problem.nullReference=warning
+org.eclipse.jdt.core.compiler.problem.overridingPackageDefaultMethod=warning
+org.eclipse.jdt.core.compiler.problem.parameterAssignment=warning
+org.eclipse.jdt.core.compiler.problem.possibleAccidentalBooleanAssignment=warning
+org.eclipse.jdt.core.compiler.problem.potentialNullReference=warning
+org.eclipse.jdt.core.compiler.problem.rawTypeReference=warning
+org.eclipse.jdt.core.compiler.problem.redundantNullCheck=warning
+org.eclipse.jdt.core.compiler.problem.redundantSuperinterface=warning
+org.eclipse.jdt.core.compiler.problem.reportMethodCanBePotentiallyStatic=ignore
+org.eclipse.jdt.core.compiler.problem.reportMethodCanBeStatic=ignore
+org.eclipse.jdt.core.compiler.problem.specialParameterHidingField=disabled
+org.eclipse.jdt.core.compiler.problem.staticAccessReceiver=warning
+org.eclipse.jdt.core.compiler.problem.suppressOptionalErrors=disabled
+org.eclipse.jdt.core.compiler.problem.suppressWarnings=enabled
+org.eclipse.jdt.core.compiler.problem.syntheticAccessEmulation=ignore
+org.eclipse.jdt.core.compiler.problem.typeParameterHiding=warning
+org.eclipse.jdt.core.compiler.problem.unavoidableGenericTypeProblems=enabled
+org.eclipse.jdt.core.compiler.problem.uncheckedTypeOperation=warning
+org.eclipse.jdt.core.compiler.problem.undocumentedEmptyBlock=ignore
+org.eclipse.jdt.core.compiler.problem.unhandledWarningToken=warning
+org.eclipse.jdt.core.compiler.problem.unnecessaryElse=warning
+org.eclipse.jdt.core.compiler.problem.unnecessaryTypeCheck=warning
+org.eclipse.jdt.core.compiler.problem.unqualifiedFieldAccess=ignore
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownException=warning
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionExemptExceptionAndThrowable=enabled
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionIncludeDocCommentReference=enabled
+org.eclipse.jdt.core.compiler.problem.unusedDeclaredThrownExceptionWhenOverriding=disabled
+org.eclipse.jdt.core.compiler.problem.unusedImport=warning
+org.eclipse.jdt.core.compiler.problem.unusedLabel=warning
+org.eclipse.jdt.core.compiler.problem.unusedLocal=warning
+org.eclipse.jdt.core.compiler.problem.unusedObjectAllocation=ignore
+org.eclipse.jdt.core.compiler.problem.unusedParameter=warning
+org.eclipse.jdt.core.compiler.problem.unusedParameterIncludeDocCommentReference=enabled
+org.eclipse.jdt.core.compiler.problem.unusedParameterWhenImplementingAbstract=disabled
+org.eclipse.jdt.core.compiler.problem.unusedParameterWhenOverridingConcrete=disabled
+org.eclipse.jdt.core.compiler.problem.unusedPrivateMember=warning
+org.eclipse.jdt.core.compiler.problem.unusedWarningToken=warning
+org.eclipse.jdt.core.compiler.problem.varargsArgumentNeedCast=warning
+org.eclipse.jdt.core.compiler.source=1.5
diff --git a/com.google.eclipse.protobuf/META-INF/MANIFEST.MF b/com.google.eclipse.protobuf/META-INF/MANIFEST.MF
new file mode 100644
index 0000000..cbe1187
--- /dev/null
+++ b/com.google.eclipse.protobuf/META-INF/MANIFEST.MF
@@ -0,0 +1,32 @@
+Manifest-Version: 1.0

+Bundle-ManifestVersion: 2

+Bundle-Name: %Bundle-Name

+Bundle-Vendor: %Bundle-Vendor

+Bundle-Version: 1.0.0

+Bundle-SymbolicName: com.google.eclipse.protobuf; singleton:=true

+Bundle-ActivationPolicy: lazy

+Require-Bundle: org.eclipse.xtext,

+ org.eclipse.xtext.generator;resolution:=optional,

+ org.eclipse.emf.codegen.ecore;resolution:=optional,

+ org.eclipse.emf.mwe.utils;resolution:=optional,

+ org.eclipse.emf.mwe2.launch;resolution:=optional,

+ com.ibm.icu;resolution:=optional,

+ org.eclipse.xtext.util,

+ org.eclipse.emf.ecore,

+ org.eclipse.emf.common,

+ org.antlr.runtime,

+ org.eclipse.core.runtime;bundle-version="3.7.0"

+Import-Package: org.apache.log4j,

+ org.apache.commons.logging

+Bundle-RequiredExecutionEnvironment: J2SE-1.5

+Export-Package: com.google.eclipse.protobuf,

+ com.google.eclipse.protobuf.parseTreeConstruction,

+ com.google.eclipse.protobuf.parser.antlr,

+ com.google.eclipse.protobuf.parser.antlr.internal,

+ com.google.eclipse.protobuf.protobuf,

+ com.google.eclipse.protobuf.protobuf.impl,

+ com.google.eclipse.protobuf.protobuf.util,

+ com.google.eclipse.protobuf.scoping,

+ com.google.eclipse.protobuf.services,

+ com.google.eclipse.protobuf.util,

+ com.google.eclipse.protobuf.validation

diff --git a/com.google.eclipse.protobuf/OSGI-INF/l10n/bundle.properties b/com.google.eclipse.protobuf/OSGI-INF/l10n/bundle.properties
new file mode 100644
index 0000000..a8b3cdb
--- /dev/null
+++ b/com.google.eclipse.protobuf/OSGI-INF/l10n/bundle.properties
@@ -0,0 +1,3 @@
+#Properties file for com.google.eclipse.protobuf
+Bundle-Vendor = Google Inc.
+Bundle-Name = com.google.eclipse.protobuf
\ No newline at end of file
diff --git a/com.google.eclipse.protobuf/build.properties b/com.google.eclipse.protobuf/build.properties
new file mode 100644
index 0000000..300fb1a
--- /dev/null
+++ b/com.google.eclipse.protobuf/build.properties
@@ -0,0 +1,6 @@
+source.. = src/,\
+          src-gen/
+bin.includes = META-INF/,\
+               .,\
+               plugin.xml,\
+               OSGI-INF/
diff --git a/com.google.eclipse.protobuf/global.proto b/com.google.eclipse.protobuf/global.proto
new file mode 100644
index 0000000..294cd0e
--- /dev/null
+++ b/com.google.eclipse.protobuf/global.proto
@@ -0,0 +1,22 @@
+message FileOptions {
+
+  optional string java_package = 1;
+  optional string java_outer_classname = 8;
+  optional bool java_multiple_files = 10 [default=false];
+  optional bool java_generate_equals_and_hash = 20 [default=false];
+
+  enum OptimizeMode {
+    SPEED = 1;        // Generate complete code for parsing, serialization, etc.
+    CODE_SIZE = 2;    // Use ReflectionOps to implement these methods.
+    LITE_RUNTIME = 3; // Generate code using MessageLite and the lite runtime.
+  }
+
+  optional OptimizeMode optimize_for = 9 [default=SPEED];
+  optional bool cc_generic_services = 16 [default=false];
+  optional bool java_generic_services = 17 [default=false];
+  optional bool py_generic_services = 18 [default=false];
+
+  repeated UninterpretedOption uninterpreted_option = 999;
+
+  extensions 1000 to max;
+}
diff --git a/com.google.eclipse.protobuf/plugin.xml b/com.google.eclipse.protobuf/plugin.xml
new file mode 100644
index 0000000..c9018bb
--- /dev/null
+++ b/com.google.eclipse.protobuf/plugin.xml
@@ -0,0 +1,17 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<?eclipse version="3.0"?>
+
+<plugin>
+
+  <extension point="org.eclipse.emf.ecore.generated_package">
+    <package 
+       uri = "http://www.google.com/eclipse/protobuf/Protobuf" 
+       class = "com.google.eclipse.protobuf.protobuf.ProtobufPackage"
+       genModel = "com/google/eclipse/protobuf/Protobuf.genmodel" /> 
+	
+  </extension>
+
+
+
+
+</plugin>
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/GenerateProtobuf.mwe2 b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/GenerateProtobuf.mwe2
new file mode 100644
index 0000000..a954940
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/GenerateProtobuf.mwe2
@@ -0,0 +1,109 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ * 
+ * Generated by Xtext
+ */
+module com.google.eclipse.protobuf.Protobuf
+
+import org.eclipse.emf.mwe.utils.*
+import org.eclipse.xtext.generator.*
+import org.eclipse.xtext.ui.generator.*
+
+var grammarURI = "classpath:/com/google/eclipse/protobuf/Protobuf.xtext"
+var file.extensions = "proto"
+var projectName = "com.google.eclipse.protobuf"
+var runtimeProject = "../${projectName}"
+
+Workflow {
+    bean = StandaloneSetup {
+		platformUri = "${runtimeProject}/.."
+	}
+
+	component = DirectoryCleaner {
+		directory = "${runtimeProject}/src-gen"
+	}
+
+	component = DirectoryCleaner {
+		directory = "${runtimeProject}.ui/src-gen"
+	}
+
+	component = Generator {
+		pathRtProject = runtimeProject
+		pathUiProject = "${runtimeProject}.ui"
+		projectNameRt = projectName
+		projectNameUi = "${projectName}.ui"
+		language = {
+			uri = grammarURI
+			fileExtensions = file.extensions
+
+			// Java API to access grammar elements (required by several other fragments)
+			fragment = grammarAccess.GrammarAccessFragment {}
+
+			// generates Java API for the generated EPackages 
+			fragment = ecore.EcoreGeneratorFragment {
+			// referencedGenModels = "uri to genmodel, uri to next genmodel"
+			}
+
+			// the serialization component
+			fragment = parseTreeConstructor.ParseTreeConstructorFragment {}
+
+			// a custom ResourceFactory for use with EMF 
+			fragment = resourceFactory.ResourceFactoryFragment {
+				fileExtensions = file.extensions
+			}
+
+			// The antlr parser generator fragment.
+			fragment = parser.antlr.XtextAntlrGeneratorFragment {
+			//  options = {
+			//		backtrack = true
+			//	}
+			}
+
+			// java-based API for validation 
+			fragment = validation.JavaValidatorFragment {
+				composedCheck = "org.eclipse.xtext.validation.ImportUriValidator"
+				composedCheck = "org.eclipse.xtext.validation.NamesAreUniqueValidator"
+				// registerForImportedPackages = true
+			}
+
+			// scoping and exporting API
+			// fragment = scoping.ImportURIScopingFragment {}
+			// fragment = exporting.SimpleNamesFragment {}
+
+			// scoping and exporting API 
+			fragment = scoping.ImportNamespacesScopingFragment {}
+			fragment = exporting.QualifiedNamesFragment {}
+			fragment = builder.BuilderIntegrationFragment {}
+
+			// formatter API 
+			fragment = formatting.FormatterFragment {}
+
+			// labeling API 
+			fragment = labeling.LabelProviderFragment {}
+
+			// outline API 
+			fragment = outline.OutlineTreeProviderFragment {}
+			fragment = outline.QuickOutlineFragment {}
+
+			// quickfix API 
+			fragment = quickfix.QuickfixProviderFragment {}
+
+			// content assist API  
+			fragment = contentAssist.JavaBasedContentAssistFragment {}
+
+			// generates a more lightweight Antlr parser and lexer tailored for content assist  
+			fragment = parser.antlr.XtextAntlrUiGeneratorFragment {}
+
+			// project wizard (optional) 
+			// fragment = projectWizard.SimpleProjectWizardFragment {
+			// 		generatorProjectName = "${projectName}.generator" 
+			//		modelFileExtension = file.extensions
+			// }
+		}
+	}
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/Protobuf.xtext b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/Protobuf.xtext
new file mode 100644
index 0000000..944e1b2
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/Protobuf.xtext
@@ -0,0 +1,126 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ * 
+ * Author: alruiz@google.com (Alex Ruiz)
+ */
+
+grammar com.google.eclipse.protobuf.Protobuf with org.eclipse.xtext.common.Terminals
+
+import "http://www.eclipse.org/emf/2002/Ecore" as ecore
+generate protobuf "http://www.google.com/eclipse/protobuf/Protobuf"
+
+Protobuf:
+  (package=Package)?
+  (imports+=Import)*
+  (options+=Option)*
+  (elements+=ProtobufElement)*;
+  
+Package:
+  'package' name=QualifiedName ';';
+
+Import: 
+  'import' importURI=STRING ';';
+
+QualifiedName:
+  ID ('.' ID)*;
+
+Option:
+  'option' name=ID '=' value=ValueRef ';';
+
+ProtobufElement:
+  Type | ExtendMessage;
+
+Type:
+  Message | Enum;
+
+Message:
+  'message' name=ID '{'
+  elements+=MessageElement*
+  ('extensions' extensionsFrom=INT 'to' (extensionsTo=INT | 'max') ';')?
+  '}';
+
+MessageElement:
+  Type | Property;
+
+Property:
+  modifier=Modifier type=AbstractTypeReference name=ID '=' index=INT (('[' 'default' '=' default=ValueRef
+  ']') | ('[' 'packed' '=' packed=BooleanRef ']'))? ';';
+
+enum Modifier:
+  required
+  | optional
+  | repeated;
+
+AbstractTypeReference:
+  ScalarTypeReference | TypeReference;
+
+ScalarTypeReference:
+  scalar=ScalarType;
+
+enum ScalarType:
+  double
+  | float
+  | int32
+  | int64
+  | uint32
+  | uint64
+  | sint32
+  | sint64
+  | fixed32
+  | fixed64
+  | sfixed32
+  | sfixed64
+  | bool
+  | string
+  | bytes;
+
+TypeReference:
+  type=[Type | QualifiedName];
+
+ValueRef:
+  LiteralRef
+  | BooleanRef
+  | IntRef
+  | FloatRef
+  | StringRef;
+
+LiteralRef:
+  literal=[Literal];
+
+BooleanRef:
+  boolean=BOOL;
+
+enum BOOL:
+  true
+  | false;
+
+IntRef:
+  int=INT;
+
+FloatRef:
+  float=FLOAT;
+
+StringRef:
+  string=STRING;
+
+Enum:
+  'enum' name=ID '{'
+  literals+=Literal*
+  '}' ';'?;
+
+Literal:
+  name=ID '=' index=INT ';';
+
+ExtendMessage:
+  'extend' message=[Message] '{'
+  elements+=MessageElement*
+  '}';
+
+terminal INT returns ecore::EInt: ('0'..'9')+;  
+terminal FLOAT returns ecore::EFloat: ('0'..'9')* ('.' ('0'..'9')+)?;
+  
\ No newline at end of file
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufRuntimeModule.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufRuntimeModule.java
new file mode 100644
index 0000000..5492360
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufRuntimeModule.java
@@ -0,0 +1,37 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf;
+
+import org.eclipse.xtext.naming.IQualifiedNameProvider;
+import org.eclipse.xtext.scoping.IGlobalScopeProvider;
+import org.eclipse.xtext.scoping.impl.*;
+
+import com.google.eclipse.protobuf.naming.ProtobufQualifiedNameProvider;
+import com.google.eclipse.protobuf.scoping.SimpleImportUriResolver;
+import com.google.inject.Binder;
+
+/**
+ * Use this class to register components to be used at runtime / without the Equinox extension registry.
+ */
+public class ProtobufRuntimeModule extends com.google.eclipse.protobuf.AbstractProtobufRuntimeModule {
+
+  /** {@inheritDoc} */
+  @Override public Class<? extends IGlobalScopeProvider> bindIGlobalScopeProvider() {
+    return ImportUriGlobalScopeProvider.class;
+  }
+
+  /** {@inheritDoc} */
+  @Override public Class<? extends IQualifiedNameProvider> bindIQualifiedNameProvider() {
+    return ProtobufQualifiedNameProvider.class;
+  }
+
+  public void configureImportUriResolver(Binder binder) {
+    binder.bind(ImportUriResolver.class).to(SimpleImportUriResolver.class);
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufStandaloneSetup.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufStandaloneSetup.java
new file mode 100644
index 0000000..ac0af7c
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/ProtobufStandaloneSetup.java
@@ -0,0 +1,20 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf;
+
+/**
+ * Initialization support for running Xtext languages without Equinox extension registry.
+ */
+public class ProtobufStandaloneSetup extends ProtobufStandaloneSetupGenerated{
+
+	public static void doSetup() {
+		new ProtobufStandaloneSetup().createInjectorAndDoEMFRegistration();
+	}
+}
+
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/formatting/ProtobufFormatter.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/formatting/ProtobufFormatter.java
new file mode 100644
index 0000000..10d911a
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/formatting/ProtobufFormatter.java
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.formatting;
+
+import org.eclipse.xtext.formatting.impl.*;
+
+import com.google.eclipse.protobuf.services.ProtobufGrammarAccess;
+
+/**
+ * This class contains custom formatting description.
+ *
+ * @see http://www.eclipse.org/Xtext/documentation/latest/xtext.html#formatting on how and when to use it
+ * @see {@link org.eclipse.xtext.xtext.XtextFormattingTokenSerializer} as an example
+ */
+public class ProtobufFormatter extends AbstractDeclarativeFormatter {
+
+	@Override
+	protected void configureFormatting(FormattingConfig c) {
+	  ProtobufGrammarAccess g = (ProtobufGrammarAccess) getGrammarAccess();
+	  c.setLinewrap(0, 1, 2).before(g.getSL_COMMENTRule());
+	}
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/naming/ProtobufQualifiedNameProvider.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/naming/ProtobufQualifiedNameProvider.java
new file mode 100644
index 0000000..2019edc
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/naming/ProtobufQualifiedNameProvider.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.naming;
+
+import java.util.*;
+
+import org.eclipse.emf.ecore.EObject;
+import org.eclipse.xtext.naming.*;
+
+import com.google.eclipse.protobuf.protobuf.Package;
+import com.google.eclipse.protobuf.util.EObjectFinder;
+import com.google.inject.Inject;
+
+/**
+ * Provides fully-qualified names for protobuf elements.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufQualifiedNameProvider extends DefaultDeclarativeQualifiedNameProvider {
+
+  @Inject private EObjectFinder finder;
+
+  /** {@inheritDoc} */
+  @Override public QualifiedName getFullyQualifiedName(EObject obj) {
+    QualifiedName fqn = super.getFullyQualifiedName(obj);
+    if (fqn == null || obj instanceof Package) return fqn;
+    Package p = finder.findPackage(obj);
+    if (p == null) return fqn;
+    List<String> segments = new ArrayList<String>(fqn.getSegments());
+    String packageName = p.getName();
+    if (packageName != null && !segments.isEmpty() && !packageName.equals(segments.get(0))) 
+      segments.add(0, packageName);
+    return QualifiedName.create(segments.toArray(new String[segments.size()]));
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/GlobalScope.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/GlobalScope.java
new file mode 100644
index 0000000..0e3dfc2
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/GlobalScope.java
@@ -0,0 +1,114 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.scoping;
+
+import static java.util.Collections.unmodifiableCollection;
+import static org.eclipse.core.runtime.FileLocator.*;
+
+import java.io.*;
+import java.net.URL;
+import java.util.*;
+
+import org.eclipse.core.runtime.Path;
+import org.eclipse.core.runtime.Platform;
+import org.eclipse.emf.common.util.URI;
+import org.eclipse.xtext.parser.IParseResult;
+import org.eclipse.xtext.parser.IParser;
+import org.eclipse.xtext.resource.XtextResource;
+import org.osgi.framework.Bundle;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.protobuf.Enum;
+import com.google.eclipse.protobuf.util.EObjectFinder;
+import com.google.inject.Inject;
+import com.google.inject.Singleton;
+
+/**
+ * Protobuf elements accesible to any .proto file.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class GlobalScope {
+
+  private Protobuf root;
+
+  private boolean initialized;
+  private Map<String, Property> fileOptions = new LinkedHashMap<String, Property>();
+  private Enum optimizedMode;
+
+  @Inject EObjectFinder finder;
+
+  @Inject public GlobalScope(IParser parser) {
+    try {
+      XtextResource resource = new XtextResource(URI.createURI(""));
+      IParseResult result = parser.parse(new InputStreamReader(globalScopeContents(), "UTF-8"));
+      root = (Protobuf) result.getRootASTElement();
+      resource.getContents().add(root);
+    } catch (IOException e) {
+      throw new IllegalStateException("Unable to parse global scope", e);
+    }
+  }
+
+  private static InputStream globalScopeContents() throws IOException {
+    Bundle bundle = Platform.getBundle("com.google.eclipse.protobuf");
+    Path originPath = new Path("global.proto");
+    URL bundledFileURL = find(bundle, originPath, null);
+    return (InputStream) resolve(bundledFileURL).getContent();
+  }
+
+  private void init() {
+    if (initialized) return;
+    initialized = true;
+    Message m = fileOptionsMessage();
+    for (MessageElement e : m.getElements()) {
+      if (e instanceof Property) {
+        addFileOption((Property) e);
+        continue;
+      }
+      if (e instanceof Enum && "OptimizeMode".equals(e.getName())) {
+        optimizedMode = (Enum) e;
+        continue;
+      }
+    }
+  }
+
+  private Message fileOptionsMessage() {
+    for (ProtobufElement e : root.getElements()) {
+      if (!(e instanceof Message)) continue;
+      Message m = (Message) e;
+      if ("FileOptions".equals(m.getName())) return m;
+    }
+    return null;
+  }
+
+  private void addFileOption(Property p) {
+    fileOptions.put(p.getName(), p);
+  }
+
+  public Collection<Property> fileOptions() {
+    init();
+    return unmodifiableCollection(fileOptions.values());
+  }
+
+  public Enum optimizedMode() {
+    init();
+    return optimizedMode;
+  }
+
+  public boolean isOptimizeForOption(Option option) {
+    init();
+    return "optimize_for".equals(option.getName());
+  }
+
+  public Property lookupFileOption(String name) {
+    init();
+    return fileOptions.get(name);
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/ProtobufScopeProvider.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/ProtobufScopeProvider.java
new file mode 100644
index 0000000..2a32b85
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/ProtobufScopeProvider.java
@@ -0,0 +1,64 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.scoping;
+
+import static org.eclipse.xtext.resource.EObjectDescription.create;
+
+import java.util.*;
+
+import org.eclipse.emf.ecore.*;
+import org.eclipse.xtext.resource.IEObjectDescription;
+import org.eclipse.xtext.scoping.IScope;
+import org.eclipse.xtext.scoping.impl.*;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.protobuf.Enum;
+import com.google.eclipse.protobuf.util.EObjectFinder;
+import com.google.inject.Inject;
+
+/**
+ * Custom scoping description.
+ *
+ * @author alruiz@google.com (Alex Ruiz)
+ *
+ * @see http://www.eclipse.org/Xtext/documentation/latest/xtext.html#scoping
+ */
+public class ProtobufScopeProvider extends AbstractDeclarativeScopeProvider {
+
+  private static final boolean DO_NOT_IGNORE_CASE = false;
+
+  @Inject private EObjectFinder finder;
+  @Inject private GlobalScope globalScope;
+
+  @SuppressWarnings("unused")
+  IScope scope_LiteralRef_literal(LiteralRef literalRef, EReference reference) {
+    EObject container = literalRef.eContainer();
+    if (container instanceof Property) {
+      Enum enumType = finder.enumTypeOfProperty((Property) container);
+      if (enumType != null) return scopeForLiteralsIn(enumType);
+    }
+    if (container instanceof Option && globalScope.isOptimizeForOption((Option) container)) {
+      Enum optimizedMode = globalScope.optimizedMode();
+      return scopeForLiteralsIn(optimizedMode);
+    }
+    return null;
+  }
+
+  private static IScope scopeForLiteralsIn(Enum enumType) {
+    List<IEObjectDescription> descriptions = descriptionsFrom(enumType);
+    return new SimpleScope(descriptions, DO_NOT_IGNORE_CASE);
+  }
+
+  private static List<IEObjectDescription> descriptionsFrom(Enum enumType) {
+    List<IEObjectDescription> descriptions = new ArrayList<IEObjectDescription>();
+    for (Literal literal : enumType.getLiterals())
+      descriptions.add(create(literal.getName(), literal));
+    return descriptions;
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/SimpleImportUriResolver.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/SimpleImportUriResolver.java
new file mode 100644
index 0000000..4e57e85
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/scoping/SimpleImportUriResolver.java
@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.scoping;
+
+import java.util.List;
+
+import org.eclipse.emf.common.util.URI;
+import org.eclipse.emf.ecore.EObject;
+import org.eclipse.xtext.scoping.impl.ImportUriResolver;
+
+import com.google.eclipse.protobuf.protobuf.Import;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class SimpleImportUriResolver extends ImportUriResolver {
+
+  private static final String PREFIX = "platform:/resource";
+
+  /**
+   * Prefix used by EMF for resource URIs: "platform:/resource/".
+   */
+  public static final String URI_PREFIX = PREFIX + "/";
+
+  /** {@inheritDoc} */
+  @Override public String apply(EObject from) {
+    if (from instanceof Import)
+      fixUri((Import) from);
+    return super.apply(from);
+  }
+
+  /*
+   * The import URI is relative to the file where the import is. Protoc works fine, but the editor doesn't.
+   * In order for the editor to see the import, we need to add to the import URI "platform:resource" and the parent
+   * folder of the file containing the import.
+   *
+   * For example: given the following file hierarchy:
+   *
+   * - protobuf-test (project)
+   *   - folder
+   *     - proto2.proto
+   *   - proto1.proto
+   *
+   * If we import "folder/proto2.proto" into proto1.proto, proto1.proto will compile fine, but the editor will complain.
+   * We need to have the import URI as "platform:/resource/protobuf-test/folder/proto2.proto" for the editor to see it.
+   */
+  private void fixUri(Import i) {
+    String prefix = uriPrefix(i.eResource().getURI());
+    String uri = i.getImportURI();
+    if (!uri.startsWith(prefix)) {
+      if (!uri.startsWith(prefix)) prefix += "/";
+      i.setImportURI(prefix + uri);
+    }
+  }
+
+  private String uriPrefix(URI containerUri) {
+    StringBuilder prefix = new StringBuilder();
+    prefix.append(PREFIX);
+    int start = (containerUri.scheme() == null) ? 0 : 1; // ignore the scheme if present (e.g. "resource")
+    List<String> segments = containerUri.segmentsList();
+    int end = segments.size() - 1; // ignore file name (at the end of the URI)
+    for (int j = start; j < end; j++)
+      prefix.append("/").append(segments.get(j));
+    return prefix.toString();
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/util/EObjectFinder.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/util/EObjectFinder.java
new file mode 100644
index 0000000..82c470a
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/util/EObjectFinder.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.util;
+
+import org.eclipse.emf.ecore.EObject;
+
+import com.google.eclipse.protobuf.protobuf.*;
+import com.google.eclipse.protobuf.protobuf.Enum;
+import com.google.eclipse.protobuf.protobuf.Package;
+import com.google.inject.Singleton;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+@Singleton
+public class EObjectFinder {
+
+  public Enum enumTypeOfProperty(Property p) {
+    AbstractTypeReference aTypeRef = (p).getType();
+    if (aTypeRef instanceof TypeReference) {
+      Type type = ((TypeReference) aTypeRef).getType();
+      if (type instanceof Enum) return (Enum) type;
+    }
+    return null;
+  }
+
+  public ScalarType scalarTypeOfProperty(Property p) {
+    AbstractTypeReference aTypeRef = (p).getType();
+    if (aTypeRef instanceof ScalarTypeReference)
+      return ((ScalarTypeReference) aTypeRef).getScalar();
+    return null;
+  }
+
+  public Package findPackage(EObject o) {
+    EObject current = o;
+    while (current != null) {
+      if (current instanceof Protobuf) return ((Protobuf) current).getPackage();
+      current = current.eContainer();
+    }
+    return null;
+  }
+}
diff --git a/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/validation/ProtobufJavaValidator.java b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/validation/ProtobufJavaValidator.java
new file mode 100644
index 0000000..5932ad4
--- /dev/null
+++ b/com.google.eclipse.protobuf/src/com/google/eclipse/protobuf/validation/ProtobufJavaValidator.java
@@ -0,0 +1,23 @@
+/*
+ * Copyright (c) 2011 Google Inc.
+ *
+ * All rights reserved. This program and the accompanying materials are made available under the terms of the Eclipse
+ * Public License v1.0 which accompanies this distribution, and is available at
+ *
+ * http://www.eclipse.org/legal/epl-v10.html
+ */
+package com.google.eclipse.protobuf.validation;
+
+/**
+ * @author alruiz@google.com (Alex Ruiz)
+ */
+public class ProtobufJavaValidator extends AbstractProtobufJavaValidator {
+
+//	@Check
+//	public void checkGreetingStartsWithCapital(Greeting greeting) {
+//		if (!Character.isUpperCase(greeting.getName().charAt(0))) {
+//			warning("Name should start with a capital", MyDslPackage.GREETING__NAME);
+//		}
+//	}
+
+}