Skip to content

Instantly share code, notes, and snippets.

@smhmic
Last active March 14, 2016 13:15
Show Gist options
  • Save smhmic/4a97af7ecfe87b55d2a8 to your computer and use it in GitHub Desktop.
Save smhmic/4a97af7ecfe87b55d2a8 to your computer and use it in GitHub Desktop.
HTML-only social share buttons generator w/ utm auto-tagging and bitly shortening
<?php
seer_html_social_share_links( array(
'url' => THE URL TO SHARE, // The URL to share
'text' => '', // Default share text
'image' => IMAGE URL, // URL of the image for the share
//'text_before' => '', // The text to show before the buttons
//'text_after' => '', // The text to show after the buttons
'class' => '', // HTML wrapper class(es)
'_use_the_post' => false, // If true, uses Wordpress functions to populate default options
//'shorten' => false, // Whether or not to use bit.ly to shorten URL
'utm' => array( // UTM tagging parameters
//'source' => '',
//'medium' => 'social',
'campaign' => 'people+asset',
'term' => FOR PEOPLE-SPECIFIC SHARE BUTTONS, SET TO PERSON NAME,
//'content' => '',
),
//'networks_exclude' => array('pinterest'), // Social networks not to provide buttons for
) );
<?php
/* returns the shortened url */
function get_bitly_short_url($url,$login,$appkey,$format='txt') {
$connectURL = 'http://api.bit.ly/v3/shorten?login='.$login.'&apiKey='.$appkey.'&uri='.urlencode($url).'&format='.$format;
return curl_get_result($connectURL);
}
/* returns expanded url */
function get_bitly_long_url($url,$login,$appkey,$format='txt') {
$connectURL = 'http://api.bit.ly/v3/expand?login='.$login.'&apiKey='.$appkey.'&shortUrl='.urlencode($url).'&format='.$format;
return curl_get_result($connectURL);
}
/* returns the response body for a url */
function curl_get_result($url) {
$ch = curl_init();
$timeout = 5;
curl_setopt($ch,CURLOPT_URL,$url);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
curl_setopt($ch,CURLOPT_CONNECTTIMEOUT,$timeout);
$data = curl_exec($ch);
curl_close($ch);
return $data;
}
/* wrapper for get_bitly_short_url(), with caching and leveraging constants:
define('BITLY_USERNAME','');
define('BITLY_API_KEY','');
define('BITLY_CUSTOM_DOMAIN','');
*/
function seer_get_short_url($url) {
$cached = get_option('seer_shorturls'); // Wordpress-specific
$cached = $cached ? $cached : array();
if( empty( $cached[$url] ) ){
$shorturl = get_bitly_short_url($url,BITLY_USERNAME,BITLY_API_KEY);
if( $shorturl && stripos($shorturl,BITLY_CUSTOM_DOMAIN) ){
$cached[$url] = trim($shorturl);
update_option('seer_shorturls', $cached, true ); // Wordpress-specific
}else{
$shorturl = '';
}
}else{
$shorturl = $cached[$url];
}
return $shorturl;
}
function seer_html_social_share_links__tag_url( $network, $opts, $default_campaign = '' ){
if( empty($opts->url) ) return;
if( false === @$opts->utm ) return;
if( true === @$opts->utm_tagged ) return;
switch( $network ){
case 'twitter': $network = 'twitter.com'; break;
case 'facebook': $network = 'facebook.com'; break;
case 'google-plus': $network = 'plus.google.com'; break;
case 'pinterest': $network = 'pinterest.com'; break;
case 'linkedin': $network = 'linkedin.com'; break;
case 'buffer':
// Don't tag for buffer
$network = '';
break;
default:
//TODO: throw error
continue;
}
$req = array(
'source' => $network,
'medium' => 'social',
'campaign' => $default_campaign,
//'term' => '',
//'content' => '',
);
foreach( $req as $k => $default ){
$opts->utm->$k = trim( empty($opts->utm->$k) ? $default : $opts->utm->$k );
if( empty($opts->utm->$k))
return;
}
$opts->orig_url = empty($opts->orig_url) ? $opts->url : $opts->orig_url;
$opts->utm_tagged = true;
preg_match( '|^([^#]+)(#.*)$|', $opts->url, $matches );
if( !empty($matches[2]) ){
$opts->url = $matches[1];
$hash = $matches[2];
}
$glue = strpos($opts->url,'?') ? '&' : '?';
foreach( $opts->utm as $k => $v ){
if( $v ){
$v = strtolower( str_replace(array(' ','&'),array('+','%26'), trim( trim( strip_tags( $v ) ), ':' ) ) );
$opts->url .= $glue."utm_$k=$v";
$glue = '&';
}
}
if( !empty($hash)){
$opts->url .= $hash;
}
}
function seer_html_social_share_links__shorten_url( $opts ){
if( empty($opts->url) ) return;
if( false === @$opts->shorten ) return;
if( true === @$opts->shortened ) return;
$short_url = seer_get_short_url($opts->url);
if( $short_url && 0 === stripos($short_url,'http') ){
$opts->orig_url = empty($opts->orig_url) ? $opts->url : $opts->orig_url;
$opts->url = $short_url;
$opts->shortened = true;
}
}
/**
$opts = (object)array(
'url' => get_the_permalink().'#',
'utm'=>(object)array(
'source' => '',
'medium' => '',
'campaign' => '',
'term' => '',
'content' => '',
)
);
*/
function seer_html_social_share_links__tag_and_shorten_url( $network, &$opts, $default_campaign = '' ){
if( empty($opts->url) ) return;
if( ! empty($opts->{"url_$network"}) ){
$url_bak = $opts->url;
$opts->url = $opts->{"url_$network"};
}else{
#pfx_dump($opts);
$opts->url = empty($opts->orig_url) ? $opts->url : $opts->orig_url;
}
$opts->shortened = null;
$opts->utm_tagged = null;
seer_html_social_share_links__tag_url( $network, $opts, $default_campaign );
#pfx_dump($opts);
seer_html_social_share_links__shorten_url( $opts );
#pfx_dump($opts);
$r = $opts->url;
if( !empty($url_bak) ){
$opts->url = $url_bak;
}
return $r;
}
function seer_html_social_share_links( $opts = array() ){
if( is_string( $opts )) {
$opts = array( 'text' => $opts );
}
$opts = (object)$opts;
if( ! isset( $opts->_use_the_post ) ){
$opts->_use_the_post = true;
}
if( empty( $opts->url ) ){
if( $opts->_use_the_post ){
$opts->url = get_the_permalink();
}
}elseif( substr($opts->url, 0 , 1 ) == '#' ) {
$opts->url = get_the_permalink().$opts->url;
}
if( empty( $opts->text ) && $opts->_use_the_post ){
$opts->text = get_the_title();
}
if( empty( $opts->image ) ){
$opts->image = '';
if( $opts->_use_the_post ){
if( function_exists('get_hero_img') ){
$__img=get_hero_img();
if( $__img && !empty($__img['url']) ){
$opts->image = $__img['url'];
}
}
}
}
if( empty( $opts->networks ) ){
$opts->networks = array(
'facebook',
'twitter',
'google-plus',
'pinterest',
'linkedin',
'buffer'
);
}
if( !empty($opts->networks_exclude)){
$opts->networks = array_diff($opts->networks,$opts->networks_exclude);
}
#$opts->url = esc_attr( urlencode($opts->url));
#$opts->text = esc_attr( urlencode($opts->text));
if( empty($opts->text) ) $opts->text = null;
if( !isset( $opts->shorten ) ){
$opts->shorten = true;
}
if( false !== @$opts->utm ){
$opts->utm = empty($opts->utm) ? (object)array(
'source' => '',
'medium' => '',
'campaign' => '',
'term' => '',
'content' => '',
) : (object)$opts->utm;
if( empty($opts->utm->medium))
$opts->utm->medium = 'social';
if( empty($opts->utm->term)){
$opts->utm->term = array();
if( strpos($opts->url,'#')){
$opts->utm->term[] = 'contextual';
}
if( !empty($opts->text_before)){
$opts->utm->term[] = $opts->text_before;
}
elseif( !empty($opts->text_after)){
$opts->utm->term[] = $opts->text_after;
}
$opts->utm->term = join('+',$opts->utm->term);
}
if( empty($opts->utm->content)){
if( !empty($opts->text)){
$opts->utm->content = $opts->text;
}
}
}
ob_start();
foreach( $opts->networks as $network ){
$_href = '';
$_href_params = (object)array();
$extra_attr = '';
switch( $network ){
case 'twitter':
$network = 'twitter';
$_href = 'http://www.twitter.com/intent/tweet';
if( $opts->url ) $_href_params->url = seer_html_social_share_links__tag_and_shorten_url( $network, $opts ); //A fully-qualified URL with a HTTP or HTTPS scheme, URL-encoded. The provided URL will be shortened with Twitter’s t.co to the number of characters specified by short_url_length.
if( $opts->text ) $_href_params->text = $opts->text; //Pre-populated UTF-8 and URL-encoded Tweet text. The passed text will appear pre-selected for a Twitter user to delete or edit before posting. The length of your passed Tweet text should not exceed 140 characters when combined with any passed hashtags, via, or url parameters. Passed Tweet text which causes the Tweet to exceed 140 characters in length will require additional editing by a Twitter user before he or she can successfully post.
$_href_params->via = 'SeerInteractive'; //A Twitter username to associate with the Tweet, such as your site’s Twitter account. The provided username will be appended to the end of the Tweet with the text “via @username”. A logged-out Twitter user will be encouraged to sign-in or join Twitter to engage with the via account’s Tweets. The account may be suggested as an account to follow after the user posts a Tweet
//$_href_params->hashtags = ''; //Allow easy discovery of Tweets by topic by including a comma-separated list of hashtag values without the preceding # character.
//$_href_params->related = ''; //Suggest additional Twitter usernames related to the Tweet as comma-separated values. Twitter may suggest these accounts to follow after the user posts his or her Tweet. You may provide a brief description of how the account relates to the Tweet with a URL-encoded comma and text after the username
//$_href_params->in-reply-to = ''; //The Tweet ID of a parent Tweet in a conversation, such as the initial Tweet from your site or author account.
break;
case 'facebook':
$network = 'facebook';
$_href = 'http://www.facebook.com/sharer/sharer.php';
if( $opts->url ) $_href_params->u = seer_html_social_share_links__tag_and_shorten_url( $network, $opts );
//TODO
//if( $opts->text ) $_href_params->text = $opts->text;
break;
case 'google-plus':
$network = 'google-plus';
//https://developers.google.com/+/web/share/#sharelink
$_href = 'https://plus.google.com/share';
if( $opts->url ) $_href_params->url = seer_html_social_share_links__tag_and_shorten_url( $network, $opts );
//TODO if( $opts->text ) $_href_params->text = $opts->text;
break;
case 'pinterest':
if( $opts->image )
$_href_params->media = $opts->image;//image url
else
continue; // no point posting to pinterest with no image, right?
$network = 'pinterest';
$_href = 'https://pinterest.com/pin/create/button/';//TODO: or https://pinterest.com/pin/create/bookmarklet/
if( $opts->url ) $_href_params->url = seer_html_social_share_links__tag_and_shorten_url( $network, $opts );
if( $opts->text ) $_href_params->description = $opts->text;
//TODO
// TODO prevent pinit.js from making this into pinit button
//$extra_attr .= ' data-pin-no-hover="true" nopin="nopin"';
//$extra_attr .= ' data-href="'.esc_attr($_href.'?'.http_build_query( $_href_params ) ).'"';
//$_href = 'data';
break;
case 'linkedin':
$network = 'linkedin';
$_href = 'https://www.linkedin.com/shareArticle';
if( $opts->url ) $_href_params->url = seer_html_social_share_links__tag_and_shorten_url( $network, $opts );
$_href_params->mini = 'true'; //A required argument who's value must always be: true
//TODO
$_href_params->source = 'Seer Interactive';
if( $opts->text ) $_href_params->title = $opts->text;
//if( $opts->text ) $_href_params->summary = $opts->text;
break;
case 'buffer':
$network = 'buffer';
$_href = 'http://bufferapp.com/add/';
if( $opts->url ) $_href_params->url = seer_html_social_share_links__tag_and_shorten_url( $network, $opts );
if( $opts->text ) $_href_params->text = $opts->text;
//TODO
//https://github.com/bufferapp/sharejs
//preferred-login //Preferred service to share to or login from
//partner-source // Partner name.
//partner-placement //Partner specified identifier (ie. mini)
$_href_params->via = 'SeerInteractive'; // Via (twitter screen_name)
break;
default:
//TODO: throw error
continue;
}
// moved to main.js
//$onclick = "window.open(this.href,'_blank','toolbar=no,location=no,status=no,menubar=no,scrollbars=yes,resizable=yes,width=760px,height=570px'); if('cancelable' in event && event.cancelable)event.preventDefault();else event.returnValue = false;return event";
$onclick = '';
echo '<a class="sharelink fa fa-'.$network.'" href="'.(esc_attr( $_href.'?'.http_build_query( $_href_params ) )).'" target="_blank"'.($onclick?' onclick="'.esc_attr( $onclick ).'"':'').($extra_attr?' '.trim($extra_attr):'').'></a>';
}
$sharelinks = ob_get_clean();
ob_start();
?>
<aside class="sharelinks<?= empty($opts->class) ?'':' '.$opts->class ?>" data-url="<?=esc_attr($opts->url)?>" data-text="<?=esc_attr($opts->text)?>">
<?php if( !empty($opts->text_before) ){ ?>
<span class="sharelinks-text"><?= $opts->text_before ?></span>
<?php } ?>
<?= $sharelinks ?>
<?php if( !empty($opts->text_after) ){ ?>
<span class="sharelinks-text"><?= $opts->text_after ?></span>
<?php } ?>
</aside>
<?php
return ob_get_clean();
}
@mixin retina() {
@media (-webkit-min-device-pixel-ratio: 1.5), (min--moz-device-pixel-ratio: 1.5), (-o-min-device-pixel-ratio: 1.5/1), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi), (min-resolution: 1.5dppx) {
@content;
}
}
.sharelink {
width: 40px;
height: 40px;
display: block;
background-image: url('');
background-repeat: no-repeat;
background-size: 40px 240px;
transition: opacity 0.2s;
@include retina {
background-image: url('');
}
&:hover {
opacity: 0.7;
}
&.facebook {background-position: 0px 0;}
&.twitter {background-position: 0px -40px;}
&.google-plus {background-position: 0px -80px;}
&.linkedin {background-position: 0px -120px;}
&.pinterest {background-position: 0px -160px;}
&.buffer {background-position: 0px -200px;}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment