- Twig Templating Engine
- Gulp for SASS compiling, file concatination, image minifying, javascript uglifying, and livereload
- Bower for front-end package management
- Bootstrap
- HTML5 Ready
- Tons of useful functions and theme activation thanks to Roots
- Clone this repo -
git clone [email protected]:kurtschaltzer/twiggy-theme.gitor download the zip file and install it like a normal WordPress theme. - Go to the theme directory and
- Run
sudo npm installornpm install - Run
bower installto install dependencies - Run
gulp devto compile the initial css and js ~ or ~ Rungulpto compile initial css and js and then run watch task ~ or ~ Rungulp buildwhen you're ready for your assets to be concatinated and minified
Twig is a flexible, fast, and secure template engine for PHP. It allows developers to write and structure their themes quickly and understandably.
Here's the WordPress loop in Twig:
{% for post in posts() %} {{ the_post(post) }}
<a href="{{ wp.the_permalink }}">{{ wp.the_title }}</a>
{% endfor %}Twig is loaded in the theme functions in the twigpress.php file in the inc/ directory. After it has been loaded the function twigpress_render_twig_template is available for us to use. If we look at all the top-level theme files you'll notice all they have in theme is that function. What that function does is tell WordPress to look into the twigs/ directory and find the equivalent filename except with .twig as an extension. It then tells WordPress that Twig will handle the rendering of this file.
You call functions like normal except with wp prepended to them like (wp.the_title) due to reasons explained in Caveats section. Functions can be called like normal:
wp.comment_form_title('Leave a Reply', 'Leave a Reply to')Or simply:
wp.comment_form_titleIf you need to give it an array as an argument you can like so:
wp.get_comments({'post_id':wp.get_the_ID,'status':'approve'})This equates to
$args = array(
'post_id' => get_the_ID(),
'status' => 'approve'
);
get_comments($args);Note: Twig can not instantiate objects. So I made a function to do it for you. You can make an object like so:
wp.returnObject('twiggy_Walker_Comment')
{{ wp.wp_list_comments({'walker':wp.returnObject('twiggy_Walker_Comment')}, comments) }}You can access the WordPress loop with the posts() function. The posts() function without any arguments will return the original WordPress loop. Instead of using the_post() like normal you need to pass in the post object so it will setup correctly.
{% for post in posts() %} {{ the_post(post) }}
{{ wp.the_title }}
{% endfor %}Instead of using an empty posts function you can pass in arguments like you would with WP_Query.
{% for post in posts({
'orderby':'name',
'order':'ASC'}) %} {{ the_post(post) }}
{% include 'content/content-excerpt.twig' %}
{% endfor %}You can get Repeater loops like this:
{% for row in wp.get_field('images') %}
<h1>{{ row.title }}</h1>
{{ wp.wp_get_attachment_image(row.image, 'full') }}
{% endfor %}There's always a catch. There are some interesting hacks I had to include in order for Twig to play nice with WordPress.
- All Non-Twig functions must be preceded by 'wp' (e.x. wp.the_title, wp.the_content, etc.). Normally in Twig you'd tell it which functions and variables you'd like to be able to use in the environment, however it would get tedious to add all the WordPress functions to the Twig Loader. So instead I added a proxy function
wpwhich is just a wrapper forcall_user_func_array. - Some WordPress functions don't like to be echoed (e.x. dynamic_sidebar). Instead you can just use Twig's set to not echo but still have the function run (e.x {%
set sidebar = dynamic_sidebar('primary') %}) - Accessing Global Variables. Twig does NOT like accessing global variables which WordPress relies on. Instead you'll have to make the global variables. I've already added two,
wp_queryfor the wp_query global variable andposts()function which returns all the posts necessary for the WordPress loop to work.
+-- dist/ - Distribution/Production files
| +-- fonts/ - Font Files
| +-- img/ - Images (images optimized by gulp in `gulp imagemin`)
+-- inc/ - Various helpful functions and Twig code. All included in function.php
| +-- Twig/ - Twig Engine and init code
| +-- activation.php - Code to run on theme activation
| +-- comments.php - Custom comments walker optimized for Bootstrap
| +-- config.php - Theme configuration options
| +-- extras.php - Some extra functions and important Twig WordPress helper functions
| +-- gallery.php - Cleans up the gallery shortcode and optimizes it for Bootstrap
| +-- init.php - Code to run on theme init
| +-- scripts.php - Scripts queueing
| +-- titles.php - Better titles function (Thanks to _s!)
| +-- twigpress.php - Loader for the Twig Engine
| +-- utils.php - Utility functions
| +-- wp_bootstrap_navwalker.php - Navwalker optimized for Bootstrap
+-- src/ - Development Files
| +-- js/ - Javascript files
| +-- sass/ - Default SASS directory
| | +-- base/ - Basic CSS styles for HTML, Typography, Colors, etc.
| | +-- components/ - WordPress Specific code, tables, buttons, etc.
| | +-- helpers/ - SASS helpers, variables, mixins, paths
| | +-- layout/ - Header, Footer, Navigation, Site, etc.
| | +-- pages/ - Page specific code (home, contact, etc.)
+-- vendor/ - Vendor files, bower installs here
+-- twigs/ - Twig templates go here
| +-- content/ - Main content for the templates
| +-- includes/ - Various includes
| +-- layouts/ - Fundamental layouts of the templates
Install Gulp with npm install -g gulp and Bower with npm install -g bower
gulp dev- Compiles SASS (without minification), concatinates CSS included with Bower (read in Bower section), copies main.jsgulp build- Compiles SASS (with minifcation), concatinates and minifies CSS included with Bower (read in Bower section), copies main.jsgulp watch- Watches src/ and dist/ folders for changes (as well as all PHP and Twig files) and triggers livereload when it detects onegulp imagemin- Minifies images in /dist/imggulp- Runsgulp devthengulp watch
Read more about bower here. Bower installs to the vendor/ directory.
Gulp has a plugin called main-bower-files that can read the main files in each bower install, determining which one you're looking for from that. Most of it should happen automatically as you install Bower packages, however there may be times where you don't want packages included in the vendor.css or vendor.js or you wish to alter the files that are included by default. Here's how to do that.
- Open your bower.json
- Add the "overrides" section like so:
{
"overrides": {
"BOWER-PACKAGE-NAME-GOES-HERE": {
"main": "**/*.js",
ignore": true
}
}
}
- Put in the name of the Bower Package
- main is the name of the Javascript files that are passed to Gulp to be minified, you can edit which one Bower chooses by default
- ignore, if set to true, will set the package to be ignored by Gulp when it looks