FillInFrom doesn't just take a string - point it at a GPALFile and GPAL treats each line as a token to type, run, and react to in turn. This tutorial searches a site once per line in a text file, using CallAfterFillIn to scrape and save results for each search term.
Here's the whole workflow, start to finish. Each piece is broken down and explained below.
using GenerallyPositive;
using GenerallyPositive.Browser;
using static GenerallyPositive.Enums;
using System;
using System.Diagnostics;
GPALFile inputFile = @"c:datasearch.words.txt";
Selector searchSelector = (Selector)GPAL.Selector
.WithSelectorName("SearchInput")
.WithCSS("#gh-ac")
.WithXPath("//*[@id='gh-ac']")
.MatchPlaceholder("Search for anything");
GPAL.WithExceptionHandler(ExceptionEventHandler)
.WithInformationHandler(ExceptionEventHandler)
.WithTypingDelay(50);
GPAL.Browser
.WithBrowserType(BrowserType.Edge)
.WithDriverLocation(@"c:drivers")
.GoTo("https://www.example.com/")
.WithSelector(searchSelector)
.CallAfterFillIn(CallAfterFillIn)
.FillInFrom(inputFile)
.Close(true);
static IGPALGrid<string> resultGrid;
static CallIfStatus CallAfterFillIn(IBrowser myBrowser, IGPALGrid<string> tokens, int tokenIdx)
{
GPALFile outputFile = $@"c:data{tokens[tokenIdx][0]}.txt";
Selector searchButton = (Selector)GPAL.Selector
.WithSelectorName("SearchButton")
.WithCSS("#gh-btn")
.MatchText("Search");
Selector descriptionColumn = (Selector)GPAL.Selector
.WithSelectorName("Description")
.WithCSS("li .item-title");
Selector priceColumn = (Selector)GPAL.Selector
.WithSelectorName("Price")
.WithCSS("li .item-price");
myBrowser
.WithSelector(searchButton)
.LeftClick()
.WithSelector(descriptionColumn)
.WithSelector(priceColumn)
.WithAllThatMatch()
.GetGrid(out resultGrid)
.SaveToTabbedText(outputFile);
Debug.WriteLine($"File saved to {outputFile.ReturnFilenames[0]}.");
return CallIfStatus.Handled;
}
static void ExceptionEventHandler(object sender, EventArgs e)
{
Debug.WriteLine(((GPAL.GPALEventArgs)e).Message);
Debug.WriteLine(((GPAL.GPALEventArgs)e).ExceptionRaised?.Message);
}
Assigning a path string to a GPALFile gives you a value object, not just a filename. Pass it to FillInFrom and GPAL reads the file, treats each line (or token) as one value to type into the selected element, and runs the rest of the chain once per token.
GPALFile inputFile = @"c:datasearch.words.txt";
Selector searchSelector = (Selector)GPAL.Selector
.WithSelectorName("SearchInput")
.WithCSS("#gh-ac")
.WithXPath("//*[@id='gh-ac']")
.MatchPlaceholder("Search for anything");
FillInFrom can take a literal string, a GPALFile, or (as in the GPALDatabase tutorial) a database query result. GPAL treats all three the same way: a source of tokens to type one at a time.
FillInFrom drives the loop; CallAfterFillIn is your hook into each iteration. tokens[tokenIdx][0] gives you the search term that was just typed, which is used here to name the output file after the search.
GPAL.Browser
.WithBrowserType(BrowserType.Edge)
.WithDriverLocation(@"c:drivers")
.GoTo("https://www.example.com/")
.WithSelector(searchSelector)
.CallAfterFillIn(CallAfterFillIn)
.FillInFrom(inputFile)
.Close(true);
Inside the callback, build a fresh chain: click the search button, collect description and price columns for every matching result with WithAllThatMatch, pull them into a grid, and write that grid to a file named for the current search term.
static CallIfStatus CallAfterFillIn(IBrowser myBrowser, IGPALGrid<string> tokens, int tokenIdx)
{
GPALFile outputFile = $@"c:data{tokens[tokenIdx][0]}.txt";
Selector searchButton = (Selector)GPAL.Selector
.WithSelectorName("SearchButton")
.WithCSS("#gh-btn")
.MatchText("Search");
Selector descriptionColumn = (Selector)GPAL.Selector
.WithSelectorName("Description")
.WithCSS("li .item-title");
Selector priceColumn = (Selector)GPAL.Selector
.WithSelectorName("Price")
.WithCSS("li .item-price");
myBrowser
.WithSelector(searchButton)
.LeftClick()
.WithSelector(descriptionColumn)
.WithSelector(priceColumn)
.WithAllThatMatch()
.GetGrid(out resultGrid)
.SaveToTabbedText(outputFile);
return CallIfStatus.Handled;
}
CallIfStatus.Handled signals that your callback fully processed this iteration. Returning NotHandled lets GPAL fall back to its own default behavior for the token, which is rarely what you want once you've taken over the chain.
Showing off some plain text in these paragraphs eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Lorem ipsum dolor sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Lorem ipsum dolor sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Lorem ipsum dolor sit amet consectetur adipisicing elit. Quo veniam mollitia excepturi animi eum illum non libero sapiente provident assumenda, delectus voluptatum nobis sed dolorem adipisci laudantium incidunt. Error, ratione?
Lorem ipsum dolor sit amet consectetur adipisicing elit. Quo veniam mollitia excepturi animi eum illum non libero sapiente provident assumenda, delectus voluptatum nobis sed dolorem adipisci laudantium incidunt. Error, ratione?
Lorem ipsum dolor sit amet consectetur adipisicing elit. Quo veniam mollitia excepturi animi eum illum non libero sapiente provident assumenda, delectus voluptatum nobis sed dolorem adipisci laudantium incidunt. Error, ratione?
Lorem ipsum dolor sit amet consectetur adipisicing elit. Quo veniam mollitia excepturi animi eum illum non libero sapiente provident assumenda, delectus voluptatum nobis sed dolorem adipisci laudantium incidunt. Error, ratione?
Here you can find different accents and emphasis sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
This is a link and how it could look like bestlinkinthebeautifulworld. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Here's just some classic bold text adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam notBoldSecondbestlinkinthebeautifulworld illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Obcaecati, iste distinctio veritatis eligendi laboriosam adipisicing elit illo nostrum corporis at adipisicing elit libero vel voluptas? Expedita, adipisicing facere dolores voluptatem ad ab rem assumenda soluta!
Other cuple of colors in case we want to emphasize several ways adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam adipisicing elit illo nostrum corporis at voluptatem libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Lorem ipsum dolor sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta! Lorem ipsum dolor, sit amet consectetur adipisicing elit. Quod veniam, quam ad expedita laborum sed at voluptates culpa ipsam ut vel. Ullam temporibus a mollitia quod aliquam ratione exercitationem nesciunt.
Lorem ipsum dolor sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta! Lorem ipsum dolor, sit amet consectetur adipisicing elit. Quod veniam, quam ad expedita laborum sed at voluptates culpa ipsam ut vel. Ullam temporibus a mollitia quod aliquam ratione exercitationem nesciunt.
Lorem ipsum dolor sit amet consectetur adipisicing elit. Obcaecati, iste distinctio veritatis eligendi laboriosam illo nostrum corporis at libero vel voluptas? Expedita, facere dolores voluptatem ad ab rem assumenda soluta!
Lorem ipsum dolor sit amet consectetur adipisicing elit. Repudiandae quas consequuntur illo numquam assumenda autem exercitationem distinctio perspiciatis in natus. Eius dicta similique ipsam ipsa minima, nemo quae enim tempore.
GPAL
.CallIfNotFound(GenericCallIfNotFound)
.WithPublishToConsole();
//System.Drawing.Rectangle windowSize = new System.Drawing.Rectangle(10, 10, 1500, 1024);
// NOTE: we have to set browser = before we execute any steps
// this is due to the 'GenericCallIfNotFound' which might throw an exception, and BankScraper will not have the browser set when it calls scraper.Close()
// until the complete fluent line gets executed (meaning every step, meaning browser is not set until everything else succeeds)
browser = GPAL.Browser
.WithBrowserType(Enums.BrowserType.Chrome)
.WithProfileDataDirectory(ChromeProfileLocation)
.WithUseAutomationEngine(AutomationEngine.Selenium)
.WithWindowSize(new System.Drawing.Rectangle(0,0,1920,1080))
.ToGPALObject();