diff --git a/app/App.php b/app/App.php
index 1641655..1b64e50 100644
--- a/app/App.php
+++ b/app/App.php
@@ -69,10 +69,23 @@ class App extends Application {
}
/**
- * @return \Noodlehaus\Config
+ * Registers media icons
+ *
+ * @param \Sikofitt\Image\Profile\ProfileIconInterface $icon
*/
- public function conf2() {
- return Noodlehaus\Config::load($this->getConfDirectory());
+ public function registerIcon(\Sikofitt\Image\Profile\ProfileIconInterface $icon)
+ {
+ $this->config(sprintf('app.icons.%s', $icon->getName()), ['icon' => $icon->getIcon(), 'url' => $icon->getDefaultUrl()]);
+ }
+
+ public function boot() {
+ // register default icons
+ $this->registerIcon(new \Sikofitt\Image\Profile\TwitterProfileIcon());
+ $this->registerIcon(new \Sikofitt\Image\Profile\FacebookProfileIcon());
+ $this->registerIcon(new \Sikofitt\Image\Profile\GithubProfileIcon());
+ $this->registerIcon(new \Sikofitt\Image\Profile\GitlabProfileIcon());
+ $this->registerIcon(new \Sikofitt\Image\Profile\LinkedinProfileIcon());
+ return parent::boot();
}
}
diff --git a/app/views/resume.html.twig b/app/views/resume.html.twig
index 57852cf..f4a8671 100644
--- a/app/views/resume.html.twig
+++ b/app/views/resume.html.twig
@@ -4,64 +4,162 @@
{{ app.config.app.title | default('Resume') }}
{% endblock %}
{% block body %}
- {{ dump(basics) }}
+
+
-
-
- {% if basics.email is not empty %}
- - {{ basics.email }}
- {% endif %}
- {% if basics.phone is not empty %}
- - {{ basics.phone }}
- {% endif %}
- {% if basics.location|length > 0 and basics.location is not empty %}
-
- {% for location in basics.location %}
- {% if location.address is not empty %}
- {{ location.address }}
- {% endif %}
- {% if location.city is not empty %}
- {{ location.city }}
- {% if location.postalCode is not empty %}
- {{ location.postalCode }}
- {% endif %}
- {% if location.countryCode is not empty %}
- {{ location.countryCode }}
- {% endif %}
- {% endif %}
-
- {% endfor %}
- {% endif %}
-
-
-
{% endblock %}
\ No newline at end of file
diff --git a/composer.json b/composer.json
index 457fedb..f39529b 100644
--- a/composer.json
+++ b/composer.json
@@ -38,7 +38,8 @@
"symfony/monolog-bridge": "^3.1",
"google/recaptcha": "^1.1",
"hassankhan/config": "^0.10.0",
- "silex/web-profiler": "^2.0"
+ "silex/web-profiler": "^2.0",
+ "twig/extensions": "^1.3"
},
"require-dev": {
"symfony/debug": "~2.8|^3.0",
diff --git a/composer.lock b/composer.lock
index d851f76..23478f8 100644
--- a/composer.lock
+++ b/composer.lock
@@ -4,8 +4,8 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
"This file is @generated automatically"
],
- "hash": "b95f44a2d8bcbac673a62298f16f4ad9",
- "content-hash": "d0a7cfefe30599f33ddbe16cd3cce33d",
+ "hash": "d47e1d8d6470d4d29deb55649ecc969b",
+ "content-hash": "3c41da3f56c72d6c1bc86d0a1889d6d9",
"packages": [
{
"name": "google/recaptcha",
@@ -2187,6 +2187,58 @@
"homepage": "https://symfony.com",
"time": "2016-06-29 05:41:56"
},
+ {
+ "name": "twig/extensions",
+ "version": "v1.3.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/twigphp/Twig-extensions.git",
+ "reference": "449e3c8a9ffad7c2479c7864557275a32b037499"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/twigphp/Twig-extensions/zipball/449e3c8a9ffad7c2479c7864557275a32b037499",
+ "reference": "449e3c8a9ffad7c2479c7864557275a32b037499",
+ "shasum": ""
+ },
+ "require": {
+ "twig/twig": "~1.20|~2.0"
+ },
+ "require-dev": {
+ "symfony/translation": "~2.3"
+ },
+ "suggest": {
+ "symfony/translation": "Allow the time_diff output to be translated"
+ },
+ "type": "library",
+ "extra": {
+ "branch-alias": {
+ "dev-master": "1.3-dev"
+ }
+ },
+ "autoload": {
+ "psr-0": {
+ "Twig_Extensions_": "lib/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Fabien Potencier",
+ "email": "fabien@symfony.com"
+ }
+ ],
+ "description": "Common additional features for Twig that do not directly belong in core",
+ "homepage": "http://twig.sensiolabs.org/doc/extensions/index.html",
+ "keywords": [
+ "i18n",
+ "text"
+ ],
+ "time": "2015-08-22 16:38:35"
+ },
{
"name": "twig/twig",
"version": "v1.24.1",
diff --git a/data/resume.json b/data/resume.json
index 1707ef6..23d7f63 100644
--- a/data/resume.json
+++ b/data/resume.json
@@ -6,7 +6,7 @@
"email": "eric@ericwheeler.net",
"phone": "",
"website": "https://code.reric.me",
- "summary": "",
+ "summary": "My name is Eric Wheeler. I enjoy programming, working on computer hardware/electronics, music and gardening.",
"location": {
"address": "",
"postalCode": "CA 94578",
@@ -18,7 +18,7 @@
{
"network": "Twitter",
"username": "sikofitt",
- "url": ""
+ "url": "https://twitter.com/sikofitt"
},
{
"network": "Github",
@@ -26,7 +26,7 @@
"url": ""
},
{
- "network": "Private Git",
+ "network": "Gitlab",
"username": "sikofitt",
"url": "https://repos.bgemi.net/u/sikofitt"
},
@@ -43,12 +43,12 @@
"position": "Developer / Systems Administrator",
"website": "https://ee.stanford.edu",
"startDate": "2007-05-27",
- "endDate": "2016-07-02",
- "summary": "",
+ "summary": "I have done many different things during my employment at Stanford University.
A few of things things have been : Desktop Support, Computer cluster administration for the NSF and NNIN using Scyld Clusterware, General programing/debugging and frontend as well as backend web development using a number of different technologies.",
"highlights": [
- "Build an algorithm for artist to detect if their music was violating copy right infringement laws",
- "Successfully won Techcrunch Disrupt",
- "Optimized an algorithm that holds the current world record for Weisman Scores"
+ "Upgraded and Maintained a 64 node research computing cluster for the NNIN funded by the
NSF.",
+ "Converted Electrical Engineering's static web site to Drupal 7.",
+ "Created many Drupal 7 modules for the EE website including a custom events, lecture and committe minutes content type, a CDN module for uploading and recieving images/videos/documents, an Orglist module for displaying Faculty, Students, and Staff from a custom API.",
+ "Created a custom Drupal 7 site for the SystemX organization with integrated login for affiliates as well as Stanford staff."
]
},
{
@@ -80,6 +80,22 @@
"Let's Encrypt"
]
},
+ {
+ "name": "PHP",
+ "level": "",
+ "keywords": [
+ "PHP5",
+ "PHP7",
+ "
Drupal 7",
+ "Symfony 2",
+ "Symfony 3",
+ "OOP",
+ "Silex",
+ "Slim",
+ "Doctrine",
+ "Composer"
+ ]
+ },
{
"name": "PHP",
"level": "",
@@ -95,6 +111,7 @@
"Composer"
]
}
+
],
"languages": [],
"interests": [
diff --git a/src/Sikofitt/Image/Profile/FacebookProfileIcon.php b/src/Sikofitt/Image/Profile/FacebookProfileIcon.php
new file mode 100644
index 0000000..a0b0231
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/FacebookProfileIcon.php
@@ -0,0 +1,31 @@
+
+ *
+ * 7/7/16 / 7:39 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+
+class FacebookProfileIcon implements ProfileIconInterface {
+
+ public function getName() {
+ return 'facebook';
+ }
+
+ public function getDefaultUrl() {
+ return 'https://facebook.com';
+ }
+
+ public function getIcon() {
+ return '';
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Image/Profile/GithubProfileIcon.php b/src/Sikofitt/Image/Profile/GithubProfileIcon.php
new file mode 100644
index 0000000..35269f5
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/GithubProfileIcon.php
@@ -0,0 +1,40 @@
+
+ *
+ * 7/7/16 / 9:18 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+/**
+ * Class GithubProfileIcon
+ * @package Sikofitt\Image\Profile
+ */
+class GithubProfileIcon implements ProfileIconInterface
+{
+ /**
+ * {@inheritdoc}
+ */
+ public function getName() {
+ return 'github';
+ }
+
+ public function getDefaultUrl() {
+ return 'https://github.com';
+ }
+
+ /**
+ * {@inheritdoc}
+ */
+ public function getIcon() {
+ return '';
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Image/Profile/GitlabProfileIcon.php b/src/Sikofitt/Image/Profile/GitlabProfileIcon.php
new file mode 100644
index 0000000..11cad28
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/GitlabProfileIcon.php
@@ -0,0 +1,40 @@
+
+ *
+ * 7/7/16 / 9:36 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+/**
+ * Class GitlabProfileIcon
+ * @package Sikofitt\Image\Profile
+ */
+class GitlabProfileIcon implements ProfileIconInterface {
+
+ /**
+ * {@inheritdoc}
+ */
+ public function getName() {
+ return 'gitlab';
+ }
+
+ public function getDefaultUrl() {
+ return 'https://gitlab.com';
+ }
+
+ /**
+ * {@inheritdoc}
+ */
+ public function getIcon() {
+ return '';
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Image/Profile/LinkedinProfileIcon.php b/src/Sikofitt/Image/Profile/LinkedinProfileIcon.php
new file mode 100644
index 0000000..31cd0c6
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/LinkedinProfileIcon.php
@@ -0,0 +1,31 @@
+
+ *
+ * 7/7/16 / 10:51 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+
+class LinkedinProfileIcon implements ProfileIconInterface {
+
+ public function getName() {
+ return 'linkedin';
+ }
+
+ public function getDefaultUrl() {
+ return 'https://linkedin.com/in';
+ }
+
+ public function getIcon() {
+ return '';
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Image/Profile/ProfileIconInterface.php b/src/Sikofitt/Image/Profile/ProfileIconInterface.php
new file mode 100644
index 0000000..1489c36
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/ProfileIconInterface.php
@@ -0,0 +1,42 @@
+
+ *
+ * 7/7/16 / 7:16 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+/**
+ * Interface ProfileIconInterface
+ * @package Sikofitt\Image\Profile
+ */
+interface ProfileIconInterface {
+
+ /**
+ * @return string
+ * The name of the network.
+ */
+ public function getName();
+
+ /**
+ * @return string
+ * The default url prefix if no url was given.
+ * (e.g. https://twitter.com for twitter, https://facebook.com for facebook.)
+ */
+ public function getDefaultUrl();
+ /**
+ * @return string
+ * The social icon.
+ * Can be a url, a relative path, or base64 encoded uri.
+ */
+ public function getIcon();
+
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Image/Profile/TwitterProfileIcon.php b/src/Sikofitt/Image/Profile/TwitterProfileIcon.php
new file mode 100644
index 0000000..7392aa6
--- /dev/null
+++ b/src/Sikofitt/Image/Profile/TwitterProfileIcon.php
@@ -0,0 +1,32 @@
+
+ *
+ * 7/7/16 / 7:20 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Image\Profile;
+
+
+class TwitterProfileIcon implements ProfileIconInterface {
+
+ public function getName()
+ {
+ return 'twitter';
+ }
+ public function getDefaultUrl() {
+ return 'https://twitter.com';
+ }
+
+ public function getIcon() {
+ return '';
+ }
+
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Twig/Date.php b/src/Sikofitt/Twig/Date.php
new file mode 100644
index 0000000..bb38309
--- /dev/null
+++ b/src/Sikofitt/Twig/Date.php
@@ -0,0 +1,50 @@
+
+ *
+ * 7/7/16 / 6:57 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Twig;
+
+
+class Date extends \Twig_Extension {
+
+ public function getName() {
+ return 'date_diff';
+ }
+
+ /**
+ * @return array
+ * Array of Twig simple filters.
+ */
+ public function getFilters()
+ {
+ return [
+ new \Twig_SimpleFilter('date_diff', [$this, 'dateDiff']),
+ ];
+ }
+
+ /**
+ * Calculates the difference in time.
+ *
+ * @param $startDate
+ * @param string $endDate
+ *
+ * @return \DateTimeInterface
+ */
+ public function dateDiff($startDate, $endDate = 'now')
+ {
+ $dt = new \DateTimeImmutable($startDate);
+ $dateDifference = $dt->diff(new \DateTime($endDate));
+ return $dateDifference->format('%y years');
+
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/Twig/RenderProfile.php b/src/Sikofitt/Twig/RenderProfile.php
new file mode 100644
index 0000000..6eb7b66
--- /dev/null
+++ b/src/Sikofitt/Twig/RenderProfile.php
@@ -0,0 +1,54 @@
+
+ *
+ * 7/7/16 / 9:39 AM
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Sikofitt\Twig;
+
+
+class RenderProfile extends \Twig_Extension {
+
+ public function getName() {
+ return 'render_profile';
+ }
+
+ public function getFunctions() {
+ return [
+ new \Twig_SimpleFunction('render_profile', [$this, 'renderProfile'], ['needs_context' => true]),
+ ];
+ }
+
+ public function renderProfile($context, $iconData)
+ {
+ $imageData = '';
+ //network": "Twitter" +"username": "sikofitt" +"url": ""
+ $icons = $context['app']->config('app.icons');
+
+ $network = strtolower($iconData->network);
+ $haveNetwork = isset($icons[$network]);
+ if($haveNetwork) {
+ $imageData = $icons[strtolower($iconData->network)];
+ } else {
+ return;
+ }
+ if(!isset($iconData->url) || empty($iconData->url))
+ {
+ $iconData->url = $icons[strtolower($iconData->network)]['url'] . '/' . $iconData->username;
+ }
+ $imageUrl = sprintf('
', $imageData['icon'], $iconData->network);
+ if(isset($iconData->url) && !empty($iconData->url)) {
+ return sprintf('
%s', $iconData->url, $iconData->network,$imageUrl);
+ } else {
+ return $imageUrl;
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/Sikofitt/less/resume.less b/src/Sikofitt/less/resume.less
index ef0a943..e202587 100644
--- a/src/Sikofitt/less/resume.less
+++ b/src/Sikofitt/less/resume.less
@@ -2,7 +2,49 @@
@import 'https://fonts.googleapis.com/css?family=Lato:300|Eczar';
@base-body-font-family : 'Lato';
-@base-body-font-size : 26px;
+@base-body-font-size : 16px;
@base-heading-font-family : 'Eczar';
@base-body-line-height : 1.2em;
-@thumbnail-mini-width : 100px;
\ No newline at end of file
+@thumbnail-mini-width : 100px;
+@panel-box-background : #fff;
+
+
+.uk-panel.uk-panel-box {
+ border: 1px solid @base-hr-border;
+ border-radius:2px;
+}
+hr.uk-hr-light {
+ border-top-color:#fafafa;
+}
+
+a.profile-link {
+ filter: grayscale(100%);
+ -webkit-transition: filter 500ms;
+ -moz-transition: filter 500ms;
+ -ms-transition: filter 500ms;
+ -o-transition: filter 500ms;
+ transition: filter 500ms;
+ &:hover {
+ filter: grayscale(0%);
+ }
+}
+
+.uk-panel-image {
+ position: absolute;
+ top: 10px;
+ right: 16px;
+ > img {
+ width:34px;
+ padding: 2px;
+ }
+
+}
+
+.uk-text-lead {
+ font-size: 24px;
+ line-height: 38px;
+ font-weight: normal;
+ margin-bottom: 16px;
+ border-bottom: 1px solid #ccc;
+ padding-bottom: 12px;
+}
\ No newline at end of file
diff --git a/web/index.php b/web/index.php
index 631ee64..001497d 100644
--- a/web/index.php
+++ b/web/index.php
@@ -1,37 +1,40 @@
register(new Sikofitt\Config\ConfigServiceProvider(), [
- 'config.path' => $app->getConfDirectory(),
+ 'config.path' => $app->getConfDirectory(),
]);
-$app->register (new TwigServiceProvider(), [
- 'twig.path' => [APP_ROOT . 'app/views', APP_ROOT .'vendor/symfony/web-profiler-bundle/Resources/views/Profiler'],
+$app->register(new TwigServiceProvider(), [
+ 'twig.path' => [
+ APP_ROOT . 'app/views',
+ APP_ROOT . 'vendor/symfony/web-profiler-bundle/Resources/views/Profiler'
+ ],
]);
-$app->register (new WhoopsServiceProvider());
-$app->register (new \Silex\Provider\AssetServiceProvider());
-$app->register (new \Silex\Provider\MonologServiceProvider());
+$app->register(new WhoopsServiceProvider());
+$app->register(new \Silex\Provider\AssetServiceProvider());
+$app->register(new \Silex\Provider\MonologServiceProvider());
-$app->register (new \Silex\Provider\HttpKernelServiceProvider());
+$app->register(new \Silex\Provider\HttpKernelServiceProvider());
$app->register(new \Silex\Provider\AssetServiceProvider());
$app->register(new \Silex\Provider\FormServiceProvider());
$app->register(new \Silex\Provider\HttpFragmentServiceProvider());
$app->register(new \Silex\Provider\ServiceControllerServiceProvider());
$app->register(new \Silex\Provider\RoutingServiceProvider());
-$app->register (new \Silex\Provider\VarDumperServiceProvider(), array(
+$app->register(new \Silex\Provider\VarDumperServiceProvider(), array(
'var_dumper.dump_destination' => new \Symfony\Component\VarDumper\Cloner\VarCloner(),
));
@@ -41,35 +44,39 @@ $app->register(new \Silex\Provider\MonologServiceProvider(), array(
$app->register(new \Silex\Provider\SessionServiceProvider());
$app->register(new \Silex\Provider\WebProfilerServiceProvider(), array(
- 'profiler.cache_dir' => $app->getDataDirectory() .'/cache/profiler',
+ 'profiler.cache_dir' => $app->getDataDirectory() . '/cache/profiler',
));
-$app->extend('twig', function(\Twig_Environment $twig, $app) {
-$twig->enableDebug();
-return $twig;
+$app->extend('twig', function (\Twig_Environment $twig, $app) {
+ $twig->enableDebug();
+ $twig->addExtension(new Twig_Extensions_Extension_Date());
+ $twig->addExtension(new Sikofitt\Twig\Date());
+ $twig->addExtension(new Sikofitt\Twig\RenderProfile());
+ $twig->addGlobal('config', $app['config']->all());
+ return $twig;
});
-$app->error(function (\Exception $e, $code) use($app) {
- switch ($code) {
- case 404:
- $message = $app['twig']->render('error404.html.twig');
- break;
- default:
- $message = $app['twig']->render('error500.html.twig');
- }
- return new Response($message, $code);
+$app->error(function (\Exception $e, $code) use ($app) {
+ switch ($code) {
+ case 404:
+ $message = $app['twig']->render('error404.html.twig');
+ break;
+ default:
+ $message = $app['twig']->render('error500.html.twig');
+ }
+ return new Response($message, $code);
});
$app['json.decoder'] = function ($app) {
- return new Webmozart\Json\JsonDecoder();
+ return new Webmozart\Json\JsonDecoder();
};
$app['json.encoder'] = function ($app) {
- return new Webmozart\Json\JsonEncoder();
+ return new Webmozart\Json\JsonEncoder();
};
$app['json.validator'] = function ($app) {
- return new Webmozart\Json\JsonValidator();
+ return new Webmozart\Json\JsonValidator();
};
-
-$app->get ('/', function () use ($app) {
- $resumeData = $app['json.decoder']->decodeFile($app->getDataDirectory() .'/resume.json', $app->getDataDirectory() . '/resume.schema.json');
+$app->registerIcon(new \Sikofitt\Image\Profile\TwitterProfileIcon());
+$app->get('/', function () use ($app) {
+ $resumeData = $app['json.decoder']->decodeFile($app->getDataDirectory() . '/resume.json', $app->getDataDirectory() . '/resume.schema.json');
$basics = (isset($resumeData->basics) && count($resumeData->basics) > 0) ? $resumeData->basics : null;
$work = (isset($resumeData->work) && count($resumeData->work) > 0) ? $resumeData->work : null;
$volunteer = (isset($resumeData->volunteer) && count($resumeData->volunteer) > 0) ? $resumeData->volunteer : null;
@@ -80,7 +87,7 @@ $app->get ('/', function () use ($app) {
$languages = (isset($resumeData->languages) && count($resumeData->languages) > 0) ? $resumeData->languages : null;
$interests = (isset($resumeData->interests) && count($resumeData->interests) > 0) ? $resumeData->interests : null;
$references = (isset($resumeData->references) && count($resumeData->references) > 0) ? $resumeData->references : null;
-
+
return $app['twig']->render('resume.html.twig', [
'basics' => $basics,
@@ -96,4 +103,4 @@ $app->get ('/', function () use ($app) {
]);
});
-$app->run ();
+$app->run();