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.name is not empty %} {{ basics.name }} {% else %} {{ app.config.app.title|default('Resume') }} {% endif %} {% if basics.label is not empty %} - {{ basics.label }} - {% endif %} -

-
- {% if basics.summary is not empty %} -

- - {{ basics.summary }} -

+ {{ basics.label }} {% endif %} + +
+ {% if basics.summary is not empty %} +

{{ basics.summary }}

+ {% endif %} +
- {% set position = work|first %} - {{ dump(position) }} -

Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Aenean commodo ligula eget dolor. - Aenean massa. Cum sociis natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. - Donec quam felis, ultricies nec, pellentesque eu, pretium quis, sem. Nulla consequat massa quis enim. -

+

Experience

+
+
+ {% for position in work %} +

+ {{ position.company }} + + {{ position.startDate|date('Y') }} - + {% if position.endDate is not defined %} + Current + {% else %} + {{ position.endDate|date('Y') }} + {% endif %} + + (~ + {% if position.endDate is not defined %} + {{ position.startDate|date_diff }} + {% else %} + {{ position.startDate|date_diff(position.endDate) }} + {% endif %} + ) + +

+
{{ position.position }}
+

+ {{ position.summary|raw }} +

+ + + {% if position.highlights is not empty and position.highlights is defined %} +
+
Highlights
+ {% for highlight in position.highlights %} +
{{ highlight|raw }}
+ {% endfor %} +
+ {% endif %} +
+ {% endfor %} + +
+
-
+
-

Contact

-
    + +
    + +
    +

    Contact

    + +
      {% if basics.email is not empty %} -
    • [Click to view email]
    • - {% endif %} - {% if basics.phone is not empty %} -
    • {{ basics.phone }}
    • +
    • [Click to view email]
    • + {% endif %} + {% if basics.phone is not empty %} +
    • {{ basics.phone }}
    • {% endif %} {% if basics.website is not empty %} -
    • {{ basics.website }}
    • - {% endif %} +
    • {{ basics.website }}
    • + {% endif %} {% if basics.location|length > 0 and basics.location is not empty %} -
    • -
      - {% set location = basics.location %} +
    • +
      + {% set location = basics.location %} + {% if location.address is not empty %} + {{ location.address }}
      + {% endif %} + {% if location.city is not empty %} + {{ location.city }}, + {% endif %} + {% if location.region is not empty %} +  {{ location.region }}, + {% endif %} + {% if location.countryCode is not empty %} +  {{ location.countryCode }} + {% endif %}
      + +
      +
    • + {% endif %} +
    • + {% for profile in basics.profiles %} + {{ render_profile(profile)|raw }} + {% endfor %} + +
    • +
    • +
    + {% if skills is defined and skills is not empty %} + +

    Skills

    + +
    + {% for skill in skills %} +
    {{ skill.name }}
    +
    {{ skill.keywords|join(', ')|raw }}
    + {% 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('%s', $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();