Ads
Sunday, January 31, 2021
Mostly Clear today!
With a high of F and a low of 24F. Currently, it's 28F and Clear outside.
Current wind speeds: 8 from the Southwest
Pollen: 0
Sunrise: January 31, 2021 at 07:59PM
Sunset: February 1, 2021 at 06:11AM
UV index: 0
Humidity: 64%
via https://ift.tt/2livfew
February 1, 2021 at 10:01AM
Saturday, January 30, 2021
Clear today!
With a high of F and a low of 25F. Currently, it's 34F and Clear outside.
Current wind speeds: 15 from the Northwest
Pollen: 0
Sunrise: January 30, 2021 at 07:59PM
Sunset: January 31, 2021 at 06:10AM
UV index: 0
Humidity: 71%
via https://ift.tt/2livfew
January 31, 2021 at 10:00AM
Friday, January 29, 2021
Partly Cloudy today!
With a high of F and a low of 32F. Currently, it's 41F and Clear outside.
Current wind speeds: 15 from the Southwest
Pollen: 0
Sunrise: January 29, 2021 at 08:00PM
Sunset: January 30, 2021 at 06:09AM
UV index: 0
Humidity: 47%
via https://ift.tt/2livfew
January 30, 2021 at 10:00AM
Bulletproof flag components
A clever use of CSS grid from Jay Freestone to accomplish a particular variation of the media object design pattern (where the image is centered with the title) without any magic numbers anything that isn’t flexible and resiliant.
The trick is to use an “extra” row above and below the title:
The image goes on the first three rows in the first column, and the content goes in the last three rows in the second column using named grid areas:
grid-template-areas:
'signifier .'
'signifier content'
'signifier content'
'. content';
Read Jay’s post for a little more trickery required to make it entirely resilient.
I love the kind of post that zeroes in on the mental model behind CSS grid like this. It’s like… how can I slice up this design with arbitrary columns and rows, knowing that I can place things on arbitrary rectangular combinations of cells with any type of alignment, to best suit this design?
Direct Link to Article — Permalink
The post Bulletproof flag components appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/3o9gk1u
via IFTTT
Styling Web Components
Nolan Lawson has a little emoji-picker-element that is awfully handy and incredibly easy to use. But considering you’d probably be using it within your own app, it should be style-able so it can incorporated nicely anywhere. How to allow that styling isn’t exactly obvious:
What wasn’t obvious to me, though, was how to allow users to style it. What if they wanted a different background color? What if they wanted the emoji to be bigger? What if they wanted a different font for the input field?
Nolan list four possibilities (I’ll rename them a bit in a way that helps me understand them).
- CSS Custom Properties: Style things like
background: var(--background, white);
. Custom properties penetrate the Shadow DOM, so you’re essentially adding styling hooks. - Pre-built variations: You can add a
class
attribute to the custom elements, which are easy to access within CSS inside the Shadow DOM thanks to the pseudo selectors, like:host(.dark) { background: black; }
. - Shadow parts: You add attributes to things you want to be style-able, like
<span part="foo">
, then CSS from the outside can reach in likecustom-component::part(foo) { }
. - User forced: Despite the nothing in/nothing out vibe of the Shadow DOM, you can always reach the
element.shadowRoot
and inject a<style>
, so there is always a way to get styles in.
It’s probably worth a mention that the DOM you slot
into place is style-able from “outside” CSS as it were.
This is such a funky problem. I like the Shadow DOM because it’s the closest thing we have on the web platform to scoped styles which are definitely a good idea. But I don’t love any of those styling solutions. They all seem to force me into thinking about what kind of styling API I want to offer and document it, while not encouraging any particular consistency across components.
To me, the DOM already is a styling API. I like the scoped protection, but there should be an easy way to reach in there and style things if I want to. Seems like there should be a very simple CSS-only way to reach inside and still use the cascade and such. Maybe the dash-separated custom-element name is enough? my-custom-elemement li { }
. Or maybe it’s more explicit, like @shadow my-custom-element li { }
. I just think it should be easier. Constructable Stylesheets don’t seem like a step toward make it easier, either.
Last time I was thinking about styling web components, I was just trying to figure out how to it works in the first place, not considering how to expose styling options to consumers of the component.
Does this actually come up as a problem in day-to-day work? Sure does.
I don’t see any particularly good options in that thread (yet) for the styling approach. If I was Dave, I’d be tempted to just do nothing. Offer minimal styling, and if people wanna style it, they can do it however they want from their copy of the component. Or they can “force” the styles in, meaning you have complete freedom.
The post Styling Web Components appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/39rBL8X
via IFTTT
Thursday, January 28, 2021
Partly Cloudy today!
With a high of F and a low of 31F. Currently, it's 34F and Clear outside.
Current wind speeds: 10 from the Southwest
Pollen: 0
Sunrise: January 28, 2021 at 08:01PM
Sunset: January 29, 2021 at 06:07AM
UV index: 0
Humidity: 64%
via https://ift.tt/2livfew
January 29, 2021 at 10:00AM
GreenSock ScrollTrigger
High five to the Greensock gang for the ScrollTrigger release. The point of this new plugin is triggering animation when a page scrolls to certain positions, as well as when certain elements are in the viewport. Anything you’d want configurable about it, is. There’s been plenty of scroll-position libraries over the years, but Greensock has a knack for getting the APIs and performance just right — not to mention that because what you want is to trigger animations, now you’ve got Greensock at your fingertips making sure you’re in good hands. It’s tightly integrated with all the other animation possibilities of GSAP (e.g. animating a timeline based on scroll position).
They’ve got docs and a bunch of examples. I particularly like how they have a mistakes section with ways you can screw it up. Every project should do that.
CodePen is full of examples too, so I’ll take the opportunity to drop some here for your viewing pleasure. Note that while this is a paid plugin, you can play with it on CodePen for free (search for it).
If you’re worried about too much motion, that’s something that you can do responsibly through prefers-reduced-motion
, which is available both as a CSS media query and in JavaScript.
The post GreenSock ScrollTrigger appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/3cjRFnz
via IFTTT
A Whole Website in a Single HTML File
I can’t stop thinking about this site. It looks like a pretty standard fare; a website with links to different pages. Nothing to write home about except that… the whole website is contained within a single HTML file.
What about clicking the navigation links, you ask? Each link merely shows and hides certain parts of the HTML.
<section id="home">
<!-- home content goes here -->
</section>
<section id="about">
<!-- about page goes here -->
</section>
Each <section>
is hidden with CSS:
section { display: none; }
Each link in the main navigation points to an anchor on the page:
<a href="#home">Home</a>
<a href="#about">About</a>
And once you click a link, the <section>
for that particular link is displayed via:
section:target { display: block; }
See that :target
pseudo selector? That’s the magic! Sure, it’s been around for years, but this is a clever way to use it for sure. Most times, it’s used to highlight the anchor on the page once an anchor link to it has been clicked. That’s a handy way to help the user know where they’ve just jumped to.
Anyway, using :target
like this is super smart stuff! It ends up looking like just a regular website when you click around:
Direct Link to Article — Permalink
The post A Whole Website in a Single HTML File appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/2J1XoBL
via IFTTT
Mind the gap: E-commerce marketers should revise their TAM and SAM estimates
2021 is going to be another glorious year for e-commerce.
It is that time of the year when most of us are looking back at the “total addressable market” estimates to plan for specific campaigns. Unlike us, if you had your 2021 kick off in Q3, bless your soul. You are an enlightened being.
For the rest of you, for whom e-commerce is a strategic market, I have a question — have you built your total addressable market (TAM) and serviceable addressable market (SAM) estimates for 2021 considering how things evolved in 2020?
It’s important to understand the underlying business model dynamics of companies and visualize TAM from those perspectives.
For most of us, research is a mind-numbing, repetitive exercise of clicking through links on Google until they all turn purple — at which point we start seeking the simplest possible explanation. For e-commerce, addressable market estimates come in the form of headlines from platforms like Shopify. The company quotes a merchant count number in its earnings calls and that becomes the basis for guesstimating the current TAM of e-commerce companies.
The other, rather simplistic approach is to look at the user-base count from several databases that publish tech platform-level user stats.
In reality, the simplest answer is not the right answer.
Mind the gap
Let’s take e-commerce shopping cart installations. Shopify, Magento, WooCommerce, BigCommerce and others publish installation numbers that run into millions.
Here is the dichotomy that should frame your TAM discussions.
E-commerce is long-tail heavy. Yes, there are millions of merchants, but e-commerce revenue is a fat-tail phenomenon — meaning, a disproportionate amount of e-commerce revenue comes from a few tens of thousands of companies.
PipeCandy publishes bottom-up TAM estimates with detailed data cuts by technology, logistics and payment system adoptions by firms across revenue tiers across all major markets. One of the common misconceptions we see in how firms misinterpret TAM estimates is that they equate revenue to spend potential.
from Amazon – TechCrunch https://ift.tt/2MgJ3n0
via IFTTT
Components: Server-Side vs. Client-Side
Building a website in 2021? I’m guessing you’re going to take a component-driven approach. It’s all the chatter these days. React and Vue are everywhere (is Angular still a thing?), while other emerging frameworks continue to attempt a push into the spotlight.
Over the last decade or so we’ve seen an explosion of frameworks and tools that help us build sites systematically using components. Early frameworks like AngularJS helped shape the generic concept of web components. Web components are also reusable bits of HTML code that are written in JavaScript and made functional by the browser. They are client-side components.
But components, in a more generic sense, have actually been around much longer. In fact, they go back to the early days of the web. They just haven’t typically been called components, though they still function as such. Server components are also reusable bits of code, but are compiled into HTML before the browser sees them. They are server-side components, and they are still very much a thing today.
Even in a world in which all it seems like we hear is “React, React, React,” both types of components are still relevant and can help us build super awesome websites. Let’s explore how client and server components differ from one another. That will give us a clearer picture of where we came from. And then we’ll have the information we need to dream about the future.
Rendering
Perhaps the biggest difference between client-side and server-side components is what makes them what they are. That is the thing that is responsible for rendering them.
Server components are rendered by — you guessed it! — the server. They aren’t typically referred to as components. They’re often called partials, includes, snippets, or templates, depending on the framework in which they are used.
Server components can take two flavors. The first is the classic approach, which is to render components in real-time based on a request from the client. See here:
The second flavor is the Jamstack approach. In this case, the entire site is compiled during a build a process, and static HTML is already available when requested by the client. See here:
In both cases, the client (i.e. your browser) never sees the distinction between your components. It simply receives a bunch of HTML from the server.
Client components, on the other hand, are rendered by — you are two-for-two and on a ROLL! — the client. They are written in JavaScript and rendered by the client (your browser). Because the server is the server and it knows all, it can know about your client components, but whether it cares enough to do anything with them depends on the framework you’re using.
Like server components, there are also two flavors of client components. The first is the more official web component, which makes use of the shadow DOM. The shadow DOM helps with encapsulating styles and other functionality (we’ll talk more about this later). Frameworks like Polymer and Stencil make use of the shadow DOM.
The more popular frameworks, like React and Vue, represent the second flavor of component, which handles DOM manipulation and scoping on their own.
Interactivity
Because server components are just HTML when they are sent to the client, if they are to be interactive on the front end, the application must load JavaScript code separately.
Consider a countdown timer. Its presentation is determined by HTML and CSS (we‘ll come back to the CSS part). But if it is to do its thing (count), it also needs some JavaScript. That means not just bringing in that JavaScript, but also having a means by which the JavaScript can attach itself to the countdown’s HTML element(s), which must either be done manually or with (yet) another framework.
Though this may feel unnecessarily tedious (especially if you’ve been around long enough to have been forced into this approach), there are benefits to it. It is a clear separation of concerns, where server-side code lives in one place, while the functionality lives in another. And it brings only the code it needs for the interactivity (theoretically), which can lessen the burden on the browser.
With client components, the markup and interactivity tend to be tightly coupled, often in the same file or directory. While this can quickly become a mess if you’re not diligent about staying organized, one major benefit to client components is that they already have access to the client. And because they are written in JavaScript, their functionality can ship right alongside their markup (and styles).
Performance
In a one-to-one comparison, server-side components tend to perform better. When the page that a browser receives contains everything it needs for presentation, it’s going to be able to deliver that presentation to the user much quicker.
Because client-side components require JavaScript, the browser must download or process additional information (often in separate files) to be able to render the component.
That said, client-side components are often used within the context of a larger framework. React has Gatsby and Next, while Vue has Nuxt. These frameworks have mechanisms for creating a superior in-app experience. What I mean is that, while they may be slower to load the first page you visit on a site, they can then focus their energy on delivering subsequent views extremely fast — often faster than a server-side rendered site can deliver its content.
If you’re thinking, Yeah but what about pre-rendering and…
Yes, you’re right. We’ll get there. Also, no more spoilers, please. The rest of us are along for the ride.
Languages
Server components can be written in (almost) any server-side language. This enables you to write your templates in the same language as your application’s logic. For example, applications written with Ruby on Rails use ERB templating by default, which is a form of Ruby. Thus, Rails apps use the same language for the application itself as it does for its components.
The reason client components are written in JavaScript is because that’s the language browsers parse for interactivity on a website. However, JavaScript also has server-based runtimes, the most popular of which is Node.js. That means code for client components could be written in the same language as the application, as long as the application is written with Node (or similar).
Styling (CSS)
When it comes to styling components, server-side components run into the same trouble they face with JavaScript. The styles are typically detached from the components, and require a bit of extra effort to tie styles to the elements on the page.
However, there are frameworks like Tailwind CSS that are working to make this process less painful.
Many client-side component libraries come with CSS support (or at least a pattern for styling) right out of the box. That often means including the styles in the same file as the markup and logic, which can get messy. But typically, with a little effort, you can adjust that approach to your liking.
Welcome to the (hybrid) future
Neither type of component is the answer by itself. Server-side components require additional effort in styling and interactivity that feels unnecessary when we look at the offerings of client components. But then client components have a tendency to take away from performance on the front end. And because the success of a website often depends on user engagement, a lack of performance can hurt the end result and be enough not to want to use client components.
What does that mean for a future that demands both performance and a good developer experience? More than likely, a hybrid approach.
Components are going to have to be rendered on the server side. They just are. That‘s how we optimize performance, and good performance is going to continue to be an attribute of successful websites. But, now that we’ve seen the ease of front-end logic and interactivity using frameworks, again, like React and Vue, those frameworks are here to stay (at least for awhile).
So where are we going?
I think we’re going to see these components come together in three ways in the very near future.
1. Advancement of JavaScript framework frameworks
Remember when you thought up that spoiler about pre-rendering? Well, let’s talk about it now.
Frameworks like Gatsby, Next, and Nuxt act as front-end engines built on top of component frameworks, like React and Vue. They bring together tooling to build a comprehensive front-end experience using their preferred framework. One such feature is pre-rendering, which means these engines will introspect components and then write static HTML on the page while the site is being built. Then, when users view that page, it‘s actually already there. They don’t need JavaScript to view it.
However, JavaScript comes into play through a process called hydration. After the page loads and your user sees all the (static) content, that’s when JavaScript goes to work. It takes over the components to make them interactive. This provides the opportunity to build a client-side, component-based website with some of the benefits of the server, namely performance and SEO.
These tools have gotten super popular because of this approach, and I suspect we’ll see them continue to advance.
2. Baked-in client-side pre-rendering
That’s a lot of compound words.
What I‘ve been thinking about a lot the last couple years is: Why doesn’t React (or Vue) take on server-side rendering? They do, it’s just not super easy to understand or implement without another framework to help.
On one hand, I understand the single-responsibility principle, and that these component frameworks are just ways to build client-side components. But it felt like a huge miss to delegate server-side rendering to bigger, more complex tools like Gatsby and Next (among others).
Well, React has started moving that way. Vue is already there. And Svelte has made this approach a priority from the beginning.
I think we‘re going to see a lot more development while these traditionally client-side-focused tools solve for server-side rendering. I suspect that also means we‘ll hear a little more from Svelte in the future, which seems like it’s ahead of the game in this regard.
That may also lead to the development of more competitors to bulkier tools like Gatsby and Next. For example, look at what Netlify is doing with their website. It‘s an Eleventy project that pulls in Vue components and renders them for use on the server. What it’s missing is the hydration and interactivity piece. I expect that to come together in the very near future.
3. Server-side component interactivity
And still, we can‘t discount the continued use of server-side components. The one side effect of both of the other two advancements is that they’re still using JavaScript frameworks that can feel unnecessary when you only need just a little interactivity.
There must be a simpler way to add just a little JavaScript to make a server-side component that are written in a server-side language more interactive.
Solving that problem seems to be the approach from the folks at Basecamp, who just released Hotwire, which is a means to bring some of the gains of client components to the server, using (almost) any server-side language.
I don‘t know if that means we‘re going to see competition to Hotwire emerge right away. But I do think Hotwire is going to get some attention. And that might just bring folks back to working with full-stack monolithic frameworks like Rails. (Personally, I love that Rails hasn’t become obsolete in this JavaScript-focused world. The more competition we have, the better the web gets.)
Where do you think all this component business is going? Let’s talk about it.
The post Components: Server-Side vs. Client-Side appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/3cljgou
via IFTTT
Embedding an Interactive Analytics Component with Cumul.io and Any Web Framework
In this article, we explain how to build an integrated and interactive data visualization layer into an application with Cumul.io. To do so, we’ve built a demo application that visualizes Spotify Playlist analytics! We use Cumul.io as our interactive dashboard as it makes integration super easy and provides functionality that allow interaction between the dashboard and applications (i.e. custom events). The app is a simple JavaScript web app with a Node.js server, although you can, if you want, achieve the same with Angular, React and React Native while using Cumul.io dashboards too.
Here, we build dashboards that display data from the The Kaggle Spotify Dataset 1921–2020, 160k+ Tracks and also data via the Spotify Web API when a user logs in. We’ve built dashboards as an insight into playlist and song characteristics. We’ve added some Cumul.io custom events that will allow any end user visiting these dashboards to select songs from a chart and add them to one of their own Spotify playlists. They can also select a song to display more info on them, and play them from within the application. The code for the full application is also publicly available in an open repository.
Here’s a sneak peak into what the end result for the full version looks like:
What are Cumul.io custom events and their capabilities?
Simply put, Cumul.io custom events are a way to trigger events from a dashboard, to be used in the application that the dashboard is integrated in. You can add custom events into selected charts in a dashboard, and have the application listen for these events.
Why? The cool thing about this tool is in how it allows you to reuse data from an analytics dashboard, a BI tool, within the application it’s built into. It gives you the freedom to define actions based on data, that can be triggered straight from within an integrated dashboard, while keeping the dashboard, analytics layer a completely separate entity to the application, that can be managed separately to it.
What they contain: Cumul.io custom events are attached to charts rather than dashboards as a whole. So the information an event has is limited to the information a chart has.
An event is simply put a JSON object. This object will contain fields such as the ID of the dashboard that triggered it, the name of the event and a number of other fields depending on the type of chart that the event was triggered from. For example, if the event was triggered from a scatter plot, you will receive the x-axis and y-axis values of the point it was triggered from. On the other hand, if it were triggered from a table, you would receive column values for example. See examples of what these events will look like from different charts:
// 'Add to Playlist' custom event from a row in a table
{
"type":"customEvent",
"dashboard":"xxxx",
"name":"xxxx",
"object":"xxxx",
"data":{
"language":"en",
"columns":[
{"id":"Ensueno","value":"Ensueno","label":"Name"},
{"id":"Vibrasphere","value":"Vibrasphere","label":"Artist"},
{"value":0.406,"formattedValue":"0.41","label":"Danceability"},
{"value":0.495,"formattedValue":"0.49","label":"Energy"},
{"value":180.05,"formattedValue":"180.05","label":"Tempo (bpm)"},
{"value":0.568,"formattedValue":"0.5680","label":"Accousticness"},
{"id":"2007-01-01T00:00:00.000Z","value":"2007","label":"Release Date (Yr)"},
],
"event":"add_to_playlist"
}
}
//'Song Info' custom event from a point in a scatter plot
{
"type":"customEvent",
"dashboard":"xxxx",
"name":"xxxx",
"object":"xxxx",
"data":{
"language":"en",
"x-axis":{"id":0.601,"value":"0.601","label":"Danceability"},
"y-axis":{"id":0.532,"value":"0.532","label":"Energy"},
"name":{"id":"xxxx","value":"xxx","label":"Name"},
"event":"song_info"
}
}
The possibilities with this functionality are virtually limitless. Granted, depending on what you want to do, you may have to write a couple more lines of code, but it is unarguably quite a powerful tool!
The dashboard
We won’t actually go through the dashboard creation process here and we’ll focus on the interactivity bit once it’s integrated into the application. The dashboards integrated in this walk through have already been created and have custom events enabled. You can, of course create your own ones and integrate those instead of the one we’ve pre-built (you can create an account with a free trial). But before, some background info on Cumul.io dashboards;
Cumul.io offers you a way to create dashboards from within the platform, or via its API. In either case, dashboards will be available within the platform, decoupled from the application you want to integrate it into, so can be maintained completely separately.
On your landing page you’ll see your dashboards and can create a new one:
You can open one and drag and drop any chart you want:
You can connect data which you can then drag and drop into those charts:
And, that data can be one of a number of things. Like a pre-existing database which you can connect to Cumul.io, a dataset from a data warehouse you use, a custom built plugin etc.
Enabling custom events
We have already enabled these custom events to the scatter plot and table in the dashboard used in this demo, which we will be integrating in the next section. If you want to go through this step, feel free to create your own dashboards too!
First thing you need to do will be to add custom events to a chart. To do this, first select a chart in your dashboard you’d like to add an event to. In the chart settings, select Interactivity and turn Custom Events on:
To add an event, click edit and define its Event Name and Label. Event Name is what your application will receive and Label is the one that will show up on your dashboard. In our case, we’ve added 2 events; ‘Add to Playlist’ and ‘Song Info’:
This is all the setup you need for your dashboard to trigger an event on a chart level. Before you leave the editor, you will need your dashboard ID to integrate the dashboard later. You can find this in the Settings tab of your dashboard. The rest of the work remains on application level. This will be where we define what we actually want to do once we receive any of these events.
Takeaway points
- Events work on a chart level and will include information within the limits of the information on the chart
- To add an event, go to the chart settings on the chart you want to add them to
- Define name and label of event. And you’re done!
- (Don’t forget to take note of the dashboard ID for integration)
Using custom events in your own platform
Now that you’ve added some events to the dashboard, the next step is to use them. The key point here is that, once you click an event in your dashboard, your application that integrates the dashboard receives an event. The Integration API provides a function to listen to these events, and then it’s up to you to define what you do with them. For more information on the API and code examples for your SDK, you can also check out the relevant developer docs.
For this section, we’re also providing an open GitHub repository (separate to the repository for the main application) that you can use as a starting project to add custom events to.
The cumulio-spotify-datatalks repository is structured so that you can checkout on the commit called skeleton to start from the beginning. All the following commits will represent a step we go through here. It’s a boiled down version of the full application, focusing on the main parts of the app that demonstrates Custom Events. I’ll be skipping some steps such as the Spotify API calls which are in src/spotify.js
, so as to limit this tutorial to the theme of ‘adding and using custom events’.
Useful info for following steps
- You can use the cumulio-spotify-datatalks repository and checkout on the ‘skeleton’ commit as a starting point.
- All of the changes and code additions will be made in
src/app.js
. - Dependencies and instructions to run are at the end of this post.
Let’s have a look at what happens in our case. We had created two events; add_to_playlist
and song_info
. We want visitors of our dashboard to be able to add a song to their own playlist of choice in their own Spotify account. In order to do so, we take the following steps:
Integrate the dashboard with your app
First, we need to add a dashboard to our application. Here we use the Cumul.io Spotify Playlist dashboard as the main dashboard and the Song Info dashboard as the drill through dashboard (meaning we create a new dashboard within the main one that pops up when we trigger an event). If you have checked out on the commit called skeleton and npm run start
, the application should currently just open up an empty ‘Cumul.io Favorites’ tab, with a Login button at the top right. For instructions on how to locally run the project, go to the bottom of the article:
To integrate a dashboard, we will need to use the Cumulio.addDashboard()
function. This function expects an object with dashboard options. Here’s what we do to add the dashboard:
In src/app.js
, we create an object that stores the dashboard IDs for the main dashboard and the drill through dashboard that displays song info alongside a dashboardOptions
object:
// create dashboards object with the dashboard ids and dashboardOptions object
// !!!change these IDs if you want to use your own dashboards!!!
const dashboards = {
playlist: 'f3555bce-a874-4924-8d08-136169855807',
songInfo: 'e92c869c-2a94-406f-b18f-d691fd627d34',
};
const dashboardOptions = {
dashboardId: dashboards.playlist,
container: '#dashboard-container',
loader: {
background: '#111b31',
spinnerColor: '#f44069',
spinnerBackground: '#0d1425',
fontColor: '#ffffff'
}
};
We create a loadDashboard()
function that calls Cumulio.addDashboard()
. This function optionally receives a container and modifies the dashboardOptions
object before adding dashboard to the application.
// create a loadDashboard() function that expects a dashboard ID and container
const loadDashboard = (id, container) => {
dashboardOptions.dashboardId = id;
dashboardOptions.container = container || '#dashboard-container';
Cumulio.addDashboard(dashboardOptions);
};
Finally, we use this function to add our playlist dashboard when we load the Cumul.io Favorites tab:
export const openPageCumulioFavorites = async () => {
ui.openPage('Cumul.io playlist visualized', 'cumulio-playlist-viz');
/**************** INTEGRATE DASHBOARD ****************/
loadDashboard(dashboards.playlist);
};
At this point, we’ve integrated the playlist dashboard and when we click on a point in the Energy/Danceability by Song scatter plot, we get two options with the custom events we added earlier. However, we’re not doing anything with them yet.
Listen to incoming events
Now that we’ve integrated the dashboard, we can tell our app to do stuff when it receives an event. The two charts that have ‘Add to Playlist’ and ‘Song Info’ events here are:
First, we need to set up our code to listen to incoming events. To do so, we need to use the Cumulio.onCustomEvent()
function. Here, we chose to wrap this function in a listenToEvents()
function that can be called when we load the Cumul.io Favorites tab. We then use if statements to check what event we’ve received:
const listenToEvents = () => {
Cumulio.onCustomEvent((event) => {
if (event.data.event === 'add_to_playlist'){
//DO SOMETHING
}
else if (event.data.event === 'song_info'){
//DO SOMETHING
}
});
};
This is the point after which things are up to your needs and creativity. For example, you could simply print a line out to your console, or design your own behaviour around the data you receive from the event. Or, you could also use some of the helper functions we’ve created that will display a playlist selector to add a song to a playlist, and integrate the Song Info dashboard. This is how we did it;
Add song to playlist
Here, we will make use of the addToPlaylistSelector()
function in src/ui.js
. This function expects a Song Name and ID, and will display a window with all the available playlists of the logged in user. It will then post a Spotify API request to add the song to the selected playlist. As the Spotify Web API requires the ID of a song to be able to add it, we’ve created a derived Name & ID
field to be used in the scatter plot.
An example event we receive on add_to_playlist
will include the following for the scatter plot:
"name":{"id":"So Far To Go&id=3R8CATui5dGU42Ddbc2ixE","value":"So Far To Go&id=3R8CATui5dGU42Ddbc2ixE","label":"Name & ID"}
And these columns for the table:
"columns":[
{"id":"Weapon Of Choice (feat. Bootsy Collins) - Remastered Version","value":"Weapon Of Choice (feat. Bootsy Collins) - Remastered Version","label":"Name"},
{"id":"Fatboy Slim","value":"Fatboy Slim","label":"Artist"},
// ...
{"id":"3qs3aHNUcqFGv7jMYJJCYa","value":"3qs3aHNUcqFGv7jMYJJCYa","label":"ID"}
]
We extract the Name and ID of the song from the event via the getSong()
function, then call the ui.addToPlaylistSelector()
function:
/*********** LISTEN TO CUSTOM EVENTS AND ADD EXTRAS ************/
const getSong = (event) => {
let songName;
let songArtist;
let songId;
if (event.data.columns === undefined) {
songName = event.data.name.id.split('&id=')[0];
songId = event.data.name.id.split('&id=')[1];
}
else {
songName = event.data.columns[0].value;
songArtist = event.data.columns[1].value;
songId = event.data.columns[event.data.columns.length - 1].value;
}
return {id: songId, name: songName, artist: songArtist};
};
const listenToEvents = () => {
Cumulio.onCustomEvent(async (event) => {
const song = getSong(event);
console.log(JSON.stringify(event));
if (event.data.event === 'add_to_playlist'){
await ui.addToPlaylistSelector(song.name, song.id);
}
else if (event.data.event === 'song_info'){
//DO SOMETHING
}
});
};
Now, the ‘Add to Playlist’ event will display a window with the available playlists that a logged in user can add the song to:
Display more song info
The final thing we want to do is to make the ‘Song Info’ event display another dashboard when clicked. It will display further information on the selected song, and include an option to play the song. It’s also the step where we get into more some more complicated use cases of the API which may need some background knowledge. Specifically, we make use of Parameterizable Filters. The idea is to create a parameter on your dashboard, for which the value can be defined while creating an authorization token. We include the parameter as metadata while creating an authorization token.
For this step, we have created a songId
parameter that is used in a filter on the Song Info dashboard:
Then, we create a getDashboardAuthorizationToken()
function. This expects metadata which it then posts to the /authorization
endpoint of our server in server/server.js
:
const getDashboardAuthorizationToken = async (metadata) => {
try {
const body = {};
if (metadata && typeof metadata === 'object') {
Object.keys(metadata).forEach(key => {
body[key] = metadata[key];
});
}
/*
Make the call to the backend API, using the platform user access credentials in the header
to retrieve a dashboard authorization token for this user
*/
const response = await fetch('/authorization', {
method: 'post',
body: JSON.stringify(body),
headers: { 'Content-Type': 'application/json' }
});
// Fetch the JSON result with the Cumul.io Authorization key & token
const responseData = await response.json();
return responseData;
}
catch (e) {
return { error: 'Could not retrieve dashboard authorization token.' };
}
};
Finally, we use the load the songInfo
dashboard when the song_info
event is triggered. In order to do this, we create a new authorization token using the song ID:
const loadDashboard = (id, container, key, token) => {
dashboardOptions.dashboardId = id;
dashboardOptions.container = container || '#dashboard-container';
if (key && token) {
dashboardOptions.key = key;
dashboardOptions.token = token;
}
Cumulio.addDashboard(dashboardOptions);
};
We make some modifications to the loadDashboard()
function so as to use the new token:
const loadDashboard = (id, container, key, token) =u003e {n dashboardOptions.dashboardId = id;n dashboardOptions.container = container || '#dashboard-container'; nn if (key u0026u0026 token) {n dashboardOptions.key = key;n dashboardOptions.token = token;n }nn Cumulio.addDashboard(dashboardOptions);n};
Then call the ui.displaySongInfo()
. The final result looks as follows:
const listenToEvents = () => {
Cumulio.onCustomEvent(async (event) => {
const song = getSong(event);
if (event.data.event === 'add_to_playlist'){
await ui.addToPlaylistSelector(song.name, song.id);
}
else if (event.data.event === 'song_info'){
const token = await getDashboardAuthorizationToken({ songId: [song.id] });
loadDashboard(dashboards.songInfo, '#song-info-dashboard', token.id, token.token);
await ui.displaySongInfo(song);
}
});
};
And voilá! We are done! In this demo we used a lot of helper functions I haven’t gone through in detail, but you are free clone the demo repository and play around with them. You can even disregard them and build your own functionality around the custom events.
Conclusion
For any one intending to have a layer of data visualisation and analytics integrated into their application, Cumul.io provides a pretty easy way of achieving it as I’ve tried to demonstrate throughout this demo. The dashboards remain decoupled entities to the application that can then go on to be managed separately. This becomes quite an advantage if say you’re looking at integrated analytics within a business setting and you’d rather not have developers going back and fiddling with dashboards all the time.
Events you can trigger from dashboards and listen to in their host applications on the other hand allows you to define implementations based off of the information in those decoupled dashboards. This can be anything from playing a song in our case to triggering a specific email to be sent. The world is your oyster in this sense, you decide what to do with the data you have from your analytics layer. In other words, you get to reuse the data from your dashboards, it doesn’t have to just stay there in its dashboard and analytics world 🙂
Steps to run this project
Before you start:
- You will need a Cumul.io account.
- You will need to register your application on the Spotify Developer dashboard
- Clone the cumulio-spotify-datatalks repository with
npm install
- Create a
.env
file in the root directory and add the following from your Cumul.io and Spotify Developer accounts: - From Cumul.io:
CUMULIO_API_KEY=xxx CUMULIO_API_TOKEN=xxx
- From Spotify:
SPOTIFY_CLIENT_ID=xxx SPOTIFY_CLIENT_SECRET=xxx ACCESS_TOKEN=xxx REFRESH_TOKEN=xxxnpm run start
- On your browser, go to
http://localhost:3000/
and log into your Spotify account 🥳
The post Embedding an Interactive Analytics Component with Cumul.io and Any Web Framework appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/3iW0KV9
via IFTTT
Ula raises $20 million to expand its e-commerce marketplace in Indonesia
Tokopedia, Lazada, Shopee, and other firms created an e-commerce market in Indonesia in the past decade, making it possible for consumers to shop online in the island nation. But as is true in other Asian markets, most small retailers and mom-and-pop stores in the Southeast Asian country still face a myriad of challenges in sourcing inventory and working capital, and continue to rely on an age-old supply chain network.
Nipun Mehra (a former executive of Flipkart in India), Alan Wong (who previously worked with Amazon), Derry Sakti (who oversaw consumer goods giant P&G’s operations in Indonesia), and Riky Tenggara (former Lazada and aCommerce), began to explore opportunities to address this in 2019.
“Much like India, much of the Indonesian retail market is unorganized. In the food and vegetable category, for instance, there are lots of farmers who sell to agents, who then sell to markets. From these markets, the inventory goes to small wholesalers, and so on. There are lots of players in the chain,” said Mehra, whose previous stints include working at Sequoia Capital India, in an interview with TechCrunch.
Mehra, Wong, Sakti and Tenggara co-founded Ula in January of 2020. With Ula, they are trying to organize this sourcing and supply chain for small retailers so that there is a one-stop shop for everybody.
Despite the pandemic, Ula made inroads in the Indonesian market last year and today serves more than 20,000 stores. And naturally, investors have noticed.
On Thursday, Ula announced it has raised $20 million in a Series A financing round. The round was led by existing investor Quona Capital and B Capital Group. Other existing investors including Sequoia Capital India and Lightspeed — that financed Ula’s $10.5 million Seed round in June last year — have also participated in the Series A.
“If you look at the whole retail value chain, especially for essential goods, FMCG, staple, and fresh produce, it’s significantly fragmented,” said Ganesh Rengaswamy, Managing Partner at Quona Capital, in an interview. “Whereas the market has moved on in terms of being able to more efficiently consolidate, demand and supply. Ula is trying to redo the retail distribution ecosystem with a significant technology overlay. It’s connecting some of the largest players in the supply side to the smallest retailers and consumers.”
Additionally, Ula is providing these micro retailers, who usually operate from small shops that are extensions of their homes, with working capital so that they don’t have to wait to be paid by their customers to buy the new batch of inventory. (It’s a serious challenge that micro-retailers face in Asian markets. These shops have strong bonds with their customers, so often they sell them items without getting paid upfront. Collecting this payment often takes longer than it should.)
“Frictionless payment and offering credit to retailers so that they can more efficiently manage their cashflow are critical components of modern digital commerce,” said Rengaswamy. For Quona, which has backed several e-commerce and fintech startups in Asia, Ula checks both the boxes.
Mehra said last year was largely about expanding the Ula team and building the technology stack. The startup now plans to deploy the capital to reach more small retailers and expand within the nation.
Indonesia will remain Ula’s focus market. The opportunity in the region itself is very large. The retail spend is expected to surpass $0.5 trillion over the next 4 years, said Kabir Narang, Founding General Partner at B Capital Group, in a statement. Traditional in-store retail accounts for nearly 80% of the total retail market, according to some estimates.
Ula currently operates in the FMCG and food and vegetable spaces, but it intends to broaden its offerings to include apparel and eventually electronics.
A few more things from my notes:
- Like many other startups in Asia, Ula largely relies on feet-and-street sales people to spread the word out about its offerings and onboarding new shops. The key to growing, said Mehra, is to get a few retailers who are very happy with the services and see its value and then tell their friends about it. It’s a learning he credited to Indian business-to-business e-commerce platform Udaan co-founders Amod Malviya, Vaibhav Gupta and Sujeet Kumar, with whom he worked at Flipkart back in the day. Udaan co-founders have backed Ula.
- Electronics is a category that is very popular among B2C and B2B e-commerce platforms. Mehra said he has always known that the startup could expand to electronics, so it has chosen to focus on other categories first that test the supply chain network.
- Indonesia comprises of more than 17,000 islands, but only a handful of islands including Java and Sumatra contributes most to the GDP.
- I asked Quona’s Rengaswamy to draw parallels between e-commerce and payments markets of India and Indonesia. He said India has made more inroads with creating frictionless payments. But on the flip side, this has created potential for startups in Indonesia to solve additional challenges.
from Amazon – TechCrunch https://ift.tt/39pIfFd
via IFTTT
Wednesday, January 27, 2021
Mostly Clear today!
With a high of F and a low of 12F. Currently, it's 14F and Clear outside.
Current wind speeds: 13 from the South
Pollen: 0
Sunrise: January 27, 2021 at 08:02PM
Sunset: January 28, 2021 at 06:06AM
UV index: 0
Humidity: 70%
via https://ift.tt/2livfew
January 28, 2021 at 10:00AM
The Holy Grail Layout with CSS Grid
A reader wrote in asking specifically how to build this layout in CSS Flexbox:
My answer: That’s not really a layout for CSS Flexbox. You could pull it off if you had to, but you’d need some kind of conceit, like grouping the nav and article together in a parent element (if not more grouping). CSS Grid was born to describe this kind of layout and it will be far easier to work with, not to mention that the browser support for both is largely the same these days.
What do you mean by “Holy Grail”?
See, kids, layout on the web used to be so janky that the incredible simple diagram above was relatively difficult to pull off, particularly if you needed the “columns” there to match heights. I know, ridiculous, but that was the deal. We used super weird hacks to get it done (like huge negative margins paired with positive padding), which evolved over time to cleaner tricks (like background images that mimicked columns). Techniques that did manage to pull it off referred to it as the holy grail. (Just for extra clarity, usually, holy grail meant a three-column layout with content in the middle, but the main point was equal height columns).
CSS is much more robust now, so we can use it without resorting to hacks to do reasonable things, like accomplish this basic layout.
Here it is in CSS Grid
This grid is set up both with grid-template-columns
and grid-template-rows
. This way we can be really specific about where we want these major site sections to fall.
I slipped in some extra stuff
- I had another question come my way the other day about doing 1px lines between grid areas. The trick there is as simple as the parent having a background color and using
gap: 1px;
, so I’ve done that in the demo above. - It’s likely that small screens move down to a single-column layout. I’ve done that at a media query above. Sometimes I use
display: block;
on the parent, turning off the grid, but here I’ve leftgrid
on and reset the columns and rows. This way, we still get the gap, and we can shuffle things around if needed. - Another recent question I was asked about is the subtle “body border” effect you can see in the demo above. I did it about as simple as possible, with a smidge of padding between the body and the grid wrapper. I originally did it between the body and the HTML element, but for full-page grids, I think it’s smarter to use a wrapper div than use the body for the grid. That way, third-party things that inject stuff into the body won’t cause layout weirdness.
The post The Holy Grail Layout with CSS Grid appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/2Md8tlv
via IFTTT
Ring’s new video doorbell is $60
The top-line feature for Ring’s latest is no doubt its price. No way around that. At $60, it’s $40 cheaper than the standard Video Doorbell – and prices from there go up significantly, with the “Elite” running $350.
Perhaps the company is feeling some pressure from the race the bottom for smart home hardware pricing. Wyze, notably, has done the Wyze thing, launching a $30 doorbell along with a slew of other products in September. Though as of this writing, that device is still listed as a “pre-order.”
The Wyze device was expected to be available this month, but has since been pushed back to February. The Ring Video Doorbell Wired is also currently slated for next month, with a shipping date of the 24th. As the name suggests, the new product is only available in a hardwired option – which could be a deal breaker for some. Other standard features here include 1080p video with night vision, motion zones that trigger notifications and two-way audio with noise cancelation built-in. It’s also the company’s smallest doorbell to date.
The Amazon-owned company is, of course, not without its share of controversy. Earlier this month, we noted a security flaw that exposed the locations and home addresses of people using its Neighbors app. There has also been plenty of concern around the brand’s willingness to partner with the law enforcement. A number of civil rights penned an open letter in 2019. Earlier this year, Ring finally enabled end-to-end encryption that requires user opt-in.
The new doorbell will be available through Amazon (naturally) and will be a Home Depot in-store exclusively through late-March.
from Amazon – TechCrunch https://ift.tt/3t47x3q
via IFTTT
Tuesday, January 26, 2021
Snow Showers Early today!
With a high of F and a low of 4F. Currently, it's 7F and Snow Shower outside.
Current wind speeds: 3 from the East
Pollen: 0
Sunrise: January 26, 2021 at 08:03PM
Sunset: January 27, 2021 at 06:05AM
UV index: 0
Humidity: 96%
via https://ift.tt/2livfew
January 27, 2021 at 10:00AM
Monorepo
I’m not exactly a large-scale DevOps guy, but I can tell ya we’ve been moving back toward a monorepo at CodePen and it’s rife with advantages over a system with lots of smaller repos. For us, I mean. It’s very likely that you have entirely different challenges and have come to entirely different conclusions at your place. 🤙
I was thinking about this after reading Ben Nadel’s “Why I’ve Been Merging Microservices Back Into The Monolith At InVision.” Even though our conclusions are similar, I can tell he faces an entirely different set of problems.
Microservices Solve Both Technical and People Problems
A technical problem is one in which an aspect of the application is putting an undue burden on the infrastructure; which, in turn, is likely causing a poor user experience (UX). For example, image processing requires a lot of CPU. If this CPU load becomes too great, it could start starving the rest of the application of processing resources. This could affect system latency. And, if it gets bad enough, it could start affecting system availability.
A people problem, on the other hand, has little to do with the application at all and everything to do with how your team is organized. The more people you have working in any given part of the application, the slower and more error-prone development and deployment becomes. For example, if you have 30 engineers all competing to “Continuously Deploy” (CD) the same service, you’re going to get a lot of queuing; which means, a lot of engineers that could otherwise be shipping product are actually sitting around waiting for their turn to deploy.
Advantages of the Monorepo (for us)
- One ring to rule them all. You
git pull
one repo and you are 100% up to date with everyone else and have everything you need for a complete dev environment. - No stray puppies. There is no confusion on where the action happens on GitHub. You do pull requests against the monorepo. You open issues on the monorepo. This avoids scattered activity that gets lost.
- Kumbaya. You can share code. It can be particularly helpful to share utilities or components anywhere in the codebase. We poked at ideas like publishing shared bits to npm for other repos to use, but that workflow was janky compared to having the code together in on place.
- Growing old together. There are no old and neglected repos, because it’s just one. For our small team, having dozens of repos meant some of them had old outdated dependencies, ancient versions of Node, linting and formatting rules that were out of sync with other repos, etc.
Disadvantages of the Monorepo (for us)
- Deployment trickiness. I think the main reason we split off repos originally is that the code in those repos needed to go to unique places. They might have represented an individual Lambda or individual service on some other server. An individual repo means it’s easier to hook up stuff that is unique to that server/service, like CI/CD.
Yes, I get that this is controversial.
I actually don’t care that much. I’m not gonna get all intense about this like air fryer people and CrossFit zealots. Here’s a full-throated argument against monorepos from Matt Klein.
I’m just saying: it’s been clearly useful for us. I can see how things play out differently for other companies. I can see how a company that works with contractors might want to limit their access to something less than an entire monorepo. I can see how a git repo might become unwieldy and large. Those aren’t problems for us at CodePEn right now, so the advantages of a monorepo win.
The post Monorepo appeared first on CSS-Tricks.
You can support CSS-Tricks by being an MVP Supporter.
from CSS-Tricks https://ift.tt/2YgEHPl
via IFTTT
Mostly Clear today!
With a high of F and a low of 15F. Currently, it's 14F and Clear outside. Current wind speeds: 13 from the Southwest Pollen: 0 S...
-
So you want an auto-playing looping video without sound? In popular vernacular this is the very meaning of the word GIF . The word has stuck...
-
With a high of F and a low of 31F. Currently, it's 37F and Cloudy outside. Current wind speeds: 7 from the Northeast Pollen: 0 S...
-
Last year , we kicked out a roundup of published surveys, research, and other findings from around the web. There were some nice nuggets in ...