forked from mirrors/gecko-dev
		
	Bug 1699515 - Switching to react for Save to Pocket panels. r=gvn
Differential Revision: https://phabricator.services.mozilla.com/D128753
This commit is contained in:
		
							parent
							
								
									1aa25b7d0b
								
							
						
					
					
						commit
						a56ce14062
					
				
					 25 changed files with 1930 additions and 138 deletions
				
			
		|  | @ -42,6 +42,7 @@ browser/components/enterprisepolicies/schemas/schema.jsm | |||
| browser/components/translation/cld2/ | ||||
| # generated or library files in pocket | ||||
| browser/components/pocket/content/panels/js/tmpl.js | ||||
| browser/components/pocket/content/panels/js/main.bundle.js | ||||
| 
 | ||||
| # Ignore newtab files | ||||
| browser/components/newtab/aboutwelcome/content/aboutwelcome.bundle.js | ||||
|  |  | |||
							
								
								
									
										26
									
								
								browser/components/pocket/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								browser/components/pocket/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | |||
| /* This Source Code Form is subject to the terms of the Mozilla Public | ||||
|  * License, v. 2.0. If a copy of the MPL was not distributed with this | ||||
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | ||||
| 
 | ||||
| module.exports = { | ||||
|   plugins: [ | ||||
|     "react", // require("eslint-plugin-react")
 | ||||
|   ], | ||||
|   settings: { | ||||
|     react: { | ||||
|       version: "17.0.2", | ||||
|     }, | ||||
|   }, | ||||
|   overrides: [ | ||||
|     { | ||||
|       files: ["content/**/*.js", "content/**/*.jsx"], | ||||
|       parserOptions: { | ||||
|         sourceType: "module", | ||||
|       }, | ||||
|     }, | ||||
|   ], | ||||
|   rules: { | ||||
|     "react/jsx-uses-react": 2, | ||||
|     "react/jsx-uses-vars": 2, | ||||
|   }, | ||||
| }; | ||||
							
								
								
									
										1
									
								
								browser/components/pocket/.nvmrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								browser/components/pocket/.nvmrc
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | |||
| 12.22.7 | ||||
							
								
								
									
										39
									
								
								browser/components/pocket/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								browser/components/pocket/README.md
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,39 @@ | |||
| # Pocket Integration | ||||
| 
 | ||||
| This is mostly everything related to Pocket in the browser. There are a few exceptions, like newtab, some reader mode integration, and some JSWindowActors, that live in other places. | ||||
| 
 | ||||
| Primarily though this directory includes code for setting up the save to Pocket button, setting up the panels that are used once the Pocket button is clicked, setting up the Pocket context menu item, and a little scaffolding for the reader mode Pocket integration. | ||||
| 
 | ||||
| ## Basic Code Structure | ||||
| 
 | ||||
| We have three primary areas code wise. | ||||
| 
 | ||||
| There are some JSMs that handle communication with the browser. This includes some telemetry, some API functions usable by other parts of the browser, like newtab, and some initialization and setup code. These files live in `/content` | ||||
| 
 | ||||
| There is also some standard js, html, and css that run inside the panels. Panels are the contents inside the drop downs if you click the save to Pocket button. Panels run in their own browser process, and have their own js. This js also has a build/bundle step. These files live in `/content/panels`. We have three panels. There is a sign up panel that is displayed if you click the save to Pocket button while not signed in. There is a saved panel, if you click the save to Pocket button while signed in, and on a page that is savable. Finally there is a home panel, if you click the save to Pocket button while signed in, on a page that is not savable, like about:home. | ||||
| 
 | ||||
| ## Build Panels | ||||
| 
 | ||||
| We use webpack and node to build the panel bundle. So if you change anything in `/content/panels/js` or `/content/panels/css`, you probably need to build the bundle. | ||||
| 
 | ||||
| The build step makes changes to the bundle files, that need to be included in your patch. | ||||
| 
 | ||||
| ### Prerequisites | ||||
| 
 | ||||
| You need node.js installed, and a working local build of Firefox. The current or active version of node is probably fine. At the time of this writing, node version 14 and up is active, and is recommended. | ||||
| 
 | ||||
| ### How to Build | ||||
| 
 | ||||
| From `/browser/components/pocket` | ||||
| 
 | ||||
| If you're making a patch that's ready for review: | ||||
| run `npm install` | ||||
| then `npm run build` | ||||
| 
 | ||||
| For active development instead of `npm run build` use `npm run watch`, which should update bundles as you work. | ||||
| 
 | ||||
| ## React and JSX | ||||
| 
 | ||||
| We use React and JSX for most of the panel html and js. You can find the React components in `/content/panels/js/components`. | ||||
| 
 | ||||
| We are trying to keep the React implementation and dependencies as small as possible. | ||||
|  | @ -13,8 +13,7 @@ | |||
|         <script src="js/vendor/jquery-2.1.1.min.js"></script> | ||||
|         <script src="js/vendor/handlebars.runtime.js"></script> | ||||
|         <script src="js/tmpl.js"></script> | ||||
|         <script src="js/messages.js"></script> | ||||
|         <script src="js/home.js"></script> | ||||
|         <script src="js/main.js"></script> | ||||
|         <script src="js/main.bundle.js"></script> | ||||
|         <script src="js/home/entry.js"></script> | ||||
|     </body> | ||||
| </html> | ||||
|  |  | |||
|  | @ -0,0 +1,33 @@ | |||
| /* This Source Code Form is subject to the terms of the Mozilla Public | ||||
|  * License, v. 2.0. If a copy of the MPL was not distributed with this file, | ||||
|  * You can obtain one at http://mozilla.org/MPL/2.0/. */ | ||||
| 
 | ||||
| import React from "react"; | ||||
| 
 | ||||
| function PopularTopics(props) { | ||||
|   return ( | ||||
|     <> | ||||
|       <h3 data-l10n-id="pocket-panel-home-explore-popular-topics"></h3> | ||||
|       <ul> | ||||
|         {props.topics.map(item => ( | ||||
|           <li key={`item-${item.topic}`}> | ||||
|             <a | ||||
|               className="pkt_ext_topic" | ||||
|               href={`https://${props.pockethost}/explore/${item.topic}?utm_source=${props.utmsource}`} | ||||
|             > | ||||
|               {item.title} | ||||
|               <span className="pkt_ext_chevron_right"></span> | ||||
|             </a> | ||||
|           </li> | ||||
|         ))} | ||||
|       </ul> | ||||
|       <a | ||||
|         className="pkt_ext_discover" | ||||
|         href={`https://${props.pockethost}/explore?utm_source=${props.utmsource}`} | ||||
|         data-l10n-id="pocket-panel-home-discover-more" | ||||
|       /> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export default PopularTopics; | ||||
							
								
								
									
										17
									
								
								browser/components/pocket/content/panels/js/home/entry.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								browser/components/pocket/content/panels/js/home/entry.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | |||
| /* global PKT_PANEL:false */ | ||||
| 
 | ||||
| function onDOMLoaded() { | ||||
|   if (!window.thePKT_PANEL) { | ||||
|     var thePKT_PANEL = new PKT_PANEL(); | ||||
|     /* global thePKT_PANEL */ | ||||
|     window.thePKT_PANEL = thePKT_PANEL; | ||||
|     thePKT_PANEL.initHome(); | ||||
|   } | ||||
|   window.thePKT_PANEL.create(); | ||||
| } | ||||
| 
 | ||||
| if (document.readyState != `loading`) { | ||||
|   onDOMLoaded(); | ||||
| } else { | ||||
|   document.addEventListener(`DOMContentLoaded`, onDOMLoaded); | ||||
| } | ||||
|  | @ -1,13 +1,16 @@ | |||
| /* global Handlebars:false */ | ||||
| /* import-globals-from messages.js */ | ||||
| /* import-globals-from main.js */ | ||||
| 
 | ||||
| /* | ||||
| PKT_PANEL_OVERLAY is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| HomeOverlay is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| It does not contain any logic for saving or communication with the extension or server. | ||||
| */ | ||||
| 
 | ||||
| var PKT_PANEL_OVERLAY = function(options) { | ||||
| import React from "react"; | ||||
| import ReactDOM from "react-dom"; | ||||
| import PopularTopics from "../components/PopularTopics"; | ||||
| import pktPanelMessaging from "../messages.js"; | ||||
| 
 | ||||
| var HomeOverlay = function(options) { | ||||
|   this.inited = false; | ||||
|   this.active = false; | ||||
|   this.pockethost = "getpocket.com"; | ||||
|  | @ -17,15 +20,15 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|   }; | ||||
| 
 | ||||
|   this.setupClickEvents = function() { | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_mylist`), { | ||||
|     pktPanelMessaging.clickHelper(document.querySelector(`.pkt_ext_mylist`), { | ||||
|       source: `home_view_list`, | ||||
|     }); | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_discover`), { | ||||
|     pktPanelMessaging.clickHelper(document.querySelector(`.pkt_ext_discover`), { | ||||
|       source: `home_discover`, | ||||
|     }); | ||||
| 
 | ||||
|     document.querySelectorAll(`.pkt_ext_topic`).forEach((el, position) => { | ||||
|       thePKT_PANEL.clickHelper(el, { | ||||
|       pktPanelMessaging.clickHelper(el, { | ||||
|         source: `home_topic`, | ||||
|         position, | ||||
|       }); | ||||
|  | @ -33,7 +36,7 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|   }; | ||||
| }; | ||||
| 
 | ||||
| PKT_PANEL_OVERLAY.prototype = { | ||||
| HomeOverlay.prototype = { | ||||
|   create() { | ||||
|     var host = window.location.href.match(/pockethost=([\w|\.]*)&?/); | ||||
|     if (host && host.length > 1) { | ||||
|  | @ -73,19 +76,19 @@ PKT_PANEL_OVERLAY.prototype = { | |||
|     // We only have topic pages in English,
 | ||||
|     // so ensure we only show a topics section for English browsers.
 | ||||
|     if (this.locale.startsWith("en")) { | ||||
|       const data = { | ||||
|         pockethost: templateData.pockethost, | ||||
|         utmsource: templateData.utmsource, | ||||
|         topics: [ | ||||
|           { title: "Self Improvement", topic: "self-improvement" }, | ||||
|           { title: "Food", topic: "food" }, | ||||
|           { title: "Entertainment", topic: "entertainment" }, | ||||
|           { title: "Science", topic: "science" }, | ||||
|         ], | ||||
|       }; | ||||
|       document | ||||
|         .querySelector(`.pkt_ext_more`) | ||||
|         .append(this.parseHTML(Handlebars.templates.popular_topics(data))); | ||||
|       ReactDOM.render( | ||||
|         <PopularTopics | ||||
|           pockethost={templateData.pockethost} | ||||
|           utmsource={templateData.utmsource} | ||||
|           topics={[ | ||||
|             { title: "Self Improvement", topic: "self-improvement" }, | ||||
|             { title: "Food", topic: "food" }, | ||||
|             { title: "Entertainment", topic: "entertainment" }, | ||||
|             { title: "Science", topic: "science" }, | ||||
|           ]} | ||||
|         />, | ||||
|         document.querySelector(`.pkt_ext_more`) | ||||
|       ); | ||||
|     } else if (enableLocalizedExploreMore) { | ||||
|       // For non English, we have a slightly different component to the page.
 | ||||
|       document | ||||
|  | @ -100,3 +103,5 @@ PKT_PANEL_OVERLAY.prototype = { | |||
|     pktPanelMessaging.sendMessage("PKT_show_home"); | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
| export default HomeOverlay; | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							|  | @ -0,0 +1,32 @@ | |||
| /* | ||||
| object-assign | ||||
| (c) Sindre Sorhus | ||||
| @license MIT | ||||
| */ | ||||
| 
 | ||||
| /** @license React v0.20.2 | ||||
|  * scheduler.production.min.js | ||||
|  * | ||||
|  * Copyright (c) Facebook, Inc. and its affiliates. | ||||
|  * | ||||
|  * This source code is licensed under the MIT license found in the | ||||
|  * LICENSE file in the root directory of this source tree. | ||||
|  */ | ||||
| 
 | ||||
| /** @license React v17.0.2 | ||||
|  * react-dom.production.min.js | ||||
|  * | ||||
|  * Copyright (c) Facebook, Inc. and its affiliates. | ||||
|  * | ||||
|  * This source code is licensed under the MIT license found in the | ||||
|  * LICENSE file in the root directory of this source tree. | ||||
|  */ | ||||
| 
 | ||||
| /** @license React v17.0.2 | ||||
|  * react.production.min.js | ||||
|  * | ||||
|  * Copyright (c) Facebook, Inc. and its affiliates. | ||||
|  * | ||||
|  * This source code is licensed under the MIT license found in the | ||||
|  * LICENSE file in the root directory of this source tree. | ||||
|  */ | ||||
|  | @ -1,14 +1,31 @@ | |||
| /* global PKT_PANEL_OVERLAY:false */ | ||||
| /* import-globals-from messages.js */ | ||||
| /* This Source Code Form is subject to the terms of the Mozilla Public | ||||
|  * License, v. 2.0. If a copy of the MPL was not distributed with this file, | ||||
|  * You can obtain one at http://mozilla.org/MPL/2.0/. */
 | ||||
| 
 | ||||
| import HomeOverlay from "./home/overlay.js"; | ||||
| import SignupOverlay from "./signup/overlay.js"; | ||||
| import SavedOverlay from "./saved/overlay.js"; | ||||
| import pktPanelMessaging from "./messages.js"; | ||||
| 
 | ||||
| var PKT_PANEL = function() {}; | ||||
| 
 | ||||
| PKT_PANEL.prototype = { | ||||
|   init() { | ||||
|     if (this.inited) { | ||||
|       return; | ||||
|     } | ||||
|     this.overlay = new PKT_PANEL_OVERLAY(); | ||||
|   initHome() { | ||||
|     this.overlay = new HomeOverlay(); | ||||
|     this.init(); | ||||
|   }, | ||||
| 
 | ||||
|   initSignup() { | ||||
|     this.overlay = new SignupOverlay(); | ||||
|     this.init(); | ||||
|   }, | ||||
| 
 | ||||
|   initSaved() { | ||||
|     this.overlay = new SavedOverlay(); | ||||
|     this.init(); | ||||
|   }, | ||||
| 
 | ||||
|   setupObservers() { | ||||
|     this.setupMutationObserver(); | ||||
|     // Mutation observer isn't always enough for fast loading, static pages.
 | ||||
|     // Sometimes the mutation observer fires before the page is totally visible.
 | ||||
|  | @ -17,7 +34,13 @@ PKT_PANEL.prototype = { | |||
|     // So in this case, we have a backup intersection observer that fires when
 | ||||
|     // the page is first visible, and thus, the page is going to guarantee a height.
 | ||||
|     this.setupIntersectionObserver(); | ||||
|   }, | ||||
| 
 | ||||
|   init() { | ||||
|     if (this.inited) { | ||||
|       return; | ||||
|     } | ||||
|     this.setupObservers(); | ||||
|     this.inited = true; | ||||
|   }, | ||||
| 
 | ||||
|  | @ -38,21 +61,6 @@ PKT_PANEL.prototype = { | |||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   // Click helper to reduce bugs caused by oversight
 | ||||
|   // from different implementations of similar code.
 | ||||
|   clickHelper(element, { source = "", position }) { | ||||
|     element?.addEventListener(`click`, event => { | ||||
|       event.preventDefault(); | ||||
| 
 | ||||
|       pktPanelMessaging.sendMessage("PKT_openTabWithUrl", { | ||||
|         url: event.currentTarget.getAttribute(`href`), | ||||
|         activate: true, | ||||
|         source, | ||||
|         position, | ||||
|       }); | ||||
|     }); | ||||
|   }, | ||||
| 
 | ||||
|   setupIntersectionObserver() { | ||||
|     const observer = new IntersectionObserver(entries => { | ||||
|       if (entries.find(e => e.isIntersecting)) { | ||||
|  | @ -97,18 +105,5 @@ PKT_PANEL.prototype = { | |||
|   }, | ||||
| }; | ||||
| 
 | ||||
| function onDOMLoaded() { | ||||
|   if (!window.thePKT_PANEL) { | ||||
|     var thePKT_PANEL = new PKT_PANEL(); | ||||
|     /* global thePKT_PANEL */ | ||||
|     window.thePKT_PANEL = thePKT_PANEL; | ||||
|     thePKT_PANEL.init(); | ||||
|   } | ||||
|   window.thePKT_PANEL.create(); | ||||
| } | ||||
| 
 | ||||
| if (document.readyState != `loading`) { | ||||
|   onDOMLoaded(); | ||||
| } else { | ||||
|   document.addEventListener(`DOMContentLoaded`, onDOMLoaded); | ||||
| } | ||||
| window.PKT_PANEL = PKT_PANEL; | ||||
| window.pktPanelMessaging = pktPanelMessaging; | ||||
|  |  | |||
|  | @ -1,43 +1,51 @@ | |||
| /* global RPMRemoveMessageListener:false, RPMAddMessageListener:false, RPMSendAsyncMessage:false */ | ||||
| 
 | ||||
| var pktPanelMessaging = (function() { | ||||
|   function removeMessageListener(messageId, callback) { | ||||
| var pktPanelMessaging = { | ||||
|   removeMessageListener(messageId, callback) { | ||||
|     RPMRemoveMessageListener(messageId, callback); | ||||
|   } | ||||
|   }, | ||||
| 
 | ||||
|   function addMessageListener(messageId, callback = () => {}) { | ||||
|   addMessageListener(messageId, callback = () => {}) { | ||||
|     RPMAddMessageListener(messageId, callback); | ||||
|   } | ||||
|   }, | ||||
| 
 | ||||
|   function sendMessage(messageId, payload = {}, callback) { | ||||
|   sendMessage(messageId, payload = {}, callback) { | ||||
|     if (callback) { | ||||
|       // If we expect something back, we use RPMSendAsyncMessage and not RPMSendQuery.
 | ||||
|       // Even though RPMSendQuery returns something, our frame could be closed at any moment,
 | ||||
|       // and we don't want to close a RPMSendQuery promise loop unexpectedly.
 | ||||
|       // So instead we setup a response event.
 | ||||
|       const responseMessageId = `${messageId}_response`; | ||||
|       var responseListener = function(responsePayload) { | ||||
|       var responseListener = responsePayload => { | ||||
|         callback(responsePayload); | ||||
|         removeMessageListener(responseMessageId, responseListener); | ||||
|         this.removeMessageListener(responseMessageId, responseListener); | ||||
|       }; | ||||
| 
 | ||||
|       addMessageListener(responseMessageId, responseListener); | ||||
|       this.addMessageListener(responseMessageId, responseListener); | ||||
|     } | ||||
| 
 | ||||
|     // Send message
 | ||||
|     RPMSendAsyncMessage(messageId, payload); | ||||
|   } | ||||
|   }, | ||||
| 
 | ||||
|   function log() { | ||||
|   // Click helper to reduce bugs caused by oversight
 | ||||
|   // from different implementations of similar code.
 | ||||
|   clickHelper(element, { source = "", position }) { | ||||
|     element?.addEventListener(`click`, event => { | ||||
|       event.preventDefault(); | ||||
| 
 | ||||
|       this.sendMessage("PKT_openTabWithUrl", { | ||||
|         url: event.currentTarget.getAttribute(`href`), | ||||
|         activate: true, | ||||
|         source, | ||||
|         position, | ||||
|       }); | ||||
|     }); | ||||
|   }, | ||||
| 
 | ||||
|   log() { | ||||
|     RPMSendAsyncMessage("PKT_log", arguments); | ||||
|   } | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
|   /** | ||||
|    * Public functions | ||||
|    */ | ||||
|   return { | ||||
|     log, | ||||
|     addMessageListener, | ||||
|     sendMessage, | ||||
|   }; | ||||
| })(); | ||||
| export default pktPanelMessaging; | ||||
|  |  | |||
							
								
								
									
										17
									
								
								browser/components/pocket/content/panels/js/saved/entry.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								browser/components/pocket/content/panels/js/saved/entry.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | |||
| /* global PKT_PANEL:false */ | ||||
| 
 | ||||
| function onDOMLoaded() { | ||||
|   if (!window.thePKT_PANEL) { | ||||
|     var thePKT_PANEL = new PKT_PANEL(); | ||||
|     /* global thePKT_PANEL */ | ||||
|     window.thePKT_PANEL = thePKT_PANEL; | ||||
|     thePKT_PANEL.initSaved(); | ||||
|   } | ||||
|   window.thePKT_PANEL.create(); | ||||
| } | ||||
| 
 | ||||
| if (document.readyState != `loading`) { | ||||
|   onDOMLoaded(); | ||||
| } else { | ||||
|   document.addEventListener(`DOMContentLoaded`, onDOMLoaded); | ||||
| } | ||||
|  | @ -1,13 +1,13 @@ | |||
| /* global Handlebars:false */ | ||||
| /* import-globals-from messages.js */ | ||||
| /* import-globals-from main.js */ | ||||
| 
 | ||||
| /* | ||||
| PKT_PANEL_OVERLAY is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| SavedOverlay is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| It does not contain any logic for saving or communication with the extension or server. | ||||
| */ | ||||
| 
 | ||||
| var PKT_PANEL_OVERLAY = function(options) { | ||||
| import pktPanelMessaging from "../messages.js"; | ||||
| 
 | ||||
| var SavedOverlay = function(options) { | ||||
|   var myself = this; | ||||
| 
 | ||||
|   this.inited = false; | ||||
|  | @ -442,9 +442,12 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|       }); | ||||
|   }; | ||||
|   this.initOpenListInput = function() { | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_openpocket`), { | ||||
|       source: `view_list`, | ||||
|     }); | ||||
|     pktPanelMessaging.clickHelper( | ||||
|       document.querySelector(`.pkt_ext_openpocket`), | ||||
|       { | ||||
|         source: `view_list`, | ||||
|       } | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   this.showTagsLocalizedError = function(l10nId) { | ||||
|  | @ -568,9 +571,12 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|         .querySelector(`.pkt_ext_item_recs`) | ||||
|         .append(myself.parseHTML(renderedRecs)); | ||||
| 
 | ||||
|       thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_learn_more`), { | ||||
|         source: `recs_learn_more`, | ||||
|       }); | ||||
|       pktPanelMessaging.clickHelper( | ||||
|         document.querySelector(`.pkt_ext_learn_more`), | ||||
|         { | ||||
|           source: `recs_learn_more`, | ||||
|         } | ||||
|       ); | ||||
| 
 | ||||
|       document | ||||
|         .querySelectorAll(`.pkt_ext_item_recs_link`) | ||||
|  | @ -645,7 +651,7 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|   }; | ||||
| }; | ||||
| 
 | ||||
| PKT_PANEL_OVERLAY.prototype = { | ||||
| SavedOverlay.prototype = { | ||||
|   create() { | ||||
|     if (this.active) { | ||||
|       return; | ||||
|  | @ -746,3 +752,5 @@ PKT_PANEL_OVERLAY.prototype = { | |||
|     pktPanelMessaging.sendMessage("PKT_show_saved"); | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
| export default SavedOverlay; | ||||
							
								
								
									
										17
									
								
								browser/components/pocket/content/panels/js/signup/entry.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								browser/components/pocket/content/panels/js/signup/entry.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | |||
| /* global PKT_PANEL:false */ | ||||
| 
 | ||||
| function onDOMLoaded() { | ||||
|   if (!window.thePKT_PANEL) { | ||||
|     var thePKT_PANEL = new PKT_PANEL(); | ||||
|     /* global thePKT_PANEL */ | ||||
|     window.thePKT_PANEL = thePKT_PANEL; | ||||
|     thePKT_PANEL.initSignup(); | ||||
|   } | ||||
|   window.thePKT_PANEL.create(); | ||||
| } | ||||
| 
 | ||||
| if (document.readyState != `loading`) { | ||||
|   onDOMLoaded(); | ||||
| } else { | ||||
|   document.addEventListener(`DOMContentLoaded`, onDOMLoaded); | ||||
| } | ||||
|  | @ -1,27 +1,33 @@ | |||
| /* global Handlebars:false */ | ||||
| /* import-globals-from messages.js */ | ||||
| /* import-globals-from main.js */ | ||||
| 
 | ||||
| /* | ||||
| PKT_PANEL_OVERLAY is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| SignupOverlay is the view itself and contains all of the methods to manipute the overlay and messaging. | ||||
| It does not contain any logic for saving or communication with the extension or server. | ||||
| */ | ||||
| 
 | ||||
| var PKT_PANEL_OVERLAY = function(options) { | ||||
| import pktPanelMessaging from "../messages.js"; | ||||
| 
 | ||||
| var SignupOverlay = function(options) { | ||||
|   this.inited = false; | ||||
|   this.active = false; | ||||
| 
 | ||||
|   this.setupClickEvents = function() { | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_learnmore`), { | ||||
|       source: `learn_more`, | ||||
|     }); | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.signup-btn-firefox`), { | ||||
|       source: `sign_up_1`, | ||||
|     }); | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.signup-btn-email`), { | ||||
|     pktPanelMessaging.clickHelper( | ||||
|       document.querySelector(`.pkt_ext_learnmore`), | ||||
|       { | ||||
|         source: `learn_more`, | ||||
|       } | ||||
|     ); | ||||
|     pktPanelMessaging.clickHelper( | ||||
|       document.querySelector(`.signup-btn-firefox`), | ||||
|       { | ||||
|         source: `sign_up_1`, | ||||
|       } | ||||
|     ); | ||||
|     pktPanelMessaging.clickHelper(document.querySelector(`.signup-btn-email`), { | ||||
|       source: `sign_up_2`, | ||||
|     }); | ||||
|     thePKT_PANEL.clickHelper(document.querySelector(`.pkt_ext_login`), { | ||||
|     pktPanelMessaging.clickHelper(document.querySelector(`.pkt_ext_login`), { | ||||
|       source: `log_in`, | ||||
|     }); | ||||
|   }; | ||||
|  | @ -71,3 +77,5 @@ var PKT_PANEL_OVERLAY = function(options) { | |||
|     pktPanelMessaging.sendMessage("PKT_show_signup"); | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| export default SignupOverlay; | ||||
|  | @ -14,8 +14,7 @@ | |||
|         <script src="js/vendor/handlebars.runtime.js"></script> | ||||
|         <script src="js/vendor/jquery.tokeninput.min.js"></script> | ||||
|         <script src="js/tmpl.js"></script> | ||||
|         <script src="js/messages.js"></script> | ||||
|         <script src="js/saved.js"></script> | ||||
|         <script src="js/main.js"></script> | ||||
|         <script src="js/main.bundle.js"></script> | ||||
|         <script src="js/saved/entry.js"></script> | ||||
|     </body> | ||||
| </html> | ||||
|  |  | |||
|  | @ -13,8 +13,7 @@ | |||
|     <body class="pkt_ext_containersignup" aria-live="polite"> | ||||
|         <script src="js/vendor/handlebars.runtime.js"></script> | ||||
|         <script src="js/tmpl.js"></script> | ||||
|         <script src="js/messages.js"></script> | ||||
|         <script src="js/signup.js"></script> | ||||
|         <script src="js/main.js"></script> | ||||
|         <script src="js/main.bundle.js"></script> | ||||
|         <script src="js/signup/entry.js"></script> | ||||
|     </body> | ||||
| </html> | ||||
|  |  | |||
|  | @ -1,12 +0,0 @@ | |||
| <h3 data-l10n-id="pocket-panel-home-explore-popular-topics"></h3> | ||||
| <ul> | ||||
|   {{#each topics}} | ||||
|     <li> | ||||
|       <a class="pkt_ext_topic" href="https://{{@root.pockethost}}/explore/{{this.topic}}?utm_source={{@root.utmsource}}"> | ||||
|         {{this.title}} | ||||
|         <span class="pkt_ext_chevron_right"></span> | ||||
|       </a> | ||||
|     </li> | ||||
|   {{/each}} | ||||
| </ul> | ||||
| <a class="pkt_ext_discover" href="https://{{pockethost}}/explore?utm_source={{utmsource}}" data-l10n-id="pocket-panel-home-discover-more"></a> | ||||
|  | @ -4,5 +4,20 @@ | |||
| 
 | ||||
| browser.jar: | ||||
| % content pocket %content/pocket/ contentaccessible=yes | ||||
|   content/pocket/  (content/*) | ||||
| 
 | ||||
|   content/pocket/pktApi.jsm (content/pktApi.jsm) | ||||
|   content/pocket/pktTelemetry.jsm (content/pktTelemetry.jsm) | ||||
|   content/pocket/pktUI.js (content/pktUI.js) | ||||
|   content/pocket/Pocket.jsm (content/Pocket.jsm) | ||||
|   content/pocket/SaveToPocket.jsm (content/SaveToPocket.jsm) | ||||
|   content/pocket/panels/css (content/panels/css/*) | ||||
|   content/pocket/panels/fonts (content/panels/fonts/*) | ||||
|   content/pocket/panels/img (content/panels/img/*) | ||||
|   content/pocket/panels/home.html (content/panels/home.html) | ||||
|   content/pocket/panels/saved.html (content/panels/saved.html) | ||||
|   content/pocket/panels/signup.html (content/panels/signup.html) | ||||
|   content/pocket/panels/js/main.bundle.js (content/panels/js/main.bundle.js) | ||||
|   content/pocket/panels/js/tmpl.js (content/panels/js/tmpl.js) | ||||
|   content/pocket/panels/js/vendor (content/panels/js/vendor/*) | ||||
|   content/pocket/panels/js/home/entry.js (content/panels/js/home/entry.js) | ||||
|   content/pocket/panels/js/saved/entry.js (content/panels/js/saved/entry.js) | ||||
|   content/pocket/panels/js/signup/entry.js (content/panels/js/signup/entry.js) | ||||
|  |  | |||
							
								
								
									
										1543
									
								
								browser/components/pocket/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1543
									
								
								browser/components/pocket/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -6,19 +6,30 @@ | |||
|     "start": "npm run build && npm run watch", | ||||
|     "build": "npx run-p build:*", | ||||
|     "build:handlebars": "node compile_handlebars.js", | ||||
|     "build:webpack": "webpack --config webpack.config.js", | ||||
|     "build:sass": "npx sass content/panels/css/main.scss content/panels/css/main.compiled.css --style=compressed", | ||||
|     "watch": "npx run-p watch:*", | ||||
|     "watch:handlebars": "npx chokidar \"content/panels/tmpl/**/*.handlebars\" -c \"npm run build:handlebars\"", | ||||
|     "watch:webpack": "npm run build:webpack -- --env development -w", | ||||
|     "watch:sass": "npx chokidar \"content/panels/css/**/*.scss\" -c \"npm run build:sass\"", | ||||
|     "test": "echo \"Error: no test specified\" && exit 1" | ||||
|   }, | ||||
|   "author": "Mozilla (https://mozilla.org/)", | ||||
|   "license": "MPL-2.0", | ||||
|   "devDependencies": { | ||||
|     "@babel/core": "7.15.8", | ||||
|     "@babel/preset-react": "7.14.5", | ||||
|     "babel-loader": "8.2.2", | ||||
|     "chokidar-cli": "2.1.0", | ||||
|     "handlebars": "3.0.0", | ||||
|     "npm-run-all": "4.1.5", | ||||
|     "sass": "1.32.8" | ||||
|     "sass": "1.32.8", | ||||
|     "webpack": "5.58.2", | ||||
|     "webpack-cli": "4.9.0" | ||||
|   }, | ||||
|   "repository": "https://hg.mozilla.org/" | ||||
|   "repository": "https://hg.mozilla.org/", | ||||
|   "dependencies": { | ||||
|     "react": "17.0.2", | ||||
|     "react-dom": "17.0.2" | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -7,7 +7,10 @@ function test_runner(test) { | |||
|     // Before each
 | ||||
|     const sandbox = sinon.createSandbox(); | ||||
|     try { | ||||
|       await test({ sandbox, main: testGlobal.window.thePKT_PANEL }); | ||||
|       await test({ | ||||
|         sandbox, | ||||
|         pktPanelMessaging: testGlobal.window.pktPanelMessaging, | ||||
|       }); | ||||
|     } finally { | ||||
|       // After each
 | ||||
|       sandbox.restore(); | ||||
|  | @ -19,24 +22,24 @@ function test_runner(test) { | |||
|   add_task(testTask); | ||||
| } | ||||
| 
 | ||||
| test_runner(async function test_clickHelper({ sandbox, main }) { | ||||
| test_runner(async function test_clickHelper({ sandbox, pktPanelMessaging }) { | ||||
|   // Create a button to test the click helper with.
 | ||||
|   const button = document.createElement("button"); | ||||
|   button.setAttribute("href", "http://example.com"); | ||||
| 
 | ||||
|   // Setup a stub for the click itself.
 | ||||
|   sandbox.stub(testGlobal.pktPanelMessaging, "sendMessage"); | ||||
|   sandbox.stub(pktPanelMessaging, "sendMessage"); | ||||
| 
 | ||||
|   // Create the click helper and trigger the click.
 | ||||
|   main.clickHelper(button, { source: "test-click", position: 2 }); | ||||
|   pktPanelMessaging.clickHelper(button, { source: "test-click", position: 2 }); | ||||
|   button.click(); | ||||
| 
 | ||||
|   Assert.ok( | ||||
|     testGlobal.pktPanelMessaging.sendMessage.calledOnce, | ||||
|     pktPanelMessaging.sendMessage.calledOnce, | ||||
|     "Should fire sendMessage once with clickHelper click" | ||||
|   ); | ||||
|   Assert.ok( | ||||
|     testGlobal.pktPanelMessaging.sendMessage.calledWith("PKT_openTabWithUrl", { | ||||
|     pktPanelMessaging.sendMessage.calledWith("PKT_openTabWithUrl", { | ||||
|       url: "http://example.com", | ||||
|       activate: true, | ||||
|       source: "test-click", | ||||
|  |  | |||
|  | @ -11,10 +11,6 @@ const testGlobal = { | |||
| }; | ||||
| 
 | ||||
| Services.scriptloader.loadSubScript( | ||||
|   "chrome://pocket/content/panels/js/messages.js", | ||||
|   testGlobal | ||||
| ); | ||||
| Services.scriptloader.loadSubScript( | ||||
|   "chrome://pocket/content/panels/js/main.js", | ||||
|   "chrome://pocket/content/panels/js/main.bundle.js", | ||||
|   testGlobal | ||||
| ); | ||||
|  |  | |||
							
								
								
									
										30
									
								
								browser/components/pocket/webpack.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								browser/components/pocket/webpack.config.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,30 @@ | |||
| /* This Source Code Form is subject to the terms of the Mozilla Public | ||||
|  * License, v. 2.0. If a copy of the MPL was not distributed with this file, | ||||
|  * You can obtain one at http://mozilla.org/MPL/2.0/. */
 | ||||
| /* eslint-env node */ | ||||
| 
 | ||||
| module.exports = { | ||||
|   mode: "production", | ||||
|   entry: { | ||||
|     main: "./content/panels/js/main.js", | ||||
|   }, | ||||
|   output: { | ||||
|     filename: "[name].bundle.js", | ||||
|     path: `${__dirname}/content/panels/js/`, | ||||
|   }, | ||||
|   module: { | ||||
|     rules: [ | ||||
|       { | ||||
|         test: /\.jsx?$/, | ||||
|         exclude: /node_modules/, | ||||
|         loader: "babel-loader", | ||||
|         options: { | ||||
|           presets: ["@babel/preset-react"], | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   resolve: { | ||||
|     extensions: [".js", ".jsx"], | ||||
|   }, | ||||
| }; | ||||
		Loading…
	
		Reference in a new issue
	
	 Scott
						Scott