Added custom Twig globals, functions. Added Profile icon renderers.

This commit is contained in:
R. Eric Wheeler 2016-07-07 14:50:58 -07:00
parent 85e463c82f
commit f94d82770e
15 changed files with 636 additions and 119 deletions

View File

@ -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();
}
}

View File

@ -4,59 +4,100 @@
{{ app.config.app.title | default('Resume') }}
{% endblock %}
{% block body %}
{{ dump(basics) }}
<div class="uk-grid" data-uk-grid-margin>
<div class="uk-width-1-1">
<h1 class="uk-heading-large" data-uk-sticky="{top:35}">
<h1 class="uk-heading-large">
{% if basics.name is not empty %}
{{ basics.name }}
{% else %}
{{ app.config.app.title|default('Resume') }}
{% endif %}
{% if basics.label is not empty %}
<small class="uk-h2">{{ basics.label }}</small>
<small class="uk-h2 uk-align-right uk-text-bottom">{{ basics.label }}</small>
{% endif %}
</h1>
<hr/>
{% if basics.summary is not empty %}
<p class="uk-text-large">
{{ basics.summary }}
</p>
<p class="uk-text-lead">{{ basics.summary }}</p>
{% endif %}
</div>
</div>
<div class="uk-grid" data-uk-grid-margin>
<div class="uk-width-medium-3-4">
{% set position = work|first %}
{{ dump(position) }}
<p class="resume-positions">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.
<h1>Experience</h1>
<div class="uk-panel uk-panel-header">
<div class="resume-positions">
{% for position in work %}
<h3 class="company uk-panel-title uk-animation-slide-left">
{{ position.company }}
<small class="uk-align-right">
{{ 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 %}
)
</small>
</h3>
<h5>{{ position.position }}</h5>
<p class="summary uk-text-primary uk-text-large">
{{ position.summary|raw }}
</p>
{% if position.highlights is not empty and position.highlights is defined %}
<dl class="uk-description-list-horizontal uk-animation-slide-right">
<dt>Highlights</dt>
{% for highlight in position.highlights %}
<dd class="uk-margin-small-top uk-margin-small-bottom">{{ highlight|raw }}</dd>
{% endfor %}
</dl>
{% endif %}
<hr class="uk-hr-light"/>
{% endfor %}
</div>
</div>
</div>
<div class="uk-width-medium-1-4">
<div class="uk-panel uk-panel-box" data-uk-sticky="{top:35}">
<div class="uk-panel uk-panel-header uk-panel-box" data-uk-sticky="{top:35}">
<div class="uk-panel-header"><h3 class="uk-panel-title">Contact<img class="uk-align-right uk-thumbnail uk-border-circle uk-thumbnail-mini uk-animation-scale-up uk-img-preserve" src="{{ basics.picture }}"/></h3></div>
<!--
--><ul class="uk-list uk-list-line uk-h5">
<div class="uk-panel-image">
<img class="uk-align-right uk-thumbnail uk-border-circle uk-thumbnail-mini uk-animation-scale-up uk-img-preserve"
src="{{ basics.picture }}"/>
</div>
<h3 class="uk-panel-title">Contact</h3>
<ul class="uk-list uk-list-line">
{% if basics.email is not empty %}
<li><a href="#" class="hidden-email">[Click to view email]</a></li>
<li class="uk-list-space"><a href="#" class="hidden-email">[Click to view email]</a></li>
{% endif %}
{% if basics.phone is not empty %}
<li>{{ basics.phone }}</li>
<li class="uk-list-space">{{ basics.phone }}</li>
{% endif %}
{% if basics.website is not empty %}
<li><a href="{{ basics.website }}" target="_blank" title="Home page">{{ basics.website }}</a></li>
<li class="uk-list-space"><a href="{{ basics.website }}" target="_blank"
title="Home page">{{ basics.website }}</a></li>
{% endif %}
{% if basics.location|length > 0 and basics.location is not empty %}
<li>
<li class="uk-list-space">
<address>
{% set location = basics.location %}
@ -64,26 +105,46 @@
{{ location.address }}<br/>
{% endif %}
{% if location.city is not empty %}
{{ location.city }}
{% if location.postalCode is not empty %}
&nbsp;{{ location.postalCode }}
{{ location.city }},
{% endif %}
{% if location.region is not empty %}
&nbsp;{{ location.region }},
{% endif %}
{% if location.countryCode is not empty %}
&nbsp;{{ location.countryCode }}
{% endif %}<br/>
{% endif %}
</address>
</li>
{% endif %}
<li class="uk-list-space">
{% for profile in basics.profiles %}
{{ render_profile(profile)|raw }}
{% endfor %}
</li>
<li class="uk-list-divider"></li>
</ul>
{% if skills is defined and skills is not empty %}
<h3 class="uk-panel-title">Skills</h3>
<dl class="uk-description-list-line">
{% for skill in skills %}
<dt class="uk-text-bold">{{ skill.name }}</dt>
<dd>{{ skill.keywords|join(', ')|raw }}</dd>
{% endfor %}
</dl>
{% endif %}
</div>
</div>
</div>
</div>
<div id="offcanvas" class="uk-offcanvas">
<div class="uk-offcanvas-bar">
<ul class="uk-list uk-list-line">
@ -94,6 +155,7 @@
<li>{{ basics.phone }}</li>
{% endif %}
{% if basics.location|length > 0 and basics.location is not empty %}
<li>
<address>
{% for location in basics.location %}
{% if location.address is not empty %}
@ -108,10 +170,13 @@
&nbsp;{{ location.countryCode }}
{% endif %}<br/>
{% endif %}
{% endfor %}
{% endif %}
</address>
</li>
{% endif %}
<li>
</li>
</ul>
</div>

View File

@ -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",

56
composer.lock generated
View File

@ -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",

View File

@ -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.<br />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 <b>NSF</b>.",
"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",
"<b>Drupal 7</b>",
"Symfony 2",
"Symfony 3",
"OOP",
"Silex",
"Slim",
"Doctrine",
"Composer"
]
},
{
"name": "PHP",
"level": "",
@ -95,6 +111,7 @@
"Composer"
]
}
],
"languages": [],
"interests": [

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,40 @@
<?php
/**
* This file is part of test.
*
* @file GithubProfileIcon.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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 '';
}
}

View File

@ -0,0 +1,40 @@
<?php
/**
* This file is part of test.
*
* @file GitlabProfileIcon.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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 '';
}
}

View File

@ -0,0 +1,31 @@
<?php
/**
* This file is part of test.
*
* @file LinkedinProfileIcon.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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 '';
}
}

View File

@ -0,0 +1,42 @@
<?php
/**
* This file is part of resume.
*
* @file ProfileIconInterface.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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();
}

View File

@ -0,0 +1,32 @@
<?php
/**
* This file is part of resume.
*
* @file TwitterProfileIcon.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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 '';
}
}

View File

@ -0,0 +1,50 @@
<?php
/**
* This file is part of resume.
*
* @file Date.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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');
}
}

View File

@ -0,0 +1,54 @@
<?php
/**
* This file is part of test.
*
* @file Icon.php
*
* R. Eric Wheeler <reric@ee.stanford.edu>
*
* 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('<img src="%s" alt="%s" />', $imageData['icon'], $iconData->network);
if(isset($iconData->url) && !empty($iconData->url)) {
return sprintf('<a class="profile-link" href="%s" title="%s" target="_blank">%s</a>', $iconData->url, $iconData->network,$imageUrl);
} else {
return $imageUrl;
}
}
}

View File

@ -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;
@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;
}

View File

@ -1,12 +1,12 @@
<?php
use Sikofitt\Json\ResumeBuilder;
use Symfony\Component\HttpFoundation\Response;
use Silex\Provider\TwigServiceProvider;
use WhoopsSilex\WhoopsServiceProvider;
use Symfony\Component\HttpKernel\DataCollector\DumpDataCollector;
require_once __DIR__ . '/../vendor/autoload.php';
require_once __DIR__ . '/../app/App.php';
@ -18,7 +18,10 @@ $app->register(new Sikofitt\Config\ConfigServiceProvider(), [
'config.path' => $app->getConfDirectory(),
]);
$app->register(new TwigServiceProvider(), [
'twig.path' => [APP_ROOT . 'app/views', APP_ROOT .'vendor/symfony/web-profiler-bundle/Resources/views/Profiler'],
'twig.path' => [
APP_ROOT . 'app/views',
APP_ROOT . 'vendor/symfony/web-profiler-bundle/Resources/views/Profiler'
],
]);
$app->register(new WhoopsServiceProvider());
@ -45,6 +48,10 @@ $app->register(new \Silex\Provider\WebProfilerServiceProvider(), array(
));
$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) {
@ -67,7 +74,7 @@ $app['json.validator'] = function ($app) {
return new Webmozart\Json\JsonValidator();
};
$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;