Skip to content

SEO friendly Vue.js + WordPress Single Page Application Starter Theme. With Hot Module Replacement (HMR) & PHP rendering.

Notifications You must be signed in to change notification settings

vladpedan/wue-theme

ย 
ย 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 

History

17 Commits
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

update 14.06.2019

You'll find the starter theme demo now at starter.wue-theme.app. The main domain wue-theme.app will contain more general content and information and will have additianal features. So I've seperated the demo to not confuse you.

Wue Theme - A SEO friendly Vue.js WordPress single page application starter theme

with all the goodness of both worlds:

- Webpacks Hot Module Replacement (HMR) inside php files!

- PHP rendering

(keep using usual WordPress templates to organise your theme!)

- lightning fast!

(though the more plugins you add and the more data will be safed in the database the more performance optimisation you will have to apply as with all WordPress sites)

- Full SEO support without Nuxt.js and phpv8/v8js! (using Dynamic Rendering)

- Thus keep using the window object in your Vue templates!

How cooooool is this! ๐Ÿ˜Š

- Thus a plug-and-play theme without any effort from the user side

- keep-alive is working! No need for local storage or to use Vuex to handle all the loaded data!

That's actually the reason why I've stopped working on a Nuxt.js implimentation in January 2019. Nuxt.js doesn't support (properly?) the native keep-alive feature of Vue - at least I couldn't get it work and actually a bunch of other people, too. And how Sebastien Chopin treated the requests was totally disapointing to me: nuxt/nuxt#431

Here he's promising to have solved the issue: nuxt/nuxt#1706, but it didn't work for me and it's not documented in the official docs at all. So to me it's quite untransparent how the Nuxt.js team is working / communicating.

Though I am actually thankfull it didn't work with Nuxt.js and I've developed this approach here ๐Ÿ˜Š

How it works:

โ€žConventionalโ€œ Single Page Applications use the headless approach where the backend is reduced to output the JSON data for the frontend app. While this is great for performance it is bad for SEO. So you need to pre-render the app or render it on each request on the server side. The most famous tool for Server Side Rendering (SSR) Vue.js apps is Nuxt.js. But it has some disadvantages. First it requires to remove all the window object references from the Vue components. Then the router has to be adapted to the Nuxt-Workflow. For me the killer argument was that the keep-alive feature doesnโ€™t work (properly) with Nuxt.js. And on the top of this you need to learn one technology more and setup an extra node.js server for serving the rendered files. I don't have any problems to learn new things. Otherwise this theme wouldn't exist :) But why should I use and setup a Node.js server to get a WordPress site work?! No way. In some cases (some supa dupa high tech sites with a bunch of micro services and 50 different backend technologies and a team of 20 people for both backend and frontend - but why are you using WordPress then? ๐Ÿ˜) it might be ok. But not in the most of the WordPress use cases. Especially when you don't have the option to use Node.js on the server (keyword shared hosting) or when it's unaccaptable to bother the client to prerender the app each time they update or create post/page/product. I wish I would have clients who can spend 20.000โ‚ฌ on a WordPress site, then I wouldn't need this theme and go probably with the headless approach. But I have deal with clients who don't have such huge budgets and have to deliver cheap and quickly. One other use case for this theme is if you want to sell plug and play single page application themes. The "headless" dogma is the reason why there is still no official Vue.js Single Page Application Theme to buy. Unitll now ๐Ÿ˜Š

So after the "keep-alive" troubles I've tried it with phpv8/v8js. But after I fought myself through the installation I've found out you have to eliminate all the window object reference from all the node_modules used in your package.json! In my simple app the app.bundle.js has contained over 70 window object references even I didn't use any inside my components! Fuck that shit! ๐Ÿ˜ 

Though you can probably get SSR running with v8js when using Vue.js without any webpack stuff, just as a drop-in-library (did not test it). There might be also easy ways to remove the node_module's window objects - let me know, please if you know some.

Then after this problem has arised and when reading through Vue.js SSR docs I've found there a claim that Google crawls a JavaScript rendered site, but does not wait for the Ajax responses from the server upon the initial App request. So why you not just send the initial data with the HTML as I do all the time when localizing a script in my WordPress themes?! And that's what I did as next. I've added a global window object / variable in the header.php named "technomad" and appended all the data I need at the start of rendering the app by Vue.js. And each template (page.php, index.php or whatever.php) adds more stuff to this object. For example here the header-output.php where I output all the global data regardless the current template/component/route:

<?php
$menus = [];

$menu_locations = get_theme_mod( 'nav_menu_locations' );
asort( $menu_locations );
foreach ( $menu_locations as $menu_name => $menu_id ) {
	if ( ! empty( wp_get_nav_menu_items( $menu_id ) ) ) {

		foreach ( wp_get_nav_menu_items( $menu_id ) as $index => $menu_item ) {

			$menus[ $menu_name ][ $index ]["title"]   = $menu_item->title;
			$menus[ $menu_name ][ $index ]["slug"]    = str_replace( site_url(), "", $menu_item->url );
			$menus[ $menu_name ][ $index ]["classes"] = $menu_item->classes;
		}

	}
}

$header_output = array(
	"host"    => site_url(),
	"bloginfo" => array(
		"name"        => wp_specialchars_decode( get_bloginfo( 'name' ) ),
		"description" => wp_specialchars_decode( get_bloginfo( 'description' ) ),
	),
	"archives" => array(
		"news" => array(
			"excerpt" => get_option( "rss_use_excerpt" ),
			"title"   => get_post( get_option( "page_for_posts" ) )->post_title,
		)
	),
	"nonce"    => wp_create_nonce( 'wp_rest' ),
	"menus"    => $menus,
	"blogpath" => get_post( get_option( 'page_for_posts' ) )->post_name,
);
?>

<script>
    var technomad = <?php
    print_r( json_encode( $header_output ) )
    ?>
</script> 

This way you can continue to use your usual template structure on the PHP side. For any reason it worked indeed. Untill last week ๐Ÿ˜Š Google stopped render the Vue.js app. First I thought it would be related to the Lazy Loading Routes I've implemented in the router.js:

routes: [
        {
            path: '/',
            component: {
                template: "<router-view></router-view>"
            },
            children: [
                {
                    path: '',
                    name: "Homepage",
                    component: () => import('~/components/main/pages/Home.vue')
                },
                {
                    /**
                     *
                     * List Posts
                     *
                     * */
                    path: technomad.blogpath,
                    component: {
                        template: "<router-view></router-view>"
                    },
                    children: [
                        {
                            /**
                             *
                             * Blog Post List
                             *
                             * */
                            path: '',
                            name: "Posts",
                            component: () => import("~/components/main/lists/Posts"),
                        }, {
                            /**
                             *
                             * Single Blog Post
                             *
                             * */
                            path: ':slug',
                            name: "Post",
                            component: () => import("~/components/main/single/Post"),
                        },
                    ]
                },
                {
                    path: ':slug',
                    name: 'Page',
                    component: () => import( '~/components/main/pages/Page.vue')
                }
            ]
        }
    ] 

But after some testing I've found out, it was not the case - google crawlers don't reliably render JavaScript content. Suddenly, after a new page index request from within Google Search Console for the demo site the crawler stopped indexing at the point when everything was loaded but not rendered - for all pages. So I've almost gave up this project. But then after a lot of sleep I could solve the problem by detecting the bots and output a simple markup for them on the php side and load the app only if the useragent is clean of some bot/crawler specific keywords. This approach I was actually pursuing in February of this year until I've read in the Vue.js SSR docs a claim, according to which Google Bots can render synchornious JavaScript: Vue JS docs on rendering JavaScript by Google bots

So now, with dynamic rendiring, the Demo site is back in the top results for "vue.js wordpress theme seo" using browserstack (to disable biased search results) ๐Ÿ˜Š Search results after refactoring

That's actually it. There are no any other "secrets". All the other difficulties (pagenation, comments etc) are not related to the kind of app rendering but will appear in each case regardless whether you use this or a headless approach. I will also provide here some solutions regarding those other difficulties in future. But the main goal of this theme was and is to provide a simple SEO friendly WordPress+Vue.js Plug-And-Play Single Page Application starter theme (sorry, need my SEO keywords ๐Ÿ˜Š ) as there were no proper documentation of my approach at the time I've started to work on it (January 2019). Later in April, when I've finished all the main conceptual work, I've found out I am not the only one: this repo pursue a similar concept and has proper documentation. Though I would consider my theme as one more simple to use. That's all you can find right now on the internet with this approach. Let me know if you know some other PHP served, SEO friendly Vue.js WordPress themes, I will mention them here.

SEO

Dev Environment Setup

  1. download this repo

  2. setup local Apache VirtualHost named dev.wue-theme-public (on Mac):

    • open hosts file:
    sudo vim /etc/hosts
    
    • and add this line
    127.0.0.1       dev.wue-theme-public
    
    • add a new VirtualHost to your Apache config. If you are using Homebrew (which you should as Mojave's Apache doesn't work properly or at least didn't work porperly until recently with Mojave's PHP7.2) then you will find the configs here:
    vim /usr/local/etc/httpd/httpd.conf
    
    • and add thes lines anywhere at the bottom of the file
    <VirtualHost dev.wue-theme-public:80>
       ServerName dev.wue-theme-public
       DocumentRoot "/Users/YourUserName/path/to/project/wue-theme-public/wordpress"
       ServerAdmin your@adminemail.com
       ErrorLog "/Users/YourUserName/path/to/project/wue-theme-public/wordpress/error.log"
       CustomLog "/Users/YourUserName/path/to/project/wue-theme-public/wordpress/access.log" combined
    </VirtualHost>
    
    <VirtualHost dev.wue-theme-public:443>
        ServerAdmin your@adminemail.com
        ServerName dev.wue-theme-public
        DocumentRoot "/Users/YourUserName/path/to/project/wue-theme-public/wordpress"
        ErrorLog "/Users/YourUserName/path/to/project/wue-theme-public/wordpress/error.log"
        CustomLog "/Users/YourUserName/path/to/project/wue-theme-public/wordpress/access.log" combined
        
        #create a self signed certificate on mac:
        #https://ksearch.wordpress.com/2017/08/22/generate-and-import-a-self-signed-ssl-certificate-on-mac-osx-sierra/
        #if you don't need SSL for any reason you will have to search-replace the database via wp-cli or using a plugin like wp migrate db 
        https://dev.wue-theme-public by http://dev.wue-theme-public 
        SSLEngine on
        SSLCertificateFile "/private/etc/apache2/server.crt"
        SSLCertificateKeyFile "/private/etc/apache2/server.key"
    </VirtualHost>
    
  3. rename wp-config-sample.php to wp-config.php 3.1 add "salts" wp-config.php

  4. create a database named wue_theme_public and a database user wuethemepublic (or just change the db credentials in the wp-config.php):

    mysql -uroot -p 
    CREATE DATABASE wue_theme_public CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
    CREATE USER wuethemepublic@localhost IDENTIFIED BY 'cornhulio';
    GRANT ALL PRIVILEGES ON wue_theme_public . * TO wuethemepublic@localhost;
    
  5. import database into your MySQL server:

    cd /Users/YourUserName/path/to/project/wue-theme-public
    mysql -uwuethemepublic -pcornhulio wue_theme_public < database.sql
    
  6. Install dev node modules (I assume you have Node.js installed):

    cd /Users/YourUserName/path/to/project/wue-theme-public
    npm install
    
  7. start dev server:

    cd /Users/YourUserName/path/to/project/wue-theme-public
    npm run start    
    
    • if port 9200 is already in use change it to any other in the webpack.dev.js
  8. Login into WordPress Backend:

    ๐Ÿ˜

Now everything should run as in the demo

You can delete the port number in the browser address bar so the main requests are handeled by Apache. Express.js writes the app assets and hot module replacement chunks physically (as configured in the webpack.dev.js) to the wue-public-theme/app directory so Apache can access them without any proxieng

If you see after "npm run start" a blank screen and the Wue logo, then just reload the page and then everything should work, if not then something went wrong :)

Generate production app assets:

  1. just simply run:
    cd /Users/YourUserName/path/to/project/wue-theme-public
    npm run build
    

Use the theme only

  1. download the repo and use wuetheme.zip to install Wue Theme as a regular theme.

    Though there is no data within the theme. You would have to setup the menus, the permalinks, the pages and other settings by yourself or just import the database.sql via PhpMyAdmin

Permalinks

You have to sync the route structure of Vue Router and of WordPress. For News it's easy - you just have to set the permalinks for blog posts to custom structure and prepend "/news/" to the permalink in the settings:

/news/%postname%/    

For custom post types you might have to rewrite rules in the themes functions.php using the add_rewrite_rule() function.

More documentation is coming soon. Meanwhile you can check my other repo where the webpack configs are commented properly.

Roadmap

  • Youtube tutorial
  • Proper documentation and comments within the code

About

SEO friendly Vue.js + WordPress Single Page Application Starter Theme. With Hot Module Replacement (HMR) & PHP rendering.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • PHP 66.6%
  • JavaScript 25.3%
  • CSS 7.9%
  • Other 0.2%